Aviso is a system developed at ECMWF with the aim of:
- Notifying for data availability
- Real-Time Model Output Data
- Product dissemination via ECPDS
- Meant for automated systems
- Based on an Event system supporting a mechanism When <this> … Do <that> …
It allows users to:
- Define which events to be notified of.
- Define the triggers to be executed once a notification is received.
- Dispatch notifications to the notification server.
Content
Table of Contents |
---|
Children Display |
---|
Install AVISO and configure for ECMWF events
Aviso can be used as a Python API or as Command-Line Interface (CLI) application. Here a few steps to quickly get a working configuration listening to notifications.
Note, this guide assumes the user to have a ECMWF account.
- Install AVISO following instructions on: https://pyaviso.readthedocs.io/en/latest/guide/getting_started.html#installing
- Create ~/.marsrc/mars.email and
Install AVISO and configure for ECMWF events
Aviso can be used as a Python API or as Command-Line Interface (CLI) application. Here a few steps to quickly get a working configuration listening to notifications.
Note, this guide assumes the user to have a ECMWF account.
- Install AVISO following instructions on: https://pyaviso.readthedocs.io/en/latest/guide/getting_started.html#installing
- Configure MARS credentials following instructions on Access MARS
Create a configuration file in the default location
~
/.aviso/config.yaml
Code Block username_file: ~/.marsrc/mars.email key_file: ~/.marsrc/mars.token files. mars.email file should contain “email” listed here: https://api.ecmwf.int/v1/key and mars.token file should contain “key” shown on the same page.language yml Create a configuration file in the default location ~/.aviso/config.yaml
Code Block language yml username_file: ~/.marsrc/mars.email key_file: ~/.marsrc/mars.token notification_engine: notification_engine: type: etcd_rest host: aviso.ecmwf.int port: 443 https: true configuration_engine: type: etcd_rest host: aviso.ecmwf.int port: 443 https: true schemaconfiguration_parserengine: ecmwf remote_schema type: True auth_typeetcd_rest host: aviso.ecmwf
- Create ~/.marsrc/mars.email and ~/.marsrc/mars.token files. mars.email file should contain “email” listed here: https://api.ecmwf.int/v1/key and mars.token should contain “key” shown on the same page.
.int port: 443 https: true schema_parser: ecmwf remote_schema: True auth_type: ecmwf
- Create your listener configuration file(s) Crate a listener configuration file with following content:
- Example for dissemination:
Note theCode Block language yml title listener_diss.yaml listeners: - event: dissemination request: destination: <user_destination> class: od expver: 1 domain: g stream: enfo step: [1,2,3] triggers: - type: echo
dissemination
event listener.request
describes for which dissemination event users want to execute the triggers. It is made of a set of fields. Users have to specify only the fields they wants to use as filters.destination
is a mandatory field and it is associated to one or more destinations which are linked to the user's ECMWF account. Only the notifications complying with all the fields defined will execute the trigger. The trigger in this example isecho
. This will simply print out the notification to the console output. - Example for MARS:
Code Block language yml title listener_mars.yaml listeners: - event: mars request: class: od expver: 1 domain: g stream: enfo step: [1,2,3] triggers: - type: echo
Note, in the MARS event listener, destination field is not present. You can see the the full list of allowed fields for MARS/ dissemination event listeners here.Warning Aviso will notify when data exists in MARS, but most users will not be able to retrieve this data before the scheduled time (Dissemination schedule). Triggering an automated retrieval of data will require adding logic to ensure you only retrieve data when you have respective permissions to do so. For automation of downstream time-critical workflows, this should mainly be based on dissemination events.
- Example for dissemination:
- If AVISO is installed in Python virtual environment, activate the environment first:
If AVISO is installed in conda or Python virtual environment, activate the environment first:Python virtual environment: "myenv"Code Block language bash source {PATH_TO_
Code Block source {PATH_TO_language bash
Conda:MY_ENV}/myenv/bin/activate
Code Block language bash conda activate
- Launch the AVISO application Launch the AVISO application to listen ECMWF defined in your configuration file:
Code Block language bash aviso listen listener_diss.yaml #or aviso listen listener_mars.yaml
Once in execution this command will create a process waiting for notifications. Users can terminate the application by typing CTRL+C .
Note, the configuration file is only read at start time, therefore every time users make changes to it they need to restart the listening process.
...
Defining your listeners
Aviso configuration file allows the definition of multiple listeners. Alternatively, the listeners configuration can be indicated as an independent file or multiple files:
Code Block | ||||
---|---|---|---|---|
|
...
conda activate
...
Launch the aviso application
Code Block | ||||
---|---|---|---|---|
| ||||
aviso listen |
Once in execution this command will create a process waiting for notifications. Users can terminate the application by typing CTRL+C
.
Note, the configuration file is only read at start time, therefore every time users make changes to it they need to restart the listening process.
Upgrading pre-existent installation
...
aviso listen listener1.yaml listener2.yaml |
Regardless where is defined, each listener is composed of:
- an
event
type - a
request
block - a
triggers
block
Event
Aviso offers notifications for the following types of events:
- The
dissemination
event is submitted by the product generation. The related listener configuration must define thedestination
field. A notification related to adissemination
event will have the fieldlocation
.By default, the location field contains the URL to the product notified and can be used to download the file. However, it is also possible to configure your Dissemination Host in ECPDS to send full path to the local copy of transferred file. This can be done by adding following option to the Dissemination Host
...
- :
Code Block language
...
pip3 install --upgrade pyaviso
Defining your listeners
...
With this option, the location field will have value:java ectrans.location = "file:"
Code Block language bash
...
aviso listen listener1.yaml listener2.yaml
Regardless where is defined, each listener is composed of:
- an
event
type - a
request
block - a
triggers
block
Event
Aviso offers notifications for the following types of events:
file:/{Full Path on the Destination File System}/{File Name}
The
mars
event is designed for real-time data from the model output. The related listener configuration has no mandatory fields. Moreover the related notification will not contain thelocation
field because users will have to access to it by the- The
dissemination
event is submitted by the product generation. The related listener configuration must define thedestination
field. A notification related to adissemination
event will have the fieldlocation
containing the URL to the product notified. The
mars
event is designed for real-time data from the model output. The related listener configuration has no mandatory fields. Moreover the related notification will not contain thelocation
field because users will have to access to it by the conventional MARS API.
Request
...
Field | Type | Event | Optional/Mandatory |
---|---|---|---|
destination | String, uppercase | dissemination | Mandatory |
target | String, uppercase | dissemination | Optional |
class | Enum | All | Optional |
stream | Enum | All | Optional |
domain | Enum | All | Optional |
expver | Integer | All | Optional |
date | Date (e.g.20190810) | All | Optional |
time | Values: [0,6,12,18] | All | Optional |
step | Integer | All | Optional |
...
Code Block | ||
---|---|---|
| ||
triggers: - type: command working_dir: $HOME/aviso/examples command: ./script.sh --datestream ${request.datestream} -s-date ${request.streamdate} --time ${request.time} --step ${request.step} environment: STEP: ${request.step} TIME: "The time is ${request.time}" |
...
A notification is a dictionary whose keys can be used in the parameter substitution mechanism described above. Here an example of a notification:a notification:
Code Block | ||
---|---|---|
| ||
{
"event": "dissemination",
"request": {
"class": "od",
"date": "20191112", | ||
Code Block | ||
| ||
{ "eventdestination": "disseminationFOO", "requestdomain": {"g", "classexpver": "od0001", "datestep": "20191112001", "destinationstream": "FOOenfo", "domaintime": "18", "target": "gE1", }, "expverlocation": "0001", "step": "001", "stream": "enfo", "time": "18", "target": "E1" }, "location": "/home/ecpds/xxx.xx" } |
Here an example file of a listener command triggering a bash script executing a MARS request.
Post
This trigger allows the user to send as HTTP POST message the notification received and formatted accordingly to the CloudEvents specification
Code Block | ||
---|---|---|
| ||
triggers:
- type: post
protocol:
type: cloudevents_http
url: http://my.notification.system/api |
This is the basic configuration. More parameters can be specified to customise the CloudEvents message. More info the reference documentation.
The CloudEvents message sent would look like the following:
/home/ecpds/xxx.xx"
} |
Here an example file of a listener command triggering a bash script executing a MARS request.
Post
This trigger allows the user to send as HTTP POST message the notification received and formatted accordingly to the CloudEvents specification
Code Block | ||
---|---|---|
| ||
triggers:
- type: post
protocol:
type: cloudevents_http
url: http://my.notification.system/api |
This is the basic configuration. More parameters can be specified to customise the CloudEvents message. More info the reference documentation.
The CloudEvents message sent would look like the following:
Code Block | ||
---|---|---|
| ||
{
"type" : "aviso", # this is customisable by the user
"data": { | ||
Code Block | ||
| ||
{ "type" : "aviso", # this is customisableaviso specific by the user "event": "datadissemination":, { "request": { "target": "E1", "class": "od", # this is aviso specific "date": "20190810", "eventdestination": "disseminationFOO", "domain": "g", "expver": "1", "step": "1", "requeststream": { "enfo", "targettime": "E10", }, "classlocation": "od", /home/diss/foo/bar/20190810/xyz", # location on ceph or s3 }, "datedatacontenttype": "20190810application/json", "destinationid": "FOO0c02fdc5-148c-43b5-b2fa-cb1f590369ff", # UUID random generated by aviso "domainsource": "ghttps://aviso.ecmwf.int", "expver": "1", "step": "1", # this is customisable by the user "streamspecversion": "enfo1.0", "time": "02020-03-02T13:34:40.245Z", }, "location": "/home/diss/foo/bar/20190810/xyz", # locationTimestamp onof cephwhen orthis s3 message is }, "datacontenttype": "application/json", "id": "0c02fdc5-148c-43b5-b2fa-cb1f590369ff", # UUID random generated by aviso "source": "https://aviso.ecmwf.int", # this is customisable by the user "specversion": "1.0", "time": "2020-03-02T13:34:40.245Z", # Timestamp of when this message is created } |
...
created
} |
Aviso can also post events to a AWS Simple Notification Service (SNS) topic. See the relevant documentation page for more details: https://pyaviso.readthedocs.io/en/latest/reference/triggers.html#post
Combining multiple trigger types into sequence
An example of the triggers
block with multiple sequence of triggers is given below:
Code Block | ||
---|---|---|
| ||
triggers:
- type: echo
- type: log
path: testLog.log
- type: command
working_dir: $HOME/aviso/examples
command: ./script.sh --stream ${request.stream} --date ${request.date} --time ${request.time} --step ${request.step}
environment:
STEP: ${request.step}
TIME: "The time is ${request.time}" |
Aviso as a Python API
Aviso can be used as a Python API. This is intended for users that want to integrate Aviso in a bigger workflow written in Python or that simply have their trigger defined as a Python function. Below an example of a python script that defines a function to be executed once a notification is received, creates a listener that references to this function trigger and finally passes it to aviso to execute.
...
Here an example file of a Python script running Aviso and executing a MARS request after a notification is received.
Dealing with past notifications
Before listening to new notifications, Aviso by default checks what was the last notification received and it will then return all the notifications that have been missed since. It will then carry on by listening to new ones. The first ever time the application runs however no previous notification will be returned. This behaviour allows users not to miss any notifications in case of machine reboots.
script running Aviso and executing a MARS request after a notification is received.
Dealing with past notifications
Before listening to new notifications, Aviso by default checks what was the last notification received and it will then return all the notifications that have been missed since. It will then carry on by listening to new ones. The first ever time the application runs however no previous notification will be returned. This behaviour allows users not to miss any notifications in case of machine reboots.
To override this behaviour by ignoring the missed notifications while listening only to the new ones, run the following:
Code Block | ||||
---|---|---|---|---|
| ||||
aviso listen --now |
This command will also reset the previous history.
Users can also explicitly replay past notifications. Aviso can deliver notifications from the ECMWF server up to 14 days in the past. This can also be used to test the listener configuration with real notifications.
Here an example, launch Aviso with the following options:To override this behaviour by ignoring the missed notifications while listening only to the new ones, run the following:
Code Block | ||||
---|---|---|---|---|
| ||||
aviso listen --now |
This command will also reset the previous history.
Users can also explicitly replay past notifications. Aviso can deliver notifications from the ECMWF server up to 14 days in the past. This can also be used to test the listener configuration with real notifications.
Here an example, launch Aviso with the following options:
Code Block | ||||
---|---|---|---|---|
| ||||
aviso listen --from 2020-01-20T00:00:00.0Z --to 2020-01-21T00:00:00.0Z |
It will replay all the notifications sent from 20 January to 21 January and the ones complying with the listener request will execute the triggers.
Note, the dates must be in the past and --to
can only be defined together with --from
. The dates are defined in ISO format and they are in UTC.
...
| |||
aviso listen --from 2020-01-20T00:00:00.0Z --to 2020-01-21T00:00:00.0Z |
It will replay all the notifications sent from 20 January to 21 January and the ones complying with the listener request will execute the triggers.
Note, the dates must be in the past and --to
can only be defined together with --from
. The dates are defined in ISO format and they are in UTC.
In absence of --to
, the system after having retrieved the past notifications, it will continue listening to future notifications. If --to
is defined Aviso will terminate once retrieved all the past notifications.
Examples
There is a GitHub repository with collection of example Python scripts and listener configuration files specifically designed for the AVISO software and notifications generated by ECMWF. These examples are intended to illustrate how AVISO can be effectively used with data distributed by ECMWF and we strongly recommend going through the examples if you intend to use AVISO to get notifications about availability of ECMWF data.
Upgrading AVISO installation
To upgrade Aviso to a newer version once it is available run the following command:
Code Block | ||||
---|---|---|---|---|
| ||||
pip3 install --upgrade pyaviso |
Running as a service on Linux machines
...