defmodule Z.DateTime do
@moduledoc """
A module for validating a DateTime
"""
alias Z.{Result, Error, Issue, Any}
use Z.Type,
options: Z.Any.__z__(:options) ++ [:parse, :allow_int, :shift, :trunc, :min, :max]
def check(result, :conversions, rules, context) do
result
|> Any.check(:conversions, rules, context)
|> maybe_check(:parse, rules, context)
|> maybe_check(:allow_int, rules, context)
end
def check(result, :mutations, rules, context) do
result
|> Any.check(:mutations, rules, context)
|> maybe_check(:shift, rules, context)
|> maybe_check(:trunc, rules, context)
end
def check(result, :assertions, rules, context) do
result
|> Any.check(:assertions, rules, context)
|> maybe_check(:min, rules, context)
|> maybe_check(:max, rules, context)
end
def check(result, _rule, _options, _context) when result.value == nil do
result
end
def check(result, :parse, _format, _context) when not is_binary(result.value) do
result
end
def check(result, :parse, false, _context) do
result
end
def check(result, :parse, true, context) do
check(result, :parse, :iso8601, context)
end
def check(result, :parse, format, context) when format not in [:iso8601] do
message = "unable to parse DateTime with format: #{inspect(format)}, format must be :iso8601"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :parse, _format, context) do
case DateTime.from_iso8601(result.value) do
{:ok, datetime, _offset} ->
result |> Result.set_value(datetime)
{:error, _} ->
message = "unable to parse input as a DateTime"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_string(),
message,
context
)
)
end
end
def check(result, :allow_int, _mode, _context) when not is_integer(result.value) do
result
end
def check(result, :allow_int, false, _context) do
result
end
def check(result, :allow_int, true, context) do
check(result, :allow_int, :unix, context)
end
def check(result, :allow_int, mode, context) when mode not in [:unix, :gregorian] do
message =
"unable to convert to DateTime with mode: #{inspect(mode)}, mode must be :unix or :gregorian"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :allow_int, mode, context) do
case mode do
:unix ->
from_unix(result, context)
:gregorian ->
from_gregorian(result, context)
end
end
def check(result, :type, options, context) when not is_struct(result.value, DateTime) do
message = Keyword.get(options, :message, "input is not a DateTime")
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_type(),
message,
context
)
)
end
def check(result, :type, _options, _context) do
result
end
def check(result, _rule, _options, _context) when not is_struct(result.value, DateTime) do
result
end
def check(result, :shift, zone, context) when not is_binary(zone) do
message = "timezone must be a string"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :shift, zone, context) do
case DateTime.shift_zone(result.value, zone) do
{:ok, datetime} ->
result |> Result.set_value(datetime)
{:error, _} ->
message = "unable to shift input to #{inspect(zone)} timezone"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
end
def check(result, :trunc, false, _context) do
result
end
def check(result, :trunc, true, context) do
check(result, :trunc, :second, context)
end
def check(result, :trunc, precision, context)
when precision not in [:second, :millisecond, :microsecond] do
message =
"unable to truncate with precision: #{inspect(precision)}, precision must be :second, :millisecond, or :microsecond"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :trunc, precision, _context) do
result |> Result.set_value(result.value |> DateTime.truncate(precision))
end
def check(result, :min, {value, _options}, context) when not is_struct(value, DateTime) do
message = "min value must be a DateTime"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :min, {value, options}, context) do
case DateTime.compare(result.value, value) do
:lt ->
message = Keyword.get(options, :message, "input is too early")
result
|> Result.add_issue(
Issue.new(
Error.Codes.too_small(),
message,
context
)
)
_ ->
result
end
end
def check(result, :min, value, context) do
check(result, :min, {value, []}, context)
end
def check(result, :max, {value, _options}, context) when not is_struct(value, DateTime) do
message = "max value must be a DateTime"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_arguments(),
message,
context
)
)
end
def check(result, :max, {value, options}, context) do
case DateTime.compare(result.value, value) do
:gt ->
message = Keyword.get(options, :message, "input is too late")
result
|> Result.add_issue(
Issue.new(
Error.Codes.too_big(),
message,
context
)
)
_ ->
result
end
end
def check(result, :max, value, context) do
check(result, :max, {value, []}, context)
end
defp from_unix(result, context) do
case DateTime.from_unix(result.value) do
{:ok, datetime} ->
result |> Result.set_value(datetime)
{:error, _} ->
message = "unable to convert unix input to a DateTime"
result
|> Result.add_issue(
Issue.new(
Error.Codes.invalid_date(),
message,
context
)
)
end
end
defp from_gregorian(result, _context) do
datetime = DateTime.from_gregorian_seconds(result.value)
result |> Result.set_value(datetime)
end
end