defprotocol Timex.Protocol do
@moduledoc """
This protocol defines the API for functions which take a `Date`,
`NaiveDateTime`, or `DateTime` as input.
"""
@fallback_to_any true
alias Timex.Types
@doc """
Convert a date/time value to a Julian calendar date number
"""
@spec to_julian(Types.valid_datetime()) :: integer | {:error, term}
def to_julian(datetime)
@doc """
Convert a date/time value to gregorian seconds (seconds since start of year zero)
"""
@spec to_gregorian_seconds(Types.valid_datetime()) :: non_neg_integer | {:error, term}
def to_gregorian_seconds(datetime)
@doc """
Convert a date/time value to gregorian microseconds (microseconds since the start of year zero)
"""
@spec to_gregorian_microseconds(Types.valid_datetime()) :: non_neg_integer | {:error, term}
def to_gregorian_microseconds(datetime)
@doc """
Convert a date/time value to seconds since the UNIX Epoch
"""
@spec to_unix(Types.valid_datetime()) :: non_neg_integer | {:error, term}
def to_unix(datetime)
@doc """
Convert a date/time value to a Date
"""
@spec to_date(Types.valid_datetime()) :: Date.t() | {:error, term}
def to_date(datetime)
@doc """
Convert a date/time value to a DateTime.
An optional timezone can be provided, UTC will be assumed if one is not provided.
"""
@spec to_datetime(Types.valid_datetime()) :: DateTime.t() | {:error, term}
@spec to_datetime(Types.valid_datetime(), Types.valid_timezone()) ::
DateTime.t() | Timex.AmbiguousDateTime.t() | {:error, term}
def to_datetime(datetime, timezone \\ :utc)
@doc """
Convert a date/time value to a NaiveDateTime
"""
@spec to_naive_datetime(Types.valid_datetime()) :: NaiveDateTime.t() | {:error, term}
def to_naive_datetime(datetime)
@doc """
Convert a date/time value to it's Erlang tuple variant
i.e. Date becomes `{y,m,d}`, DateTime/NaiveDateTime become
`{{y,m,d},{h,mm,s}}`
"""
@spec to_erl(Types.valid_datetime()) :: Types.date() | Types.datetime() | {:error, term}
def to_erl(datetime)
@doc """
Get the century a date/time value is in
"""
@spec century(Types.year() | Types.valid_datetime()) :: non_neg_integer | {:error, term}
def century(datetime)
@doc """
Return a boolean indicating whether the date/time value is in a leap year
"""
@spec is_leap?(Types.valid_datetime() | Types.year()) :: boolean | {:error, term}
def is_leap?(datetime)
@doc """
Shift a date/time value using a list of shift unit/value pairs
"""
@spec shift(Types.valid_datetime(), Timex.shift_options()) ::
Types.valid_datetime() | Timex.AmbiguousDateTime.t() | {:error, term}
def shift(datetime, options)
@doc """
Set fields on a date/time value using a list of unit/value pairs
"""
@spec set(Types.valid_datetime(), Timex.set_options()) ::
Types.valid_datetime() | {:error, term}
def set(datetime, options)
@doc """
Get a new version of the date/time value representing the beginning of the day
"""
@spec beginning_of_day(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def beginning_of_day(datetime)
@doc """
Get a new version of the date/time value representing the end of the day
"""
@spec end_of_day(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def end_of_day(datetime)
@doc """
Get a new version of the date/time value representing the beginning of it's week,
providing a weekday name (as an atom) for the day which starts the week, i.e. `:mon`.
"""
@spec beginning_of_week(Types.valid_datetime(), Types.weekstart()) ::
Types.valid_datetime() | {:error, term}
def beginning_of_week(datetime, weekstart)
@doc """
Get a new version of the date/time value representing the ending of it's week,
providing a weekday name (as an atom) for the day which starts the week, i.e. `:mon`.
"""
@spec end_of_week(Types.valid_datetime(), Types.weekstart()) ::
Types.valid_datetime() | {:error, term}
def end_of_week(datetime, weekstart)
@doc """
Get a new version of the date/time value representing the beginning of it's year
"""
@spec beginning_of_year(Types.year() | Types.valid_datetime()) ::
Types.valid_datetime() | {:error, term}
def beginning_of_year(datetime)
@doc """
Get a new version of the date/time value representing the ending of it's year
"""
@spec end_of_year(Types.year() | Types.valid_datetime()) ::
Types.valid_datetime() | {:error, term}
def end_of_year(datetime)
@doc """
Get a new version of the date/time value representing the beginning of it's quarter
"""
@spec beginning_of_quarter(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def beginning_of_quarter(datetime)
@doc """
Get a new version of the date/time value representing the ending of it's quarter
"""
@spec end_of_quarter(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def end_of_quarter(datetime)
@doc """
Get a new version of the date/time value representing the beginning of it's month
"""
@spec beginning_of_month(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def beginning_of_month(datetime)
@doc """
Get a new version of the date/time value representing the ending of it's month
"""
@spec end_of_month(Types.valid_datetime()) :: Types.valid_datetime() | {:error, term}
def end_of_month(datetime)
@doc """
Get the quarter for the given date/time value
"""
@spec quarter(Types.month() | Types.valid_datetime()) :: 1..4 | {:error, term}
def quarter(datetime)
@doc """
Get the number of days in the month for the given date/time value
"""
@spec days_in_month(Types.valid_datetime()) :: Types.num_of_days() | {:error, term}
def days_in_month(datetime)
@doc """
Get the week number of the given date/time value, starting at 1
"""
@spec week_of_month(Types.valid_datetime()) :: Types.week_of_month() | {:error, term}
def week_of_month(datetime)
@doc """
Get the ordinal weekday number of the given date/time value
"""
@spec weekday(Types.valid_datetime()) :: Types.weekday() | {:error, term}
def weekday(datetime)
@doc """
Get the ordinal weekday number of the given date/time value and relative to the given weekstart
"""
@spec weekday(Types.valid_datetime(), Types.weekday_name()) ::
Types.weekday() | {:error, term}
def weekday(datetime, weekstart)
@doc """
Get the ordinal day number of the given date/time value
"""
@spec day(Types.valid_datetime()) :: Types.daynum() | {:error, term}
def day(datetime)
@doc """
Determine if the provided date/time value is valid.
"""
@spec is_valid?(Types.valid_datetime()) :: boolean | {:error, term}
def is_valid?(datetime)
@doc """
Return a pair {year, week number} (as defined by ISO 8601) that the given date/time value falls on.
"""
@spec iso_week(Types.valid_datetime()) :: {Types.year(), Types.weeknum()} | {:error, term}
def iso_week(datetime)
@doc """
Shifts the given date/time value to the ISO day given
"""
@spec from_iso_day(Types.valid_datetime(), non_neg_integer) ::
Types.valid_datetime() | {:error, term}
def from_iso_day(datetime, day)
end
defimpl Timex.Protocol, for: Any do
def to_julian(%{__struct__: _} = d), do: Timex.to_julian(Map.from_struct(d))
def to_julian(_datetime), do: {:error, :invalid_date}
def to_gregorian_seconds(%{__struct__: _} = d),
do: Timex.to_gregorian_seconds(Map.from_struct(d))
def to_gregorian_seconds(_datetime), do: {:error, :invalid_date}
def to_gregorian_microseconds(%{__struct__: _} = d),
do: Timex.to_gregorian_microseconds(Map.from_struct(d))
def to_gregorian_microseconds(_datetime), do: {:error, :invalid_date}
def to_unix(%{__struct__: _} = d), do: Timex.to_unix(Map.from_struct(d))
def to_unix(_datetime), do: {:error, :invalid_date}
def to_date(%{__struct__: _} = d), do: Timex.to_date(Map.from_struct(d))
def to_date(_datetime), do: {:error, :invalid_date}
def to_datetime(%{__struct__: _} = d, timezone),
do: Timex.to_datetime(Map.from_struct(d), timezone)
def to_datetime(_datetime, _timezone), do: {:error, :invalid_date}
def to_naive_datetime(%{__struct__: _} = d), do: Timex.to_naive_datetime(Map.from_struct(d))
def to_naive_datetime(_datetime), do: {:error, :invalid_date}
def to_erl(%{__struct__: _} = d), do: Timex.to_erl(Map.from_struct(d))
def to_erl(_datetime), do: {:error, :invalid_date}
def century(%{__struct__: _} = d), do: Timex.century(Map.from_struct(d))
def century(_datetime), do: {:error, :invalid_date}
def is_leap?(%{__struct__: _} = d), do: Timex.is_leap?(Map.from_struct(d))
def is_leap?(_datetime), do: {:error, :invalid_date}
def shift(%{__struct__: _} = d, options), do: Timex.shift(Map.from_struct(d), options)
def shift(_datetime, _options), do: {:error, :invalid_date}
def set(%{__struct__: _} = d, options), do: Timex.set(Map.from_struct(d), options)
def set(_datetime, _options), do: {:error, :invalid_date}
def beginning_of_day(%{__struct__: _} = d), do: Timex.beginning_of_day(Map.from_struct(d))
def beginning_of_day(_datetime), do: {:error, :invalid_date}
def end_of_day(%{__struct__: _} = d), do: Timex.end_of_day(Map.from_struct(d))
def end_of_day(_datetime), do: {:error, :invalid_date}
def beginning_of_week(%{__struct__: _} = d, weekstart),
do: Timex.beginning_of_week(Map.from_struct(d), weekstart)
def beginning_of_week(_datetime, _weekstart), do: {:error, :invalid_date}
def end_of_week(%{__struct__: _} = d, weekstart),
do: Timex.end_of_week(Map.from_struct(d), weekstart)
def end_of_week(_datetime, _weekstart), do: {:error, :invalid_date}
def beginning_of_year(%{__struct__: _} = d), do: Timex.beginning_of_year(Map.from_struct(d))
def beginning_of_year(_datetime), do: {:error, :invalid_date}
def end_of_year(%{__struct__: _} = d), do: Timex.end_of_year(Map.from_struct(d))
def end_of_year(_datetime), do: {:error, :invalid_date}
def beginning_of_quarter(%{__struct__: _} = d),
do: Timex.beginning_of_quarter(Map.from_struct(d))
def beginning_of_quarter(_datetime), do: {:error, :invalid_date}
def end_of_quarter(%{__struct__: _} = d), do: Timex.end_of_quarter(Map.from_struct(d))
def end_of_quarter(_datetime), do: {:error, :invalid_date}
def beginning_of_month(%{__struct__: _} = d), do: Timex.beginning_of_month(Map.from_struct(d))
def beginning_of_month(_datetime), do: {:error, :invalid_date}
def end_of_month(%{__struct__: _} = d), do: Timex.end_of_month(Map.from_struct(d))
def end_of_month(_datetime), do: {:error, :invalid_date}
def quarter(%{__struct__: _} = d), do: Timex.quarter(Map.from_struct(d))
def quarter(_datetime), do: {:error, :invalid_date}
def days_in_month(%{__struct__: _} = d), do: Timex.days_in_month(Map.from_struct(d))
def days_in_month(_datetime), do: {:error, :invalid_date}
def week_of_month(%{__struct__: _} = d), do: Timex.week_of_month(Map.from_struct(d))
def week_of_month(_datetime), do: {:error, :invalid_date}
def weekday(%{__struct__: _} = d), do: Timex.weekday(Map.from_struct(d))
def weekday(_datetime), do: {:error, :invalid_date}
def weekday(%{__struct__: _} = d, weekstart), do: Timex.weekday(Map.from_struct(d), weekstart)
def weekday(_datetime, _weekstart), do: {:error, :invalid_date}
def day(%{__struct__: _} = d), do: Timex.day(Map.from_struct(d))
def day(_datetime), do: {:error, :invalid_date}
def is_valid?(%{__struct__: _} = d), do: Timex.is_valid?(Map.from_struct(d))
def is_valid?(_datetime), do: false
def iso_week(%{__struct__: _} = d), do: Timex.iso_week(Map.from_struct(d))
def iso_week(_datetime), do: {:error, :invalid_date}
def from_iso_day(%{__struct__: _} = d, _day), do: Timex.from_iso_day(Map.from_struct(d))
def from_iso_day(_datetime, _day), do: {:error, :invalid_date}
end