base/Timer
Timers for one-off or periodic tasks. Applicable as part of the default mechanism.
The resolution of the timers is in the order of the block rate, so durations should be chosen well above that. For frequent canister wake-ups the heartbeat mechanism should be considered.
The functionality described below is enabled only when the actor does not override it by declaring an explicit system func timer.
Timers are not persisted across upgrades. One possible strategy
to re-establish timers after an upgrade is to walk stable variables
in the post_upgrade hook and distill necessary timer information
from there.
Basing security (e.g. access control) on timers is almost always the wrong choice. Be sure to inform yourself about state-of-the-art dApp security. If you must use timers for security controls, be sure to consider reentrancy issues, and the vanishing of timers on upgrades and reinstalls.
If moc is invoked with -no-timer, the importing will fail.
Type Duration
type Duration = {#seconds : Nat; #nanoseconds : Nat}
Type TimerId
type TimerId = Nat
Function setTimer
func setTimer(d : Duration, job : () -> async ()) : TimerId
Installs a one-off timer that upon expiration after given duration d
executes the future job().
let now = Time.now();
let thirtyMinutes = 1_000_000_000 * 60 * 30;
func alarmUser() : async () {
  // ...
};
appt.reminder = setTimer(#nanoseconds (Int.abs(appt.when - now - thirtyMinutes)), alarmUser);
Function recurringTimer
func recurringTimer(d : Duration, job : () -> async ()) : TimerId
Installs a recurring timer that upon expiration after given duration d
executes the future job() and reinserts itself for another expiration.
A duration of 0 will only expire once.
func checkAndWaterPlants() : async () {
  // ...
};
let daily = recurringTimer(#seconds (24 * 60 * 60), checkAndWaterPlants);
Function cancelTimer
func cancelTimer(_ : TimerId) : ()
Cancels a still active timer with (id : TimerId). For expired timers
and not recognized ids nothing happens.
func deleteAppointment(appointment : Appointment) {
  cancelTimer (appointment.reminder);
  // ...
};