Contrail Forecast API

Table of Contents

Overview

This document specifies API endpoints for contrail forecast data provided to air traffic planners in support of navigational contrail avoidance systems.

At present, this document specifies two endpoints for general availability.

  1. grids: The first endpoint surfaces gridded netCDF content at 0.25 degree grid spacing, across standardized flight levels, on a one hour interval, at all longitudes and latitudes ranging from [-80, 80] (this is expected to expand and encompass the poles, with an update at a later time). These grids are available across several aircraft classes; only the default aircraft class is served until there is sufficient evidence and agreement to implement additional aircraft classes. The variable returned in the netCDF is contrails, a domain-agnostic quantity of arbitrary units, existing in the range of [0, 4], 0 being a location with no contrail impact, and 4 being a location with severe contrail impact. The value of contrails is intended to scale, roughly, with the CO2 equivalence of contrail impact, and by extension, the appropriate fuel investment for avoiding a given region. i.e. one should be willing to burn 2x as much fuel to avoid a location with contrails of 3 over a region with contrails of 1.5.

  2. regions: The second endpoint surfaces contrail avoidance regions (aka polygons) served as geoJSON content, at the same flight levels and temporal intervals as the grid above. Regions are available at pre-defined contrails thresholds of [1, 2, 3, 4].

These two endpoints represent a minimal yet complete set of capabilities for air traffic planners to explore and prototype contrail avoidance in air traffic decision support systems.

The signature, behavior and design considerations of these two endpoints are detailed below.

If/when multiple aircraft classes are implemented, future work may include a discovery/recommendation endpoint to may include adding a discovery/recommendation endpoint, to help consumers map aircraft configuration attributes to a recommended aircraft class. The materialization of such an endpoint requires additional product discovery, and should not block or hinder adoption of the endpoints currently in GA.

grids

The v1/grids route returns pre-rendered netCDF assets.

signature

Request

GET /v1/grids?aircraft_class={ac_id}&time={ts}&flight_level={fl} HTTP/2
Host: {TBD}
Headers:
    x-api-key: {key}

Response

Headers:
    content-type: application/netcdf

auth

The {key} passed in x-api-key is a static (no expiry) token.

aircraft classes

{ac_id}, which defines the aircraft_class, is an optional query parameter. If provided, a singular value, default, is supported.

Possible future classes include the following: low_e, high_e. These aircraft classes correspond to the following pairs of representative aircraft type and engine configurations. See Engberg et al 2024 for more details on the creation of these aircraft classes.

low_e

aircraft_type_icao: B789

engine_uid: 01P17GE211

default

aircraft_type_icao: B738

engine_uid: 01P11CM116

high_e

aircraft_type_icao: A320

engine_uid: 01P10IA021

The aircraft_class values were chosen such that they be self-describing and ordinal yet leave sufficient room for future expansion.

time

The {ts} value must be a valid ISO8601 datetime object. It is a required query parameter.

This time represents the target time of the model prediction.

Given that ECMWF delivers 73 hours of forecast data every 6 hours, it is expected that a given ts will have multiple candidate URIs to serve to a client.

If multiple candidate URIs exist for a given time at the instance of the client request, the API should return the URI whose outputs were generated using the nearest forecast reference time.

time-handling behavior

If the caller provides a timezone designator for the {ts} param, the API should handle the timezone designator, converting it to a target time in UTC when fetching (⚠️ please note that response data is always in UTC).

If no timezone designator is provided, it is implied UTC.

If the {ts} value does not fall exactly on a matching time for a forecast resource, then the API will floor the {ts} value to the nearest interval time increment for which a a resource is expected to exist. For example, forecasts are currently available on an hourly basis. If a caller were to provide &time=2024-01-12T13:04:11, the API should attempt to retrieve a forecast resource entity matching 2024-01-12T13:00:00.

flight level

The {fl} value represents a single flight level (hectofeet) of the returned netCDF global grid. It is an optional query parameter. If no value is passed to {fl}, then all flight levels are included.

The fl value must be in the following:

[270, 280, 290, 300, 310, 320, 330, 340, 350, 360, 370, 380, 390, 400, 410, 420, 430, 440]

error handling

A 400 status code and informative message should be returned if:

  • the provided flight level is not recognized

  • the provided time string is malformed

  • the provided aircraft class is not recognized

  • the request is properly formed and interpretable, but the requested resource does not exist

response headers

No custom response headers.

response object

The netCDF object returned from the API represents contrail climate forcing in an arbitrary contrails unit, on a 0.25 degree by 0.25 degree basis, for a given flight level. The longitude range extends around the entire globe. The latitude range extends from 80N to 80S.

The data types for the coordinates and data variables are outlined below:

<xarray.Dataset> Size: 4MB
Dimensions:    (longitude: 1440, latitude: 641, flight_level: 1, time: 1)
Coordinates:
  * longitude  (longitude) float32 6kB -180.0 -179.75 -179.5 ... 179.5 179.75
  * latitude   (latitude) float32 3kB -80.0 -79.75 -79.5 ... 79.5 79.75 80.0
  * flight_level      (flight_level) int16 2B 300
  * time       (time) datetime64[ns] 8B 2024-07-01T12:00:00
    forecast_reference_time (time) datetime64[ns] 8B 2024-07-01T06:00:00
Data variables:
    contrails   (longitude, latitude, flight_level, time) float32 4MB ...
Attributes:
    aircraft_class: "default"

The contrails data variable has the following Attributes:

<xarray.DataArray 'contrails' (longitude: 1440, latitude: 641, level: 1, time: 1)> Size: 4MB
[923040 values with dtype=float32]
Coordinates:
  * longitude  (longitude) float32 6kB -180.0 -179.8 -179.5 ... 179.5 179.8
  * latitude   (latitude) float32 3kB -80.0 -79.75 -79.5 ... 79.5 79.75 80.0
  * flight_level      (flight_level) int16 2B 270
  * time       (time) datetime64[ns] 8B 2024-07-01T12:00:00
    forecast_reference_time (time) datetime64[ns] 8B 2024-07-01T06:00:00
Attributes:
    long_name:  'Contrail forcing index'
    units:      ''
    valid_min:  0
    valid_max:  4

regions

The v1/regions route returns pre-rendered geojson assets. The URI of the specific asset is fully defined by the suffixed path parameters. This endpoint effectively acts as a wrapper for content distribution of static assets in cloud storage.

signature

Request

GET /v1/regions?aircraft_class={ac_id}&time={ts}&flight_level={fl}&threshold={threshold} HTTP/2
Host: {TBD}
Headers:
    x-api-key: {key}

Response

Headers:
    content-type: application/geo+json

auth

Same as grids.auth

aircraft classes

Same as grids.aircraft_classes

time

Same as grids.time

flight level

Same as grids.flight_level

threshold

The threshold value is the contrails value used in generating the regions polygons. A polygon of a given threshold will surround a region of contrails values equal or greater than the threshold.

This is an optional query parameter. If no value is passed, then all thresholds will be included in the response.

The threshold value provided by the client must be in the following: [ 1, 2, 3, 4]

error handling

Same as grids.error_handling

response headers

Same as grids.response_headers

response object

The geoJSON object returned is a FeatureCollection, with the following format:

{
  "type": "FeatureCollection",
  "features": [
      {
        "type": "Feature",
        "properties": {
            "aircraft_class": "default",
            "time": "<%Y-%m-%dT%H:%M:%SZ>",
            "flight_level": <int>,
            "threshold": <int>,
            "forecast_reference_time": <%Y-%m-%dT%H:%M:%SZ>
        },
        "geometry": {
          "type": "MultiPolygon",
          "coordinates": []
        }
      }
  ]
}

The positions defined in the polygon coordinates include the third optional value of altitude (ref)

e.g. [165.5,63.12, 10363]. The altitude value is in units of meters.

Changelog

2025.01.08

  • /grids endpoint now supports returning all flight levels in the response netCDF. This is default behavior when no fl query parameter value is passed.

  • /regions endpoint now supports returning Features for all flight levels in the response netCDF. This is default behavior when no fl query parameter value is passed.

  • /regions endpoint now supports returning Features for all threshold (contrail severity) values. This is default behavior when no threshold query param value is passed.

2024.10.09

  • /grids endpoint updated to return a modified netCDF file, replacing the variable ef_per_m with the variable contrails, and adding forcast_reference_time and aircraft_class to the netCDF global attributes.

  • /regions endpoint updated to return geoJSON polygons rendered with a threshold based on contrails rather than ef_per_m

    • new supported threshold include [1, 2, 3, 4]

  • /regions geoJSON response object format updated from .features.* to .features[].*.

2024.11.14

  • /grids endpoint reworked to move all path parameters to query parameters. aircraft_class is documented, but optional.

  • /grids move forecast_reference_time to a non-dimension coordinate; remove forecast_reference_time from data-array attributes.

  • /regions endpoint reworked to remove header k-vs. aircraft_class is documented, but optional.

  • /regions geoJSON response object updated to include resource-identifying attributes in the "properties" object for the geoJSON Feature.

  • /grids and /regions to take a query param time (renamed from timestamp), with updated handling/interpretation (see time section)