Timestamps

Learn how to work effectively with timestamps in the Synnax TypeScript client.

Working with high-resolution timestamps in JavaScript is tricky. We provide several utility classes to make it easier.

JavaScript’s Limitations

Synnax stores timestamps as 64-bit integers representing the number of nanoseconds elapsed since the unix epoch in UTC. This is unlike JavaScript’s native Date object, which only supports millisecond precision.

Numbers in JavaScript are represented as 64-bit floating point numbers, which means that they can only represent integers up to 2^53 accurately. This means that you may lose precision for certain timestamps. Also, doing arithmetic with timestamps expressed as floating point numbers can lead to rounding errors.

TimeStamp

Synnax provides the TimeStamp utility class to effectively work with nanosecond timestamps in JavaScript. It wraps a bigint to store a unix timestamp in nanoseconds.

Constructing a TimeStamp

There are several easy ways to construct a TimeStamp:

import { TimeStamp } from "@synnaxlabs/client";
// From the current time
const now = TimeStamp.now();
// From a Date object
const ts = new TimeStamp(new Date("2021-01-01T00:00:00Z"));
// From a string
const ts = new TimeStamp("2021-01-01T00:00:00Z");
// From a number of nanoseconds
const ts = new TimeStamp(1000000000);
// From a bigint of nanoseconds
const ts = new TimeStamp(BigInt(1000000000));
// From utility functions
const ts = TimeStamp.now().sub(TimeStamp.seconds(1));

Any of these formats can be passed to common methods used throughout the Synnax client. The union of these formats is called a CrudeTimeStamp. Examples include read, write, openIterator, openStreamer, and openWriter.

Converting to a Date

You can convert a TimeStamp to a Date object using the date method:

const ts = TimeStamp.now();
const date = ts.date();

Arithmetic

You can perform arithmetic on TimeStamp objects:

const ts1 = TimeStamp.now();
const ts2 = ts1.add(TimeStamp.seconds(1));
const diff = ts2.sub(ts1);

Comparisons

You can compare TimeStamp objects:

const ts1 = TimeStamp.now();
const ts2 = ts1.add(TimeStamp.seconds(1));
const isAfter = ts2.after(ts1);
const isAfterEq = ts2.afterEq(ts1);
const isBefore = ts1.before(ts2);
const isBeforeEq = ts1.beforeEq(ts2);

Accessing the Underlying Value

You can access the underlying bigint value using the value property:

const ts = TimeStamp.now();
const value = ts.value;

TimeSpan

TimeSpan is a utility class that represents a duration of time. It wraps a bigint to store a duration in nanoseconds.

Constructing a TimeSpan

You can construct a TimeSpan directly from a number of nanoseconds, but it’s generally easier to use the utility functions:

import { TimeSpan } from "@synnaxlabs/client";

// From a number of nanoseconds
const span = new TimeSpan(1000000000);
// From a utility function
const span = TimeSpan.hours(1);
// From multiple utility functions
const span = TimeSpan.days(1).add(TimeSpan.hours(1)).add(TimeSpan.minutes(1));

Performing Arithmetic

You can perform arithmetic on TimeSpan objects:

const span1 = TimeSpan.hours(1);
const span2 = span1.add(TimeSpan.minutes(1));
const diff = span2.sub(span1);

Accessing the Underlying Value

You can access the underlying bigint value using the value property:

const span = TimeSpan.hours(1);
const value = span.value;

TimeRange

TimeRange is another utility class that represents a range of time. It consists of two TimeStamp objects called start and end.

Constructing a TimeRange

You can construct a TimeRange from two timestamps in any of the formats that TimeStamp supports:

import { TimeRange } from "@synnaxlabs/client";

const start = TimeStamp.now();
const end = start.add(TimeStamp.seconds(1));
// From TimeStamp objects
const range = new TimeRange(start, end);
// From dates
const range = new TimeRange(
  new Date("2021-01-01T00:00:00Z"),
  new Date("2021-01-01T00:00:01Z"),
);
// From strings
const range = new TimeRange("2021-01-01T00:00:00Z", "2021-01-01T00:00:01Z");
// From numbers
const range = new TimeRange(1000000000, 2000000000);
// From bigints
const range = new TimeRange(BigInt(1000000000), BigInt(2000000000));
// From a mix of formats
const range = new TimeRange(BigInt(1000000000), "2021-01-01T00:00:01Z");
// From an object
const range = TimeRange.from({
  start: TimeStamp.now(),
  end: TimeStamp.now().add(TimeStamp.seconds(1)),
});

Checking if a TimeStamp Is in a TimeRange

You can check if a TimeStamp is in a TimeRange using the contains method:

const range = new TimeRange(
  new Date("2021-01-01T00:00:00Z"),
  new Date("2021-01-01T00:00:01Z"),
);
const ts = new TimeStamp("2021-01-01T00:00:00.5Z");
const isIn = range.contains(ts);
console.log(isIn); // true

Checking if Two TimeRanges Overlap

You can check if two TimeRange objects overlap using the overlaps method:

const range1 = new TimeRange(
  new Date("2021-01-01T00:00:00Z"),
  new Date("2021-01-01T00:00:01Z"),
);
const range2 = new TimeRange(
  new Date("2021-01-01T00:00:00.5Z"),
  new Date("2021-01-01T00:00:01.5Z"),
);
const doesOverlap = range1.overlapsWith(range2);
console.log(doesOverlap); // true

Getting the TimeSpan of a TimeRange

You can get the TimeSpan of a TimeRange using the span property:

const range = new TimeRange(
  new Date("2021-01-01T00:00:00Z"),
  new Date("2021-01-01T00:00:01Z"),
);
const span = range.span;
console.log(span.seconds); // 1