Skip to main content

Authoritative tooling for creating OGC API - EDR Part 3 Service Profiles

Project description

OGC API - EDR Part 3 Service Profile Generator

Authoritative tooling for creating OGC API - Environmental Data Retrieval (EDR) Part 3 Service Profiles, built on Pydantic and edr-pydantic.

Overview

Profile structure is defined as Pydantic models (src/ogc_edr_profile/models.py). Instantiating a ServiceProfile validates the entire profile — cross-model validators catch referential errors — before any files are written.

Collections use edr-pydantic's authoritative Collection model directly, meaning a profile config is simultaneously a valid EDR collection descriptor and a Part 3 profile definition.

Installation

pip install ogc-edr-profile

Workflow

1. Author a Profile Config

A profile config is a YAML or JSON file. Copy the example and modify it:

cp examples/nwsviz_profile.yaml my_profile.yaml

The minimal valid config:

name: my_profile
title: My EDR Profile

collections:
  - id: my_collection
    links:
      - href: https://example.com/collections/my_collection
        rel: self
        type: application/json
    extent:
      spatial:
        bbox:
          - [-180, -90, 180, 90]
        crs: "http://www.opengis.net/def/crs/OGC/1.3/CRS84"
    parameter_names: {}

See examples/nwsviz_profile.yaml for a full working config with 13 collections, 3 processes, requirements, abstract tests, collection_examples, and document_metadata.

2. Generate Profile Artifacts

ogc-edr-profile generate \
  --config my_profile.yaml \
  --output ./my_profile

Produces:

my_profile/
├── openapi.yaml
├── profile_config.json
├── document.adoc                        # Metanorma root document
├── sections/
│   ├── 00-abstract.adoc
│   ├── 01-preface.adoc
│   ├── 02-scope.adoc
│   ├── 03-conformance.adoc
│   ├── 04-references.adoc
│   ├── 05-terms.adoc
│   ├── 06-requirements.adoc
│   └── 07-abstract-tests.adoc
├── requirements/
│   ├── requirements_class_core.adoc
│   └── core/REQ_<id>.adoc
└── abstract_tests/
    ├── ATS_class_core.adoc
    └── core/ATS_<id>.adoc

Validate a config without generating output:

ogc-edr-profile validate --config my_profile.yaml

3. Compile OGC PDF

Requires Docker. Shells out to the official metanorma/metanorma image — no Ruby or LaTeX install needed.

ogc-edr-profile generate \
  --config my_profile.yaml \
  --output ./my_profile \
  --pdf

The document_metadata block in the profile config drives the Metanorma document header:

document_metadata:
  doc_number: "24-nwsviz"
  doc_subtype: implementation
  copyright_year: 2026
  editors:
    - Shane Mill
  submitting_orgs:
    - NOAA/NWS/MDL
  keywords:
    - ogcdoc
    - OGC API
    - EDR
    - NWSViz
    - service profile
  external_id: http://www.opengis.net/doc/dp/ogcapi-edr-nwsviz/1.0

Produces my_profile/document.pdf — a fully compliant OGC draft-standard PDF with Abstract, Preface, Scope, Conformance, References, Terms, Requirements class, and normative Abstract Test Suite annex.

4. Validate Against a Live Server

ogc-edr-profile validate-server \
  --config my_profile.yaml \
  --url https://edr-api-desi-c.mdl.nws.noaa.gov \
  --max-examples 3

Results:

Operations:  100 selected / 106 total
Tested:      47
Test cases:  1002 generated, 1002 passed

No issues found in 49.51s

Use --stateful to additionally test job lifecycle endpoints (/jobs/{jobId}, DELETE /jobs/{jobId}) via POST /execution chaining.

Add collection_examples to your config to supply real instanceId values for schemathesis path parameters:

collection_examples:
  my_collection:
    instanceId: "2025-04-02T00:00:00Z"

5. OGC CITE Conformance Testing

EDR Conformance Testing

Run the official OGC API - EDR Part 1 conformance test suite (ets-ogcapi-edr10):

ogc-edr-profile cite-test \
  --url https://edr-api-desi-c.mdl.nws.noaa.gov \
  --report ./cite_results

Results:

OGC API - EDR CITE Results
  Passed:  76/84
  Failed:  0
  Skipped: 8

✓ All CITE tests passed.

The tool automatically:

  • Clones and builds ets-ogcapi-edr10 from GitHub on first run
  • Caches Docker image (ogccite/ets-ogcapi-edr10:local) for subsequent runs
  • Runs TestNG tests via docker exec
  • Supports localhost testing with --network host mode
  • Generates JSON report with detailed test results

The skipped tests are optional features not implemented by the server.

Features Conformance Testing

Run the official OGC API - Features Part 1 conformance test suite (ets-ogcapi-features10):

ogc-edr-profile cite-test-features \
  --url https://api.example.com \
  --report ./cite_features_results

Results:

OGC API - Features CITE Results
  Passed:  639/712
  Failed:  18
  Skipped: 55

✗ 18 test(s) failed.

The tool automatically:

  • Pulls pre-built Docker image (ogccite/ets-ogcapi-features10:latest) from Docker Hub
  • Runs TestNG tests via docker exec
  • Supports localhost testing with --network host mode
  • Generates JSON report with detailed test results

The skipped tests are optional features not implemented by the server.


Config Reference

Top-level fields

Field Type Required Description
name string yes Lowercase identifier using only a-z, 0-9, _. Used in OGC URIs and OpenAPI operationIds. e.g. water_gauge
title string yes Human-readable profile title
version string no Profile version. Defaults to 1.0
server_url string no Base URL of the live server. Populates the OpenAPI servers block
collections list yes One or more EDR collections (see below)
processes list no OGC API Processes to include in the OpenAPI (see below)
requirements list no Normative requirements (see below)
abstract_tests list no Conformance tests — each must reference a valid requirement id (see below)
pubsub object no OGC API - EDR Part 2 PubSub configuration (see below)
collection_examples object no Map of collection id → example parameter values (e.g. instanceId) for server validation
document_metadata object no Metanorma document header fields for PDF compilation (see below)

collections[]

Each collection uses the edr-pydantic Collection schema — the same model an EDR server returns at /collections/{id}. Key fields:

Field Type Required Description
id string yes Collection identifier
title string no Human-readable collection name
description string no Collection description
links list yes At minimum a self link
extent.spatial.bbox list yes Bounding box as [[minLon, minLat, maxLon, maxLat]]
extent.spatial.crs string yes CRS URI, typically http://www.opengis.net/def/crs/OGC/1.3/CRS84
data_queries object no Which EDR query types this collection supports
output_formats list no Supported output format labels e.g. GeoJSON, CoverageJSON
parameter_names object no Map of parameter id → Parameter object

data_queries

Supported keys: items · position · area · radius · cube · trajectory · corridor · locations · instances

data_queries:
  position:
    link:
      href: https://example.com/collections/water_gauge/position
      rel: data
      variables:
        query_type: position
        output_formats:
          - CoverageJSON
  items:
    link:
      href: https://example.com/collections/water_gauge/items
      rel: data
      variables:
        query_type: items
        output_formats:
          - GeoJSON

parameter_names

parameter_names:
  gauge_height:
    type: Parameter
    observedProperty:
      label: Gauge Height
    unit:
      label: feet
      symbol: ft

processes[]

OGC API Processes to expose in the generated OpenAPI. Each entry produces /processes/{id} and /processes/{id}/execution paths, plus /processes, /jobs, /jobs/{jobId}, and /jobs/{jobId}/results.

Field Type Required Description
id string yes Process identifier e.g. edr-zarr-difference
title string no Human-readable process name
description string no Process description
output_content object no OpenAPI content map for the 200 response. Defaults to application/json
processes:
  - id: edr-zarr-difference
    title: EDR Zarr Dataset Difference
    description: Calculates the difference between two EDR Zarr datasets.
    output_content:
      application/zip:
        schema:
          type: object

requirements[]

Field Type Required Description
id string yes Lowercase, hyphen-separated. Must match ^[a-z0-9][a-z0-9\-]*$
statement string yes One-sentence normative statement
parts list[string] yes One or more SHALL/MUST clauses
requirements:
  - id: position-coveragejson
    statement: The position query SHALL return CoverageJSON.
    parts:
      - The service SHALL provide a /collections/{id}/position endpoint.
      - The response Content-Type SHALL be application/prs.coverage+json.

abstract_tests[]

Every requirement_id must match an existing requirement id — the model validator will reject the profile if not.

Field Type Required Description
id string yes Must equal requirement_id
requirement_id string yes The id of the requirement this test validates
steps list[string] yes Ordered test steps
abstract_tests:
  - id: position-coveragejson
    requirement_id: position-coveragejson
    steps:
      - Send GET request to /collections/{id}/position?coords=POINT(lon lat).
      - Verify the response Content-Type is application/prs.coverage+json.

pubsub

When present, an asyncapi.yaml is generated.

Field Type Default Description
broker_host string localhost Message broker hostname
broker_port integer 5672 Broker port (1–65535)
protocol string amqp One of amqp, mqtt, kafka
filters list [] Subscription filters

Each filter: name (required), description (required), type (one of string, number, array, boolean, default string).


document_metadata

Controls the Metanorma document header when compiling a PDF with --pdf.

Field Type Required Description
doc_number string yes OGC document number e.g. 24-nwsviz
doc_subtype string yes One of implementation, best-practice, engineering-report
editors list[string] yes Editor names
submitting_orgs list[string] yes Submitting organization names
keywords list[string] no Document keywords
copyright_year integer no Defaults to current year
external_id string no OGC external document URI

Programmatic Use

from ogc_edr_profile.models import ServiceProfile
from ogc_edr_profile.generate import generate
from pathlib import Path

profile = ServiceProfile.model_validate(config_dict)
generate(profile, Path("./output"))

Repository Structure

├── src/
│   └── ogc_edr_profile/
│       ├── models.py            # Authoritative Pydantic schema
│       ├── generate.py          # Validated model → OpenAPI, AsyncAPI, AsciiDoc
│       ├── compile.py           # PDF compilation via metanorma/metanorma Docker image
│       ├── cite.py              # OGC CITE test suite orchestration
│       └── cli.py               # CLI entry point
├── examples/
│   ├── water_gauge.yaml         # Minimal example profile config
│   └── nwsviz_profile.yaml      # Full NWSViz profile: 13 collections, 3 processes, PDF metadata
├── profile.schema.json          # Machine-readable JSON Schema for profile configs
└── pyproject.toml

Standards

  • OGC API - EDR Part 1: Core
  • OGC API - EDR Part 2: PubSub
  • OGC API - EDR Part 3: Service Profiles (draft)
  • OGC API - Processes Part 1
  • OpenAPI 3.0 / AsyncAPI 3.0
  • Metanorma/AsciiDoc documentation format

License

MIT — See LICENSE for details.

Contact

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

ogc_edr_profile-1.2.0.tar.gz (28.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

ogc_edr_profile-1.2.0-py3-none-any.whl (33.5 kB view details)

Uploaded Python 3

File details

Details for the file ogc_edr_profile-1.2.0.tar.gz.

File metadata

  • Download URL: ogc_edr_profile-1.2.0.tar.gz
  • Upload date:
  • Size: 28.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.13

File hashes

Hashes for ogc_edr_profile-1.2.0.tar.gz
Algorithm Hash digest
SHA256 d51a7f2d5c6d24395c8466777a09ce11a104109f1a1ffec385eb879019d4cdd4
MD5 e23dca278643f6bd1114d27c686b7ac8
BLAKE2b-256 3591275693719b1baa7882f3ea015bc1625c9338beaa1c0a0e91a228fed637ae

See more details on using hashes here.

File details

Details for the file ogc_edr_profile-1.2.0-py3-none-any.whl.

File metadata

File hashes

Hashes for ogc_edr_profile-1.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 ff3e2dc941212c61f0b5bc542e6aadd059dae8398d6d2279e7f8df4dc5a059e3
MD5 25080ecf6db5fa4acdbec784064206e5
BLAKE2b-256 6878f28ea63e7cdb61487e85ab35162311ab0dc325805a21de9c8ed9f89c4a76

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page