defmodule Timex.Calendar.Julian do @moduledoc """ This module contains functions for working with dates in the Julian calendar. """ require Bitwise import Timex.Macros alias Timex.Types @doc """ Returns the Julian day number for the given Erlang date (gregorian) The Julian date (JD) is a continuous count of days from 1 January 4713 BC (= -4712 January 1), Greenwich mean noon (= 12h UT). For example, AD 1978 January 1, 0h UT is JD 2443509.5 and AD 1978 July 21, 15h UT, is JD 2443711.125. This algorithm assumes a proleptic Gregorian calendar (i.e. dates back to year 0), unlike the NASA or US Naval Observatory algorithm - however they align perfectly for dates back to October 15th, 1582, which is where it starts to differ, which is due to the fact that their algorithm assumes there is no Gregorian calendar before that date. """ @spec julian_date(Types.date()) :: integer def julian_date({year, month, day}), do: julian_date(year, month, day) # Same as julian_date/1, except takes an Erlang datetime, and returns a more precise Julian date number @spec julian_date(Types.datetime()) :: integer def julian_date({{year, month, day}, {hour, minute, second}}) do julian_date(year, month, day, hour, minute, second) end def julian_date(_), do: {:error, :invalid_date} @doc """ Same as julian_date/1, except takes year/month/day as distinct arguments """ @spec julian_date(Types.year(), Types.month(), Types.day()) :: integer def julian_date(year, month, day) when is_date(year, month, day) do a = div(14 - month, 12) y = year + 4800 - a m = month + 12 * a - 3 jdn = day + trunc((153 * m + 2) / 5) + 365 * y + div(y, 4) - div(y, 100) + div(y, 400) - 32045 jdn end def julian_date(_, _, _), do: {:error, :invalid_date} @doc """ Same as julian_date/1, except takes year/month/day/hour/minute/second as distinct arguments """ @spec julian_date( Types.year(), Types.month(), Types.day(), Types.hour(), Types.minute(), Types.second() ) :: float def julian_date(year, month, day, hour, minute, second) when is_datetime(year, month, day, hour, minute, second) do jdn = julian_date(year, month, day) jdn + (hour - 12) / 24 + minute / 1440 + second / 86400 end def julian_date(_, _, _, _, _, _), do: {:error, :invalid_datetime} @doc """ Given a Julian day of year, and a year, this function returns the `Date` which that day falls on. If no options are provided, leap days are disregarded, and the valid range for the day provided is 1-365, i.e. there is no representation for Feb 29. To allow representing leap days, you may pass `leaps: true`, which in turn expands the range of the day provided to 0-365. NOTE: This is internally used for POSIX-TZ support, but may be useful to others, so it is being made public. """ def date_for_day_of_year(day, year, opts \\ []) when is_integer(day) and day >= 0 and day <= 365 do leaps? = Keyword.get(opts, :leaps, false) day = if leaps?, do: day + 1, else: day days = if leaps? and :calendar.is_leap_year(year) do [31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] else [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] end {month, day} = Enum.reduce_while(days, {1, 0}, fn eom, {last_month, jd} -> next = eom + jd cond do day > next -> {:cont, {last_month + 1, next}} day == next -> {:halt, {last_month, eom}} day < next -> {:halt, {last_month, day - jd}} end end) Timex.Date.new!(year, month, day) end @doc """ Returns the day of the week, starting with 0 for Sunday, or 1 for Monday """ @spec day_of_week(Types.date(), :sun | :mon) :: Types.weekday() def day_of_week({year, month, day}, weekstart), do: day_of_week(year, month, day, weekstart) @doc """ Same as day_of_week/1, except takes year/month/day as distinct arguments """ @spec day_of_week(Types.year(), Types.month(), Types.day(), :sun | :mon) :: Types.weekday() def day_of_week(year, month, day, weekstart) when is_date(year, month, day) and weekstart in [:sun, :mon] do cardinal = mod(trunc(julian_date(year, month, day)) + 1, 7) case weekstart do :sun -> cardinal :mon -> mod(cardinal + 6, 7) + 1 end end def day_of_week(_, _, _, weekstart) do case weekstart in [:sun, :mon] do true -> {:error, :invalid_date} false -> {:error, {:bad_weekstart_value, expected: [:sun, :mon], got: weekstart}} end end defp mod(a, b), do: rem(rem(a, b) + b, b) end