2024-12-13 02:17:30 +00:00
|
|
|
defmodule UkraineTaxidEx.BaseValidator do
|
2024-12-13 05:34:25 +00:00
|
|
|
@callback validate(code :: String.t()) ::
|
|
|
|
{:ok, code :: String.t()}
|
2024-12-13 02:17:30 +00:00
|
|
|
| {:error,
|
|
|
|
:length_too_short | :length_too_long | :invalid_length | :invalid_checksum}
|
2024-12-13 05:34:25 +00:00
|
|
|
@callback violates_length?(code :: String.t()) :: boolean
|
|
|
|
@callback violates_length_too_short?(code :: String.t()) :: boolean
|
|
|
|
@callback violates_length_too_long?(code :: String.t()) :: boolean
|
|
|
|
@callback violates_checksum?(code :: String.t()) :: boolean
|
2024-12-13 02:17:30 +00:00
|
|
|
|
|
|
|
defmacro __using__(_) do
|
|
|
|
quote do
|
|
|
|
alias UkraineTaxidEx.BaseValidator
|
|
|
|
|
|
|
|
@behaviour UkraineTaxidEx.BaseValidator
|
|
|
|
import UkraineTaxidEx.Commons, only: [digits: 1, digits_and_check_digit: 1, error: 1, ok: 1]
|
|
|
|
|
|
|
|
@impl BaseValidator
|
|
|
|
@spec validate(String.t()) ::
|
|
|
|
{:ok, String.t()}
|
|
|
|
| {:error,
|
|
|
|
:length_too_short | :length_too_long | :invalid_length | :invalid_checksum}
|
|
|
|
def validate(string) do
|
|
|
|
cond do
|
|
|
|
violates_length_too_short?(string) -> error(:length_too_short)
|
|
|
|
violates_length_too_long?(string) -> error(:length_too_long)
|
|
|
|
violates_checksum?(string) -> error(:invalid_checksum)
|
|
|
|
true -> ok(string)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@doc "Check whether a given EDRPOU violates the required length"
|
|
|
|
@impl BaseValidator
|
2024-12-13 05:34:25 +00:00
|
|
|
@spec violates_length?(code :: String.t()) :: boolean
|
|
|
|
def violates_length?(code), do: String.length(code) != length()
|
2024-12-13 02:17:30 +00:00
|
|
|
|
|
|
|
@doc "Check whether a given EDRPOU too short"
|
|
|
|
@impl BaseValidator
|
2024-12-13 05:34:25 +00:00
|
|
|
@spec violates_length_too_short?(code :: String.t()) :: boolean
|
|
|
|
def violates_length_too_short?(code), do: String.length(code) < length()
|
2024-12-13 02:17:30 +00:00
|
|
|
|
|
|
|
@doc "Check whether a given EDRPOU too long"
|
|
|
|
@impl BaseValidator
|
2024-12-13 05:34:25 +00:00
|
|
|
@spec violates_length_too_long?(code :: String.t()) :: boolean
|
|
|
|
def violates_length_too_long?(code), do: String.length(code) > length()
|
2024-12-13 02:17:30 +00:00
|
|
|
|
|
|
|
@doc "Check whether a given EDRPOU has correct checksum"
|
|
|
|
@impl BaseValidator
|
2024-12-13 05:34:25 +00:00
|
|
|
@spec violates_checksum?(code :: String.t()) :: boolean
|
|
|
|
def violates_checksum?(code) do
|
2024-12-13 02:17:30 +00:00
|
|
|
{digits, check_digit} =
|
2024-12-13 05:34:25 +00:00
|
|
|
code
|
2024-12-13 02:17:30 +00:00
|
|
|
|> digits()
|
|
|
|
|> digits_and_check_digit()
|
|
|
|
|
|
|
|
check_sum = check_sum(digits)
|
|
|
|
|
|
|
|
check_sum != check_digit
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|