software.amazon.smithy.jsonschema.JsonSchemaConverter Maven / Gradle / Ivy
Show all versions of smithy-jsonschema Show documentation
/*
* Copyright 2019 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.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.function.Predicate;
import java.util.logging.Logger;
import java.util.stream.Collectors;
import software.amazon.smithy.model.Model;
import software.amazon.smithy.model.loader.Prelude;
import software.amazon.smithy.model.neighbor.Walker;
import software.amazon.smithy.model.node.Node;
import software.amazon.smithy.model.node.ObjectNode;
import software.amazon.smithy.model.shapes.MemberShape;
import software.amazon.smithy.model.shapes.ServiceShape;
import software.amazon.smithy.model.shapes.Shape;
import software.amazon.smithy.model.shapes.ShapeId;
import software.amazon.smithy.model.shapes.ToShapeId;
import software.amazon.smithy.model.shapes.UnionShape;
import software.amazon.smithy.model.traits.UnitTypeTrait;
import software.amazon.smithy.model.transform.ModelTransformer;
import software.amazon.smithy.utils.FunctionalUtils;
import software.amazon.smithy.utils.Pair;
import software.amazon.smithy.utils.SmithyBuilder;
import software.amazon.smithy.utils.ToSmithyBuilder;
/**
* Converts a Smithy model index to a JSON schema document.
*/
public final class JsonSchemaConverter implements ToSmithyBuilder {
private static final Logger LOGGER = Logger.getLogger(JsonSchemaConverter.class.getName());
private static final PropertyNamingStrategy DEFAULT_PROPERTY_STRATEGY = PropertyNamingStrategy
.createDefaultStrategy();
/** All converters use the built-in mappers. */
private final List mappers = new ArrayList<>();
private final Model model;
private final PropertyNamingStrategy propertyNamingStrategy;
private JsonSchemaConfig config;
private final Predicate shapePredicate;
private final RefStrategy refStrategy;
private final List realizedMappers;
private final JsonSchemaShapeVisitor visitor;
private final Shape rootShape;
private final String rootDefinitionPointer;
private final int rootDefinitionSegments;
/** A workaround for including definitions for Unit; it's only included in the schema if a union targets it. */
private final boolean unitTargetedByUnion;
private JsonSchemaConverter(Builder builder) {
mappers.addAll(builder.mappers);
config = SmithyBuilder.requiredState("config", builder.config);
propertyNamingStrategy = SmithyBuilder.requiredState("propertyNamingStrategy", builder.propertyNamingStrategy);
model = SmithyBuilder.requiredState("model", builder.model);
shapePredicate = builder.shapePredicate;
LOGGER.fine("Building filtered JSON schema shape index");
if (builder.rootShape == null) {
rootShape = null;
} else {
rootShape = builder.model.getShape(builder.rootShape)
.orElseThrow(() -> new SmithyJsonSchemaException(
"Invalid root shape (shape not found): " + builder.rootShape));
}
LOGGER.fine("Creating JSON ref strategy");
Model refModel = config.isEnableOutOfServiceReferences()
? this.model : scopeModelToService(model, config.getService());
unitTargetedByUnion = refModel.shapes(UnionShape.class)
.anyMatch(u -> u.members().stream().anyMatch(m -> m.getTarget().equals(UnitTypeTrait.UNIT)));
refStrategy = RefStrategy.createDefaultStrategy(refModel, config, propertyNamingStrategy,
new FilterPreludeUnit(unitTargetedByUnion));
// Combine custom mappers with the discovered mappers and sort them.
realizedMappers = new ArrayList<>(mappers);
realizedMappers.add(new DisableMapper());
realizedMappers.add(new TimestampMapper());
realizedMappers.sort(Comparator.comparing(JsonSchemaMapper::getOrder));
LOGGER.fine(() -> "Adding the following JSON schema mappers: " + realizedMappers.stream()
.map(Object::getClass)
.map(Class::getCanonicalName)
.collect(Collectors.joining(", ")));
visitor = new JsonSchemaShapeVisitor(this.model, this, realizedMappers);
// Compute the number of segments in the root definition section.
rootDefinitionPointer = config.getDefinitionPointer();
rootDefinitionSegments = countSegments(rootDefinitionPointer);
LOGGER.fine(() -> "Using the following root JSON schema pointer: " + rootDefinitionPointer
+ " (" + rootDefinitionSegments + " segments)");
}
private static Model createUpdatedModel(
Model model,
Shape rootShape,
Predicate predicate
) {
ModelTransformer transformer = ModelTransformer.create();
if (rootShape != null) {
LOGGER.fine(() -> "Filtering out shapes that are not connected to " + rootShape);
Set connected = new Walker(model).walkShapes(rootShape);
LOGGER.fine(() -> "Only generating the following JSON schema shapes: " + connected.stream()
.map(Shape::getId)
.map(ShapeId::toString)
.collect(Collectors.joining(", ")));
model = transformer.filterShapes(model, connected::contains);
}
model = transformer.filterShapes(model, predicate);
// Traits and their shapes are not generated into the OpenAPI schema.
model = transformer.scrubTraitDefinitions(model);
return model;
}
private static Model scopeModelToService(Model model, ShapeId serviceId) {
if (serviceId == null) {
return model;
}
Set connected = new Walker(model).walkShapes(model.expectShape(serviceId));
return ModelTransformer.create().filterShapes(model, connected::contains);
}
private static int countSegments(String pointer) {
int totalSegments = 0;
for (int i = 0; i < pointer.length(); i++) {
if (pointer.charAt(i) == '/') {
totalSegments++;
}
}
return totalSegments;
}
public static Builder builder() {
return new Builder();
}
/**
* Gets the configuration object.
*
* @return Returns the config object.
*/
public JsonSchemaConfig getConfig() {
return config;
}
/**
* Set the JSON Schema configuration settings.
*
* @param config Config object to set.
*/
public void setConfig(JsonSchemaConfig config) {
this.config = config;
}
/**
* Gets the property naming strategy of the converter.
*
* @param member Member to convert to a property name.
* @return Returns the PropertyNamingStrategy.
*/
public String toPropertyName(MemberShape member) {
Shape containingShape = model.getShape(member.getContainer())
.orElseThrow(() -> new SmithyJsonSchemaException("Invalid member: " + member));
return propertyNamingStrategy.toPropertyName(containingShape, member, config);
}
/**
* Given a shape ID, returns the value used in a $ref to refer to it.
*
* The return value is expected to be a JSON pointer.
*
* @param id Shape ID to convert to a $ref string.
* @return Returns the $ref string (e.g., "#/responses/MyShape").
*/
public String toPointer(ToShapeId id) {
return refStrategy.toPointer(id.toShapeId());
}
/**
* Checks if the given JSON pointer points to a top-level definition.
*
*
Note that this expects the pointer to exactly start with the same
* string that is configured as {@link JsonSchemaConfig#getDefinitionPointer()},
* or the default value of "#/definitions". If the number of segments
* in the provided pointer is also equal to the number of segments
* in the default pointer + 1, then it is considered a top-level pointer.
*
* @param pointer Pointer to check.
* @return Returns true if this is a top-level definition pointer.
*/
public boolean isTopLevelPointer(String pointer) {
return pointer.startsWith(rootDefinitionPointer)
&& countSegments(pointer) == rootDefinitionSegments + 1;
}
/**
* Checks if the given shape is inlined into its container when targeted
* by a member.
*
* @param shape Shape to check.
* @return Returns true if this shape is inlined into its containing shape.
*/
public boolean isInlined(Shape shape) {
return refStrategy.isInlined(shape);
}
/**
* Perform the conversion of the entire shape index.
*
* @return Returns the created SchemaDocument.
*/
public SchemaDocument convert() {
LOGGER.fine("Converting to JSON schema");
SchemaDocument.Builder builder = SchemaDocument.builder();
if (rootShape != null && !(rootShape instanceof ServiceShape)) {
LOGGER.fine(() -> "Setting root schema to " + rootShape);
builder.rootSchema(rootShape.accept(visitor));
}
addExtensions(builder);
// Create a model that strips out traits and disconnected shapes.
Model updatedModel = createUpdatedModel(model, rootShape, shapePredicate);
model.shapes()
// Only generate shapes that passed through each predicate.
.filter(shape -> updatedModel.getShape(shape.getId()).isPresent())
// Don't generate members.
.filter(FunctionalUtils.not(Shape::isMemberShape))
// Don't write the root shape to the definitions.
.filter((shape -> rootShape == null || !shape.getId().equals(rootShape.getId())))
// Don't convert unsupported shapes.
.filter(FunctionalUtils.not(this::isUnsupportedShapeType))
// Ignore prelude shapes.
.filter(s -> s.getId().equals(UnitTypeTrait.UNIT) && unitTargetedByUnion || !Prelude.isPreludeShape(s))
// Do not generate inlined shapes in the definitions map.
.filter(FunctionalUtils.not(refStrategy::isInlined))
// Create a pair of pointer and shape.
.map(shape -> Pair.of(toPointer(shape), shape))
// Only add definitions if they are at the top-level and not inlined.
.filter(pair -> isTopLevelPointer(pair.getLeft()))
// Create the pointer to the shape and schema object.
.map(pair -> {
LOGGER.fine(() -> "Converting " + pair.getRight() + " to JSON schema at " + pair.getLeft());
return Pair.of(pair.getLeft(), pair.getRight().accept(visitor));
})
.forEach(pair -> builder.putDefinition(pair.getLeft(), pair.getRight()));
LOGGER.fine(() -> "Completed JSON schema document conversion (root shape: " + rootShape + ")");
return builder.build();
}
/**
* Perform the conversion of a single shape.
*
*
The root shape of the created document is set to the given shape.
* No schema extensions are added to the converted schema. This
* conversion also doesn't take the shape predicate or private
* controls into account.
*
* @param shape Shape to convert.
* @return Returns the created SchemaDocument.
*/
public SchemaDocument convertShape(Shape shape) {
SchemaDocument.Builder builder = SchemaDocument.builder();
builder.rootSchema(shape.accept(visitor));
return builder.build();
}
// We can't generate service, resource, or operation schemas.
private boolean isUnsupportedShapeType(Shape shape) {
return shape.isServiceShape() || shape.isResourceShape() || shape.isOperationShape();
}
private void addExtensions(SchemaDocument.Builder builder) {
ObjectNode extensions = config.getSchemaDocumentExtensions();
if (!extensions.isEmpty()) {
LOGGER.fine(() -> "Adding JSON schema extensions: " + Node.prettyPrintJson(extensions));
builder.extensions(extensions);
}
}
@Override
public Builder toBuilder() {
return builder()
.model(model)
.propertyNamingStrategy(propertyNamingStrategy)
.config(config)
.rootShape(rootShape == null ? null : rootShape.getId())
.shapePredicate(shapePredicate)
.mappers(mappers);
}
public static final class Builder implements SmithyBuilder {
private Model model;
private ShapeId rootShape;
private PropertyNamingStrategy propertyNamingStrategy = DEFAULT_PROPERTY_STRATEGY;
private JsonSchemaConfig config = new JsonSchemaConfig();
private Predicate shapePredicate = shape -> true;
private final List mappers = new ArrayList<>();
private Builder() {}
@Override
public JsonSchemaConverter build() {
return new JsonSchemaConverter(this);
}
/**
* Sets the shape index to convert.
*
* @param model Shape index to convert.
* @return Returns the builder.
*/
public Builder model(Model model) {
this.model = model;
return this;
}
/**
* Only generates shapes connected to the given shape and set the
* given shape as the root of the created schema document.
*
* @param rootShape ID of the shape that is used to limit
* the closure of the generated document.
* @return Returns the builder.
*/
public Builder rootShape(ToShapeId rootShape) {
this.rootShape = rootShape == null ? null : rootShape.toShapeId();
return this;
}
/**
* Sets a predicate used to filter Smithy shapes from being converted
* to JSON Schema.
*
* @param shapePredicate Predicate that returns true if a shape is to be converted.
* @return Returns the converter.
*/
public Builder shapePredicate(Predicate shapePredicate) {
this.shapePredicate = Objects.requireNonNull(shapePredicate);
return this;
}
/**
* Sets the configuration object.
*
* @param config Config to use.
* @return Returns the converter.
*/
public Builder config(JsonSchemaConfig config) {
this.config = Objects.requireNonNull(config);
return this;
}
/**
* Sets a custom property naming strategy.
*
* This method overrides an configuration values specified by
* the configuration object.
*
* @param propertyNamingStrategy Property name strategy to use.
* @return Returns the converter.
*/
public Builder propertyNamingStrategy(PropertyNamingStrategy propertyNamingStrategy) {
this.propertyNamingStrategy = Objects.requireNonNull(propertyNamingStrategy);
return this;
}
/**
* Adds a mapper used to update schema builders.
*
* @param jsonSchemaMapper Mapper to add.
* @return Returns the converter.
*/
public Builder addMapper(JsonSchemaMapper jsonSchemaMapper) {
mappers.add(Objects.requireNonNull(jsonSchemaMapper));
return this;
}
/**
* Replaces the mappers of the builder with the given mappers.
*
* @param jsonSchemaMappers Mappers to replace with.
* @return Returns the converter.
*/
public Builder mappers(List jsonSchemaMappers) {
mappers.clear();
mappers.addAll(jsonSchemaMappers);
return this;
}
}
static final class FilterPreludeUnit implements Predicate {
private final boolean includePreludeUnit;
FilterPreludeUnit(boolean includePreludeUnit) {
this.includePreludeUnit = includePreludeUnit;
}
@Override
public boolean test(Shape shape) {
return includePreludeUnit || !shape.getId().equals(UnitTypeTrait.UNIT);
}
}
}