README.md
December 17, 2025 ยท View on GitHub
DataStream Public API
Our public API uses the ISO/IEC 20802-2 Standard known as OData JSON Format v4.
Attribution/Citation
Thank you ahead of time for using this data responsibly and providing the appropriate citations when necessary when being presented to external parties. These citations must be accompanied by a link to the DOI (https://doi.org/{value}). The licence, citation, and DOI can be retrieved from the /Metadata endpoint.
Modules
We have built modules to wrap around our API to make it easier to use.
For those building their own implementation, here are some key things to keep in mind:
- Querystring parameters must be URL encoded. All languages should have a function to do this.
- Requests to Observations/Records that you expect a large amount (>1M rows) of data from should be partitioned. We recommend by Monitoring locations and/or activity start year. There is a technical database reason for this that you're welcome to ask us about.
- Each request partition should be paginated over using the
Linkheader or@odata.nextLinkwithin the body of the response. - Rate limit yourself (2 reqs/sec) and don't make requests in parallel. This will ensure you don't get
429 Too Many Requestserror response. - Use HTTP/3
Endpoints
You can test out your script by prefixing https://api.qa.datastream.org/v1/odata/v4 to the endpoints.
When you're ready to pull data from the production system you can use: https://api.datastream.org/v1/odata/v4.
For browser requests all you need to do is let us know your domain name and we can add it to the CORS whitelist, only GET requests are supported. All other application should store the API Key in the header x-api-key.
Note
Remember that your API key is for your use only. Please do not share your API key. If it does become public, please let us know, we can give you a new one.
-
GET /Metadata
- Retrieves the dataset-level metadata for the datasets that meet your query criteria.
- Select By:
DOI,Version,DatasetName,DataStewardEmail,DataCollectionOrganization,DataUploadOrganization,ProgressCode,MaintenanceFrequencyCode,Abstract,DataCollectionInformation,DataProcessing,FundingSources,DataSourceURL,OtherDataSources,Citation,Licence,Disclaimer,TopicCategoryCode,Keywords,CreateTimestamp,PublishedTimestamp - Filter By:
DOI,LocationId,ActivityMediaName,ActivityGroupType,CharacteristicName,MonitoringLocationType,ActivityStartYear,RegionId,DatasetName,CreateTimestamp
-
GET /Locations
- Retrieves the location information that meets your query criteria.
- Select By:
Id,DOI,ID(Maps toMonitoringLocationIDinternally),Name,Latitude,Longitude,HorizontalCoordinateReferenceSystem,HorizontalAccuracyMeasure,HorizontalAccuracyUnit,VerticalMeasure,VerticalUnit,MonitoringLocationType - Filter By:
DOI,LocationId,ActivityMediaName,ActivityGroupType,CharacteristicName,MonitoringLocationType,ActivityStartYear,RegionId,Name
-
GET /Observations
- Retrieves the observations that meet your query criteria.
- Select By:
Id,LocationId,DOI,ActivityType,ActivityMediaName,ActivityStartDate,ActivityStartTime,ActivityStartTimeZone,ActivityEndDate,ActivityEndTime,ActivityEndTimeZone,ActivityDepthHeightMeasure,ActivityDepthHeightUnit,SampleCollectionEquipmentName,CharacteristicName,MethodSpeciation,ResultSampleFraction,ResultValue,ResultUnit,ResultValueType,ResultDetectionCondition,ResultDetectionQuantitationLimitUnit,ResultDetectionQuantitationLimitMeasure,ResultDetectionQuantitationLimitType,ResultStatusID,ResultComment,ResultAnalyticalMethodID,ResultAnalyticalMethodContext,ResultAnalyticalMethodName,AnalysisStartDate,AnalysisStartTime,AnalysisStartTimeZone,LaboratoryName,LaboratorySampleID - Filter By:
DOI,LocationId,ActivityMediaName,ActivityGroupType,CharacteristicName,MonitoringLocationType,ActivityStartYear,RegionId
-
GET /Records
- Retrieves the data in the DataStream schema format with all columns that meet your query criteria.
- Only intended for streaming data directly into a CSV file. For all other use cases, we recommend using
/Observationsendpoint. - Select By:
Id,DOI,DatasetName,MonitoringLocationID,MonitoringLocationName,MonitoringLocationLatitude,MonitoringLocationLongitude,MonitoringLocationHorizontalCoordinateReferenceSystem,MonitoringLocationHorizontalAccuracyMeasure,MonitoringLocationHorizontalAccuracyUnit,MonitoringLocationVerticalMeasure,MonitoringLocationVerticalUnit,MonitoringLocationType,ActivityType,ActivityMediaName,ActivityStartDate,ActivityStartTime,ActivityStartTimeZone,ActivityEndDate,ActivityEndTime,ActivityEndTimeZone,ActivityDepthHeightMeasure,ActivityDepthHeightUnit,SampleCollectionEquipmentName,CharacteristicName,MethodSpeciation,ResultSampleFraction,ResultValue,ResultUnit,ResultValueType,ResultDetectionCondition,ResultDetectionQuantitationLimitMeasure,ResultDetectionQuantitationLimitUnit,ResultDetectionQuantitationLimitType,ResultStatusID,ResultComment,ResultAnalyticalMethodID,ResultAnalyticalMethodContext,ResultAnalyticalMethodName,AnalysisStartDate,AnalysisStartTime,AnalysisStartTimeZone,LaboratoryName,LaboratorySampleID - Filter By:
DOI,LocationId,ActivityMediaName,ActivityGroupType,CharacteristicName,MonitoringLocationType,ActivityStartYear,RegionId
Body Object
{
"@data.context": "odata/v4/Records/$links/Metadata(Id=@DatasetId)"
"value":[
...
]
}
URL Parameters
OData accepts certain query parameters. The ones supported by this API are:
- $select
- Fields to be selected are entered comma delimited.
- Example:
$select=DatasetName,Abstract - Default: All columns available.
- $filter
- Available operators:
in,eq,lt,gt,lte,gte,ne - Grouping:
$filter=CharacteristicName eq 'Dissolved oxygen saturation'or$filter=DOI eq '10.25976/{suffix}'where{suffix}is replaced with a value. - Temporal:
$filter=CreateTimestamp gt '2020-03-23' and CreateTimestamp lt '2020-03-25' - Spatial:
$filter=RegionId eq 'hub.atlantic'- RegionId Values (these values are subject to change):
- Partner Hubs:
hub.{atlantic,greatlakes,lakewinnipeg,mackenzie,pacific} - Countries:
admin.2.{ca} - Provinces/Territories/States:
admin.4.ca.{ab,bc,mb,nb,nl,ns,nt,nu.on,pe,,qc,sk,yt}
- Partner Hubs:
- Bounding box
$filter=Longitude gt '-102.01' and Longitude lt '-88.99' and Latitude gt '49' and Latitude lt '60'
- RegionId Values (these values are subject to change):
- Available operators:
- $top
- Maximum: 10000
- Example:
$top=10
- $skiptoken
- Return the next items after the skipped token
- Example:
$skiptoken=Id:1234 - For pagination, use
Linkheader or@odata.nextLinkin the JSON response body
- $count
- Return only the count for the request. When the value is large enough it becomes an estimate (~0.0005% accurate)
- Example:
$count=true - Default:
false
When building an integration with any API, it's important to encode all query string parameters.
Performance Tips
- Using
$selectto request only the parameters you need will decrease the amount of data needed to be transferred.
Examples
Metadata
Get all datasets in Mackenzie and Lake Winnipeg DataStream hubs
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Metadata \
--data-urlencode "\$select=DOI,DatasetName,Licence,Citation,Version" \
--data-urlencode "\$filter=RegionId in ('hub.mackenzie', 'hub.lakewinnipeg')"
Get the citation and licence for a dataset:
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Metadata \
--data-urlencode "\$select=DOI,DatasetName,Licence,Citation,Version" \
--data-urlencode "\$filter=endswith(DOI, 'xxxx-xxxx')" \
Sample Response
{
"value": [
{
"Id": "000000ab-000a-00a0-0aba-0ab00abc0a0a",
"DOI": "10.25976/xxxx-xx00",
"Version": "1.0.0",
"DatasetName": "Sample Dataset A",
"DataStewardEmail": "example@example.com",
"DataCollectionOrganization": "Sample Organization A",
"DataUploadOrganization": "Sample Organization B",
"ProgressCode": "onGoing",
"MaintenanceFrequencyCode": "asNeeded",
"Abstract": "Sample Description",
"DataCollectionInformation": "Sample Data Collection Information",
"FundingSources": "Sample Funder A",
"DataSourceURL": null,
"OtherDataSources": "",
"Citation": "Sample Organization A. 2024. \"Sample Dataset A\" (dataset). 1.0.0. DataStream. https://doi.org/10.25976/xxxx-xx00.",
"Licence": "https://opendatacommons.org/licenses/by/1-0/",
"Disclaimer": "",
"TopicCategoryCode": [
"inlandWaters"
],
"Keywords": [
"Water quality",
"temperature"
],
"CreateTimestamp": "2020-01-01T15:30:00.000Z",
"PublishedTimestamp": "2024-01-01T15:30:00.000Z"
}
],
"@odata.nextLink": "https://api.datastream.org/v1/odata/v4/Metadata?$skiptoken=Id:000000ab-000a-00a0-0aba-0ab00abc0a0a&$top=1000"
}
Locations
Get Locations from a dataset
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Locations \
--data-urlencode "\$filter=DOI eq '10.25976/xxxx-xx00'"
Get Locations from multiple datasets
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Locations \
--data-urlencode "\$filter=DOI in ('10.25976/xxxx-xx00', '10.25976/xxxx-xx00', '10.25976/xxxx-xx00')"
Sample Response
{
"value": [
{
"Id": 99999,
"DOI": "10.25976/xxxx-xx00",
"ID": "ABC123",
"Name": "Sample Location A",
"Latitude": 51.01,
"Longitude": -144.01,
"HorizontalCoordinateReferenceSystem": "WGS84",
"HorizontalAccuracyMeasure": null,
"HorizontalAccuracyUnit": null,
"VerticalMeasure": null,
"VerticalUnit": null,
"MonitoringLocationType": "River/Stream"
}
],
"@odata.nextLink": "https://api.datastream.org/v1/odata/v4/Locations?$skiptoken=Id:99999&$top=1000"
}
Observations
Get Temperature and pH observations from multiple datasets
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Observations \
--data-urlencode "\$filter=DOI in ('10.25976/xxxx-xx00', '10.25976/xxxx-xx00', '10.25976/xxxx-xx00') and CharacteristicName in ('Temperature, water', 'pH')"
Get all pH observations in Alberta:
curl -G -H 'x-api-key: PRIVATE-API-KEY' \
https://api.datastream.org/v1/odata/v4/Observations \
--data-urlencode "\$filter=CharacteristicName eq 'pH' and RegionId eq 'admin.4.ca.ab'"
Sample Response
{
"value": [
{
"Id": 99999,
"DOI": "10.25976/xxxx-xx00",
"LocationId": 999,
"ActivityType": "Field Msr/Obs-Portable Data Logger",
"ActivityMediaName": "Surface Water",
"ActivityStartDate": "2020-01-01",
"ActivityStartTime": "15:30:00",
"ActivityStartTimeZone": null,
"ActivityEndDate": null,
"ActivityEndTime": null,
"ActivityEndTimeZone": null,
"ActivityDepthHeightMeasure": -0.25,
"ActivityDepthHeightUnit": "m",
"SampleCollectionEquipmentName": null,
"CharacteristicName": "Dissolved oxygen saturation",
"MethodSpeciation": null,
"ResultSampleFraction": null,
"ResultValue": 97,
"ResultUnit": "%",
"ResultValueType": null,
"ResultDetectionCondition": null,
"ResultDetectionQuantitationLimitUnit": null,
"ResultDetectionQuantitationLimitMeasure": null,
"ResultDetectionQuantitationLimitType": null,
"ResultStatusID": null,
"ResultComment": null,
"ResultAnalyticalMethodID": null,
"ResultAnalyticalMethodContext": null,
"ResultAnalyticalMethodName": null,
"AnalysisStartDate": null,
"AnalysisStartTime": null,
"AnalysisStartTimeZone": null,
"LaboratoryName": null,
"LaboratorySampleID": null
}
],
"@odata.nextLink": "https://api.datastream.org/v1/odata/v4/Observations?$skiptoken=Id:99999&$top=1000"
}
Records
Sample Response
{
"value": [
{
"Id": 99999,
"DOI": "10.25976/xxxx-xx00",
"DatasetName": "Sample Dataset Name A",
"MonitoringLocationID": "ABC123",
"MonitoringLocationName": "Sample Location Name A",
"MonitoringLocationLatitude": 51.01,
"MonitoringLocationLongitude": -144.01,
"MonitoringLocationHorizontalCoordinateReferenceSystem": "WGS84",
"MonitoringLocationHorizontalAccuracyMeasure": null,
"MonitoringLocationHorizontalAccuracyUnit": null,
"MonitoringLocationVerticalMeasure": null,
"MonitoringLocationVerticalUnit": null,
"MonitoringLocationType": "River/Stream",
"ActivityType": "Field Msr/Obs-Portable Data Logger",
"ActivityMediaName": "Surface Water",
"ActivityStartDate": "2010-01-01",
"ActivityStartTime": "15:30:00",
"ActivityStartTimeZone": null,
"ActivityEndDate": null,
"ActivityEndTime": null,
"ActivityEndTimeZone": null,
"ActivityDepthHeightMeasure": -0.25,
"ActivityDepthHeightUnit": "m",
"SampleCollectionEquipmentName": null,
"CharacteristicName": "Dissolved oxygen saturation",
"MethodSpeciation": null,
"ResultSampleFraction": null,
"ResultValue": 97,
"ResultUnit": "%",
"ResultValueType": null,
"ResultDetectionCondition": null,
"ResultDetectionQuantitationLimitMeasure": null,
"ResultDetectionQuantitationLimitUnit": null,
"ResultDetectionQuantitationLimitType": null,
"ResultStatusID": null,
"ResultComment": null,
"ResultAnalyticalMethodID": null,
"ResultAnalyticalMethodContext": null,
"ResultAnalyticalMethodName": null,
"AnalysisStartDate": null,
"AnalysisStartTime": null,
"AnalysisStartTimeZone": null,
"LaboratoryName": null,
"LaboratorySampleID": null
}
],
"@odata.nextLink": "https://api.datastream.org/v1/odata/v4/Records?$skiptoken=Id:99999&$top=1000"
}
Response Format
{
"value": [{
"Id": "UUID",
...
}],
"@odata.nextLink": "https://api.datastream.org/v1/odata/v4/Observations?$skiptoken=Id:99999&$top=1000"
}
Errors
400 Bad Request: {"message":null}
This means your request was denied before reaching our service. This happens when the query string is not encoded properly. ie $ -> %24.
408 or 504 Timeout
This means your request was too complicated and was unable to complete within 30sec. To resolve this issue, retry once. If the issue persists, try lowering the $top and/or adding in narrower filters.
413 Payload Too Large
This means your request result was too large. Lowering $top or only requesting the values you need should resolve this issue.
429 Too Many Requests
This status code indicates that the client has sent too many requests in a given amount of time. To resolve this issue, rate limit yourself (2 reqs/sec) and don't make requests in parallel.
Disclaimer
We are currently in a Beta, changes will happen. We will do our best effort to keep you informed of any breaking changes.