Add more CI checks for OpenAPI definitions and JSON Schemas (#1656)
Signed-off-by: Kévin Commaille <zecakeh@tedomum.fr> Co-authored-by: Richard van der Hoff <1389908+richvdh@users.noreply.github.com>
This commit is contained in:
parent
c71b528148
commit
560d98ba9b
9 changed files with 315 additions and 34 deletions
|
@ -1,5 +1,10 @@
|
|||
#! /usr/bin/env python
|
||||
#
|
||||
|
||||
# Validates the OpenAPI definitions under `../data/api`. Checks the request
|
||||
# parameters and body, and response body. The schemas are validated against the
|
||||
# JSON Schema 2020-12 specification and the examples are validated against those
|
||||
# schemas.
|
||||
|
||||
# Copyright 2016 OpenMarket Ltd
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||
|
@ -47,17 +52,39 @@ def check_schema(filepath, example, schema):
|
|||
example = resolve_references(filepath, example)
|
||||
schema = resolve_references(filepath, schema)
|
||||
resolver = jsonschema.RefResolver(filepath, schema, handlers={"file": load_file})
|
||||
jsonschema.validate(example, schema, resolver=resolver)
|
||||
validator = jsonschema.Draft202012Validator(schema, resolver)
|
||||
validator.validate(example)
|
||||
|
||||
|
||||
def check_parameter(filepath, request, parameter):
|
||||
schema = parameter.get("schema")
|
||||
example = schema.get('example')
|
||||
schema = parameter.get('schema')
|
||||
example = parameter.get('example')
|
||||
|
||||
if not example:
|
||||
example = schema.get('example')
|
||||
|
||||
if example and schema:
|
||||
try:
|
||||
print("Checking request schema for: %r %r" % (
|
||||
filepath, request
|
||||
print("Checking schema for request parameter: %r %r %r" % (
|
||||
filepath, request, parameter.get("name")
|
||||
))
|
||||
check_schema(filepath, example, schema)
|
||||
except Exception as e:
|
||||
raise ValueError("Error validating JSON schema for %r" % (
|
||||
request
|
||||
), e)
|
||||
|
||||
def check_request_body(filepath, request, body):
|
||||
schema = body.get('schema')
|
||||
example = body.get('example')
|
||||
|
||||
if not example:
|
||||
example = schema.get('example')
|
||||
|
||||
if example and schema:
|
||||
try:
|
||||
print("Checking schema for request body: %r %r" % (
|
||||
filepath, request,
|
||||
))
|
||||
check_schema(filepath, example, schema)
|
||||
except Exception as e:
|
||||
|
@ -67,44 +94,59 @@ def check_parameter(filepath, request, parameter):
|
|||
|
||||
|
||||
def check_response(filepath, request, code, response):
|
||||
example = response.get('examples', {}).get('application/json')
|
||||
schema = response.get('schema')
|
||||
if example and schema:
|
||||
try:
|
||||
print ("Checking response schema for: %r %r %r" % (
|
||||
filepath, request, code
|
||||
))
|
||||
check_schema(filepath, example, schema)
|
||||
except jsonschema.SchemaError as error:
|
||||
for suberror in sorted(error.context, key=lambda e: e.schema_path):
|
||||
print(list(suberror.schema_path), suberror.message, sep=", ")
|
||||
raise ValueError("Error validating JSON schema for %r %r" % (
|
||||
request, code
|
||||
), e)
|
||||
except Exception as e:
|
||||
raise ValueError("Error validating JSON schema for %r %r" % (
|
||||
request, code
|
||||
), e)
|
||||
if schema:
|
||||
for name, example in response.get('examples', {}).items():
|
||||
value = example.get('value')
|
||||
if value:
|
||||
try:
|
||||
print ("Checking response schema for: %r %r %r %r" % (
|
||||
filepath, request, code, name
|
||||
))
|
||||
check_schema(filepath, value, schema)
|
||||
except jsonschema.SchemaError as error:
|
||||
for suberror in sorted(error.context, key=lambda e: e.schema_path):
|
||||
print(list(suberror.schema_path), suberror.message, sep=", ")
|
||||
raise ValueError("Error validating JSON schema for %r %r" % (
|
||||
request, code
|
||||
), e)
|
||||
except Exception as e:
|
||||
raise ValueError("Error validating JSON schema for %r %r" % (
|
||||
request, code
|
||||
), e)
|
||||
|
||||
|
||||
def check_openapi_file(filepath):
|
||||
with open(filepath) as f:
|
||||
openapi = yaml.safe_load(f)
|
||||
|
||||
openapi_version = openapi.get('openapi')
|
||||
if not openapi_version:
|
||||
# This is not an OpenAPI file, skip.
|
||||
return
|
||||
elif openapi_version != '3.1.0':
|
||||
raise ValueError("File %r is not using the proper OpenAPI version: expected '3.1.0', got %r" % (filepath, openapi_version))
|
||||
|
||||
for path, path_api in openapi.get('paths', {}).items():
|
||||
|
||||
for method, request_api in path_api.items():
|
||||
request = "%s %s" % (method.upper(), path)
|
||||
for parameter in request_api.get('parameters', ()):
|
||||
if parameter['in'] == 'body':
|
||||
check_parameter(filepath, request, parameter)
|
||||
check_parameter(filepath, request, parameter)
|
||||
|
||||
json_body = request_api.get('requestBody', {}).get('content', {}).get('application/json')
|
||||
if json_body:
|
||||
check_request_body(filepath, request, json_body)
|
||||
|
||||
try:
|
||||
responses = request_api['responses']
|
||||
except KeyError:
|
||||
raise ValueError("No responses for %r" % (request,))
|
||||
for code, response in responses.items():
|
||||
check_response(filepath, request, code, response)
|
||||
json_response = response.get('content', {}).get('application/json')
|
||||
|
||||
if json_response:
|
||||
check_response(filepath, request, code, json_response)
|
||||
|
||||
|
||||
def resolve_references(path, schema):
|
||||
|
@ -171,7 +213,7 @@ if __name__ == '__main__':
|
|||
|
||||
# Resolve the directory containing the OpenAPI sources,
|
||||
# relative to the script path
|
||||
source_files_directory = os.path.realpath(os.path.join(script_directory, "../data"))
|
||||
source_files_directory = os.path.realpath(os.path.join(script_directory, "../data/api"))
|
||||
|
||||
# Walk the source path directory, looking for YAML files to check
|
||||
for (root, dirs, files) in os.walk(source_files_directory):
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue