All Downloads are FREE. Search and download functionalities are using the official Maven repository.

software.amazon.smithy.jsonschema.JsonSchemaConfig Maven / Gradle / Ivy

There is a newer version: 1.54.0
Show newest version
/*
 * Copyright 2020 Amazon.com, Inc. or its affiliates. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License").
 * You may not use this file except in compliance with the License.
 * A copy of the License is located at
 *
 *  http://aws.amazon.com/apache2.0
 *
 * or in the "license" file accompanying this file. This file is distributed
 * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
 * express or implied. See the License for the specific language governing
 * permissions and limitations under the License.
 */

package software.amazon.smithy.jsonschema;

import java.util.HashSet;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import software.amazon.smithy.model.node.Node;
import software.amazon.smithy.model.node.NodeMapper;
import software.amazon.smithy.model.node.ObjectNode;
import software.amazon.smithy.model.shapes.Shape;
import software.amazon.smithy.model.shapes.ShapeId;
import software.amazon.smithy.model.traits.TimestampFormatTrait;

/**
 * JSON Schema configuration options.
 */
public class JsonSchemaConfig {

    /**
     * Configures how Smithy union shapes are converted to JSON Schema.
     */
    public enum UnionStrategy {
        /**
         * Converts to a schema that uses "oneOf".
         *
         * 

This is the default setting used if not configured. */ ONE_OF("oneOf"), /** * Converts to an empty object "{}". */ OBJECT("object"), /** * Converts to an object with properties just like a structure. */ STRUCTURE("structure"); private String stringValue; UnionStrategy(String stringValue) { this.stringValue = stringValue; } @Override public String toString() { return stringValue; } } /** * Configures how Smithy map shapes are converted to JSON Schema. */ public enum MapStrategy { /** * Converts to a schema that uses a combination of "propertyNames" * and "additionalProperties". * *

This is the default setting used if not configured. */ PROPERTY_NAMES("propertyNames"), /** * Converts to a schema that uses "patternProperties". If a map's key * member or its target does not have a {@code pattern} trait, a default * indicating one or more of any character (".+") is applied. */ PATTERN_PROPERTIES("patternProperties"); private String stringValue; MapStrategy(String stringValue) { this.stringValue = stringValue; } @Override public String toString() { return stringValue; } } private boolean alphanumericOnlyRefs; private boolean useJsonName; private TimestampFormatTrait.Format defaultTimestampFormat = TimestampFormatTrait.Format.DATE_TIME; private UnionStrategy unionStrategy = UnionStrategy.ONE_OF; private MapStrategy mapStrategy = MapStrategy.PROPERTY_NAMES; private String definitionPointer = "#/definitions"; private ObjectNode schemaDocumentExtensions = Node.objectNode(); private ObjectNode extensions = Node.objectNode(); private Set disableFeatures = new HashSet<>(); private final ConcurrentHashMap extensionCache = new ConcurrentHashMap<>(); private final NodeMapper nodeMapper = new NodeMapper(); private ShapeId service; private boolean supportNonNumericFloats = false; private boolean enableOutOfServiceReferences = false; public JsonSchemaConfig() { nodeMapper.setWhenMissingSetter(NodeMapper.WhenMissing.INGORE); } public boolean getAlphanumericOnlyRefs() { return alphanumericOnlyRefs; } /** * Creates shape name pointers that strip out non-alphanumeric characters. * *

This is necessary for compatibility with some vendors like * Amazon API Gateway that only allow alphanumeric shape names. * * @param alphanumericOnlyRefs Set to true to strip non-alphanumeric characters. */ public void setAlphanumericOnlyRefs(boolean alphanumericOnlyRefs) { this.alphanumericOnlyRefs = alphanumericOnlyRefs; } public boolean getUseJsonName() { return useJsonName; } /** * Uses the value of the jsonName trait when creating JSON schema * properties for structure and union shapes. * *

This property has no effect if a {@link PropertyNamingStrategy} is * manually configured on a {@link JsonSchemaConverter}. * * @param useJsonName Set to true to use jsonName traits when creating refs.. */ public void setUseJsonName(boolean useJsonName) { this.useJsonName = useJsonName; } public TimestampFormatTrait.Format getDefaultTimestampFormat() { return defaultTimestampFormat; } /** * Sets the assumed timestampFormat trait for timestamps with no * timestampFormat trait. The provided value is expected to be a string. * *

Defaults to "date-time" if not set. Can be set to "date-time", * "epoch-seconds", or "http-date". * * @param defaultTimestampFormat The default timestamp format to use when none is set. */ public void setDefaultTimestampFormat(TimestampFormatTrait.Format defaultTimestampFormat) { this.defaultTimestampFormat = defaultTimestampFormat; } public UnionStrategy getUnionStrategy() { return unionStrategy; } /** * Configures how Smithy union shapes are converted to JSON Schema. * * @param unionStrategy The union strategy to use. */ public void setUnionStrategy(UnionStrategy unionStrategy) { this.unionStrategy = unionStrategy; } public MapStrategy getMapStrategy() { return mapStrategy; } /** * Configures how Smithy map shapes are converted to JSON Schema. * * @param mapStrategy The map strategy to use. */ public void setMapStrategy(MapStrategy mapStrategy) { this.mapStrategy = mapStrategy; } public String getDefinitionPointer() { return definitionPointer; } /** * Configures location of where definitions are written using JSON Pointer. * *

The provided String value MUST start with "#/" and can use nested "/" * characters to place schemas in nested object properties. The provided * JSON Pointer does not support escaping. * *

Defaults to "#/definitions" if no value is specified. OpenAPI * artifacts will want to use "#/components/schemas". * * @param definitionPointer The root definition pointer to use. */ public void setDefinitionPointer(String definitionPointer) { this.definitionPointer = Objects.requireNonNull(definitionPointer); } public ObjectNode getSchemaDocumentExtensions() { return schemaDocumentExtensions; } /** * Adds custom key-value pairs to the JSON Schema document generated from * a Smithy model. * * @param schemaDocumentExtensions Custom extensions to merge into the created schema. */ public void setSchemaDocumentExtensions(ObjectNode schemaDocumentExtensions) { this.schemaDocumentExtensions = Objects.requireNonNull(schemaDocumentExtensions); } public Set getDisableFeatures() { return disableFeatures; } /** * Disables OpenAPI features by their property name (e.g., "allOf"). * * @param disableFeatures Feature names to disable. */ public void setDisableFeatures(Set disableFeatures) { this.disableFeatures = disableFeatures; } public ObjectNode getExtensions() { return extensions; } /** * Attempts to deserialize the {@code extensions} into the targeted * type using a {@link NodeMapper}. * *

Extraneous properties are ignored and not warned on * because many different plugins could be used with different * configuration POJOs. * *

The result of calling this method is cached for each type, * and the cache is cleared when any mutation is made to * extensions. * * @param as Type to deserialize extensions into. * @param Type to deserialize extensions into. * @return Returns the deserialized type. */ @SuppressWarnings("unchecked") public T getExtensions(Class as) { return (T) extensionCache.computeIfAbsent(as, t -> nodeMapper.deserialize(extensions, t)); } /** * Sets an arbitrary map of "extensions" used by plugins that need * configuration. * * @param extensions Extensions to set. */ public void setExtensions(ObjectNode extensions) { this.extensions = Objects.requireNonNull(extensions); extensionCache.clear(); } /** * Add an extension to the "extensions" object node using a POJO. * * @param extensionContainer POJO to serialize and merge into extensions. */ public void putExtensions(Object extensionContainer) { ObjectNode serialized = nodeMapper.serialize(extensionContainer).expectObjectNode(); setExtensions(extensions.merge(serialized)); } /** * Add an extension to the "extensions" object node. * * @param key Property name to set. * @param value Value to assigned. */ public void putExtension(String key, Node value) { setExtensions(extensions.withMember(key, value)); } /** * Add an extension to the "extensions" object node. * * @param key Property name to set. * @param value Value to assigned. */ public void putExtension(String key, boolean value) { putExtension(key, Node.from(value)); } /** * Add an extension to the "extensions" object node. * * @param key Property name to set. * @param value Value to assigned. */ public void putExtension(String key, String value) { putExtension(key, Node.from(value)); } /** * Gets the service shape ID that is used to contextualize the created * schemas by using things like the "rename" property of the service. * * @return the nullable Smithy service shape ID. */ public ShapeId getService() { return service; } /** * Sets the service shape ID that is used to contextualize the created * schemas by using things like the "rename" property of the service. * * @param service the Smithy service shape ID. */ public void setService(ShapeId service) { this.service = service; } /** * Detects the TimestampFormat of the given shape, falling back to the * configured default format specified in {@link #setDefaultTimestampFormat(TimestampFormatTrait.Format)}. * * @param shape Shape to extract the timestamp format from. * @return Returns the optionally detected format. */ public Optional detectJsonTimestampFormat(Shape shape) { if (shape.isTimestampShape() || shape.hasTrait(TimestampFormatTrait.class)) { return Optional.of(shape.getTrait(TimestampFormatTrait.class) .map(TimestampFormatTrait::getValue) .orElseGet(() -> getDefaultTimestampFormat().toString())); } return Optional.empty(); } public boolean getSupportNonNumericFloats() { return supportNonNumericFloats; } /** * Set to true to add support for NaN, Infinity, and -Infinity in float * and double shapes. These values will be serialized as strings. The * OpenAPI document will be updated to refer to them as a "oneOf" of number * and string. * *

By default, non-numeric values are not supported. * * @param supportNonNumericFloats True if non-numeric float values should be supported. */ public void setSupportNonNumericFloats(boolean supportNonNumericFloats) { this.supportNonNumericFloats = supportNonNumericFloats; } public boolean isEnableOutOfServiceReferences() { return enableOutOfServiceReferences; } /** * Set to true to enable references to shapes outside the service closure. * * Setting this to true means that all the shapes in the model must not conflict, whereas * leaving it at the default, false, means that only the shapes connected to the configured * service via {@link #setService(ShapeId)} must not conflict. * * @param enableOutOfServiceReferences true if out-of-service references should be allowed. default: false */ public void setEnableOutOfServiceReferences(boolean enableOutOfServiceReferences) { this.enableOutOfServiceReferences = enableOutOfServiceReferences; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy