From d601aebb91b50bbe3d5120a5e408079f7796e8cb Mon Sep 17 00:00:00 2001 From: Sebastian Toepfer <61313468+sebastian-toepfer@users.noreply.github.com> Date: Sun, 3 Dec 2023 16:44:35 +0100 Subject: [PATCH 1/3] add limit support for ref -> works in properties --- core/pom.xml | 9 ++- .../core/DefaultJsonObjectSchema.java | 51 ++------------ .../core/DefaultJsonSchemaFactory.java | 11 ++- .../jsonschema/core/DefaultJsonSubSchema.java | 38 ++++++++-- .../core/KeywordBasedValidator.java | 70 +++++++++++++++++++ .../jsonschema/core/KeywordExtractor.java | 57 +++++++++++++++ .../applicator/PropertiesKeywordType.java | 29 +++++--- .../core/vocab/core/RefKeywordType.java | 13 ++-- .../vocab/core/VocabularyKeywordType.java | 4 +- core/src/test/java/module-info.java | 1 + 10 files changed, 215 insertions(+), 68 deletions(-) create mode 100644 core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordBasedValidator.java create mode 100644 core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordExtractor.java diff --git a/core/pom.xml b/core/pom.xml index e6d27979..6a5782d1 100644 --- a/core/pom.xml +++ b/core/pom.xml @@ -114,6 +114,10 @@ **/tests/draft2020-12/enum.json **/tests/draft2020-12/exclusiveMaximum.json **/tests/draft2020-12/exclusiveMinimum.json + **/tests/draft2020-12/format.json + @@ -128,7 +132,10 @@ **/tests/draft2020-12/patternProperties.json **/tests/draft2020-12/prefixItems.json **/tests/draft2020-12/properties.json - + **/tests/draft2020-12/required.json **/tests/draft2020-12/type.json **/tests/draft2020-12/uniqueItems.json diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonObjectSchema.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonObjectSchema.java index cf90c34e..87813013 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonObjectSchema.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonObjectSchema.java @@ -26,25 +26,14 @@ import static java.util.stream.Collectors.collectingAndThen; import static java.util.stream.Collectors.toList; -import io.github.sebastiantoepfer.jsonschema.InstanceType; import io.github.sebastiantoepfer.jsonschema.JsonSubSchema; import io.github.sebastiantoepfer.jsonschema.Validator; -import io.github.sebastiantoepfer.jsonschema.core.codition.AllOfCondition; -import io.github.sebastiantoepfer.jsonschema.core.codition.ApplicatorBasedCondtion; -import io.github.sebastiantoepfer.jsonschema.core.codition.AssertionBasedCondition; -import io.github.sebastiantoepfer.jsonschema.core.codition.Condition; -import io.github.sebastiantoepfer.jsonschema.core.vocab.core.VocabularyKeywordType; import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; -import io.github.sebastiantoepfer.jsonschema.keyword.KeywordType; -import io.github.sebastiantoepfer.jsonschema.vocabulary.spi.VocabularyDefinition; -import io.github.sebastiantoepfer.jsonschema.vocabulary.spi.VocabularyDefinitions; import jakarta.json.JsonObject; -import jakarta.json.JsonValue; -import java.util.Collection; import java.util.Optional; import java.util.stream.Stream; -final class DefaultJsonObjectSchema extends AbstractJsonValueSchema { +public final class DefaultJsonObjectSchema extends AbstractJsonValueSchema { public DefaultJsonObjectSchema(final JsonObject value) { super(value); @@ -52,12 +41,7 @@ public DefaultJsonObjectSchema(final JsonObject value) { @Override public Validator validator() { - return keywords() - .map(this::asContraint) - .flatMap(Optional::stream) - .collect( - collectingAndThen(toList(), constraints -> new DefaultValidator(new AllOfCondition<>(constraints))) - ); + return keywords().collect(collectingAndThen(toList(), KeywordBasedValidator::new)); } @Override @@ -66,42 +50,15 @@ public Optional keywordByName(final String name) { } private Stream keywords() { - final Keywords keywords = new Keywords(vocabulary()); + final Keywords keywords = new KeywordExtractor(this).createKeywords(); return asJsonObject().keySet().stream().map(propertyName -> keywords.createKeywordFor(this, propertyName)); } - private Collection vocabulary() { - final KeywordType keywordType = new VocabularyKeywordType(); - return Optional - .ofNullable(asJsonObject().get(keywordType.name())) - .map(keywordValue -> keywordType.createKeyword(this)) - .filter(VocabularyDefinitions.class::isInstance) - .map(VocabularyDefinitions.class::cast) - .stream() - .flatMap(VocabularyDefinitions::definitions) - .toList(); - } - @Override public Optional asSubSchema(final String name) { return Optional .ofNullable(asJsonObject().get(name)) - .filter(value -> - Stream.of(InstanceType.BOOLEAN, InstanceType.OBJECT).anyMatch(type -> type.isInstance(value)) - ) - .map(new DefaultJsonSchemaFactory()::create) + .flatMap(new DefaultJsonSchemaFactory()::tryToCreateSchemaFrom) .map(subSchema -> new DefaultJsonSubSchema(this, subSchema)); } - - private Optional> asContraint(final Keyword keyword) { - final Condition result; - if (keyword.hasCategory(Keyword.KeywordCategory.ASSERTION)) { - result = new AssertionBasedCondition(keyword.asAssertion()); - } else if (keyword.hasCategory(Keyword.KeywordCategory.APPLICATOR)) { - result = new ApplicatorBasedCondtion(keyword.asApplicator()); - } else { - result = null; - } - return Optional.ofNullable(result); - } } diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSchemaFactory.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSchemaFactory.java index 2435ac13..385a8b04 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSchemaFactory.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSchemaFactory.java @@ -26,11 +26,16 @@ import io.github.sebastiantoepfer.jsonschema.JsonSchema; import io.github.sebastiantoepfer.jsonschema.spi.JsonSchemaFactory; import jakarta.json.JsonValue; +import java.util.Optional; public final class DefaultJsonSchemaFactory implements JsonSchemaFactory { @Override public JsonSchema create(final JsonValue schema) { + return tryToCreateSchemaFrom(schema).orElseThrow(IllegalArgumentException::new); + } + + public Optional tryToCreateSchemaFrom(final JsonValue schema) { final JsonSchema result; if (schema == JsonValue.TRUE) { result = new TrueJsonSchema(); @@ -38,9 +43,11 @@ public JsonSchema create(final JsonValue schema) { result = new FalseJsonSchema(); } else if (schema.equals(JsonValue.EMPTY_JSON_OBJECT)) { result = new EmptyJsonSchema(); - } else { + } else if (schema.getValueType() == JsonValue.ValueType.OBJECT) { result = new DefaultJsonObjectSchema(schema.asJsonObject()); + } else { + result = null; } - return result; + return Optional.ofNullable(result); } } diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSubSchema.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSubSchema.java index dd3d8787..c217eb9d 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSubSchema.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/DefaultJsonSubSchema.java @@ -23,15 +23,20 @@ */ package io.github.sebastiantoepfer.jsonschema.core; +import static java.util.stream.Collectors.collectingAndThen; +import static java.util.stream.Collectors.toList; + import io.github.sebastiantoepfer.jsonschema.JsonSchema; import io.github.sebastiantoepfer.jsonschema.JsonSubSchema; import io.github.sebastiantoepfer.jsonschema.Validator; import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; import jakarta.json.JsonObject; +import jakarta.json.JsonValue; import java.util.Objects; import java.util.Optional; +import java.util.stream.Stream; -final class DefaultJsonSubSchema implements JsonSubSchema { +public final class DefaultJsonSubSchema implements JsonSubSchema { private final JsonSchema owner; private final JsonSchema schema; @@ -48,17 +53,42 @@ public JsonSchema owner() { @Override public Validator validator() { - return schema.validator(); + final Validator result; + if (isJsonObject()) { + result = keywords().collect(collectingAndThen(toList(), KeywordBasedValidator::new)); + } else { + result = schema.validator(); + } + return result; } @Override public Optional keywordByName(final String name) { - return schema.keywordByName(name); + return keywords().filter(keyword -> keyword.hasName(name)).findAny(); + } + + private Stream keywords() { + final Stream result; + if (isJsonObject()) { + final Keywords keywords = new KeywordExtractor(schema).createKeywords(); + result = + asJsonObject().keySet().stream().map(propertyName -> keywords.createKeywordFor(this, propertyName)); + } else { + result = Stream.empty(); + } + return result; + } + + private boolean isJsonObject() { + return getValueType() == JsonValue.ValueType.OBJECT; } @Override public Optional asSubSchema(final String name) { - return schema.asSubSchema(name); + return Optional + .ofNullable(asJsonObject().get(name)) + .flatMap(new DefaultJsonSchemaFactory()::tryToCreateSchemaFrom) + .map(subSchema -> new DefaultJsonSubSchema(this, subSchema)); } @Override diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordBasedValidator.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordBasedValidator.java new file mode 100644 index 00000000..974d09b7 --- /dev/null +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordBasedValidator.java @@ -0,0 +1,70 @@ +/* + * The MIT License + * + * Copyright 2023 sebastian. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package io.github.sebastiantoepfer.jsonschema.core; + +import static java.util.stream.Collectors.collectingAndThen; +import static java.util.stream.Collectors.toList; + +import io.github.sebastiantoepfer.jsonschema.Validator; +import io.github.sebastiantoepfer.jsonschema.core.codition.AllOfCondition; +import io.github.sebastiantoepfer.jsonschema.core.codition.ApplicatorBasedCondtion; +import io.github.sebastiantoepfer.jsonschema.core.codition.AssertionBasedCondition; +import io.github.sebastiantoepfer.jsonschema.core.codition.Condition; +import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; +import jakarta.json.JsonValue; +import java.util.Collection; +import java.util.Optional; + +final class KeywordBasedValidator implements Validator { + + private final DefaultValidator validator; + + public KeywordBasedValidator(final Collection keywords) { + this.validator = + keywords + .stream() + .map(KeywordBasedValidator::asContraint) + .flatMap(Optional::stream) + .collect( + collectingAndThen(toList(), constraints -> new DefaultValidator(new AllOfCondition<>(constraints))) + ); + } + + @Override + public boolean isValid(final JsonValue data) { + return validator.isValid(data); + } + + private static Optional> asContraint(final Keyword keyword) { + final Condition result; + if (keyword.hasCategory(Keyword.KeywordCategory.ASSERTION)) { + result = new AssertionBasedCondition(keyword.asAssertion()); + } else if (keyword.hasCategory(Keyword.KeywordCategory.APPLICATOR)) { + result = new ApplicatorBasedCondtion(keyword.asApplicator()); + } else { + result = null; + } + return Optional.ofNullable(result); + } +} diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordExtractor.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordExtractor.java new file mode 100644 index 00000000..670ef102 --- /dev/null +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/KeywordExtractor.java @@ -0,0 +1,57 @@ +/* + * The MIT License + * + * Copyright 2023 sebastian. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package io.github.sebastiantoepfer.jsonschema.core; + +import io.github.sebastiantoepfer.jsonschema.JsonSchema; +import io.github.sebastiantoepfer.jsonschema.core.vocab.core.VocabularyKeywordType; +import jakarta.json.JsonValue; +import java.util.List; +import java.util.Objects; +import java.util.stream.Collectors; + +class KeywordExtractor { + + private final JsonSchema schema; + + public KeywordExtractor(final JsonSchema schema) { + this.schema = Objects.requireNonNull(schema); + } + + public Keywords createKeywords() { + final Keywords result; + final VocabularyKeywordType keywordType = new VocabularyKeywordType(); + if ( + schema.getValueType() == JsonValue.ValueType.OBJECT && schema.asJsonObject().containsKey(keywordType.name()) + ) { + result = + keywordType + .createKeyword(schema) + .definitions() + .collect(Collectors.collectingAndThen(Collectors.toList(), Keywords::new)); + } else { + result = new Keywords(List.of()); + } + return result; + } +} diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java index edd48814..0715ccb1 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java @@ -27,7 +27,9 @@ import io.github.sebastiantoepfer.jsonschema.InstanceType; import io.github.sebastiantoepfer.jsonschema.JsonSchema; -import io.github.sebastiantoepfer.jsonschema.JsonSchemas; +import io.github.sebastiantoepfer.jsonschema.JsonSubSchema; +import io.github.sebastiantoepfer.jsonschema.core.DefaultJsonSchemaFactory; +import io.github.sebastiantoepfer.jsonschema.core.DefaultJsonSubSchema; import io.github.sebastiantoepfer.jsonschema.keyword.Annotation; import io.github.sebastiantoepfer.jsonschema.keyword.Applicator; import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; @@ -39,6 +41,7 @@ import java.util.List; import java.util.Map; import java.util.Objects; +import java.util.Optional; final class PropertiesKeywordType implements KeywordType { @@ -49,14 +52,16 @@ public String name() { @Override public Keyword createKeyword(final JsonSchema schema) { - return new PropertiesKeyword(schema.asJsonObject().getJsonObject(name())); + return new PropertiesKeyword(schema, schema.asJsonObject().getJsonObject(name())); } private class PropertiesKeyword implements Applicator, Annotation { + private final JsonSchema schema; private final JsonObject schemas; - public PropertiesKeyword(final JsonObject schemas) { + public PropertiesKeyword(final JsonSchema schema, final JsonObject schemas) { + this.schema = schema; this.schemas = schemas; } @@ -80,10 +85,18 @@ private boolean propertiesMatches(final JsonObject instance) { } private boolean propertyMatches(final Map.Entry property) { - return ( - !schemas.containsKey(property.getKey()) || - JsonSchemas.load(schemas.get(property.getKey())).validator().isValid(property.getValue()) - ); + return Optional + .ofNullable(schemas.get(property.getKey())) + .flatMap(this::toSubSchema) + .map(JsonSchema::validator) + .map(validator -> validator.isValid(property.getValue())) + .orElse(true); + } + + private Optional toSubSchema(final JsonValue value) { + return new DefaultJsonSchemaFactory() + .tryToCreateSchemaFrom(value) + .map(subSchema -> new DefaultJsonSubSchema(schema, subSchema)); } @Override @@ -92,7 +105,7 @@ public JsonValue valueFor(final JsonValue instance) { .asJsonObject() .keySet() .stream() - .filter(schemas::containsKey) + .filter(schemas.asJsonObject()::containsKey) .map(Json::createValue) .collect(toJsonArray()); } diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/RefKeywordType.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/RefKeywordType.java index a1bfd8d6..cff82fb0 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/RefKeywordType.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/RefKeywordType.java @@ -33,6 +33,7 @@ import jakarta.json.JsonPointer; import jakarta.json.JsonReader; import jakarta.json.JsonString; +import jakarta.json.JsonStructure; import jakarta.json.JsonValue; import java.io.IOException; import java.net.URI; @@ -85,7 +86,7 @@ private JsonSchema retrieveJsonSchema() { if (isRemote()) { json = retrieveValueFromRemoteLocation(); } else { - json = retrievValueFromLocalSchema(); + json = retrieveValueFromLocalSchema(); } return JsonSchemas.load(json); } catch (IOException ex) { @@ -93,15 +94,19 @@ private JsonSchema retrieveJsonSchema() { } } - private JsonValue retrievValueFromLocalSchema() throws IOException { + private JsonValue retrieveValueFromLocalSchema() throws IOException { final JsonPointer pointer = createPointer(); - if (schema.getValueType() == JsonValue.ValueType.OBJECT && pointer.containsValue(schema.asJsonObject())) { - return pointer.getValue(schema.asJsonObject()); + if (pointer.containsValue(searchAnchor())) { + return pointer.getValue(searchAnchor()); } else { throw new IOException("can not find referenced value."); } } + private JsonStructure searchAnchor() { + return schema.rootSchema().asJsonObject(); + } + private JsonPointer createPointer() { final String fragment = uri.getFragment(); final JsonPointer pointer; diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/VocabularyKeywordType.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/VocabularyKeywordType.java index e701e55e..6e648af9 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/VocabularyKeywordType.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/core/VocabularyKeywordType.java @@ -48,9 +48,9 @@ public String name() { } @Override - public Keyword createKeyword(final JsonSchema schema) { + public VocabularyKeyword createKeyword(final JsonSchema schema) { final JsonValue value = schema.asJsonObject().get((name())); - final Keyword result; + final VocabularyKeyword result; if (InstanceType.OBJECT.isInstance(value)) { result = new VocabularyKeyword(value); } else { diff --git a/core/src/test/java/module-info.java b/core/src/test/java/module-info.java index 2950fd11..d1e86850 100644 --- a/core/src/test/java/module-info.java +++ b/core/src/test/java/module-info.java @@ -31,4 +31,5 @@ requires org.junit.jupiter.params; requires org.junit.jupiter.engine; requires org.hamcrest; + requires hamcrest.optional; } From a1a2dd858be869b20fd495e978bb66fc9c41f678 Mon Sep 17 00:00:00 2001 From: Sebastian Toepfer <61313468+sebastian-toepfer@users.noreply.github.com> Date: Sun, 3 Dec 2023 20:16:24 +0100 Subject: [PATCH 2/3] improve creation of properties keyword --- .../applicator/PropertiesKeywordType.java | 35 ++++++------ .../applicator/PropertiesKeywordTypeTest.java | 54 +++++++++++++++++++ 2 files changed, 74 insertions(+), 15 deletions(-) diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java index 0715ccb1..9799450b 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordType.java @@ -24,10 +24,11 @@ package io.github.sebastiantoepfer.jsonschema.core.vocab.applicator; import static jakarta.json.stream.JsonCollectors.toJsonArray; +import static java.util.stream.Collectors.collectingAndThen; +import static java.util.stream.Collectors.toMap; import io.github.sebastiantoepfer.jsonschema.InstanceType; import io.github.sebastiantoepfer.jsonschema.JsonSchema; -import io.github.sebastiantoepfer.jsonschema.JsonSubSchema; import io.github.sebastiantoepfer.jsonschema.core.DefaultJsonSchemaFactory; import io.github.sebastiantoepfer.jsonschema.core.DefaultJsonSubSchema; import io.github.sebastiantoepfer.jsonschema.keyword.Annotation; @@ -52,17 +53,28 @@ public String name() { @Override public Keyword createKeyword(final JsonSchema schema) { - return new PropertiesKeyword(schema, schema.asJsonObject().getJsonObject(name())); + final DefaultJsonSchemaFactory factory = new DefaultJsonSchemaFactory(); + return schema + .asJsonObject() + .getJsonObject(name()) + .entrySet() + .stream() + .map(entry -> + Map.entry( + entry.getKey(), + factory.tryToCreateSchemaFrom(entry.getValue()).orElseThrow(IllegalArgumentException::new) + ) + ) + .map(entry -> Map.entry(entry.getKey(), new DefaultJsonSubSchema(schema, entry.getValue()))) + .collect(collectingAndThen(toMap(Map.Entry::getKey, Map.Entry::getValue), PropertiesKeyword::new)); } private class PropertiesKeyword implements Applicator, Annotation { - private final JsonSchema schema; - private final JsonObject schemas; + private final Map schemas; - public PropertiesKeyword(final JsonSchema schema, final JsonObject schemas) { - this.schema = schema; - this.schemas = schemas; + public PropertiesKeyword(final Map schemas) { + this.schemas = Map.copyOf(schemas); } @Override @@ -87,25 +99,18 @@ private boolean propertiesMatches(final JsonObject instance) { private boolean propertyMatches(final Map.Entry property) { return Optional .ofNullable(schemas.get(property.getKey())) - .flatMap(this::toSubSchema) .map(JsonSchema::validator) .map(validator -> validator.isValid(property.getValue())) .orElse(true); } - private Optional toSubSchema(final JsonValue value) { - return new DefaultJsonSchemaFactory() - .tryToCreateSchemaFrom(value) - .map(subSchema -> new DefaultJsonSubSchema(schema, subSchema)); - } - @Override public JsonValue valueFor(final JsonValue instance) { return instance .asJsonObject() .keySet() .stream() - .filter(schemas.asJsonObject()::containsKey) + .filter(schemas::containsKey) .map(Json::createValue) .collect(toJsonArray()); } diff --git a/core/src/test/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordTypeTest.java b/core/src/test/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordTypeTest.java index efa34fb4..aca5917c 100644 --- a/core/src/test/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordTypeTest.java +++ b/core/src/test/java/io/github/sebastiantoepfer/jsonschema/core/vocab/applicator/PropertiesKeywordTypeTest.java @@ -26,9 +26,12 @@ import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.contains; import static org.hamcrest.Matchers.is; +import static org.junit.jupiter.api.Assertions.assertThrows; +import io.github.sebastiantoepfer.jsonschema.JsonSchema; import io.github.sebastiantoepfer.jsonschema.core.DefaultJsonSchemaFactory; import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; +import io.github.sebastiantoepfer.jsonschema.keyword.KeywordType; import jakarta.json.Json; import jakarta.json.JsonValue; import org.hamcrest.Matchers; @@ -36,6 +39,57 @@ class PropertiesKeywordTypeTest { + @Test + void should_not_be_createable_with_array_in_schemas() { + final JsonSchema schema = new DefaultJsonSchemaFactory() + .create( + Json + .createObjectBuilder() + .add( + "properties", + Json + .createObjectBuilder() + .add("test", JsonValue.TRUE) + .add("invalid", JsonValue.EMPTY_JSON_ARRAY) + ) + .build() + ); + final KeywordType keywordType = new PropertiesKeywordType(); + assertThrows(IllegalArgumentException.class, () -> keywordType.createKeyword(schema)); + } + + @Test + void should_not_be_createable_with_string_in_schemas() { + final JsonSchema schema = new DefaultJsonSchemaFactory() + .create( + Json + .createObjectBuilder() + .add( + "properties", + Json.createObjectBuilder().add("test", JsonValue.TRUE).add("invalid", Json.createValue("value")) + ) + .build() + ); + final KeywordType keywordType = new PropertiesKeywordType(); + assertThrows(IllegalArgumentException.class, () -> keywordType.createKeyword(schema)); + } + + @Test + void should_not_be_createable_with_number_in_schemas() { + final JsonSchema schema = new DefaultJsonSchemaFactory() + .create( + Json + .createObjectBuilder() + .add( + "properties", + Json.createObjectBuilder().add("test", JsonValue.TRUE).add("invalid", Json.createValue(3.14)) + ) + .build() + ); + final KeywordType keywordType = new PropertiesKeywordType(); + assertThrows(IllegalArgumentException.class, () -> keywordType.createKeyword(schema)); + } + @Test void should_be_know_his_name() { final Keyword keyword = new PropertiesKeywordType() From 3b0ce576fbcd057634b9d9b8541a3ada3726f32c Mon Sep 17 00:00:00 2001 From: Sebastian Toepfer <61313468+sebastian-toepfer@users.noreply.github.com> Date: Sun, 3 Dec 2023 20:49:52 +0100 Subject: [PATCH 3/3] refactor TypeKeywordType --- .../core/codition/AnyOfCondition.java | 42 ++++++++++ .../core/codition/OfTypeCondition.java | 41 ++++++++++ .../vocab/validation/TypeKeywordType.java | 79 +++++++------------ 3 files changed, 110 insertions(+), 52 deletions(-) create mode 100644 core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/AnyOfCondition.java create mode 100644 core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/OfTypeCondition.java diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/AnyOfCondition.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/AnyOfCondition.java new file mode 100644 index 00000000..194167f6 --- /dev/null +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/AnyOfCondition.java @@ -0,0 +1,42 @@ +/* + * The MIT License + * + * Copyright 2023 sebastian. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package io.github.sebastiantoepfer.jsonschema.core.codition; + +import jakarta.json.JsonValue; +import java.util.Collection; +import java.util.List; + +public class AnyOfCondition implements Condition { + + private final Collection> conditions; + + public AnyOfCondition(final Collection> conditions) { + this.conditions = List.copyOf(conditions); + } + + @Override + public boolean isFulfilledBy(final JsonValue value) { + return conditions.stream().anyMatch(c -> c.isFulfilledBy(value)); + } +} diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/OfTypeCondition.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/OfTypeCondition.java new file mode 100644 index 00000000..9c9573b7 --- /dev/null +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/codition/OfTypeCondition.java @@ -0,0 +1,41 @@ +/* + * The MIT License + * + * Copyright 2023 sebastian. + * + * Permission is hereby granted, free of charge, to any person obtaining a copy + * of this software and associated documentation files (the "Software"), to deal + * in the Software without restriction, including without limitation the rights + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + * copies of the Software, and to permit persons to whom the Software is + * furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice shall be included in + * all copies or substantial portions of the Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + * THE SOFTWARE. + */ +package io.github.sebastiantoepfer.jsonschema.core.codition; + +import io.github.sebastiantoepfer.jsonschema.InstanceType; +import jakarta.json.JsonValue; + +public final class OfTypeCondition implements Condition { + + private final InstanceType type; + + public OfTypeCondition(final InstanceType type) { + this.type = type; + } + + @Override + public boolean isFulfilledBy(final JsonValue value) { + return type.isInstance(value); + } +} diff --git a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/validation/TypeKeywordType.java b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/validation/TypeKeywordType.java index 9daae6b0..ae5ffa6a 100644 --- a/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/validation/TypeKeywordType.java +++ b/core/src/main/java/io/github/sebastiantoepfer/jsonschema/core/vocab/validation/TypeKeywordType.java @@ -23,13 +23,17 @@ */ package io.github.sebastiantoepfer.jsonschema.core.vocab.validation; +import static java.util.stream.Collectors.collectingAndThen; +import static java.util.stream.Collectors.toList; + import io.github.sebastiantoepfer.jsonschema.InstanceType; import io.github.sebastiantoepfer.jsonschema.JsonSchema; +import io.github.sebastiantoepfer.jsonschema.core.codition.AnyOfCondition; import io.github.sebastiantoepfer.jsonschema.core.codition.Condition; +import io.github.sebastiantoepfer.jsonschema.core.codition.OfTypeCondition; import io.github.sebastiantoepfer.jsonschema.keyword.Assertion; import io.github.sebastiantoepfer.jsonschema.keyword.Keyword; import io.github.sebastiantoepfer.jsonschema.keyword.KeywordType; -import jakarta.json.JsonArray; import jakarta.json.JsonString; import jakarta.json.JsonValue; import java.util.Locale; @@ -47,14 +51,32 @@ public String name() { @Override public Keyword createKeyword(final JsonSchema schema) { - return new TypeKeyword(schema.asJsonObject().get(name())); + final JsonValue typeDefinition = schema.asJsonObject().get(name()); + final Condition typeContraint = + switch (typeDefinition.getValueType()) { + case STRING -> new OfTypeCondition( + InstanceType.valueOf(((JsonString) typeDefinition).getString().toUpperCase(Locale.US)) + ); + case ARRAY -> typeDefinition + .asJsonArray() + .stream() + .map(JsonString.class::cast) + .map(JsonString::getString) + .map(String::toUpperCase) + .map(InstanceType::valueOf) + .map(OfTypeCondition::new) + .collect(collectingAndThen(toList(), AnyOfCondition::new)); + default -> throw new IllegalArgumentException(); + }; + + return new TypeKeyword(typeContraint); } private final class TypeKeyword implements Assertion { - private final JsonValue definition; + private final Condition definition; - public TypeKeyword(final JsonValue definition) { + public TypeKeyword(final Condition definition) { this.definition = Objects.requireNonNull(definition); } @@ -65,54 +87,7 @@ public boolean hasName(final String name) { @Override public boolean isValidFor(final JsonValue instance) { - return new JsonMappedTypeConstaint(definition).isFulfilledBy(instance); - } - - private static final class JsonMappedTypeConstaint implements Condition { - - private final JsonValue definition; - - public JsonMappedTypeConstaint(final JsonValue definition) { - this.definition = Objects.requireNonNull(definition); - } - - @Override - public boolean isFulfilledBy(final JsonValue value) { - final Condition typeContraint = - switch (definition.getValueType()) { - case STRING -> new JsonStringTypeConstraint((JsonString) definition); - default -> new JsonArrayTypeConstraint(definition.asJsonArray()); - }; - return typeContraint.isFulfilledBy(value); - } - } - - private static final class JsonArrayTypeConstraint implements Condition { - - private final JsonArray types; - - public JsonArrayTypeConstraint(final JsonArray types) { - this.types = Objects.requireNonNull(types); - } - - @Override - public boolean isFulfilledBy(final JsonValue value) { - return types.stream().map(JsonMappedTypeConstaint::new).anyMatch(c -> c.isFulfilledBy(value)); - } - } - - private static final class JsonStringTypeConstraint implements Condition { - - private final String type; - - public JsonStringTypeConstraint(final JsonString type) { - this.type = Objects.requireNonNull(type).getString().toUpperCase(Locale.US); - } - - @Override - public boolean isFulfilledBy(final JsonValue value) { - return InstanceType.valueOf(type).isInstance(value); - } + return definition.isFulfilledBy(instance); } } }