UNPKG

@aws-cdk/core

Version:

AWS Cloud Development Kit Core Library

273 lines 32.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.Duration = void 0; const token_1 = require("./token"); /** * Represents a length of time. * * The amount can be specified either as a literal value (e.g: `10`) which * cannot be negative, or as an unresolved number token. * * When the amount is passed as a token, unit conversion is not possible. */ class Duration { constructor(amount, unit) { if (!token_1.Token.isUnresolved(amount) && amount < 0) { throw new Error(`Duration amounts cannot be negative. Received: ${amount}`); } this.amount = amount; this.unit = unit; } /** * Create a Duration representing an amount of milliseconds. * * @param amount the amount of Milliseconds the `Duration` will represent. * @returns a new `Duration` representing `amount` ms. */ static millis(amount) { return new Duration(amount, TimeUnit.Milliseconds); } /** * Create a Duration representing an amount of seconds. * * @param amount the amount of Seconds the `Duration` will represent. * @returns a new `Duration` representing `amount` Seconds. */ static seconds(amount) { return new Duration(amount, TimeUnit.Seconds); } /** * Create a Duration representing an amount of minutes. * * @param amount the amount of Minutes the `Duration` will represent. * @returns a new `Duration` representing `amount` Minutes. */ static minutes(amount) { return new Duration(amount, TimeUnit.Minutes); } /** * Create a Duration representing an amount of hours. * * @param amount the amount of Hours the `Duration` will represent. * @returns a new `Duration` representing `amount` Hours. */ static hours(amount) { return new Duration(amount, TimeUnit.Hours); } /** * Create a Duration representing an amount of days. * * @param amount the amount of Days the `Duration` will represent. * @returns a new `Duration` representing `amount` Days. */ static days(amount) { return new Duration(amount, TimeUnit.Days); } /** * Parse a period formatted according to the ISO 8601 standard. * * @param duration an ISO-formtted duration to be parsed. * @returns the parsed `Duration`. * @see https://www.iso.org/fr/standard/70907.html */ static parse(duration) { const matches = duration.match(/^P(?:(\d+)D)?(?:T(?:(\d+)H)?(?:(\d+)M)?(?:(\d+)S)?)?$/); if (!matches) { throw new Error(`Not a valid ISO duration: ${duration}`); } const [, days, hours, minutes, seconds] = matches; if (!days && !hours && !minutes && !seconds) { throw new Error(`Not a valid ISO duration: ${duration}`); } return Duration.millis(_toInt(seconds) * TimeUnit.Seconds.inMillis + (_toInt(minutes) * TimeUnit.Minutes.inMillis) + (_toInt(hours) * TimeUnit.Hours.inMillis) + (_toInt(days) * TimeUnit.Days.inMillis)); function _toInt(str) { if (!str) { return 0; } return Number(str); } } /** * Add two Durations together. */ plus(rhs) { const targetUnit = finestUnit(this.unit, rhs.unit); const total = convert(this.amount, this.unit, targetUnit, {}) + convert(rhs.amount, rhs.unit, targetUnit, {}); return new Duration(total, targetUnit); } /** * Return the total number of milliseconds in this Duration. * * @returns the value of this `Duration` expressed in Milliseconds. */ toMilliseconds(opts = {}) { return convert(this.amount, this.unit, TimeUnit.Milliseconds, opts); } /** * Return the total number of seconds in this Duration. * * @returns the value of this `Duration` expressed in Seconds. */ toSeconds(opts = {}) { return convert(this.amount, this.unit, TimeUnit.Seconds, opts); } /** * Return the total number of minutes in this Duration. * * @returns the value of this `Duration` expressed in Minutes. */ toMinutes(opts = {}) { return convert(this.amount, this.unit, TimeUnit.Minutes, opts); } /** * Return the total number of hours in this Duration. * * @returns the value of this `Duration` expressed in Hours. */ toHours(opts = {}) { return convert(this.amount, this.unit, TimeUnit.Hours, opts); } /** * Return the total number of days in this Duration. * * @returns the value of this `Duration` expressed in Days. */ toDays(opts = {}) { return convert(this.amount, this.unit, TimeUnit.Days, opts); } /** * Return an ISO 8601 representation of this period. * * @returns a string starting with 'P' describing the period * @see https://www.iso.org/fr/standard/70907.html */ toIsoString() { if (this.amount === 0) { return 'PT0S'; } const ret = ['P']; let tee = false; for (const [amount, unit] of this.components(true)) { if ([TimeUnit.Seconds, TimeUnit.Minutes, TimeUnit.Hours].includes(unit) && !tee) { ret.push('T'); tee = true; } ret.push(`${amount}${unit.isoLabel}`); } return ret.join(''); } /** * (deprecated) Return an ISO 8601 representation of this period. * * @returns a string starting with 'P' describing the period * @see https://www.iso.org/fr/standard/70907.html * @deprecated Use `toIsoString()` instead. */ toISOString() { return this.toIsoString(); } /** * Turn this duration into a human-readable string. */ toHumanString() { if (this.amount === 0) { return fmtUnit(0, this.unit); } if (token_1.Token.isUnresolved(this.amount)) { return `<token> ${this.unit.label}`; } return this.components(false) // 2 significant parts, that's totally enough for humans .slice(0, 2) .map(([amount, unit]) => fmtUnit(amount, unit)) .join(' '); function fmtUnit(amount, unit) { if (amount === 1) { // All of the labels end in 's' return `${amount} ${unit.label.substring(0, unit.label.length - 1)}`; } return `${amount} ${unit.label}`; } } /** * Returns a string representation of this `Duration` that is also a Token that cannot be successfully resolved. * * This * protects users against inadvertently stringifying a `Duration` object, when they should have called one of the * `to*` methods instead. */ toString() { return token_1.Token.asString(() => { throw new Error('Duration.toString() was used, but .toSeconds, .toMinutes or .toDays should have been called instead'); }, { displayHint: `${this.amount} ${this.unit.label}` }); } /** * Return the duration in a set of whole numbered time components, ordered from largest to smallest * * Only components != 0 will be returned. * * Can combine millis and seconds together for the benefit of toIsoString, * makes the logic in there simpler. */ components(combineMillisWithSeconds) { const ret = new Array(); let millis = convert(this.amount, this.unit, TimeUnit.Milliseconds, { integral: false }); for (const unit of [TimeUnit.Days, TimeUnit.Hours, TimeUnit.Minutes, TimeUnit.Seconds]) { const count = convert(millis, TimeUnit.Milliseconds, unit, { integral: false }); // Round down to a whole number UNLESS we're combining millis and seconds and we got to the seconds const wholeCount = unit === TimeUnit.Seconds && combineMillisWithSeconds ? count : Math.floor(count); if (wholeCount > 0) { ret.push([wholeCount, unit]); millis -= wholeCount * unit.inMillis; } } // Remainder in millis if (millis > 0) { ret.push([millis, TimeUnit.Milliseconds]); } return ret; } } exports.Duration = Duration; class TimeUnit { constructor(label, isoLabel, inMillis) { this.label = label; this.isoLabel = isoLabel; this.inMillis = inMillis; // MAX_SAFE_INTEGER is 2^53, so by representing our duration in millis (the lowest // common unit) the highest duration we can represent is // 2^53 / 86*10^6 ~= 104 * 10^6 days (about 100 million days). } toString() { return this.label; } } TimeUnit.Milliseconds = new TimeUnit('millis', '', 1); TimeUnit.Seconds = new TimeUnit('seconds', 'S', 1000); TimeUnit.Minutes = new TimeUnit('minutes', 'M', 60000); TimeUnit.Hours = new TimeUnit('hours', 'H', 3600000); TimeUnit.Days = new TimeUnit('days', 'D', 86400000); function convert(amount, fromUnit, toUnit, { integral = true }) { if (fromUnit.inMillis === toUnit.inMillis) { return amount; } const multiplier = fromUnit.inMillis / toUnit.inMillis; if (token_1.Token.isUnresolved(amount)) { throw new Error(`Unable to perform time unit conversion on un-resolved token ${amount}.`); } const value = amount * multiplier; if (!Number.isInteger(value) && integral) { throw new Error(`'${amount} ${fromUnit}' cannot be converted into a whole number of ${toUnit}.`); } return value; } /** * Return the time unit with highest granularity */ function finestUnit(a, b) { return a.inMillis < b.inMillis ? a : b; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"duration.js","sourceRoot":"","sources":["duration.ts"],"names":[],"mappings":"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","sourcesContent":["import { Token } from './token';\n\n                                                                                                                                                                                                                                                           \nexport class Duration {\n                                                                                                                                                                                                                    \n  public static millis(amount: number): Duration {\n    return new Duration(amount, TimeUnit.Milliseconds);\n  }\n\n                                                                                                                                                                                                               \n  public static seconds(amount: number): Duration {\n    return new Duration(amount, TimeUnit.Seconds);\n  }\n\n                                                                                                                                                                                                               \n  public static minutes(amount: number): Duration {\n    return new Duration(amount, TimeUnit.Minutes);\n  }\n\n                                                                                                                                                                                                         \n  public static hours(amount: number): Duration {\n    return new Duration(amount, TimeUnit.Hours);\n  }\n\n                                                                                                                                                                                                      \n  public static days(amount: number): Duration {\n    return new Duration(amount, TimeUnit.Days);\n  }\n\n                                                                                                                                                                                                                                       \n  public static parse(duration: string): Duration {\n    const matches = duration.match(/^P(?:(\\d+)D)?(?:T(?:(\\d+)H)?(?:(\\d+)M)?(?:(\\d+)S)?)?$/);\n    if (!matches) {\n      throw new Error(`Not a valid ISO duration: ${duration}`);\n    }\n    const [, days, hours, minutes, seconds] = matches;\n    if (!days && !hours && !minutes && !seconds) {\n      throw new Error(`Not a valid ISO duration: ${duration}`);\n    }\n    return Duration.millis(\n      _toInt(seconds) * TimeUnit.Seconds.inMillis\n      + (_toInt(minutes) * TimeUnit.Minutes.inMillis)\n      + (_toInt(hours) * TimeUnit.Hours.inMillis)\n      + (_toInt(days) * TimeUnit.Days.inMillis),\n    );\n\n    function _toInt(str: string): number {\n      if (!str) { return 0; }\n      return Number(str);\n    }\n  }\n\n  private readonly amount: number;\n  private readonly unit: TimeUnit;\n\n  private constructor(amount: number, unit: TimeUnit) {\n    if (!Token.isUnresolved(amount) && amount < 0) {\n      throw new Error(`Duration amounts cannot be negative. Received: ${amount}`);\n    }\n\n    this.amount = amount;\n    this.unit = unit;\n  }\n\n                                           \n  public plus(rhs: Duration): Duration {\n    const targetUnit = finestUnit(this.unit, rhs.unit);\n    const total = convert(this.amount, this.unit, targetUnit, {}) + convert(rhs.amount, rhs.unit, targetUnit, {});\n    return new Duration(total, targetUnit);\n  }\n\n                                                                                                                                                    \n  public toMilliseconds(opts: TimeConversionOptions = {}): number {\n    return convert(this.amount, this.unit, TimeUnit.Milliseconds, opts);\n  }\n\n                                                                                                                                          \n  public toSeconds(opts: TimeConversionOptions = {}): number {\n    return convert(this.amount, this.unit, TimeUnit.Seconds, opts);\n  }\n\n                                                                                                                                          \n  public toMinutes(opts: TimeConversionOptions = {}): number {\n    return convert(this.amount, this.unit, TimeUnit.Minutes, opts);\n  }\n\n                                                                                                                                      \n  public toHours(opts: TimeConversionOptions = {}): number {\n    return convert(this.amount, this.unit, TimeUnit.Hours, opts);\n  }\n\n                                                                                                                                    \n  public toDays(opts: TimeConversionOptions = {}): number {\n    return convert(this.amount, this.unit, TimeUnit.Days, opts);\n  }\n\n                                                                                                                                                                                          \n  public toIsoString(): string {\n    if (this.amount === 0) { return 'PT0S'; }\n\n    const ret = ['P'];\n    let tee = false;\n\n    for (const [amount, unit] of this.components(true)) {\n      if ([TimeUnit.Seconds, TimeUnit.Minutes, TimeUnit.Hours].includes(unit) && !tee) {\n        ret.push('T');\n        tee = true;\n      }\n      ret.push(`${amount}${unit.isoLabel}`);\n    }\n\n    return ret.join('');\n  }\n\n                                                                                                                                                                                                                                        \n  public toISOString(): string {\n    return this.toIsoString();\n  }\n\n                                                                \n  public toHumanString(): string {\n    if (this.amount === 0) { return fmtUnit(0, this.unit); }\n    if (Token.isUnresolved(this.amount)) { return `<token> ${this.unit.label}`; }\n\n    return this.components(false)\n      // 2 significant parts, that's totally enough for humans\n      .slice(0, 2)\n      .map(([amount, unit]) => fmtUnit(amount, unit))\n      .join(' ');\n\n    function fmtUnit(amount: number, unit: TimeUnit) {\n      if (amount === 1) {\n        // All of the labels end in 's'\n        return `${amount} ${unit.label.substring(0, unit.label.length - 1)}`;\n      }\n      return `${amount} ${unit.label}`;\n    }\n  }\n\n                                                                                                                                                                                                                                                                                   \n  public toString(): string {\n    return Token.asString(\n      () => {\n        throw new Error('Duration.toString() was used, but .toSeconds, .toMinutes or .toDays should have been called instead');\n      },\n      { displayHint: `${this.amount} ${this.unit.label}` },\n    );\n  }\n\n  /**\n   * Return the duration in a set of whole numbered time components, ordered from largest to smallest\n   *\n   * Only components != 0 will be returned.\n   *\n   * Can combine millis and seconds together for the benefit of toIsoString,\n   * makes the logic in there simpler.\n   */\n  private components(combineMillisWithSeconds: boolean): Array<[number, TimeUnit]> {\n    const ret = new Array<[number, TimeUnit]>();\n    let millis = convert(this.amount, this.unit, TimeUnit.Milliseconds, { integral: false });\n\n    for (const unit of [TimeUnit.Days, TimeUnit.Hours, TimeUnit.Minutes, TimeUnit.Seconds]) {\n      const count = convert(millis, TimeUnit.Milliseconds, unit, { integral: false });\n      // Round down to a whole number UNLESS we're combining millis and seconds and we got to the seconds\n      const wholeCount = unit === TimeUnit.Seconds && combineMillisWithSeconds ? count : Math.floor(count);\n      if (wholeCount > 0) {\n        ret.push([wholeCount, unit]);\n        millis -= wholeCount * unit.inMillis;\n      }\n    }\n\n    // Remainder in millis\n    if (millis > 0) {\n      ret.push([millis, TimeUnit.Milliseconds]);\n    }\n    return ret;\n  }\n}\n\n                                                               \nexport interface TimeConversionOptions {\n                                                                                                                                                                     \n  readonly integral?: boolean;\n}\n\nclass TimeUnit {\n  public static readonly Milliseconds = new TimeUnit('millis', '', 1);\n  public static readonly Seconds = new TimeUnit('seconds', 'S', 1_000);\n  public static readonly Minutes = new TimeUnit('minutes', 'M', 60_000);\n  public static readonly Hours = new TimeUnit('hours', 'H', 3_600_000);\n  public static readonly Days = new TimeUnit('days', 'D', 86_400_000);\n\n  private constructor(public readonly label: string, public readonly isoLabel: string, public readonly inMillis: number) {\n    // MAX_SAFE_INTEGER is 2^53, so by representing our duration in millis (the lowest\n    // common unit) the highest duration we can represent is\n    // 2^53 / 86*10^6 ~= 104 * 10^6 days (about 100 million days).\n  }\n\n  public toString() {\n    return this.label;\n  }\n}\n\nfunction convert(amount: number, fromUnit: TimeUnit, toUnit: TimeUnit, { integral = true }: TimeConversionOptions) {\n  if (fromUnit.inMillis === toUnit.inMillis) { return amount; }\n  const multiplier = fromUnit.inMillis / toUnit.inMillis;\n\n  if (Token.isUnresolved(amount)) {\n    throw new Error(`Unable to perform time unit conversion on un-resolved token ${amount}.`);\n  }\n  const value = amount * multiplier;\n  if (!Number.isInteger(value) && integral) {\n    throw new Error(`'${amount} ${fromUnit}' cannot be converted into a whole number of ${toUnit}.`);\n  }\n  return value;\n}\n\n/**\n * Return the time unit with highest granularity\n */\nfunction finestUnit(a: TimeUnit, b: TimeUnit) {\n  return a.inMillis < b.inMillis ? a : b;\n}\n"]}