Naar hoofdinhoud

CBS Content Open Data - API score details

API-metagegevens

Goede metagegevens verbetert de kwaliteit van een API. Wij controleren of belangrijke metagegevens zijn ingevuld in het API-definitiebestand: specificatie, documentatie, contactgegevens en een Service Level Agreement (SLA). Het API-definitiebestand staat op GitLab (https://gitlab.com/commonground/don/don-content) en kan door iedereen worden bijgewerkt.

  • documentatie: Documentatie beschikbaar

    Het is belangrijk dat een API documentatie heeft met alle informatie die nodig is voor het eenvoudig en effectief gebruik van de API.

  • specificatie: Specificatie beschikbaar

    De API-specificatie beschrijft welke gegevens een API biedt en hoe deze gegevens kunnen worden opgevraagd.

  • contact: Contactgegevens beschikbaar

    Het is belangrijk om contactgegevens te verstrekken van degenen die verantwoordelijk zijn voor een API omdat gebruikers in contact moeten kunnen komen met iemand om problemen te rapporteren en vragen te stellen over de API.

  • sla: Service Level Agreement (SLA) ingevuld

    Een Service Level Agreement (of serviceniveau-overeenkomst) is belangrijk omdat het opgeeft wat de uptime-garantie is of er een helpdesk is, en binnen hoeveel dagen de helpdesk reageert.

NLGov REST API Design Rules 2.0.1

More and more governmental organizations offer REST APIs (henceforth abbreviated as APIs), in addition to existing interfaces like SOAP and WFS. These APIs aim to be developer-friendly and easy to implement. While this is a commendable aim, it does not shield a developer from a steep learning curve getting to know every new API, in particular when every individual API is designed using different patterns and conventions.

Documentatie
  • /core/http-methods: Only apply standard HTTP methods

    The HTTP specification rfc7231 and the later introduced PATCH method specification rfc5789 offer a set of standard methods, where every method is designed with explicit semantics. Adhering to the HTTP specification is crucial, since HTTP clients and middleware applications rely on standardized characteristics. Therefore, resources must be retrieved or manipulated using standard HTTP methods.

    Naar documentatie
  • /core/doc-openapi: Use OpenAPI Specification for documentation

    The OpenAPI Specification (OAS) defines a standard, language-agnostic interface to RESTful APIs which allowsboth humans and computers to discover and understand the capabilities of the service without access to source code, documentation, or through network traffic inspection. When properly defined, a consumer can understand and interact with the remote service with a minimal amount of implementation logic. API documentation must be provided in the form of an OpenAPI definition document which conforms to the OpenAPI Specification (from v3 onwards). As a result, a variety of tools can be used to render the documentation (e.g. Swagger UI or ReDoc) or automate tasks such as testing or code generation. The OAS document should provide clear descriptions and examples.

    Naar documentatie
  • /core/uri-version: Include the major version number in the URI

    The URI of an API (base path) must include the major version number, prefixed by the letter v. This allows the exploration of multiple versions of an API in the browser. The minor and patch version numbers are not part of the URI and may not have any impact on existing client implementations.

    Naar documentatie
  • /core/no-trailing-slash: Leave off trailing slashes from URIs

    According to the URI specification rfc3986, URIs may contain a trailing slash. However, for REST APIs this is considered as a bad practice since a URI including or excluding a trailing slash might be interpreted as different resources (which is strictly speaking the correct interpretation). To avoid confusion and ambiguity, a URI must never contain a trailing slash. When requesting a resource including a trailing slash, this must result in a 404 (not found) error response and not a redirect. This enforces API consumers to use the correct URI.

    Naar documentatie
  • /core/publish-openapi: Publish OAS document at a standard location in JSON-format

    To make the OAS document easy to find and to facilitate self-discovering clients, there should be one standard location where the OAS document is available for download. Clients (such as Swagger UI or ReDoc) must be able to retrieve the document without having to authenticate. Furthermore, the CORS policy for this URI must allow external domains to read the documentation from a browser environment. The standard location for the OAS document is a URI called openapi.json or openapi.yaml within the base path of the API. This can be convenient, because OAS document updates can easily become part of the CI/CD process. At least the JSON format must be supported. When having multiple (major) versions of an API, every API should provide its own OAS document(s).

    Naar documentatie

    OpenAPI Specification in YAML format is not the same as the JSON format

    Details

    Diff between JSON and YAML
    The 'original' keys in the changes refers to the JSON document, 'new' to the YAML document.
    {
      "paths": {
        "pathItems": {
          "/Articles({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Definitions({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Events({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Flash({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/GetArticlesByArticleType(ArticleType={ArticleType})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/GetArticlesBySeries(Series={Series})": {
            "get": {}
          },
          "/GetArticlesByTableId(TableId={TableId})": {
            "get": {}
          },
          "/GetArticlesByTaxonomyTag(Tag={Tag})": {
            "get": {}
          },
          "/GetArticlesByTheme(Theme={Theme})": {
            "get": {}
          },
          "/GetMediaByMediaType(MediaType={MediaType})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/GetPagesByPageType(PageType={PageType})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/GetPagesByResearchCode(ResourceCode={ResourceCode})": {
            "get": {}
          },
          "/GetPagesBySeries(Series={Series})": {
            "get": {}
          },
          "/GetPagesByTaxonomyTag(Tag={Tag})": {
            "get": {}
          },
          "/GetPagesByTheme(Theme={Theme})": {
            "get": {}
          },
          "/GetSearchResultsByArticleType(ArticleType={ArticleType})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/GetSearchResultsByPath(Path={Path})": {
            "get": {}
          },
          "/GetSearchResultsByPathAndTemplateId(Path={Path},TemplateId={TemplateId})": {
            "get": {}
          },
          "/GetSearchResultsByTemplate(Template={Template})": {
            "get": {}
          },
          "/GetSearchResultsByTheme(Theme={Theme})": {
            "get": {}
          },
          "/GetSearchResultsByWord(Word={Word})": {
            "get": {}
          },
          "/GetSearchResultsByWordAndTemplateId(Word={Word},TemplateId={TemplateId})": {
            "get": {}
          },
          "/Media({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Pages({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Search({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          },
          "/Vacancies({UniqueId})": {
            "get": {
              "parameters": [
                {}
              ]
            }
          }
        }
      },
      "components": {
        "schemas": {
          "CBS.Website.ODataApi.Models.ArticleType": {
            "changes": [
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Corporate",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Article",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Custom",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Publication",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Visualisation",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Any",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Background",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "News",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Unknown",
                "breaking": false
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Article",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "News",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Visualisation",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Corporate",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Unknown",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Any",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Custom",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Publication",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Background",
                "breaking": true
              }
            ]
          },
          "CBS.Website.ODataApi.Models.DefinitionType": {
            "changes": [
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "FAQ",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Explanation",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Definition",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Unknown",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Any",
                "breaking": false
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Any",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "FAQ",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Explanation",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Definition",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Unknown",
                "breaking": true
              }
            ]
          },
          "CBS.Website.ODataApi.Models.MediaType": {
            "changes": [
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Chart",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Image",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Table",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Unknown",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Any",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Video",
                "breaking": false
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Table",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Unknown",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Any",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Video",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Chart",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Image",
                "breaking": true
              }
            ]
          },
          "CBS.Website.ODataApi.Models.PageType": {
            "changes": [
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Any",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Page",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Visualisation",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Form",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "LongreadChapter",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Unknown",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Dossier",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Overview",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Vacancy",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "Longread",
                "breaking": false
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "LongreadChapter",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Dossier",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Form",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Visualisation",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Overview",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Vacancy",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Longread",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Unknown",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Any",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "Page",
                "breaking": true
              }
            ]
          },
          "ReferenceNumeric": {
            "changes": [
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "-INF",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "INF",
                "breaking": false
              },
              {
                "context": {
                  "newLine": 0,
                  "newColumn": 0
                },
                "change": 2,
                "property": "enum",
                "new": "NaN",
                "breaking": false
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "INF",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "NaN",
                "breaking": true
              },
              {
                "context": {
                  "originalLine": 0,
                  "originalColumn": 0
                },
                "change": 5,
                "property": "enum",
                "original": "-INF",
                "breaking": true
              }
            ]
          }
        }
      }
    }
  • /core/semver: Adhere to the Semantic Versioning model when releasing API changes

    Version numbering must follow the Semantic Versioning (SemVer) model to prevent breaking changes when releasing new API versions. Versions are formatted using the major.minor.patch template. When releasing a new version which contains backwards-incompatible changes, a new major version must be released. Minor and patch releases may only contain backwards compatible changes (e.g. the addition of an endpoint or an optional attribute).

    Naar documentatie
  • /core/version-header: Return the full version number in a response header

    Since the URI only contains the major version, it's useful to provide the full version number in the response headers for every API call. This information could then be used for logging, debugging or auditing purposes. In cases where an intermediate networking component returns an error response (e.g. a reverse proxy enforcing access policies), the version number may be omitted. The version number must be returned in an HTTP response header named API-Version (case-insensitive) and SHOULD not be prefixed.

    Naar documentatie

Secure connections using TLS

Secure connections using TLS following the latest National Cyber Security Center (NCSC) guidelines. Guidelines document: https://english.ncsc.nl/publications/publications/2021/january/19/it-security-guidelines-for-transport-layer-security-2.1 Note: this ruleset is in development and it does not test for the complicany of all the guidelines.

Documentatie
  • tls-version: Test for secure TLS versions

    All supported versions of TLS are 'Good' or 'Sufficient' (B1-1).

  • cipher-suites: Test for secure ciphers suites (algorithm selections)

    All supported algorithm selections contain a 'Good' or 'Sufficient' algorithm for certificate verification, key exchange, bulk encryption and hashing (B2-1 through B2-4).