R/iso8601_regexp.R

Defines functions make_full_pattern pattern_ISO8601_truncated_helper pattern_ISO8601_timezone pattern_ISO8601_time pattern_ISO8601_ordinal_day pattern_ISO8601_week_day pattern_ISO8601_week_week pattern_ISO8601_calendar_day pattern_ISO8601_calendar_month pattern_ISO8601_calendar_year pattern_ISO8601_date_builder pattern_ISO8601_ordinal_date pattern_ISO8601_week_date pattern_ISO8601_calendar_date pattern_ISO8601_any_date pattern_ISO8601_datetime_builder pattern_ISO8601_ordinal_datetime pattern_ISO8601_week_datetime pattern_ISO8601_calendar_datetime pattern_ISO8601_any_datetime is_ISO8601_timezone is_ISO8601_time is_ISO8601_ordinal_date is_ISO8601_week_date is_ISO8601_calendar_date is_ISO8601_ordinal_datetime is_ISO8601_week_datetime is_ISO8601_calendar_datetime

Documented in is_ISO8601_calendar_date is_ISO8601_calendar_datetime is_ISO8601_ordinal_date is_ISO8601_ordinal_datetime is_ISO8601_time is_ISO8601_timezone is_ISO8601_week_date is_ISO8601_week_datetime pattern_ISO8601_any_date pattern_ISO8601_any_datetime pattern_ISO8601_calendar_date pattern_ISO8601_calendar_datetime pattern_ISO8601_calendar_year pattern_ISO8601_date_builder pattern_ISO8601_ordinal_date pattern_ISO8601_ordinal_datetime pattern_ISO8601_time pattern_ISO8601_timezone pattern_ISO8601_week_date pattern_ISO8601_week_datetime

# Boolean functions ####

#' Determine if a string matches an ISO 8601 date/time using calendar date
#' notation
#'
#' @param x A vector of character strings to test
#' @inheritDotParams pattern_ISO8601_calendar_datetime
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_calendar_datetime <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_calendar_datetime(...))
  )
}

#' Determine if a string matches an ISO 8601 date/time using week date notation
#'
#' @param x A vector of character strings to test
#' @inheritDotParams pattern_ISO8601_week_datetime
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_week_datetime <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_week_datetime(...))
  )
}

#' Determine if a string matches an ISO 8601 date/time using ordinal date notation
#'
#' @param x A vector of character strings to test
#' @inheritDotParams pattern_ISO8601_ordinal_datetime
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_ordinal_datetime <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_ordinal_datetime(...))
  )
}

#' Determine if a string is an ISO 8601 calendar date
#'
#' @inheritParams is_ISO8601_calendar_datetime
#' @inheritDotParams pattern_ISO8601_calendar_date
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_calendar_date <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_calendar_date(...))
  )
}

#' @describeIn is_ISO8601_calendar_date Determine if a string is an ISO 8601
#'   week date
#' @export
is_ISO8601_week_date <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_week_date(...))
  )
}

#' @describeIn is_ISO8601_calendar_date Determine if a string is an ISO 8601
#'   ordinal date
#' @export
is_ISO8601_ordinal_date <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_ordinal_date(...))
  )
}

#' Determine if a string is an ISO 8601 time
#'
#' @inheritParams is_ISO8601_calendar_datetime
#' @inheritDotParams pattern_ISO8601_time
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_time <- function(x, ...) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_time(...))
  )
}

#' Determine if a string is an ISO 8601 timezone
#'
#' @inheritParams is_ISO8601_calendar_datetime
#' @family ISO8601 String checking
#' @family Date management and conversion
#' @export
is_ISO8601_timezone <- function(x) {
  grepl(
    x=as.character(x),
    pattern=make_full_pattern(pattern_ISO8601_timezone())
  )
}

# Date/time functions ####

#' Generate a regular expression matching any ISO8601 date format with time
#'
#' @param truncated Should the date/time be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=second is required, 1=minute,
#'   2=hour, 3=day, 4=month or week, 5=year, 6=none).  A value of 6 will allow an empty
#'   string to match.
#' @param ... Passed to \code{pattern_ISO8601_any_date()} and
#'   \code{pattern_ISO8601_time()}
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_any_datetime <- function(truncated=0, ...) {
  pattern_ISO8601_datetime_builder(
    truncated=truncated,
    ...,
    pattern_date_fun=pattern_ISO8601_any_date
  )
}

#' Generate a regular expression matching an ISO8601 calendar date with time
#'
#' @param truncated Should the date/time be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=second is required, 1=minute,
#'   2=hour, 3=day, 4=month, 5=year, 6=none).  A value of 6 will allow an empty
#'   string to match.
#' @param ... Passed to \code{pattern_ISO8601_calendar_date()} and
#'   \code{pattern_ISO8601_time()}
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_calendar_datetime <- function(truncated=0, ...) {
  pattern_ISO8601_datetime_builder(
    truncated=truncated,
    ...,
    pattern_date_fun=pattern_ISO8601_calendar_date
  )
}

#' Generate a regular expression matching an ISO8601 week date with time
#'
#' @param truncated Should the date/time be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=second is required, 1=minute,
#'   2=hour, 3=day, 4=week, 5=year, 6=none).  A value of 6 will allow an empty
#'   string to match.
#' @param ... Passed to \code{pattern_ISO8601_week_date()} and
#'   \code{pattern_ISO8601_time()}
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_week_datetime <- function(truncated=0, ...) {
  pattern_ISO8601_datetime_builder(
    truncated=truncated,
    ...,
    pattern_date_fun=pattern_ISO8601_week_date
  )
}

#' Generate a regular expression matching an ISO8601 ordinal date with time
#'
#' @param truncated Should the date/time be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=second is required, 1=minute,
#'   2=hour, 3 or 4=day, 5=year, 6=none).  A value of 6 will allow an empty
#'   string to match.
#' @param ... Passed to \code{pattern_ISO8601_week_date()} and
#'   \code{pattern_ISO8601_time()}
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_ordinal_datetime <- function(truncated=0, ...) {
  pattern_ISO8601_datetime_builder(
    truncated=truncated,
    ...,
    pattern_date_fun=pattern_ISO8601_ordinal_date
  )
}

pattern_ISO8601_datetime_builder <- function(truncated, ..., pattern_date_fun) {
  stopifnot(is.numeric(truncated))
  stopifnot(length(truncated) == 1)
  stopifnot(truncated >= 0 & truncated <= 6)
  # Min truncated is 2 because 3 will be handled below with the addition of the
  # "T"
  pattern_time <- pattern_ISO8601_time(truncated=min(truncated, 3), require_T=TRUE, ...)
  pattern_time_aug <-
    pattern_ISO8601_truncated_helper(
      pattern_time,
      allow_truncation=truncated >= 3
    )
  pattern_date_fun(
    truncated=max(truncated - 3, 0),
    pattern_time=pattern_time_aug,
    ...
  )
}

# Date functions ####

#' Generate a regular expression matching any ISO8601 date format
#'
#' @details This matches YYYY-MM-DD (year-month-day), YYYY-Www-d, or YYYY-DDD
#'   formats. Basic format (without dashes) is not supported.
#'
#' @param truncated Should the date be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=day is required, 1=month or week, and
#'   2=year).  A value of 3 will allow an empty string to match.
#' @inheritParams pattern_ISO8601_calendar_year
#' @inheritParams pattern_ISO8601_date_builder
#' @param ... Ignored
#' @references https://en.wikipedia.org/wiki/ISO_8601
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_any_date <- function(truncated=0, allow_before_year_1583=FALSE, pattern_time="", ...) {
  pattern_calendar <-
    pattern_ISO8601_date_builder(
      truncated=pmin(truncated, 1),
      pattern_year=NULL,
      pattern_middle=pattern_ISO8601_calendar_month(),
      # Not confirming that the date is valid for the month
      pattern_day=pattern_ISO8601_calendar_day(),
      pattern_time=pattern_time
    )
  pattern_week <-
    pattern_ISO8601_date_builder(
      truncated=pmin(truncated, 1),
      pattern_year=NULL,
      pattern_middle=pattern_ISO8601_week_week(),
      # Not confirming that the date is valid for the month
      pattern_day=pattern_ISO8601_week_day(),
      pattern_time=pattern_time
    )
  pattern_ordinal <-
    pattern_ISO8601_date_builder(
      truncated=pmin(truncated, 1),
      pattern_year=NULL,
      pattern_day=pattern_ISO8601_ordinal_day(),
      pattern_time=pattern_time
    )
  pattern_less_than_year <-
    pattern_ISO8601_truncated_helper(
      sprintf("(?:%s|%s|%s)", pattern_calendar, pattern_week, pattern_ordinal),
      allow_truncation=truncated >= 2
    )
  pattern_year_prep <- pattern_ISO8601_calendar_year(allow_before_year_1583=allow_before_year_1583)
  pattern_ISO8601_truncated_helper(
    paste0(pattern_year_prep, pattern_less_than_year),
    allow_truncation=truncated >= 3
  )
}

#' Generate a regular expression matching an ISO8601 calendar date
#'
#' @details This matches the general pattern of YYYY-MM-DD (year-month-day).
#'   Basic format (without dashes) is not supported.
#'
#' @param truncated Should the date be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=day is required, 1=month, and
#'   2=year).  A value of 3 will allow an empty string to match.
#' @inheritParams pattern_ISO8601_calendar_year
#' @inheritParams pattern_ISO8601_date_builder
#' @param ... Ignored
#' @references https://en.wikipedia.org/wiki/ISO_8601
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_calendar_date <- function(truncated=0, allow_before_year_1583=FALSE, pattern_time="", ...) {
  pattern_ISO8601_date_builder(
    truncated=truncated,
    pattern_middle=pattern_ISO8601_calendar_month(),
    # Not confirming that the date is valid for the month
    pattern_day=pattern_ISO8601_calendar_day(),
    pattern_time=pattern_time,
    allow_before_year_1583=allow_before_year_1583
  )
}

#' Generate a regular expression matching an ISO8601 week date
#'
#' @details This matches the general pattern of yyyy-Www-d (year-week of
#'   year-day of week). Basic format (without dashes) is not supported.
#'
#' @param truncated Should the date be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=day is required, 1=week, and
#'   2=year).  A value of 3 will allow an empty string to match.
#' @inheritParams pattern_ISO8601_calendar_date
#' @references https://en.wikipedia.org/wiki/ISO_8601
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_week_date <- function(truncated=0, allow_before_year_1583=FALSE, pattern_time="", ...) {
  pattern_ISO8601_date_builder(
    truncated=truncated,
    pattern_middle=pattern_ISO8601_week_week(),
    pattern_day=pattern_ISO8601_week_day(),
    pattern_time=pattern_time,
    allow_before_year_1583=allow_before_year_1583
  )
}

#' Generate a regular expression matching an ISO8601 ordinal date
#'
#' @param truncated Should the date be allowed to be truncated?  An integer
#'   indicating the highest required precision (0 or 1=day is required and
#'   2=year).  A value of 3 will allow an empty string to match.
#' @details This matches the general pattern of yyyy-ddd (year-day of year).
#'   Leap days are allowed, but the year is not confirmed to be a leap year.
#'   Basic format (without dashes) is not supported.
#'
#' @inheritParams pattern_ISO8601_calendar_date
#' @references https://en.wikipedia.org/wiki/ISO_8601
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_ordinal_date <- function(truncated=0, allow_before_year_1583=FALSE, pattern_time="", ...) {
  truncated_prep <-
    ifelse(
      truncated == 1,
      0,
      truncated
    )
  pattern_ISO8601_date_builder(
    truncated=truncated_prep,
    pattern_day=pattern_ISO8601_ordinal_day(),
    pattern_time=pattern_time,
    allow_before_year_1583=allow_before_year_1583
  )
}

#' Build an ISO8601 date pattern from the year, middle (month or week), day, and
#' time patterns.
#'
#' @param pattern_year The pattern for the ISO8601 year (ignored if
#'   \code{allow_before_year_1583} is given); if null, only the middle and day
#'   part will be returned without a year part.
#' @param pattern_middle The pattern for the ISO8601 month or week (or NULL for ordinal dates)
#' @param pattern_day The pattern for the ISO8601 day
#' @param pattern_time A string to add to the day for including time with date
#'   (see \code{pattern_ISO8601_time()})
#' @inheritParams pattern_ISO8601_calendar_year
#' @inheritParams pattern_ISO8601_any_datetime
#' @keywords Internal
pattern_ISO8601_date_builder <- function(truncated, pattern_year, pattern_middle, pattern_day, pattern_time, allow_before_year_1583) {
  stopifnot(is.numeric(truncated))
  stopifnot(length(truncated) == 1)
  stopifnot(truncated >= 0 & truncated <= 3)

  stopifnot(is.character(pattern_time))
  stopifnot(length(pattern_time) == 1)
  stopifnot(!is.na(pattern_time))

  pattern_day_prep <-
    paste0(
      "-",
      if (nchar(pattern_time) == 0) {
        pattern_day
      } else {
        sprintf("(?:%s%s)", pattern_day, pattern_time)
      }
    )
  pattern_day_aug <- pattern_ISO8601_truncated_helper(pattern_day_prep, allow_truncation=truncated >= 1)
  if (!missing(pattern_middle)) {
    pattern_middle_prep <- paste0("-", pattern_middle)
    pattern_middle_aug <-
      pattern_ISO8601_truncated_helper(
        paste0(pattern_middle_prep, pattern_day_aug),
        allow_truncation=truncated >= 2
      )
  } else {
    pattern_middle_aug <- pattern_day_aug
  }
  if (missing(allow_before_year_1583)) {
    pattern_year_prep <- pattern_year
  } else {
    pattern_year_prep <- pattern_ISO8601_calendar_year(allow_before_year_1583=allow_before_year_1583)
  }
  if (is.null(pattern_year_prep)) {
    ret <- pattern_middle_aug
  } else {
    ret <-
      pattern_ISO8601_truncated_helper(
        paste0(pattern_year_prep, pattern_middle_aug),
        allow_truncation=truncated >= 3
      )
  }
  ret
}

## Individual date pattern parts ####

#' Generate a regular expression matching an ISO8601 calendar year
#'
#' @details   Sign on the year (+ or -) is not supported (therefore years before
#'   0000 are not supported). Years after 9999 are not supported.
#'
#' @param allow_before_year_1583 Should years between 0 and 1582 be allowed
#'   (they are only allowed in ISO 8601 with mutual agreement)
#' @references https://en.wikipedia.org/wiki/ISO_8601
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_calendar_year <- function(allow_before_year_1583=FALSE) {
  stopifnot(is.logical(allow_before_year_1583))
  stopifnot(!is.na(allow_before_year_1583))

  if (allow_before_year_1583) {
    "([0-9]{4})"
  } else {
    "(158[3-9]|159[0-9]|1[6-9][0-9]{2}|[2-9][0-9]{3})"
  }
}

pattern_ISO8601_calendar_month <- function() {
  "(0[1-9]|1[0-2])"
}

pattern_ISO8601_calendar_day <- function() {
  "(0[1-9]|[12][0-9]|3[01])"
}

pattern_ISO8601_week_week <- function() {
  "W(0[1-9]|[1-4][0-9]|5[0-3])"
}

pattern_ISO8601_week_day <- function() {
  "([1-7])"
}

pattern_ISO8601_ordinal_day <- function() {
  "(00[1-9]|0[1-9][0-9]|[12][0-9]{2}|3[0-5][0-9]|36[0-6])"
}

# Time functions ####

#' Generate a regular expression for ISO 8601 times
#'
#' Fractional hours and minutes are not allowed in the generated regular
#' expressions even though they are allowed in the ISO 8601 standard.
#'
#' Leap seconds occur at 23:59:60Z at the end of the month.  Given timezone
#' differences which could occur on different hours and at the end of any
#' 15-minute point, \code{allow_leap_seconds} only makes a change to allow for
#' 60 seconds in any minute.
#'
#' @param truncated Should the date be allowed to be truncated?  An integer
#'   indicating the highest required precision (0=second is required, 1=minute, and
#'   2=hour).  A value of 3 will allow an empty string to match.
#' @param allow_fractional_hours,allow_fractional_minutes Not yet supported
#' @param allow_fractional_seconds Should fractional seconds be allowed?
#' @param fractional_digits How many digits are allowed in fractional time
#'   units?  \code{0} indicates that they are not allowed, and \code{Inf}
#'   indicates any number are allowed)
#' @param pattern_decimal_mark What regular expression patterns should be
#'   allowed for the decimal mark?  Comma and period are both allowed in the
#'   standard and by default.
#' @param timezone Should the timezone be optional (NA, default), required
#'   (TRUE), or omitted (FALSE)?
#' @param allow_leap_second Should leap seconds (61 seconds in a minute) be
#'   allowed?
#' @param require_T Require the "T" at the beginning of the time?  TRUE, FALSE,
#'   or NA (optional)
#' @param ... Passed to \code{pattern_ISO8601_timezone()}
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @export
pattern_ISO8601_time <- function(truncated=0,
                                 allow_fractional_hours=FALSE,
                                 allow_fractional_minutes=FALSE,
                                 allow_fractional_seconds=TRUE,
                                 fractional_digits=Inf, pattern_decimal_mark=c("\\.", ","),
                                 timezone=NA, allow_leap_second=FALSE, require_T=NA,
                                 ...) {
  stopifnot(is.numeric(truncated))
  stopifnot(length(truncated) == 1)
  stopifnot(!is.na(truncated))
  stopifnot(truncated >= 0 & truncated <= 3)

  stopifnot(is.logical(allow_fractional_hours))
  stopifnot(length(allow_fractional_hours) == 1)
  stopifnot(isFALSE(allow_fractional_hours))
  stopifnot(is.logical(allow_fractional_minutes))
  stopifnot(length(allow_fractional_minutes) == 1)
  stopifnot(isFALSE(allow_fractional_minutes))
  stopifnot(is.logical(allow_fractional_seconds))
  stopifnot(length(allow_fractional_seconds) == 1)
  stopifnot(!is.na(allow_fractional_seconds))

  stopifnot(is.numeric(fractional_digits))
  stopifnot(length(fractional_digits) == 1)
  stopifnot(fractional_digits >= 0)

  pattern_decimal_mark <- match.arg(pattern_decimal_mark, several.ok=TRUE)
  if (length(pattern_decimal_mark) > 1) {
    pattern_decimal_mark <- paste0("[", paste0(pattern_decimal_mark, collapse=""), "]")
  }

  stopifnot(is.logical(timezone))
  stopifnot(length(timezone) == 1)

  stopifnot(is.logical(allow_leap_second))
  stopifnot(length(allow_leap_second) == 1)
  stopifnot(!is.na(allow_leap_second))

  stopifnot(is.logical(require_T))
  stopifnot(length(require_T) == 1)

  pattern_hour_number <- "([01][0-9]|2[0-3])"
  pattern_hour_t <- paste0("T", pattern_hour_number)
  if (is.na(require_T)) {
    pattern_hour <- paste0("T?", pattern_hour_number)
  } else if (require_T) {
    pattern_hour <- pattern_hour_t
  } else {
    pattern_hour <- pattern_hour_number
  }
  pattern_minute <- ":([0-5][0-9])"
  partial_pattern_second_whole <-
    if (allow_leap_second) {
      "[0-5][0-9]|60"
    } else {
      "[0-5][0-9]"
    }
  pattern_fraction <-
    if (is.infinite(fractional_digits)) {
      paste0(pattern_decimal_mark, "[0-9]+")
    } else {
      sprintf("%s[0-9]{1,%g}", pattern_decimal_mark, fractional_digits)
    }
  if (fractional_digits == 0) {
    pattern_fraction <- ""
  } else {
    # fractions are always optional
    pattern_fraction <- sprintf("(?:%s)?", pattern_fraction)
  }
  pattern_second <-
    if (allow_fractional_seconds) {
      sprintf(
        ":((?:%s)%s)",
        partial_pattern_second_whole,
        pattern_fraction
      )
    } else {
      sprintf(":(%s)", partial_pattern_second_whole)
    }

  # Put it together
  pattern_second_aug <- pattern_ISO8601_truncated_helper(pattern_second, allow_truncation=truncated >= 1)
  # To support the requirement of "T", pattern_minute_aug is not truncated by
  # itself.
  pattern_minute_aug <- paste0(pattern_minute, pattern_second_aug)
  # T is required at the beginning for hour-only times
  if (truncated >= 2) {
    pattern_hour_aug <-
      sprintf(
        "(?:%s|%s)",
        pattern_hour_t,
        paste0(pattern_hour, pattern_minute_aug)
      )
  } else {
    pattern_hour_aug <- paste0(pattern_hour, pattern_minute_aug)
  }
  if (!isFALSE(timezone)) {
    pattern_timezone <-
      pattern_ISO8601_truncated_helper(
        pattern_ISO8601_timezone(),
        allow_truncation=is.na(timezone)
      )
    ret <-
      pattern_ISO8601_truncated_helper(
        paste0(pattern_hour_aug, pattern_timezone),
        allow_truncation=truncated >= 3
      )
  } else {
    ret <-
      pattern_ISO8601_truncated_helper(
        paste0(pattern_hour, pattern_minute_aug),
        allow_truncation=truncated >= 3
      )
  }
  ret
}

#' Generate a regular expression for ISO 8601 timezones.
#'
#' @details Offsets are always multiples of 15 minutes, so if the minute part is
#'   provided, it must be 00, 15, 30, or 45.  The range allowed by this function
#'   is a bit wider than the current range as of the writing of this function.
#'   The allowed range is -14:45 to +14:45 while the actual widest range is
#'   currently -12:00 to +14:00.
#'
#'   According to the ISO standard, zero must always be preceded by a + and
#'   never a negative.  And, "Z" is allowed.
#'
#' @family ISO8601 patterns
#' @family Date management and conversion
#' @references https://en.wikipedia.org/wiki/UTC_offset
#' @export
pattern_ISO8601_timezone <- function() {
  zulu <- "Z"
  zero <- "\\+00(?::00)?"
  zero_with_minutes <- "[\\+-]00:(?:15|30|45)"
  nonzero <- "[\\+-](?:0[1-9]|1[1-4])(?::(?:00|15|30|45))?"
  paste0(
    "(",
    zulu, "|",
    zero, "|",
    zero_with_minutes, "|",
    nonzero,
    ")"
  )
}

# Helper functions ####

# Add optionality outside of the regular expression
pattern_ISO8601_truncated_helper <- function(x, allow_truncation) {
  ret <- x
  if (allow_truncation) {
    ret <- paste0("(?:", x, ")?")
  }
  ret
}

# A helper function to make the pattern bound the start and end of the string
make_full_pattern <- function(x) {
  sprintf("^%s$", x)
}
billdenney/Rsdtm documentation built on Dec. 7, 2022, 2:11 a.m.