defmodule Tempus.Guards do
@moduledoc since: "0.9.0"
@moduledoc "Handy guards to simplify pattern matching slots"
alias Tempus.Slot
@dialyzer :no_contracts
@doc """
Guard to validate that the term given is actually a `t:Tempus.Slot.origin/0`
## Examples
iex> import Tempus.Guards, only: [is_origin: 1]
...> is_origin(Date.utc_today())
true
...> is_origin(nil)
true
...> is_origin(:ok)
false
"""
@spec is_origin(Slot.origin() | any()) :: boolean()
defguard is_origin(term)
when is_nil(term) or
(is_map(term) and is_map_key(term, :__struct__) and
:erlang.map_get(:__struct__, term) in [Date, DateTime, Time, Slot])
defguardp is_date(term) when is_struct(term, Date)
# defguardp is_time(term) when is_struct(term, Time)
defguardp is_datetime(term) when is_struct(term, DateTime)
defguardp is_slot(term) when is_struct(term, Tempus.Slot)
defguardp is_date_equal(d1, d2)
when :erlang.map_get(:calendar, d1) == :erlang.map_get(:calendar, d2) and
:erlang.map_get(:year, d1) == :erlang.map_get(:year, d2) and
:erlang.map_get(:month, d1) == :erlang.map_get(:month, d2) and
:erlang.map_get(:day, d1) == :erlang.map_get(:day, d2)
defguardp is_time_equal(t1, t2)
when :erlang.map_get(:calendar, t1) == :erlang.map_get(:calendar, t2) and
:erlang.map_get(:hour, t1) == :erlang.map_get(:hour, t2) and
:erlang.map_get(:minute, t1) == :erlang.map_get(:minute, t2) and
:erlang.map_get(:second, t1) == :erlang.map_get(:second, t2) and
elem(:erlang.map_get(:microsecond, t1), 0) ==
elem(:erlang.map_get(:microsecond, t2), 0)
defguardp is_datetime_equal(dt1, dt2) when is_date_equal(dt1, dt2) and is_time_equal(dt1, dt2)
defguardp is_microsecond_coming_before(m1, m2) when elem(m1, 0) < elem(m2, 0)
defguardp is_date_coming_before(d1, d2)
when :erlang.map_get(:calendar, d1) == :erlang.map_get(:calendar, d2) and
(:erlang.map_get(:year, d1) < :erlang.map_get(:year, d2) or
(:erlang.map_get(:year, d1) == :erlang.map_get(:year, d2) and
:erlang.map_get(:month, d1) < :erlang.map_get(:month, d2)) or
(:erlang.map_get(:year, d1) == :erlang.map_get(:year, d2) and
:erlang.map_get(:month, d1) == :erlang.map_get(:month, d2) and
:erlang.map_get(:day, d1) < :erlang.map_get(:day, d2)))
defguardp is_time_coming_before(t1, t2)
when :erlang.map_get(:calendar, t1) == :erlang.map_get(:calendar, t2) and
(:erlang.map_get(:hour, t1) < :erlang.map_get(:hour, t2) or
(:erlang.map_get(:hour, t1) == :erlang.map_get(:hour, t2) and
:erlang.map_get(:minute, t1) < :erlang.map_get(:minute, t2)) or
(:erlang.map_get(:hour, t1) == :erlang.map_get(:hour, t2) and
:erlang.map_get(:minute, t1) == :erlang.map_get(:minute, t2) and
:erlang.map_get(:second, t1) < :erlang.map_get(:second, t2)) or
(:erlang.map_get(:hour, t1) == :erlang.map_get(:hour, t2) and
:erlang.map_get(:minute, t1) == :erlang.map_get(:minute, t2) and
:erlang.map_get(:second, t1) == :erlang.map_get(:second, t2) and
is_microsecond_coming_before(
:erlang.map_get(:microsecond, t1),
:erlang.map_get(:microsecond, t2)
)))
defguardp is_datetime_coming_before(dt1, dt2)
when is_date_coming_before(dt1, dt2) or
(is_date_equal(dt1, dt2) and is_time_coming_before(dt1, dt2))
defguardp is_slot_coming_before(s1, s2)
when is_datetime_coming_before(:erlang.map_get(:to, s1), :erlang.map_get(:from, s2))
defguardp is_datetime_covered(dt, dt1, dt2)
when (is_nil(dt1) and not is_nil(dt2) and not is_datetime_coming_before(dt2, dt)) or
(is_nil(dt2) and not is_nil(dt1) and not is_datetime_coming_before(dt, dt1)) or
(not is_nil(dt1) and not is_nil(dt2) and
not is_datetime_coming_before(dt, dt1) and
not is_datetime_coming_before(dt2, dt))
defguardp is_datetime_covered(dt, s)
when is_datetime_covered(dt, :erlang.map_get(:from, s), :erlang.map_get(:to, s))
defguardp is_slot_covered(s1, s2)
when is_datetime_covered(:erlang.map_get(:from, s1), s2) and
is_datetime_covered(:erlang.map_get(:to, s1), s2)
defguardp is_slot_from_equal(s, dt)
when is_slot(s) and is_datetime_equal(dt, :erlang.map_get(:from, s))
defguardp is_slot_to_equal(s, dt)
when is_slot(s) and is_datetime_equal(dt, :erlang.map_get(:to, s))
@doc """
Guard to validate whether the slot is `nil` (has neither end set.)
"""
@spec is_slot_nil(Slot.t()) :: boolean()
defguard is_slot_nil(s)
when is_slot(s) and is_nil(:erlang.map_get(:from, s)) and
is_nil(:erlang.map_get(:to, s))
@doc """
Guard to validate whether the slot is open (has either end not set.)
Please note, that the slot having both ends set to `nil` is considered
a special case and is not reported as _open_.
"""
@spec is_slot_open(Slot.t()) :: boolean()
defguard is_slot_open(s)
when is_slot(s) and not is_slot_nil(s) and
(is_nil(:erlang.map_get(:from, s)) or
is_nil(:erlang.map_get(:to, s)))
@doc """
Guard to validate whether the `t:DateTime.t/0` given as the first argument
is the border of the slot.
"""
@spec is_slot_border(DateTime.t(), Slot.t()) :: boolean()
defguard is_slot_border(dt, s)
when is_slot(s) and is_datetime(dt) and
(is_slot_from_equal(s, dt) or is_slot_to_equal(s, dt))
@doc """
Guard to validate one slot ovelaps another
## Examples
iex> import Tempus.Guards, only: [is_joint: 2]
...> import Tempus.Sigils
...> s1 = ~I[2023-04-09 23:00:00Z|2023-04-10 00:59:59Z]
...> s2 = Tempus.Slot.wrap(~D|2023-04-10|)
...> is_joint(s1, s2)
true
...> s1 = ~I[2023-04-09 23:00:00Z|2023-04-10 00:00:00Z]
...> s2 = Tempus.Slot.wrap(~D|2023-04-10|)
...> is_joint(s1, s2)
true
"""
@spec is_joint(Slot.t(), Slot.t()) :: boolean()
defguard is_joint(s1, s2)
when is_slot(s1) and is_slot(s2) and
(is_datetime_covered(:erlang.map_get(:from, s1), s2) or
is_datetime_covered(:erlang.map_get(:to, s1), s2))
@doc """
Guard to validate the slot covers the origin passed as the first argument
## Examples
iex> import Tempus.Guards, only: [is_covered: 2]
...> import Tempus.Sigils
...> {from, to} = {~U[2023-04-10 00:00:00Z], ~U[2023-04-10 00:59:59Z]}
...> s = %Tempus.Slot{from: from, to: to}
...> is_covered(from, s) and is_covered(to, s)
true
...> s1 = ~I[2023-04-10 00:00:00Z|2023-04-11 00:00:00Z]
...> s2 = Tempus.Slot.wrap(~D|2023-04-10|)
...> is_covered(s1, s2)
false
...> s1 = ~I[2023-04-10 00:00:00Z|2023-04-11 00:00:00Z]
...> s2 = Tempus.Slot.wrap(~D|2023-04-10|)
...> is_covered(s1, s2)
false
"""
@spec is_covered(Slot.origin(), Slot.t()) :: boolean()
defguard is_covered(o, s)
when is_slot(s) and
((is_slot(o) and is_slot_covered(o, s)) or
(is_datetime(o) and is_datetime_covered(o, s)))
@doc """
Guard to compare two instances of `t:Tempus.Slot.origin/0`
## Examples
iex> import Tempus.Guards, only: [is_coming_before: 2]
...> is_coming_before(~D[2023-04-10], ~U[2023-04-11T00:00:00.000000Z])
true
...> is_coming_before(~D[2023-04-10], ~D[2023-04-10])
false
"""
@spec is_coming_before(Date.t() | DateTime.t(), Date.t() | DateTime.t()) :: boolean()
@spec is_coming_before(Slot.t(), Slot.t()) :: boolean()
defguard is_coming_before(o1, o2)
# (is_datetime(o1) and is_slot(o2) and is_datetime_coming_before(o1, o2.from)) or
# (is_slot(o1) and is_datetime(o2) and is_datetime_coming_before(o1.to, o2)) or
when (is_date(o1) and is_date(o2) and is_date_coming_before(o1, o2)) or
(is_datetime(o1) and is_date(o2) and is_date_coming_before(o1, o2)) or
(is_date(o1) and is_datetime(o2) and is_date_coming_before(o1, o2)) or
(is_datetime(o1) and is_datetime(o2) and is_datetime_coming_before(o1, o2)) or
(is_slot(o1) and is_slot(o2) and is_slot_coming_before(o1, o2))
@spec joint_in_delta?(
Slot.t(),
Slot.t(),
non_neg_integer() | {non_neg_integer(), non_neg_integer()}
) :: boolean()
@doc """
Helper to validate one slot overlaps another in delta. Unlike guards,
this function does not expect arguments in the correct order, and would return
`true` if the slots overlap even if `s2` comes _before_ `s1`.
## Examples
iex> import Tempus.Guards, only: [joint_in_delta?: 3]
...> import Tempus.Sigils
...> s1 = ~I[2023-04-09 23:00:00Z|2023-04-09 23:59:59Z]
...> s2 = Tempus.Slot.wrap(~D|2023-04-10|)
...> joint_in_delta?(s1, s2, 1)
true
...> joint_in_delta?(s2, s1, 1)
true
...> joint_in_delta?(s1, s2, {0, 500})
false
"""
def joint_in_delta?(s1, s2, _delta) when is_joint(s1, s2), do: true
def joint_in_delta?(s1, s2, {delta_secs, delta_msecs})
when is_coming_before(s1, s2) do
{secs_from, msecs_from} = DateTime.to_gregorian_seconds(s2.from)
{secs_to, msecs_to} = DateTime.to_gregorian_seconds(s1.to)
1_000_000 * (secs_to - secs_from - delta_secs) + msecs_to - msecs_from - delta_msecs >= 0
end
def joint_in_delta?(s1, s2, delta) when is_coming_before(s1, s2) do
joint_in_delta?(s1, s2, {delta, 0})
end
def joint_in_delta?(s1, s2, delta) when is_coming_before(s2, s1) do
joint_in_delta?(s2, s1, {delta, 0})
end
end