From a2f119441f9b049713d4e991b3a536e220bbb45b Mon Sep 17 00:00:00 2001 From: Sorin Sbarnea Date: Wed, 14 Sep 2022 09:36:44 +0100 Subject: [PATCH] meta: refactor schema versioning to improve errors This greatly improves the quality of the validating error messages. Related: https://github.com/python-jsonschema/jsonschema/issues/1002 --- Taskfile.yml | 10 +-- f/ansible-meta.json | 37 ++++++--- negative_test/roles/empty_meta/meta/main.yml | 0 .../roles/empty_meta/meta/main.yml.md | 58 +++++++++++++ negative_test/roles/meta/main.yml | 1 + negative_test/roles/meta/main.yml.md | 46 ++-------- .../meta_invalid_collection/meta/main.yml | 2 +- .../meta_invalid_collection/meta/main.yml.md | 50 ++--------- .../meta_invalid_collections/meta/main.yml | 2 +- .../meta_invalid_collections/meta/main.yml.md | 50 ++--------- .../meta_invalid_role_namespace/meta/main.yml | 1 + .../meta/main.yml.md | 46 ++-------- .../role_with_bad_deps_in_meta/meta/main.yml | 1 + .../meta/main.yml.md | 83 ++----------------- test/roles/empty-meta/meta/main.yml.md | 58 +++++++++++++ 15 files changed, 193 insertions(+), 252 deletions(-) create mode 100644 negative_test/roles/empty_meta/meta/main.yml create mode 100644 negative_test/roles/empty_meta/meta/main.yml.md create mode 100644 test/roles/empty-meta/meta/main.yml.md diff --git a/Taskfile.yml b/Taskfile.yml index d0003db9..60abd678 100644 --- a/Taskfile.yml +++ b/Taskfile.yml @@ -57,7 +57,7 @@ tasks: - ".github" - ".vscode" - data - - f + - f/** - negative_test - src/** - test/** @@ -70,10 +70,10 @@ tasks: - npm run test - task: summary sources: - - f/ - - negative_test/ - - test/ - - src/ + - f/** + - negative_test/** + - test/** + - src/** - package.json - package-lock.json - Taskfile.yml diff --git a/f/ansible-meta.json b/f/ansible-meta.json index b5dfb18c..d57e8efc 100644 --- a/f/ansible-meta.json +++ b/f/ansible-meta.json @@ -1323,7 +1323,9 @@ "const": 1 } }, - "title": "Meta Schema v1 (standalone role)" + "required": ["version"], + "title": "Meta Schema v1 (standalone role)", + "type": "object" }, "v2": { "additionalProperties": false, @@ -1349,7 +1351,8 @@ "const": 2 } }, - "title": "Meta Schema v2 (role inside collection)" + "title": "Meta Schema v2 (role inside collection)", + "type": "object" }, "vCenterPlatformModel": { "properties": { @@ -1392,14 +1395,26 @@ }, "$id": "https://raw.githubusercontent.com/ansible/schemas/main/f/ansible-meta.json", "$schema": "http://json-schema.org/draft-07/schema", - "anyOf": [ - { - "$ref": "#/$defs/v1" - }, - { - "$ref": "#/$defs/v2" - } - ], + "else": { + "$ref": "#/$defs/v2" + }, "examples": ["meta/main.yml"], - "title": "Ansible Meta Schema v1/v2" + "if": { + "properties": { + "version": { + "const": 1 + } + } + }, + "properties": { + "version": { + "enum": [1, 2], + "type": "integer" + } + }, + "then": { + "$ref": "#/$defs/v1" + }, + "title": "Ansible Meta Schema v1/v2", + "type": "object" } diff --git a/negative_test/roles/empty_meta/meta/main.yml b/negative_test/roles/empty_meta/meta/main.yml new file mode 100644 index 00000000..e69de29b diff --git a/negative_test/roles/empty_meta/meta/main.yml.md b/negative_test/roles/empty_meta/meta/main.yml.md new file mode 100644 index 00000000..d1abae01 --- /dev/null +++ b/negative_test/roles/empty_meta/meta/main.yml.md @@ -0,0 +1,58 @@ +# ajv errors + +```json +[ + { + "instancePath": "", + "keyword": "type", + "message": "must be object", + "params": { + "type": "object" + }, + "schemaPath": "#/type" + }, + { + "instancePath": "", + "keyword": "if", + "message": "must match \"then\" schema", + "params": { + "failingKeyword": "then" + }, + "schemaPath": "#/if" + }, + { + "instancePath": "", + "keyword": "type", + "message": "must be object", + "params": { + "type": "object" + }, + "schemaPath": "#/type" + } +] +``` + +# check-jsonschema + +stdout: + +```json +{ + "status": "fail", + "errors": [ + { + "filename": "negative_test/roles/empty_meta/meta/main.yml", + "path": "$", + "message": "None is not of type 'object'", + "has_sub_errors": false + }, + { + "filename": "negative_test/roles/empty_meta/meta/main.yml", + "path": "$", + "message": "None is not of type 'object'", + "has_sub_errors": false + } + ], + "parse_errors": [] +} +``` diff --git a/negative_test/roles/meta/main.yml b/negative_test/roles/meta/main.yml index 3ed9a8c2..6ca7dfe9 100644 --- a/negative_test/roles/meta/main.yml +++ b/negative_test/roles/meta/main.yml @@ -1,3 +1,4 @@ +version: 1 galaxy_info: description: bar min_ansible_version: "2.9" diff --git a/negative_test/roles/meta/main.yml.md b/negative_test/roles/meta/main.yml.md index ca2a267c..bf950d60 100644 --- a/negative_test/roles/meta/main.yml.md +++ b/negative_test/roles/meta/main.yml.md @@ -12,29 +12,13 @@ "schemaPath": "#/properties/galaxy_tags/type" }, { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", - "params": { - "additionalProperty": "min_ansible_version" - }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", + "instancePath": "", + "keyword": "if", + "message": "must match \"then\" schema", "params": { - "additionalProperty": "galaxy_tags" + "failingKeyword": "then" }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" + "schemaPath": "#/if" } ] ``` @@ -49,23 +33,9 @@ stdout: "errors": [ { "filename": "negative_test/roles/meta/main.yml", - "path": "$", - "message": "{'galaxy_info': {'description': 'bar', 'min_ansible_version': '2.9', 'company': 'foo', 'license': 'MIT', 'galaxy_tags': 'database', 'platforms': [{'name': 'Alpine', 'versions': ['all']}]}} is not valid under any of the given schemas", - "has_sub_errors": true, - "best_match": { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('galaxy_tags', 'min_ansible_version' were unexpected)" - }, - "sub_errors": [ - { - "path": "$.galaxy_info.galaxy_tags", - "message": "'database' is not of type 'array'" - }, - { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('galaxy_tags', 'min_ansible_version' were unexpected)" - } - ] + "path": "$.galaxy_info.galaxy_tags", + "message": "'database' is not of type 'array'", + "has_sub_errors": false } ], "parse_errors": [] diff --git a/negative_test/roles/meta_invalid_collection/meta/main.yml b/negative_test/roles/meta_invalid_collection/meta/main.yml index 1d822352..56cac591 100644 --- a/negative_test/roles/meta_invalid_collection/meta/main.yml +++ b/negative_test/roles/meta_invalid_collection/meta/main.yml @@ -1,9 +1,9 @@ +version: 2 # <-- role inside a collection collections: - foo # invalid pattern galaxy_info: description: foo license: bar - min_ansible_version: "2.10" platforms: - name: Fedora versions: diff --git a/negative_test/roles/meta_invalid_collection/meta/main.yml.md b/negative_test/roles/meta_invalid_collection/meta/main.yml.md index ab980d0b..93c459c1 100644 --- a/negative_test/roles/meta_invalid_collection/meta/main.yml.md +++ b/negative_test/roles/meta_invalid_collection/meta/main.yml.md @@ -12,29 +12,13 @@ "schemaPath": "#/$defs/collections/items/pattern" }, { - "instancePath": "/collections/0", - "keyword": "pattern", - "message": "must match pattern \"^[a-z_]+\\.[a-z_]+$\"", - "params": { - "pattern": "^[a-z_]+\\.[a-z_]+$" - }, - "schemaPath": "#/$defs/collections/items/pattern" - }, - { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", + "instancePath": "", + "keyword": "if", + "message": "must match \"else\" schema", "params": { - "additionalProperty": "min_ansible_version" + "failingKeyword": "else" }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" + "schemaPath": "#/if" } ] ``` @@ -49,27 +33,9 @@ stdout: "errors": [ { "filename": "negative_test/roles/meta_invalid_collection/meta/main.yml", - "path": "$", - "message": "{'collections': ['foo'], 'galaxy_info': {'description': 'foo', 'license': 'bar', 'min_ansible_version': '2.10', 'platforms': [{'name': 'Fedora', 'versions': ['all']}]}} is not valid under any of the given schemas", - "has_sub_errors": true, - "best_match": { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" - }, - "sub_errors": [ - { - "path": "$.collections[0]", - "message": "'foo' does not match '^[a-z_]+\\\\.[a-z_]+$'" - }, - { - "path": "$.collections[0]", - "message": "'foo' does not match '^[a-z_]+\\\\.[a-z_]+$'" - }, - { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" - } - ] + "path": "$.collections[0]", + "message": "'foo' does not match '^[a-z_]+\\\\.[a-z_]+$'", + "has_sub_errors": false } ], "parse_errors": [] diff --git a/negative_test/roles/meta_invalid_collections/meta/main.yml b/negative_test/roles/meta_invalid_collections/meta/main.yml index ae9766e4..d3b84fc9 100644 --- a/negative_test/roles/meta_invalid_collections/meta/main.yml +++ b/negative_test/roles/meta_invalid_collections/meta/main.yml @@ -1,9 +1,9 @@ +version: 2 # <-- role inside a collection collections: - FOO.BAR # invalid pattern, need to use lowercase galaxy_info: description: foo license: bar - min_ansible_version: "2.10" platforms: - name: Fedora versions: diff --git a/negative_test/roles/meta_invalid_collections/meta/main.yml.md b/negative_test/roles/meta_invalid_collections/meta/main.yml.md index d3cb6033..899180e8 100644 --- a/negative_test/roles/meta_invalid_collections/meta/main.yml.md +++ b/negative_test/roles/meta_invalid_collections/meta/main.yml.md @@ -12,29 +12,13 @@ "schemaPath": "#/$defs/collections/items/pattern" }, { - "instancePath": "/collections/0", - "keyword": "pattern", - "message": "must match pattern \"^[a-z_]+\\.[a-z_]+$\"", - "params": { - "pattern": "^[a-z_]+\\.[a-z_]+$" - }, - "schemaPath": "#/$defs/collections/items/pattern" - }, - { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", + "instancePath": "", + "keyword": "if", + "message": "must match \"else\" schema", "params": { - "additionalProperty": "min_ansible_version" + "failingKeyword": "else" }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" + "schemaPath": "#/if" } ] ``` @@ -49,27 +33,9 @@ stdout: "errors": [ { "filename": "negative_test/roles/meta_invalid_collections/meta/main.yml", - "path": "$", - "message": "{'collections': ['FOO.BAR'], 'galaxy_info': {'description': 'foo', 'license': 'bar', 'min_ansible_version': '2.10', 'platforms': [{'name': 'Fedora', 'versions': ['all']}]}} is not valid under any of the given schemas", - "has_sub_errors": true, - "best_match": { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" - }, - "sub_errors": [ - { - "path": "$.collections[0]", - "message": "'FOO.BAR' does not match '^[a-z_]+\\\\.[a-z_]+$'" - }, - { - "path": "$.collections[0]", - "message": "'FOO.BAR' does not match '^[a-z_]+\\\\.[a-z_]+$'" - }, - { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" - } - ] + "path": "$.collections[0]", + "message": "'FOO.BAR' does not match '^[a-z_]+\\\\.[a-z_]+$'", + "has_sub_errors": false } ], "parse_errors": [] diff --git a/negative_test/roles/meta_invalid_role_namespace/meta/main.yml b/negative_test/roles/meta_invalid_role_namespace/meta/main.yml index d6dcf3fd..078bba5d 100644 --- a/negative_test/roles/meta_invalid_role_namespace/meta/main.yml +++ b/negative_test/roles/meta_invalid_role_namespace/meta/main.yml @@ -1,4 +1,5 @@ --- +version: 1 # <-- old standalone role galaxy_info: description: foo min_ansible_version: "2.9" diff --git a/negative_test/roles/meta_invalid_role_namespace/meta/main.yml.md b/negative_test/roles/meta_invalid_role_namespace/meta/main.yml.md index 3e2744b6..919b5d3b 100644 --- a/negative_test/roles/meta_invalid_role_namespace/meta/main.yml.md +++ b/negative_test/roles/meta_invalid_role_namespace/meta/main.yml.md @@ -12,29 +12,13 @@ "schemaPath": "#/properties/namespace/pattern" }, { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", - "params": { - "additionalProperty": "min_ansible_version" - }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", + "instancePath": "", + "keyword": "if", + "message": "must match \"then\" schema", "params": { - "additionalProperty": "namespace" + "failingKeyword": "then" }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" + "schemaPath": "#/if" } ] ``` @@ -49,23 +33,9 @@ stdout: "errors": [ { "filename": "negative_test/roles/meta_invalid_role_namespace/meta/main.yml", - "path": "$", - "message": "{'galaxy_info': {'description': 'foo', 'min_ansible_version': '2.9', 'namespace': 'foo-bar', 'company': 'foo', 'license': 'MIT', 'platforms': [{'name': 'Alpine', 'versions': ['all']}]}} is not valid under any of the given schemas", - "has_sub_errors": true, - "best_match": { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version', 'namespace' were unexpected)" - }, - "sub_errors": [ - { - "path": "$.galaxy_info.namespace", - "message": "'foo-bar' does not match '^[a-z][a-z0-9_]+$'" - }, - { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version', 'namespace' were unexpected)" - } - ] + "path": "$.galaxy_info.namespace", + "message": "'foo-bar' does not match '^[a-z][a-z0-9_]+$'", + "has_sub_errors": false } ], "parse_errors": [] diff --git a/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml b/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml index ff4464fc..c871542f 100644 --- a/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml +++ b/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml @@ -1,3 +1,4 @@ +version: 1 # <-- old standalone role galaxy_info: description: bar min_ansible_version: "2.9" diff --git a/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml.md b/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml.md index 70ed9e5c..6409f9e4 100644 --- a/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml.md +++ b/negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml.md @@ -37,54 +37,13 @@ "schemaPath": "#/anyOf" }, { - "instancePath": "/dependencies/0", - "keyword": "required", - "message": "must have required property 'role'", - "params": { - "missingProperty": "role" - }, - "schemaPath": "#/anyOf/0/required" - }, - { - "instancePath": "/dependencies/0", - "keyword": "required", - "message": "must have required property 'src'", - "params": { - "missingProperty": "src" - }, - "schemaPath": "#/anyOf/1/required" - }, - { - "instancePath": "/dependencies/0", - "keyword": "required", - "message": "must have required property 'name'", - "params": { - "missingProperty": "name" - }, - "schemaPath": "#/anyOf/2/required" - }, - { - "instancePath": "/dependencies/0", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" - }, - { - "instancePath": "/galaxy_info", - "keyword": "additionalProperties", - "message": "must NOT have additional properties", + "instancePath": "", + "keyword": "if", + "message": "must match \"then\" schema", "params": { - "additionalProperty": "min_ansible_version" + "failingKeyword": "then" }, - "schemaPath": "#/additionalProperties" - }, - { - "instancePath": "", - "keyword": "anyOf", - "message": "must match a schema in anyOf", - "params": {}, - "schemaPath": "#/anyOf" + "schemaPath": "#/if" } ] ``` @@ -99,18 +58,14 @@ stdout: "errors": [ { "filename": "negative_test/roles/role_with_bad_deps_in_meta/meta/main.yml", - "path": "$", - "message": "{'galaxy_info': {'description': 'bar', 'min_ansible_version': '2.9', 'company': 'foo', 'license': 'MIT', 'platforms': [{'name': 'Alpine', 'versions': ['all']}]}, 'dependencies': [{'version': 'foo'}]} is not valid under any of the given schemas", + "path": "$.dependencies[0]", + "message": "{'version': 'foo'} is not valid under any of the given schemas", "has_sub_errors": true, "best_match": { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" + "path": "$.dependencies[0]", + "message": "'role' is a required property" }, "sub_errors": [ - { - "path": "$.dependencies[0]", - "message": "{'version': 'foo'} is not valid under any of the given schemas" - }, { "path": "$.dependencies[0]", "message": "'role' is a required property" @@ -122,26 +77,6 @@ stdout: { "path": "$.dependencies[0]", "message": "'name' is a required property" - }, - { - "path": "$.dependencies[0]", - "message": "{'version': 'foo'} is not valid under any of the given schemas" - }, - { - "path": "$.dependencies[0]", - "message": "'role' is a required property" - }, - { - "path": "$.dependencies[0]", - "message": "'src' is a required property" - }, - { - "path": "$.dependencies[0]", - "message": "'name' is a required property" - }, - { - "path": "$.galaxy_info", - "message": "Additional properties are not allowed ('min_ansible_version' was unexpected)" } ] } diff --git a/test/roles/empty-meta/meta/main.yml.md b/test/roles/empty-meta/meta/main.yml.md new file mode 100644 index 00000000..9e26b217 --- /dev/null +++ b/test/roles/empty-meta/meta/main.yml.md @@ -0,0 +1,58 @@ +# ajv errors + +```json +[ + { + "instancePath": "", + "keyword": "type", + "message": "must be object", + "params": { + "type": "object" + }, + "schemaPath": "#/type" + }, + { + "instancePath": "", + "keyword": "if", + "message": "must match \"then\" schema", + "params": { + "failingKeyword": "then" + }, + "schemaPath": "#/if" + }, + { + "instancePath": "", + "keyword": "type", + "message": "must be object", + "params": { + "type": "object" + }, + "schemaPath": "#/type" + } +] +``` + +# check-jsonschema + +stdout: + +```json +{ + "status": "fail", + "errors": [ + { + "filename": "test/roles/empty-meta/meta/main.yml", + "path": "$", + "message": "None is not of type 'object'", + "has_sub_errors": false + }, + { + "filename": "test/roles/empty-meta/meta/main.yml", + "path": "$", + "message": "None is not of type 'object'", + "has_sub_errors": false + } + ], + "parse_errors": [] +} +```