--- title: "Calendrical Computations Using `tind` Package" author: "Grzegorz Klima" output: rmarkdown::html_vignette vignette: > %\VignetteIndexEntry{Calendrical Computations Using `tind` Package} %\VignetteEngine{knitr::rmarkdown} \usepackage[utf8]{inputenc} --- `tind` package provides an extensive set of functions for calendrical computations. These include: extraction of time index components, determination of properties of time periods (leap years, number of subperiods within a period), computation of dates of moveable observances, and calendar arithmetic. For applications in business and finance, computations involving business days (using different rolling conventions) and computation of accrual factors are provided. ```{r echo=FALSE} options(crayon.enabled = TRUE) knitr::knit_hooks$set(output = function(x, options) { paste0( '
',
    fansi::sgr_to_html(x = htmltools::htmlEscape(x), warn = FALSE),
    '
' ) }) ``` Before proceeding, let us load the package. ```{r color} library("tind") ``` ## Current Date and Time Current date and time can be checked using functions `today()` and `now()`. ```{r} today() now() ``` By default, the system time zone (as returned by `Sys.timezone()`) is used. However, we can check the current time in other time zones. Let us do it for Tokyo, Warsaw, London, and New York. ```{r} now("Asia/Tokyo") today("Asia/Tokyo") now("Europe/Warsaw") today("Europe/Warsaw") now("Europe/London") today("Europe/London") now("America/New_York") today("America/New_York") ``` ## Time Index Components (Year, Month, Day, ...) `tind` package provides a collection of intuitively-named functions for extracting time index components, see examples below. ```{r} (nw <- now()) year(nw) quarter(nw) month(nw) month(nw, labels = TRUE) month(nw, labels = TRUE, abbreviate = FALSE) week(nw) day(nw) day_of_week(nw) day_of_week(nw, labels = TRUE) day_of_week(nw, labels = TRUE, abbreviate = FALSE) day_of_year(nw) hour(nw) am(nw) pm(nw) minute(nw) second(nw) ``` ## Properties of Time Indices The inherent irregularity of our calendar is well known. Years are divided into leap and non-leap ones, months have different numbers of days (and so do quarters and years), years have varying number of weeks (52 or 53 in ISO 8601). In many time zones Daylight Saving Time is used, which then leads to some days having more or less than 24 hours. The functions shown below allow to easily check properties of time indices at hand. ```{r} (tm <- tind(y = 2023:2024, m = c(10, 2), d = 29, H = 1, tz = "Europe/Warsaw")) is.leap_year(tm) days_in_year(tm) weeks_in_year(tm) days_in_quarter(tm) days_in_month(tm) hours_in_day(tm) is.dst(tm) ``` ## Calendar Arithmetic Time indices can be shifted using `+` and `-` operators and `-` can be used to compute time differences (returning objects of auxiliary `tdiff` class). ```{r} (x <- today()) x + 0:3 x - 0:3 x - 3:0 x - as.date("2000-01-01") ``` Operators `%+y%`, `%-y%`, `%+m%`, `%-m%`, `%+d%`, `%-d%`, etc. can be used to shift time indices by multiples of a given time unit. An alternative is to use time differences constructed by `as.tdiff` method or convenience functions `years`, `qrtrs`, `mnths`, `weeks`, `days`, `hours`, `mins`, and `secs`. (The naming was chosen in order to avoid conflicts with base functions `months` and `quarters`). ```{r} (x <- today()) x %+y% -1:1 x + years(-1:1) x %+m% -1:1 x + mnths(-1:1) x %+d% -1:1 # same as x + -1:1 (x <- now()) x %-h% 3:0 x - hours(3:0) x %-min% 3:0 x - mins(3:0) x %-s% 3:0 # same as x - 3:0 x - secs(3:0) # same as x - 3:0 ``` ## Sequences of Time Indices `seq` method for `tind` allows to easily construct series of time indices. The following code creates a sequence of months with a step of two months from November 2023 till April 2025: ```{r} seq(as.month("2023-11"), as.month("2025-04"), by = 2) ``` Consider the following problem: for a given month we need to construct a sequence of dates in that month. Note that the last day in a month is the day before the first day in the following month. The following piece of code does the job: ```{r} (m <- as.month("2025-03")) seq(as.date(m), as.date(m + 1) - 1) ``` `seq` method for `tind` also accepts different types of time indices and automatically converts them. The two lines below create sequences from the beginning of current quarter till today and from today till the end of the quarter: ```{r} (td <- today()) seq(as.quarter(td), td) seq(td, as.quarter(td)) ``` ## Rounding Time Indices `floor_t` and `ceiling_t` round time indices down or up given time unit or multiple of a unit. `round_t` returns the result of one of these two functions, whichever is closer. ```{r} (x <- tind(y = 2025, m = 3, d = 30)) floor_t(x, "w") ceiling_t(x, "w") round_t(x, "w") floor_t(x, "m") ceiling_t(x, "m") round_t(x, "m") floor_t(x, "2m") ceiling_t(x, "2m") round_t(x, "2m") floor_t(x, "q") ceiling_t(x, "q") round_t(x, "q") floor_t(x, "y") ceiling_t(x, "y") round_t(x, "y") ``` `trunc_t` truncates time indices changing type when possible. ```{r} (x <- date_time(x, H = "13:13:13.13")) trunc_t(x, "s") trunc_t(x, "min") trunc_t(x, "h") trunc_t(x, "d") trunc_t(x, "w") trunc_t(x, "m") trunc_t(x, "q") trunc_t(x, "y") ``` ## Determining Dates of Movable Observances / Calendar Events It is common to define dates of movable holidays or observances as the first, second, third, fourth, or last occurrence of a particular day of week in a given month. In order to determine dates of such observances two functions can be used: `nth_dw_in_month()` and `last_dw_in_month()`. Days of week are numbered 1--7 with 1 denoting Monday and 7 Sunday (as in ISO 8601). Thanksgiving in the US is observed on the fourth Thursday of November, which in 2019 was on: ```{r} nth_dw_in_month(4, 4, 201911) ``` Black Friday is the day after Thanksgiving: ```{r} nth_dw_in_month(4, 4, 201911) + 1 ``` Daylight Saving Time in the EU begins (at least for now) on the last Sunday in March and ends on the last Sunday of October. In 2019 the two dates were: ```{r} last_dw_in_month(7, 201903) last_dw_in_month(7, 201910) ``` During change to DST, the day is one hour shorter and during change back one hour longer: ```{r} hours_in_day(last_dw_in_month(7, 201903), "Europe/Warsaw") hours_in_day(last_dw_in_month(7, 201910), "Europe/Warsaw") ``` In the Christian world, Easter is a very important feast and other many observances are set relative to it. Easter date in a given year can be determined using `easter()` function. ```{r} easter(2020:2025) ``` ## Creating and Working with Custom Calendars The functions shown above can be used to create custom calendar functions, which can later be used for pretty printing calendars or calculations involving business days. It is expected that a calendar function: * takes vector of dates * returns one of the three: * a logical vector of the same length as argument indicating business days, * a list of two logical vectors of the same length as argument indicating business days and holidays, * a list of three logical vectors of the same length as argument indicating business days, holidays, and other observances. If a calendar function returns a list and second or third logical vector is named, the names are used by function `calendar()` when pretty printing calendar. Two examples of calendar functions are provided below and can be used as templates for creating calendar functions for any country, area, or specific applications like trading days or shopping days. ### An Example - Business Days and (Federal) Holidays in the US Public holidays in the US have either fixed dates or are on nth or last day of week in a month. The calendar function below returns a two-element list (business days and holidays). A programming trick (marked in the code) is used to name holidays. ```{r} calendar_US <- function(dd) { dd <- as.tind(dd) y <- year(dd) m <- month(dd) d <- day(dd) newyear <- (m == 1) & (d == 1) martinlking <- (y >= 2000) & (m == 1) & (dd == nth_dw_in_month(3, 1, dd)) presidentsday <- (m == 2) & (dd == nth_dw_in_month(3, 1, dd)) memorialday <- (m == 5) & (dd == last_dw_in_month(1, dd)) juneteenth <- (y >= 2021) & (m == 6) & (d == 19) independence <- (m == 7) & (d == 4) labor <- (m == 9) & (dd == nth_dw_in_month(1, 1, dd)) columbus <- (m == 10) & (dd == nth_dw_in_month(2, 1, dd)) veterans <- (m == 11) & (d == 11) thanksgiving <- (m == 11) & (dd == nth_dw_in_month(4, 4, dd)) christmas <- (m == 12) & (d == 25) holiday <- newyear | martinlking | presidentsday | memorialday | juneteenth | independence | labor | columbus | veterans | thanksgiving | christmas # holiday names - a programming trick # names of holnms should be the same as names of logical vectors above names(holiday) <- rep("", length(holiday)) holnms <- c(newyear = "New Year's Day", martinlking = "Birthday of Martin Luther King, Jr.", presidentsday = "Washington's Birthday", memorialday = "Memorial Day", juneteenth = "Juneteenth National Independence Day", independence = "Independence Day", labor = "Labor Day", columbus = "Columbus Day", veterans = "Veterans Day", thanksgiving = "Thanksgiving Day", christmas = "Christmas Day") lapply(names(holnms), function(nm) names(holiday)[get(nm)] <<- holnms[nm]) # business days business <- !holiday & (day_of_week(dd) %in% 1:5) return (list(business = business, holiday = holiday)) } ``` Now we can use our function to print the calendar. Note that to obtain coloured output you have to have package `crayon` installed. ```{r} calendar(2020, calendar_US) calendar(as.year(today()), calendar_US) calendar("2020-01", calendar_US) calendar(calendar = calendar_US) ``` ### An example - Business Days, Holidays, and Other Observances in Poland Public holidays in Poland have either fixed dates or are placed relative to Easter. The calendar function below returns a three-element list (business days, holidays, and other observances). The same programming trick as before is used to name holidays and other observances. ```{r} calendar_PL <- function(dd) { dd <- as.tind(dd) y <- year(dd) m <- month(dd) d <- day(dd) # public holidays newyear <- (m == 1L) & (d == 1L) epiphany <- (y >= 2011L) & (m == 1L) & (d == 6L) easterd <- easter(dd) == dd eastermon <- easter(dd) + 1L == dd labour <- (m == 5L) & (d == 1L) constitution <- (m == 5L) & (d == 3L) pentecost <- easter(dd) + 49L == dd corpuschristi <- easter(dd) + 60L == dd assumption <- (m == 8L) & (d == 15L) allsaints <- (m == 11L) & (d == 1L) independence <- (m == 11L) & (d == 11L) christmaseve <- (m == 12L) & (d == 24L) & (y >= 2025) christmas <- (m == 12L) & (d == 25L) christmas2 <- (m == 12L) & (d == 26L) holiday <- newyear | epiphany | easterd | eastermon | labour | constitution | pentecost | corpuschristi | assumption | allsaints | independence | christmaseve | christmas | christmas2 # holiday names names(holiday) <- rep("", length(holiday)) holnms <- c(newyear = "New Year", epiphany = "Epiphany", easterd = "Easter", eastermon = "Easter Monday", labour = "Labour Day", constitution = "Constitution Day", pentecost = "Pentecost", corpuschristi = "Corpus Christi", assumption = "Assumption of Mary", allsaints = "All Saints Day", independence = "Independence Day", christmaseve = "Christmas Eve", christmas = "Christmas", christmas2 = "Christmas (2nd day)") lapply(names(holnms), function(nm) names(holiday)[get(nm)] <<- holnms[nm]) # working/business days work <- !holiday & (day_of_week(dd) <= 5L) # other observances fatthursday <- easter(dd) - 52L == dd shrovetuesday <- easter(dd) - 47L == dd ashwednesday <- easter(dd) - 46L == dd goodfriday <- easter(dd) - 2L == dd primaaprilis <- (m == 4L) & (d == 1L) flagday <- (m == 5L) & (d == 2L) mothersday <- (m == 5L) & (d == 26L) childrensday <- (m == 6L) & (d == 1L) saintjohnseve <- (m == 6L) & (d == 23L) allsoulsday <- (m == 11L) & (d == 2L) saintandrewseve <- (m == 11L) & (d == 29L) saintnicholasday <- (m == 12L) & (d == 6L) christmaseve <- (m == 12L) & (d == 24L) & (y < 2025) newyeareve <- (m == 12L) & (d == 31L) other <- fatthursday | shrovetuesday | ashwednesday | goodfriday | primaaprilis | flagday | mothersday | childrensday | saintjohnseve | allsoulsday | saintandrewseve | saintnicholasday | christmaseve | newyeareve names(other) <- rep("", length(other)) othernms <- c(fatthursday = "Fat Thursday", shrovetuesday = "Shrove Tuesday", ashwednesday = "Ash Wednesday", goodfriday = "Good Friday", primaaprilis = "All Fool's Day", flagday = "Flag Day", mothersday = "Mother's Day", childrensday = "Children's Day", saintjohnseve = "Saint John's Eve", allsoulsday = "All Souls' Day", saintandrewseve = "Saint Andrew's Eve", saintnicholasday = "Saint Nicholas Day", christmaseve = "Christmas Eve", newyeareve = "New Year's Eve") lapply(names(othernms), function(nm) names(other)[get(nm)] <<- othernms[nm]) return (list(work = work, holiday = holiday, other = other)) } ``` ```{r} calendar(2020, calendar_PL) calendar(as.year(today()), calendar_PL) calendar("2020-06", calendar_PL) calendar(calendar = calendar_PL) ``` ## Business Days Calendar functions described above can be used for business day computations. `tind` provides the following functions that can be used in this context: * `bizday()` for finding the closest business day (using different date rolling conventions) if the given day is not a one, * `bizday_diff()` for computing number of business days between two dates, * functions determining the first and the last business day in a given period: `first_bizday_in_month()`, `last_bizday_in_month()`, `first_bizday_in_quarter()`, `last_bizday_in_quarter()`, * functions determining the number of business days in a given period: `bizdays_in_month()`, `bizdays_in_quarter()`, `bizdays_in_year()`. Some examples using `bizday()` and calendar functions presented earlier are provided below (January 16, 2023 was a public holiday in the US and August 15, 2023 in Poland): ```{r} calendar("2023-01", calendar = calendar_US) bizday("2023-01-15", "p", calendar_US) bizday("2023-01-15", "f", calendar_US) bizday("2023-01-15", "mf2", calendar_US) calendar("2023-08", calendar = calendar_PL) bizday("2023-08-15", "p", calendar_PL) bizday("2023-08-15", "f", calendar_PL) bizday("2023-08-15", "mf2", calendar_PL) ``` The code below creates a table summarising the number of business days in months in 2023 in Poland and the US. ```{r} m <- as.month("2023-01") + 0:11 data.frame(month = m, PL = bizdays_in_month(m, calendar_PL), US = bizdays_in_month(m, calendar_US)) ``` ## Year Fractions and Accrual Factors `year_frac` function returns time indices as year fractions in the form year + year fraction, where fraction is the number of periods (quarters, months, weeks, days, seconds) since the beginning of the year over total number of periods in the year. ```{r} (d <- as.date("2024-05-05")) year_frac(d) year(d) + (day_of_year(d) - 1) / days_in_year(d) as.quarter(d) year_frac(as.quarter(d)) year(d) + (quarter(d) - 1) / 4 as.month(d) year_frac(as.month(d)) year(d) + (month(d) - 1) / 12 as.week(d) year_frac(as.week(d)) year(d) + (week(d) - 1) / weeks_in_year(d) ``` `daycount_frac` allows to compute accrual factor using different day count conventions. Below we create a sequence of 5 consecutive International Money Market days and compute accrual factors between the first and the following four using different conventions: ```{r} (imm <- nth_dw_in_month(3, 3, tind(y = 2025, m = 12) + 3 * (0:4))) daycount_frac(imm[1L], imm[-1L], "30/360") daycount_frac(imm[1L], imm[-1L], "30E/360") daycount_frac(imm[1L], imm[-1L], "ACT/ACT") daycount_frac(imm[1L], imm[-1L], "ACT/365F") daycount_frac(imm[1L], imm[-1L], "ACT/360") ```