Sensorthings API Queries
How to Build SensorThings API Queries for WYSEaSON
The common building blocks for Sensorthings Odata APIs
REST Information
GET requests
The OGC SensorThings API is a RESTful API that is based on the Odata protocol. For the public facing service, only GETs are supported. Major functions and endpoint of the API:
GET | Description |
---|---|
wyseason.uwyo.edu/sta/v1.1 | Get index - provides an overview of the SensorThings response. |
wyseason.uwyo.edu/sta/v1.1/Type | Get all of type - includes Locations, Things, Datastreams, ObservedProperties, or Observations. |
wyseason.uwyo.edu/sta/v1.1/Type(id) | Get one of a type - requests information related to a specific entity type. |
wyseason.uwyo.edu/sta/v1.1/Type(id)/Entity | Get linked entity- requests a list of information associated with a specific entity type. |
wyseason.uwyo.edu/sta/v1.1/Type(id)/EntitySet | Get all linked - provides data associations between object types. |
End Points
The table below explains the entities referenced throughout the API documentation.
Core Classes | Description |
---|---|
Things | A monitoring location. |
Locations | The physical location of a Thing. It is possible for a Thing to move to a different location. |
Datastreams | Connects an observation or set of observations to their Sensor, Observed Property, and Thing. |
Sensors | A description of a sensor that provides values. |
ObservedProperties | A description of what is being measured by a Sensor at a specific Location. |
Observations | A single measurement value. |
FeatureOfInterest | The object on which the measurement was performed. Autogenerated. |
HistoricalLocation | The historical location of a Thing if it was moved. |
Example Queries
GET Recent Water Temperature Observations
GETs all Datastreams that have water temp at 1 meter depth and return past two days of Observations:
$expand=
Observations($select=phenomenonTime,result,parameters;
$filter=phenomenonTime gt now() sub duration'P2D';
$orderby=phenomenonTime asc;
$top=1000)
GET Me Everything
All Thing`, with their Location and Datastreams. For each Datastream, the Sensor, ObservedProperties, and two most recent Observations:
https://wyseason.uwyo.edu/sta/v1.1/Things?
$expand=
Locations($select=name,description,properties,location,@iot.id),
Datastreams($select=name,description,@iot.id;
$expand=
Observations($select=result,phenomenonTime,parameters,resultQuality,@iot.id;
$orderby=phenomenonTime desc;
$top=2))
GET Me Everything on Jackson Lake
All Things, with their Location and Datastreams. For each Datastrea`, the sensor, ObservedProperty, and two most recent Observations, filtered by Locations that are tied to the Thing name LB00001 Jackson Lake:
https://wyseason.uwyo.edu/sta/v1.1/Things(3)?
$expand=
Locations($select=name,description,properties,location,@iot.id),
Datastreams($select=name,description,@iot.id;
$expand=
Observations($select=result,phenomenonTime,parameters,resultQuality,@iot.id;
$orderby=phenomenonTime desc;
$top=2))
Overlapping Time Frames
The PhenomenonTime and result of the first 100 Observations of a Datastream, ordered by PhenomenonTime that overlap with the time frame from 2023-01-02 07:00:00 UTC to 2023-01-04 07:00:00 UTC (2 days):
https://wyseason.uwyo.edu/sta/v1.1/Datastreams(1)/Observations?
$orderby=
phenomenonTime asc
&$top=100
&$select=phenomenonTime, result
&$filter=overlaps(phenomenonTime, 2023-10-01T01:00:00Z/2023-10-31T01:00:00Z)
The Previous Day
The Observations of a Datastream, for the last day:
https://wyseason.uwyo.edu/sta/v1.1/Datastreams(1)/Observations?
$orderby=
phenomenonTime asc
&$filter=phenomenonTime gt now() sub duration'P1D'
Additionally
ObservedProperties that are measured at the same station as the ObservedProperty with name Temperature:
https://wyseason.uwyo.edu/sta/v1.1/ObservedProperties?
$filter=Datastreams/Thing/Datastreams/ObservedProperty/name
Ordering By Function
This function orders the Datastreams by the length of their name in descending order:
https://wyseason.uwyo.edu/sta/v1.1/Datastreams?
$orderby=length(name) desc
Query Design Overview
The SensorThings API provides 8 entity types (12 with all extensions). All data can be reached from each entity type. To get Observations one could start:
directly from Observations
from Datastreams and expand Observations
from FeaturesOfInterest and expand Observations
from Things and expand Datastreams & Observations
from ObservedProperties and expand Datastreams & Observations
from Sensors and expand Datastreams & Observations
from Locations and expand Things, Datastreams & Observations
The following options can provide the same data but with different structures. All these options can provide same data but with a different structure.
Starting From Observations
The data is least structured when starting directly from /Observations. In this case each item in the resulting list is one Observation, but Observations for all Datastreams (Things/ObservedProperties) or FeaturesOfInterest are mixed together.
Individual timeseries for Datastreams are lost and it is up to the client to sort the data again. When expanding Datastreams and from there ObservedProperties or Things, the data for the Datastreams is repeated for each Observation. This result is very much like a CSV file, and quite inefficient.
The resulting data structure looks like this:
Observation + Datastream + ObservedProperty + Thing
Observation + Datastream + ObservedProperty + Thing
Observation + Datastream + ObservedProperty + Thing
It is not possible to get “the latest Observation for each …” since there is only one list, thus only one “latest Observation.”
Starting From Things
On the other side of the spectrum is starting from /Things. Each item in the resulting list will be a Thing, and for each Thing the Datastreams can be expanded, and for each of those the Observations.
Thing 1
Datastream 1 + ObservedProperty 1
Observation 1
Observation 2
Datastream 2 + ObservedProperty 2
Observation 3
Observation 4
Thing 2
Datastream 3 + ObservedProperty 1
Observation 5
Observation 6
Datastream 4 + ObservedProperty 2
Observation 7
Observation 8
There is some duplication in the ObservedProperties when fetching data in this way, but the structure is easy to put on a map. Because of the way that queries are structured, starting with a narrow data format will often lead to better performance.