7.0.1685-testing

Time handling in GreyCat

Time is a very important concept within GreyCat, as we strive to allow developers an intuitive and fast way to time series data. Towards this goal, the standard library offers a few native types to handle the notion of time.

Time

time is a native concept and represented internally as a primitive signed int64. It is absolute, chronological, and represents the number of microseconds since 01/01/1970 UTC. Timestamps in GreyCat have no explicit timezone and are by the previous definition always given in UTC. The concept of timezone is represented by the native enum TimeZone and is required for some operations on time instances as well as for the conversion to the human-readable Date type. For printing times during development in a human-friendly way, the time::format function can be used.

Name Symbol Min Max Precision
Time (64 bits) time 21/12/-290308 10/01/294247 1 microsecond

There are several ways to initialize time in GreyCat:

fn main() {
    var t1 = time::now(); //Current system time
    var t2 = 5_time; //5 microseconds after 01/01/1970
    var t3 = -1000000_time; //1 second before 01/01/1970
    var t4 = time::new(23, DurationUnit::hours); //23 hours after 01/01/1970
    var t5 = time::new(1684163705, DurationUnit::seconds); //If you have the POSIX epoch in seconds
    var t6 = time::parse("07/06/2023 10:57:32", "%d/%m/%Y %H:%M:%S");
    // print formated time
    println(t6.format("%d/%m/%Y %H:%M:%S",TimeZone::"Europe/Luxembourg")); // 07/06/2023 12:57:32, beware timezone shift
}

Since time is represented internally as an int64, it can be compared to another time (absolute order)

if (t5 > t4) {
    //Do something
}

You can also subtract two times and return a duration

var duration = t1 - t2 // duration instance representing the passed time between t1 and t2

To get to a later timestamp, you can add a duration to a time (it is not possible to add two timeinstances together!)

var newT = t1 + 10_s

This syntax allows you to easily manipulate time, a good example of this are for loops with time

for (var t = 0_time; t < 1000_time; t = t + 1_us) {
    //Stuff
}

Date

Since dealing with epoch time is not human friendly, we have the notion of Date. Date exists purely as a human friendly representation of time and should therefore only ever be used for user interaction. Converting time to Date or vice-versa is done like this:

fn main() {
    var t = time::now(); // Current system time
    var date = t.toDate(null); // Date in UTC Timezone
    var dateLux = t.toDate(TimeZone::"Europe/Luxembourg"); // Date in Luxembourg Timezone

    println("Date in UTC: ${date}"); // Date in UTC
    println("Date in Luxembourg: ${dateLux}"); // Date in Luxembourg time

    var convert_back_to_time = date.to_time(null); //Convert date back to time
}

The conversion time to Date is resource intensive, while the conversion Date to time is straightforward

A Date can be created either by converting from time OR by parsing a String with a given format:

fn main() {
    var d1 = Date::from_time(time::now(), TimeZone::"Europe/Luxembourg");
    println(d1); // curent time as date using Lux timezone
    var d2 = Date::parse("07/06/2023 10:57:32", "%d/%m/%Y %H:%M:%S");
    println(d2); // Date{year:2023,month:6,day:7,hour:10,minute:57,second:32,microsecond:0}
}

In addition to to_time, Date also provides the to_nearest_time to gracefully deal with invalid dates. Let’s consider for instance the 30th of March, 2025, that we want to convert to a timestamp in Lebanon:

fn main() {
    var tz = TimeZone::"Asia/Beirut";
    var d = Date::parse("2025-03-30", "%Y-%m-%d");
    var t = d.to_time(tz); // throws an error
    pprint(t);
}

The reason here is the unset fields in the parse method (here hours, minutes, seconds and microseconds), all default to 0, but sharp midnight is not a valid hour in Lebanon on the 30th of March, 2025, as at this exact moment DST makes it jump to 1am instead. To deal with this, you can do:

fn main() {
    var tz = TimeZone::"Asia/Beirut";
    var d = Date::parse("2025-03-30", "%Y-%m-%d");
    var t = d.to_nearest_time(tz);
    pprint(t); // '2025-03-29T22:00:00Z'
}

Note that defaulting to to_nearest_time whenever to_time fails is almost never the way to go. Converting a date to a time in a time zone where the date is invalid should always lead you to question your assumptions first, that had you consider this specific date in this specific time zone.


Date String Format

Both time and Date offer a parse function which expects a String value and optionally a String format. The timezone will default to UTC and the format to ISO 8601. Below an overview of the specifiers that can be used to indicate a different format.

Specifier Meaning Example
%d Day of the month, zero-padded (01-31) 22
%D Short MM/DD/YY date, equivalent to %m/%d/%y 07/30/09
%e Day of the month, space-padded ( 1-31) 22
%H Hour in 24h format (00-23) 16
%I Hour in 12h format (01-12) 08
%m Month as a decimal number (01-12) 08
%M Minute (00-59) 52
%S Second (00-61) 06
%s Unix time; the number of seconds since the Unix epoch. 1455803239
%y Year, last two digits (00-99) 11
%Y Year 2016

Some examples:

    var d0 = Date::parse("31.01.2022-01:30:30", "%d.%m.%Y-%H:%M:%S") // Date{year:2022,month:1,day:31,hour:1,minute:30,second:30,microsecond:0}
    var d1 = Date::parse("31.01.22-03", "%d.%m.%y-%I"); // Date{year:2022,month:1,day:31,hour:3,minute:0,second:0,microsecond:0}


Duration

Another important concept when manipulating time is duration, it is defined as the span between 2 points in time.

fn main() {
    var t1 = time::new(1684164000, DurationUnit::seconds); //timestamp 1
    var t2 = time::new(1684165000, DurationUnit::seconds); //timestamp 2
    var d = t2 - t1; //d will automatically be a duration
    println(d.to(DurationUnit::seconds)); // Prints the integer number of seconds
    println(d.tof(DurationUnit::hours));  // Prints the fractional number of hours
}

The native unit of duration in GreyCat is microseconds (since time itself is in microseconds), just like time it is represented internally as a primitive signed int64

Some other ways to initialize duration

var d1 = duration::new(5, DurationUnit::microseconds);
var d2 = duration::new(2, DurationUnit::days);         //new is used for int
var d3 = duration::newf(1.5, DurationUnit::years);     //newf is used for float
var d4 = 5.6_s;  // 5.6 seconds duration
var d5 = 7_hour; // 7 hours duration

And like time you can compare them

if(d4 > d5){
 //Stuff
}

You can also subtract and add durations with each other

var d1 = 10_s + 10_s   //20_s
var d2 = 10_s - 5_s   //5_s

DurationUnit and CalendarUnit

GreyCat comes with two enum types to map spans of time to pre-defined units, namely DurationUnit and CalendarUnit.

A DurationUnit simply maps to a fixed number of microseconds and should be used to shift time by this exact fixed quantity. A Duration using a DurationUnit can also be accessed via shorthands as indicated in the table below (where N is the number of units).

DurationUnit Value in microseconds Explanation Shorthand
microseconds 1 lowest GreyCat time precision N_microseconds OR N_us
milliseconds 1e3 1 thousand microseconds N_milliseconds OR N_ms
seconds 1e6 1 thousand milliseconds N_seconds OR N_s
minutes 60e6 60 seconds N_minutes OR N_m
hours 3600e6 60 minutes N_hours OR N_h
days 86400e6 24 hours N_days OR N_d

CalendarUnits are used is as input parameters for functions on time instances with the goal of covering spans of time while respecting the Gregorian calendar (i.e. differing month lengths, leap year) as well as TimeZone specifics (i.e. start/end of day, summer/winter time). The CalendarUnit enum defines the following units:

  • microseconds
  • seconds
  • minutes
  • hours
  • days
  • months
  • years

Below an overview of the time functions making use of CalendarUnit. All use an optional parameter tz, which indicates the TimeZone and defaults to UTC.

Function Name What it does
calendar_add Shift timeby the indicated value of units. Negative values will shift to the past
calendar_floor Floor to the first instance of the indicated unit. I.e. floor to years will set to January 1st at 00:00:00:000000.
calendar_ceiling Ceil to the last instance of the indicated unit. I.e. floor to years will set to December 31st at 23:59:59:999999.