Time scheduling with Chime
Eclipse Vert.x executes periodic and delayed actions with periodic and one-shot timers. This is the base for time scheduling and reach feature extension must be rather interesting. Be notified at certain date / time, take into account holidays, repeat notifications until a given date, apply time zone, take into account daylight saving time etc. There are a lot of useful features time scheduler may introduce to the Vert.x stack.
Chime
Chime is time scheduler verticle which works on Vert.x event bus and provides:
- scheduling with cron-style, interval or union timers:
- at a certain time of day (to the second);
- on certain days of the week, month or year;
- with a given time interval;
- with nearly any combination of all of above;
- repeating a given number of times;
- repeating until a given time / date;
- repeating infinitely
- proxying event bus with conventional interfaces
- applying time zones available on JVM with daylight saving time taken into account
- flexible timers management system:
- grouping timers;
- defining a timer start or end times
- pausing / resuming;
- fire counting;
- listening and sending messages via event bus with JSON;
- publishing or sending timer fire event to the address of your choice.
Chime is written in Ceylon and is available at Ceylon Herd.
Running
Ceylon users
Deploy Chime using Verticle.deployVerticle
method.
import io.vertx.ceylon.core {vertx}
import herd.schedule.chime {Chime}
Chime().deploy(vertx.vertx());
Or with vertx.deployVerticle(\"ceylon:herd.schedule.chime/0.2.1\");
but ensure that Ceylon verticle factory is available at class path.
Java users
- Ensure that Ceylon verticle factory is available at class path.
- Put Ceylon versions to consistency. For instance, Vert.x 3.4.1 depends on Ceylon 1.3.0 while Chime 0.2.1 depends on Ceylon 1.3.2.
- Deploy verticle, like:
vertx.deployVerticle("ceylon:herd.schedule.chime/0.2.1")
Example with Maven is available at Github.
Schedulers
Well, Chime verticle is deployed. Let’s see its structure.
In order to provide flexible and broad ways to manage timing two level architecture is adopted.
It consists of schedulers and timers. Timer is a unit which fires at a given time.
While scheduler is a set or group of timers and provides following:
- creating and deleting timers;
- pausing / resuming all timers working within the scheduler;
- info on the running timers;
- default time zone;
- listening event bus at the given scheduler address for the requests to.
Any timer operates within some scheduler. And one or several schedulers have to be created before starting scheduling.
When Chime verticle is deployed it starts listen event bus at chime address (can be configured).
In order to create scheduler send to this address a JSON message.
{
"operation": "create",
"name": "scheduler name"
}
Once scheduler is created it starts listen event bus at scheduler name address.
Sending messages to chime address or to scheduler name address are rather equivalent,
excepting that chime address provides services for every scheduler, while scheduler address
provides services for this particular scheduler only.
The request sent to the Chime has to contain operation and name keys.
Name key provides scheduler or timer name. While operation key shows an action Chime has to perform.
There are only four possible operations:
- create - create new scheduler or timer;
- delete - delete scheduler or timer;
- info - request info on Chime or on a particular scheduler or timer;
- state - set or get scheduler or timer state (running, paused or completed).
Timers
Now we have scheduler created and timers can be run within. There are two ways to access a given timer:
- Sending message to chime address with ‘name’ field set to scheduler name:timer name.
- Sending message to scheduler name address with ‘name’ field set to either timer name or scheduler name:timer name.
Timer request is rather complicated and contains a lot of details. In this blog post only basic features are considered:
{
"operation": "create",
"name": "scheduler name:timer name",
"description": {}
};
This is rather similar to request sent to create a scheduler.
The difference is only description field is added.
This description is an JSON object which identifies particular timer type and its details.
The other fields not shown here are optional and includes:
- initial timer state (paused or running);
- start or end date-time;
- number of repeating times;
- is timer message to be published or sent;
- timer fire message and delivery options;
- time zone.
Timer descriptions
Currently, three types of timers are supported:
- Interval timer which fires after each given time period (minimum 1 second):
{
"type": "interval",
"delay": "timer delay in seconds, Integer"
};
- Cron style timer which is defined with cron-style:
{
"type": "cron",
"seconds": "seconds in cron style, String",
"minutes": "minutes in cron style, String",
"hours": "hours in cron style, String",
"days of month": "days of month in cron style, String",
"months": "months in cron style, String",
"days of week": "days of week in cron style, String, optional",
"years": "years in cron style, String, optional"
};
Cron timer is rather powerful and flexible. Investigate specification for the complete list of features.
- Union timer which combines a number of timers into a one:
{
"type": "union",
"timers": ["list of the timer descriptions"]
};
Union timer may be useful to fire at a list of specific dates / times.
Timer events
Once timer is started it sends or publishes messages to scheduler name:timer name address in JSON format. Two types of events are sent:
- fire event which occurs when time reaches next timer value:
{
"name": "scheduler name:timer name, String",
"event": "fire",
"count": "total number of fire times, Integer",
"time": "ISO formated time / date, String",
"seconds": "number of seconds since last minute, Integer",
"minutes": "number of minutes since last hour, Integer",
"hours": "hour of day, Integer",
"day of month": "day of month, Integer",
"month": "month, Integer",
"year": "year, Integer",
"time zone": "time zone the timer works in, String"
};
- complete event which occurs when timer is exhausted by some criteria given at timer create request:
{
"name": "scheduler name:timer name, String",
"event": "complete",
"count": "total number of fire times, Integer"
};
Basically, now we know everything to be happy with Chime: schedulers and requests to them, timers and timer events. Will see some examples in the next section.
Examples
Ceylon example
Let’s consider a timer which has to fire every month at 16-30 last Sunday.
// listen the timer events
eventBus.consumer (
"my scheduler:my timer",
(Throwable|Message<JsonObject?> msg) {
if (is Message<JsonObject?> msg) { print(msg.body()); }
else { print(msg); }
}
);
// create scheduler and timer
eventBus.send<JsonObject> (
"chime",
JsonObject {
"operation" -> "create",
"name" -> "my scheduler:my timer",
"description" -> JsonObject {
"type" -> "cron",
"seconds" -> "0",
"minutes" -> "30",
"hours" -> "16",
"days of month" -> "*",
"months" -> "*",
"days of week" -> "SundayL"
}
}
);
’*’ means any, ‘SundayL’ means last Sunday.
If ‘create’ request is sent to Chime address with name set to ‘scheduler name:timer name’ and corresponding scheduler hasn’t been created before then Chime creates both new scheduler and new timer.
Java example
Let’s consider a timer which has to fire every Monday at 8-30 and every Friday at 17-30.
// listen the timer events
MessageConsumer<JsonObject> consumer = eventBus.consumer("my scheduler:my timer");
consumer.handler (
message -> {
System.out.println(message.body());
}
);
// description of timers
JsonObject mondayTimer = (new JsonObject()).put("type", "cron")
.put("seconds", "0").put("minutes", "30").put("hours", "8")
.put("days of month", "*").put("months", "*")
.put("days of week", "Monday");
JsonObject fridayTimer = (new JsonObject()).put("type", "cron")
.put("seconds", "0").put("minutes", "30").put("hours", "17")
.put("days of month", "*").put("months", "*")
.put("days of week", "Friday");
// union timer - combines mondayTimer and fridayTimer
JsonArray combination = (new JsonArray()).add(mondayTimer)
.add(fridayTimer);
JsonObject timer = (new JsonObject()).put("type", "union")
.put("timers", combination);
// create scheduler and timer
eventBus.send (
"chime",
(new JsonObject()).put("operation", "create")
.put("name", "my scheduler:my timer")
.put("description", timer)
);
Ensure that Ceylon verticle factory with right version is available at class path.
At the end
herd.schedule.chime
module provides some features not mentioned here:
- convenient builders useful to fill in JSON description of various timers;
- proxying event bus with conventional interfaces;
- reading JSON timer event into an object;
- attaching JSON message to the timer fire event;
- managing time zones.
There are also some ideas for the future:
- custom or user-defined timers;
- limiting the timer fire time / date with calendar;
- extracting timer fire message from external source.
This is very quick introduction to the Chime and if you are interested in you may read more in Chime documentation or even contribute to.
Thank’s for the reading and enjoy with coding!