invisiblelines/security_identifiers

View on GitHub
lib/security_identifiers/cusip.rb

Summary

Maintainability
A
20 mins
Test Coverage
module SecurityIdentifiers
  class CUSIP < Base
    SYMBOLS = {
      '*' => [3, 6],
      '@' => [3, 7],
      '#' => [3, 8]
    }.freeze

    def initialize(str)
      raise InvalidFormat if str.nil?

      match_data = str.upcase.match(/^([A-Z0-9\*@#]{8})(\d{1})?$/)

      raise InvalidFormat if match_data.nil?

      @identifier, @original_check_digit = match_data.captures

      fix! if @original_check_digit.nil?
    end

    def check_digit
      values = even_values.map { |i| i * 2 }.zip(odd_values).flatten

      sum = values.inject(0) do |result, i|
        result + (i / 10) + i % 10
      end

      mod_10(sum)
    end

    def to_isin(iso = 'US')
      raise InvalidConversion unless %w(US CA).include?(iso)

      ISIN.new([iso, @identifier, check_digit].join)
    end

    private

    def digits_for(char)
      return SYMBOLS[char] if SYMBOLS.key?(char)

      super
    end
  end
end