2024-03-05 11:02:58 +00:00
|
|
|
defmodule IbanEx.Country.Template do
|
2024-03-07 23:33:38 +00:00
|
|
|
@moduledoc false
|
|
|
|
|
2024-03-05 11:02:58 +00:00
|
|
|
alias IbanEx.Iban
|
|
|
|
@type size() :: non_neg_integer()
|
|
|
|
@type rule() :: Regex.t()
|
|
|
|
@type country_code() :: <<_::16>> | atom()
|
|
|
|
@type joiner() :: String.t()
|
|
|
|
|
|
|
|
@callback size() :: size()
|
|
|
|
@callback rule() :: rule()
|
2024-05-16 08:55:21 +00:00
|
|
|
@callback rules() :: []
|
|
|
|
@callback rules_map() :: %{}
|
|
|
|
@callback bban_fields() :: [atom()]
|
|
|
|
@callback bban_size() :: non_neg_integer()
|
2024-03-07 23:33:38 +00:00
|
|
|
@callback to_string(Iban.t(), joiner()) :: String.t()
|
|
|
|
@callback to_string(Iban.t()) :: String.t()
|
|
|
|
|
|
|
|
defmacro __using__(_opts) do
|
|
|
|
quote do
|
|
|
|
alias IbanEx.Iban
|
|
|
|
@behaviour IbanEx.Country.Template
|
|
|
|
|
|
|
|
@impl IbanEx.Country.Template
|
|
|
|
@spec to_string(Iban.t()) :: binary()
|
|
|
|
@spec to_string(Iban.t(), binary()) :: binary()
|
|
|
|
def to_string(
|
|
|
|
%Iban{
|
|
|
|
country_code: country_code,
|
|
|
|
check_digits: check_digits,
|
|
|
|
bank_code: bank_code,
|
|
|
|
branch_code: _branch_code,
|
|
|
|
national_check: _national_check,
|
|
|
|
account_number: account_number
|
|
|
|
} = _iban,
|
|
|
|
joiner \\ " "
|
|
|
|
) do
|
|
|
|
[country_code, check_digits, bank_code, account_number]
|
|
|
|
|> Enum.join(joiner)
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl IbanEx.Country.Template
|
2024-05-14 23:15:55 +00:00
|
|
|
@spec size() :: integer()
|
2024-03-07 23:33:38 +00:00
|
|
|
def size(), do: @size
|
|
|
|
|
2024-05-15 03:09:24 +00:00
|
|
|
@doc """
|
|
|
|
Return Regex for parsing complete BBAN (part of IBAN string)
|
|
|
|
"""
|
2024-03-07 23:33:38 +00:00
|
|
|
@impl IbanEx.Country.Template
|
|
|
|
@spec rule() :: Regex.t()
|
|
|
|
def rule(), do: @rule
|
|
|
|
|
2024-05-15 03:09:24 +00:00
|
|
|
@impl IbanEx.Country.Template
|
2024-05-16 08:55:21 +00:00
|
|
|
@spec bban_size() :: integer()
|
|
|
|
def bban_size() do
|
|
|
|
{_rules, bban_size} = calculate_rules()
|
|
|
|
bban_size
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl IbanEx.Country.Template
|
|
|
|
@spec bban_fields() :: []
|
|
|
|
def bban_fields(), do: rules_map() |> Map.keys()
|
|
|
|
|
|
|
|
@impl IbanEx.Country.Template
|
|
|
|
@spec rules_map() :: %{}
|
|
|
|
def rules_map(), do: rules() |> Map.new()
|
|
|
|
|
|
|
|
@impl IbanEx.Country.Template
|
|
|
|
@spec rules() :: []
|
|
|
|
def rules() do
|
|
|
|
{rules, _bban_size} = calculate_rules()
|
|
|
|
rules
|
|
|
|
end
|
|
|
|
|
|
|
|
defp calculate_rules() do
|
|
|
|
scanner = ~r/\(\?\<([\w_]+)\>(([^{]+)\{(\d+)\})\)/i
|
|
|
|
|
2024-05-15 03:09:24 +00:00
|
|
|
source =
|
|
|
|
@rule
|
|
|
|
|> Regex.source()
|
|
|
|
|
2024-05-16 08:55:21 +00:00
|
|
|
{list, bban_length} =
|
|
|
|
Regex.scan(scanner, source)
|
|
|
|
|> Enum.reduce({[], 0}, fn [_part, k, r, _syms, l], {list, position} = acc ->
|
|
|
|
key = String.to_atom(k)
|
|
|
|
{:ok, regex} = Regex.compile(r, "i")
|
|
|
|
length = String.to_integer(l)
|
|
|
|
left = position
|
|
|
|
right = left + length - 1
|
|
|
|
{[{key, %{regex: regex, range: left..right}} | list], right + 1}
|
|
|
|
end)
|
2024-05-15 03:09:24 +00:00
|
|
|
|
2024-05-16 08:55:21 +00:00
|
|
|
{Enum.reverse(list), bban_length}
|
2024-05-15 03:09:24 +00:00
|
|
|
end
|
|
|
|
|
2024-05-16 08:55:21 +00:00
|
|
|
defoverridable to_string: 1, to_string: 2, size: 0, rule: 0
|
2024-03-07 23:33:38 +00:00
|
|
|
end
|
|
|
|
end
|
2024-03-05 11:02:58 +00:00
|
|
|
end
|