module Ibandit module LocalDetailsCleaner def self.clean(local_details) country_code = local_details[:country_code] if can_clean?(country_code, local_details) local_details = local_details.merge( public_send(:"clean_#{country_code.downcase}_details", local_details), ) end return local_details if explicit_swift_details?(local_details) swift_details_for(local_details).merge(local_details) end ########### # Helpers # ########### def self.can_clean?(country_code, local_details) Constants::SUPPORTED_COUNTRY_CODES.include?(country_code) && fields_for?(country_code, local_details) end def self.explicit_swift_details?(local_details) local_details.include?(:swift_account_number) end def self.fields_for?(country_code, opts) required_fields(country_code).all? { |argument| opts[argument] } end def self.required_fields(country_code) case country_code when "AT", "CY", "DE", "FI", "LT", "LU", "LV", "NL", "RO", "SI", "SK", "US" %i[bank_code account_number] when "BE", "CZ", "DK", "EE", "ES", "HR", "HU", "IS", "NO", "PL", "SE", "NZ" %i[account_number] when "GB", "IE", "MT" if Ibandit.bic_finder.nil? then %i[bank_code branch_code account_number] else %i[branch_code account_number] end when "AU" %i[branch_code account_number] else %i[bank_code branch_code account_number] end end ########################## # Country-specific logic # ########################## def self.clean_at_details(local_details) # Account number may be 4-11 digits long. # Add leading zeros to account number if < 11 digits. return {} unless local_details[:account_number].length >= 4 { bank_code: local_details[:bank_code], account_number: local_details[:account_number].rjust(11, "0"), } end def self.clean_au_details(local_details) # Account number may be up to 10 digits long. # Add leading zeros to account number if < 10 digits. # # Minimum account_number length is 5 return {} unless local_details[:account_number].length >= 5 { branch_code: local_details[:branch_code].delete("-"), account_number: local_details[:account_number].rjust(10, "0"), } end def self.clean_be_details(local_details) account_number = local_details[:account_number].tr("-", "") { bank_code: local_details[:bank_code] || account_number.slice(0, 3), account_number: account_number, } end def self.clean_ca_details(local_details) return {} if local_details[:account_number].length < 7 # minimum length bank_code = if local_details[:bank_code].length == 3 local_details[:bank_code].rjust(4, "0") else local_details[:bank_code] end { account_number: local_details[:account_number].rjust(12, "0"), bank_code: bank_code, } end def self.clean_us_details(local_details) return {} unless local_details[:bank_code].length == 9 account_number = local_details[:account_number].delete(" ") return {} unless (1..17).cover?(account_number.length) { bank_code: local_details[:bank_code], account_number: account_number.rjust(17, "_"), } end def self.clean_bg_details(local_details) # Bulgarian national bank details were replaced with IBANs in 2006. local_details end def self.clean_cy_details(local_details) # Account number may be 7-16 digits long. # Add leading zeros to account number if < 16 digits. cleaned_bank_code = local_details[:bank_code].gsub(/[-\s]/, "") bank_code = cleaned_bank_code.slice(0, 3) branch_code = if local_details[:branch_code] local_details[:branch_code] elsif cleaned_bank_code.length > 3 cleaned_bank_code[3..-1] end account_number = if local_details[:account_number].length >= 7 local_details[:account_number].rjust(16, "0") else local_details[:account_number] end { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_cz_details(local_details) # The SWIFT definition of a Czech IBAN includes both the account # number prefix and the account number. This method therefore supports # passing those fields concatenated. account_number = if local_details.include?(:account_number_prefix) [ local_details[:account_number_prefix].rjust(6, "0"), local_details[:account_number].rjust(10, "0"), ].join else local_details[:account_number].tr("-", "").rjust(16, "0") end { bank_code: local_details[:bank_code], account_number: account_number, } end def self.clean_de_details(local_details) # Account number may be up to 10 digits long. # Add leading zeros to account number if < 10 digits. # # There are many exceptions to the way German bank details translate # into an IBAN, detailed into a 200 page document compiled by the # Bundesbank, and handled by the GermanDetailsConverter class. converted_details = begin GermanDetailsConverter.convert(local_details) rescue UnsupportedAccountDetails local_details.dup end return {} unless converted_details[:account_number].length >= 4 { bank_code: converted_details[:bank_code], account_number: converted_details[:account_number].rjust(10, "0"), } end def self.clean_dk_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single traditional-format string split by a '-'. if local_details[:bank_code] bank_code = local_details[:bank_code].rjust(4, "0") account_number = local_details[:account_number].rjust(10, "0") elsif local_details[:account_number].include?("-") bank_code, account_number = local_details[:account_number].split("-", 2) elsif local_details[:account_number].gsub(/\s/, "").length == 14 cleaned_account_number = local_details[:account_number].gsub(/\s/, "") bank_code = cleaned_account_number.slice(0, 4) account_number = cleaned_account_number.slice(4, 10) else return {} end { bank_code: bank_code.rjust(4, "0"), account_number: account_number.delete("-").rjust(10, "0"), } end def self.clean_ee_details(local_details) # Account number may be up to 14 characters long. # Add leading zeros to account number if < 14 digits. # # Bank code can be found by extracted from the first two digits of the # account number and converted using the rules at # http://www.pangaliit.ee/en/settlements-and-standards/bank-codes-of-estonian-banks domestic_bank_code = local_details[:account_number].gsub(/\A0+/, "").slice(0, 2) iban_bank_code = case domestic_bank_code when "11" then "22" when "93" then "00" else domestic_bank_code end account_number = local_details[:account_number].rjust(14, "0") { bank_code: iban_bank_code, account_number: account_number } end def self.clean_es_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single 20 digit string. if local_details[:bank_code] && local_details[:branch_code] bank_code = local_details[:bank_code] branch_code = local_details[:branch_code] account_number = local_details[:account_number].gsub(/[-\s]/, "") else cleaned_account_number = local_details[:account_number].gsub(/[-\s]/, "") bank_code = cleaned_account_number.slice(0, 4) branch_code = cleaned_account_number.slice(4, 4) account_number = cleaned_account_number[8..-1] end { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_fi_details(local_details) # Finnish account numbers need to be expanded into "electronic format" # by adding zero-padding. The expansion method depends on the first # character of the bank code. account_number = if %w[4 5 6].include?(local_details[:bank_code][0]) [ local_details[:account_number][0], local_details[:account_number][1..-1].rjust(7, "0"), ].join else local_details[:account_number].rjust(8, "0") end { bank_code: local_details[:bank_code], account_number: account_number, } end def self.clean_fr_details(local_details) { bank_code: local_details[:bank_code], branch_code: local_details[:branch_code], account_number: local_details[:account_number].gsub(/[-\s]/, ""), } end def self.clean_gb_details(local_details) # Account number may be 6-8 digits # Add leading zeros to account number if < 8 digits. branch_code = local_details[:branch_code].gsub(/[-\s]/, "") if local_details[:bank_code] bank_code = local_details[:bank_code] else bic = Ibandit.find_bic("GB", branch_code) bank_code = bic.nil? ? nil : bic.slice(0, 4) end account_number = local_details[:account_number].gsub(/[-\s]/, "") account_number = account_number.rjust(8, "0") if account_number.length > 5 { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_gr_details(local_details) # Greek IBANs construction is idiosyncratic to the individual banks, and # no central specification is published. local_details end def self.clean_hr_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single traditional-format string split by a '-'. return local_details if local_details[:bank_code] return local_details unless local_details[:account_number].include?("-") bank_code, account_number = local_details[:account_number].split("-", 2) { bank_code: bank_code, account_number: account_number, } end def self.clean_hu_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single 16 or 24 digit string. if local_details[:bank_code] || local_details[:branch_code] return local_details end cleaned_acct_number = local_details[:account_number].gsub(/[-\s]/, "") case cleaned_acct_number.length when 16 { bank_code: cleaned_acct_number.slice(0, 3), branch_code: cleaned_acct_number.slice(3, 4), account_number: cleaned_acct_number.slice(7, 9).ljust(17, "0"), } when 24 { bank_code: cleaned_acct_number.slice(0, 3), branch_code: cleaned_acct_number.slice(3, 4), account_number: cleaned_acct_number.slice(7, 17), } else local_details end end def self.clean_ie_details(local_details) # Ireland uses the same local details as the United Kingdom branch_code = local_details[:branch_code].gsub(/[-\s]/, "") if local_details[:bank_code] bank_code = local_details[:bank_code] else bic = Ibandit.find_bic("IE", branch_code) bank_code = bic.nil? ? nil : bic.slice(0, 4) end account_number = local_details[:account_number].gsub(/[-\s]/, "") account_number = account_number.rjust(8, "0") if account_number.length > 5 { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_is_details(local_details) if local_details[:bank_code] bank_code = local_details[:bank_code] parts = local_details[:account_number].split("-") elsif local_details[:account_number].include?("-") bank_code, *parts = local_details[:account_number].split("-") else bank_code = local_details[:account_number].slice(0, 4) parts = Array(local_details[:account_number][4..-1]) end { bank_code: bank_code.rjust(4, "0"), account_number: pad_is_account_number(parts), } end def self.clean_it_details(local_details) # Add leading zeros to account number if < 12 digits. { bank_code: local_details[:bank_code], branch_code: local_details[:branch_code], account_number: local_details[:account_number].rjust(12, "0"), } end def self.clean_lt_details(local_details) # Lithuanian national bank details were replaced with IBANs in 2004. local_details end def self.clean_lu_details(local_details) # Luxembourgian national bank details were replaced with IBANs in 2002. local_details end def self.clean_lv_details(local_details) # Latvian national bank details were replaced with IBANs in 2004. local_details end def self.clean_mc_details(local_details) # Monaco uses the same local details method as France clean_fr_details(local_details) end def self.clean_mt_details(local_details) # Add leading zeros to account number if < 18 digits. branch_code = local_details[:branch_code] if local_details[:bank_code] bank_code = local_details[:bank_code] else bic = Ibandit.find_bic("MT", branch_code) bank_code = bic.nil? ? nil : bic.slice(0, 4) end account_number = local_details[:account_number].gsub(/[-\s]/, "") account_number = account_number.rjust(18, "0") { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_nl_details(local_details) # Add leading zeros to account number if < 10 digits. { bank_code: local_details[:bank_code], account_number: local_details[:account_number].rjust(10, "0"), } end def self.clean_no_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single 11 digit string. if local_details[:bank_code] bank_code = local_details[:bank_code] account_number = local_details[:account_number] else cleaned_acct_number = local_details[:account_number].gsub(/[-.\s]/, "") bank_code = cleaned_acct_number.slice(0, 4) account_number = cleaned_acct_number[4..-1] end { bank_code: bank_code, account_number: account_number, } end def self.clean_nz_details(local_details) # This method supports being passed the component parts of the NZ account # number, or a single 15/16 digit string (BBbbbb-AAAAAAA-0SS) in the # account number field. # # When given a 15/16 digit string, the component parts (i.e bank_code, # branch_code and account_number) are extracted, with the 7-digit account # number body and 2/3-digit account number suffix making up the actual # account_number field. if local_details[:bank_code] && local_details[:branch_code] bank_code = local_details[:bank_code] branch_code = local_details[:branch_code] account_number = local_details[:account_number].tr("-", "") else cleaned_account_number = local_details[:account_number].tr("-", "") bank_code = cleaned_account_number.slice(0, 2) branch_code = cleaned_account_number.slice(2, 4) account_number = cleaned_account_number[6..-1] end if account_number && account_number.length == 9 # > Some banks (such as BNZ) include three digits of the suffix in their # > presentation of the account number to the end customer. Other banks # > only show the last two digits of the suffix to the end customer. # > Technically, all banks have three digit suffixes, it's just that the # > first digit of the suffix is always 0 so it's usually ignored. # > [https://en.wikipedia.org/wiki/New_Zealand_bank_account_number] # # Here, we insert the zero in cases where it is ignored. account_number = account_number[0..6] + "0" + account_number[7..8] end { bank_code: bank_code, branch_code: branch_code, account_number: account_number, } end def self.clean_pl_details(local_details) # This method supports being passed the component IBAN parts, as defined # by SWIFT, or a single 26 digit string. if local_details[:bank_code] bank_code = local_details[:bank_code] account_number = local_details[:account_number] else cleaned_acct_number = local_details[:account_number].gsub(/[\s]/, "") bank_code = cleaned_acct_number.slice(2, 8) account_number = cleaned_acct_number[10..-1] end { bank_code: bank_code, account_number: account_number, } end def self.clean_pt_details(local_details) local_details end def self.clean_ro_details(local_details) # Romanian national bank details were replaced with IBANs in 2004. local_details end def self.clean_se_details(local_details) if local_details[:bank_code] # If a bank_code was provided without a branch code we're (probably) # dealing with SWIFT details and should just return them. { swift_account_number: local_details[:account_number], swift_bank_code: local_details[:bank_code], } else Sweden::LocalDetailsConverter.new( branch_code: local_details[:branch_code], account_number: local_details[:account_number], ).convert end end def self.clean_si_details(local_details) # Add leading zeros to account number if < 10 digits. { bank_code: local_details[:bank_code], account_number: local_details[:account_number].rjust(10, "0"), } end def self.clean_sk_details(local_details) # Slovakia uses the same local details method as the Czech Republic clean_cz_details(local_details) end def self.clean_sm_details(local_details) # San Marino uses the same local details method as France clean_it_details(local_details) end def self.pad_is_account_number(parts) hufo = parts[0].nil? ? "" : parts[0].rjust(2, "0") reikningsnumer = parts[1].nil? ? "" : parts[1].rjust(6, "0") ken_1 = parts[2].nil? ? "" : parts[2].rjust(6, "0") ken_2 = parts[3].nil? ? "" : parts[3].rjust(4, "0") kennitala = ken_1.empty? ? "" : (ken_1 + ken_2).rjust(10, "0") hufo + reikningsnumer + kennitala end private_class_method :pad_is_account_number def self.swift_details_for(local_details) { swift_bank_code: local_details[:bank_code], swift_branch_code: local_details[:branch_code], swift_account_number: local_details[:account_number], } end private_class_method :swift_details_for end end