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

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

There is a newer version: 1.53.0
Show newest version
/*
 * 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.HashMap;
import java.util.Map;
import java.util.function.Predicate;
import java.util.logging.Logger;
import java.util.regex.Pattern;
import software.amazon.smithy.model.Model;
import software.amazon.smithy.model.shapes.CollectionShape;
import software.amazon.smithy.model.shapes.MapShape;
import software.amazon.smithy.model.shapes.Shape;
import software.amazon.smithy.model.shapes.ShapeId;
import software.amazon.smithy.model.shapes.SimpleShape;
import software.amazon.smithy.model.traits.EnumTrait;
import software.amazon.smithy.utils.FunctionalUtils;
import software.amazon.smithy.utils.StringUtils;

/**
 * Automatically de-conflicts map shapes, list shapes, and set shapes
 * by sorting conflicting shapes by ID and then appending a formatted
 * version of the shape ID namespace to the colliding shape.
 *
 * 

Simple types are never generated at the top level because they * are always inlined into complex shapes; however, string shapes * marked with the enum trait are never allowed to conflict since * they can easily drift away from compatibility over time. * Structures and unions are not allowed to conflict either. */ final class DeconflictingStrategy implements RefStrategy { private static final Logger LOGGER = Logger.getLogger(DeconflictingStrategy.class.getName()); private static final Pattern SPLIT_PATTERN = Pattern.compile("\\."); private final RefStrategy delegate; private final Map pointers = new HashMap<>(); private final Map reversePointers = new HashMap<>(); DeconflictingStrategy(Model model, RefStrategy delegate, Predicate shapePredicate) { this.delegate = delegate; // Pre-compute a map of all converted shape refs. Sort the shapes // to make the result deterministic. model.shapes().filter(shapePredicate.and(FunctionalUtils.not(this::isIgnoredShape))).sorted().forEach(shape -> { String pointer = delegate.toPointer(shape.getId()); if (!reversePointers.containsKey(pointer)) { pointers.put(shape.getId(), pointer); reversePointers.put(pointer, shape.getId()); } else { String deconflictedPointer = deconflict(shape, pointer, reversePointers); LOGGER.info(() -> String.format( "De-conflicted `%s` JSON schema pointer from `%s` to `%s`", shape.getId(), pointer, deconflictedPointer)); pointers.put(shape.getId(), deconflictedPointer); reversePointers.put(deconflictedPointer, shape.getId()); } }); } // Some shapes aren't converted to JSON schema at all because they // don't have a corresponding definition. private boolean isIgnoredShape(Shape shape) { return (shape instanceof SimpleShape && !shape.hasTrait(EnumTrait.class)) || shape.isResourceShape() || shape.isServiceShape() || shape.isOperationShape() || shape.isMemberShape(); } private String deconflict(Shape shape, String pointer, Map reversePointers) { LOGGER.info(() -> String.format( "Attempting to de-conflict `%s` JSON schema pointer `%s` that conflicts with `%s`", shape.getId(), pointer, reversePointers.get(pointer))); if (!isSafeToDeconflict(shape)) { throw new ConflictingShapeNameException(String.format( "Shape %s conflicts with %s using a JSON schema pointer of %s", shape, reversePointers.get(pointer), pointer)); } // Create a de-conflicted JSON schema pointer that just appends // the PascalCase formatted version of the shape's namespace to the // resulting pointer. StringBuilder builder = new StringBuilder(pointer); for (String part : SPLIT_PATTERN.split(shape.getId().getNamespace())) { builder.append(StringUtils.capitalize(part)); } String updatedPointer = builder.toString(); if (reversePointers.containsKey(updatedPointer)) { // Note: I don't know if this can ever actually happen... but just in case. throw new ConflictingShapeNameException(String.format( "Unable to de-conflict shape %s because the de-conflicted name resolves " + "to another generated name: %s", shape, updatedPointer)); } return updatedPointer; } // We only want to de-conflict shapes that are generally not code-generated // because the de-conflicted names can potentially change over time as shapes // are added and removed. Things like structures, unions, and enums should // never be de-conflicted from this class. private boolean isSafeToDeconflict(Shape shape) { return shape instanceof CollectionShape || shape instanceof MapShape; } @Override public String toPointer(ShapeId id) { return pointers.computeIfAbsent(id, delegate::toPointer); } @Override public boolean isInlined(Shape shape) { return delegate.isInlined(shape); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy