com.kjetland.jackson.jsonSchema.JsonSchemaGeneratorVisitor Maven / Gradle / Ivy
package com.kjetland.jackson.jsonSchema;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonPropertyDescription;
import com.fasterxml.jackson.annotation.JsonSubTypes;
import com.fasterxml.jackson.annotation.JsonSubTypes.Type;
import com.fasterxml.jackson.annotation.JsonTypeInfo;
import com.fasterxml.jackson.annotation.JsonTypeInfo.As;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.databind.BeanProperty;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.introspect.AnnotatedClass;
import com.fasterxml.jackson.databind.introspect.AnnotatedClassResolver;
import com.fasterxml.jackson.databind.jsonFormatVisitors.*;
import com.fasterxml.jackson.databind.jsontype.NamedType;
import com.fasterxml.jackson.databind.jsontype.SubtypeResolver;
import com.fasterxml.jackson.databind.jsontype.TypeIdResolver;
import com.fasterxml.jackson.databind.jsontype.TypeSerializer;
import com.fasterxml.jackson.databind.jsontype.impl.MinimalClassNameIdResolver;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.JsonNodeFactory;
import com.fasterxml.jackson.databind.node.ObjectNode;
import com.kjetland.jackson.jsonSchema.DefinitionsHandler.DefinitionInfo;
import com.kjetland.jackson.jsonSchema.annotations.*;
import java.lang.annotation.Annotation;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import javax.validation.constraints.*;
import lombok.AccessLevel;
import lombok.Data;
import lombok.RequiredArgsConstructor;
import lombok.experimental.Accessors;
import lombok.experimental.FieldDefaults;
import lombok.extern.slf4j.Slf4j;
@Slf4j
@RequiredArgsConstructor
class JsonSchemaGeneratorVisitor extends AbstractJsonFormatVisitorWithSerializerProvider implements JsonFormatVisitorWrapper {
final JsonSchemaGenerator ctx;
final int level; // = 0
final ObjectNode node; // = JsonNodeFactory.instance.objectNode()
final DefinitionsHandler definitionsHandler;
final BeanProperty currentProperty; // This property may represent the BeanProperty when we're directly processing beneath the property
/** Tries to retrieve an annotation and validates that it is applicable. */
private T tryGetAnnotation(Class annotationClass) {
return ctx.selectAnnotation(currentProperty, annotationClass);
}
JsonSchemaGeneratorVisitor createChildVisitor(ObjectNode childNode, BeanProperty currentProperty) {
return new JsonSchemaGeneratorVisitor(ctx, level + 1, childNode, definitionsHandler, currentProperty);
}
String extractDefaultValue() {
// Scala way (ugly and confusing)
// return selectAnnotation(p, JsonProperty.class)
// .map(JsonProperty::defaultValue)
// .or (() ->
// selectAnnotation(p, JsonSchemaDefault.class)
// .map (JsonSchemaDefault::value));
// Plain java
JsonProperty jp = tryGetAnnotation(JsonProperty.class);
if (jp != null
&& jp.defaultValue() != null
&& !jp.defaultValue().isEmpty()) // unfortunately, defaultValue is empty by default
return jp.defaultValue();
JsonSchemaDefault jsd = tryGetAnnotation(JsonSchemaDefault.class);
if (jsd != null)
return jsd.value();
return null;
// Hypothetical null operators
// return selectAnnotation(p, JsonProperty.class)?.defaultValue()
// ?? selectAnnotation(p, JsonSchemaDefault.class)?.value();
}
// @RequiredArgsConstructor
class MyJsonValueFormatVisitor
extends AbstractJsonFormatVisitorWithSerializerProvider
implements
JsonStringFormatVisitor,
JsonNumberFormatVisitor,
JsonIntegerFormatVisitor,
JsonBooleanFormatVisitor {
@Override
public void format(JsonValueFormat format) {
node.put("format", format.toString());
}
@Override
public void enumTypes(Set enums) {
log.trace("JsonStringFormatVisitor-enum.enumTypes: {}", enums);
ArrayNode enumValuesNode = JsonNodeFactory.instance.arrayNode();
for (String e : enums)
enumValuesNode.add(e);
node.set("enum", enumValuesNode);
}
@Override public void numberType(JsonParser.NumberType type) {
log.trace("JsonNumberFormatVisitor.numberType: {}", type);
}
}
@Override
public JsonStringFormatVisitor expectStringFormat(JavaType type) {
log.trace("expectStringFormat {}", type);
node.put("type", "string");
NotBlank notBlankAnnotation = tryGetAnnotation(NotBlank.class);
if (notBlankAnnotation != null)
node.put("pattern", "^.*\\S+.*$");
Pattern patternAnnotation = tryGetAnnotation(Pattern.class);
if (patternAnnotation != null)
node.put("pattern", patternAnnotation.regexp());
Pattern.List patternListAnnotation = tryGetAnnotation(Pattern.List.class);
if (patternListAnnotation != null) {
String pattern = "^";
for (Pattern p : patternListAnnotation.value())
pattern += "(?=" + p.regexp() + ")";
pattern += ".*$";
node.put("pattern", pattern);
}
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", defaultValue);
// Look for @JsonSchemaExamples
JsonSchemaExamples examplesAnnotation = tryGetAnnotation(JsonSchemaExamples.class);
if (examplesAnnotation != null) {
ArrayNode examples = JsonNodeFactory.instance.arrayNode();
for (String example : examplesAnnotation.value())
examples.add(example);
node.set("examples", examples);
}
// Look for @Email
Email emailAnnotation = tryGetAnnotation(Email.class);
if (emailAnnotation != null)
node.put("format", "email");
// Look for a @Size annotation, which should have a set of min/max properties.
Size minAndMaxLengthAnnotation = tryGetAnnotation(Size.class);
NotNull notNullAnnotation = tryGetAnnotation(NotNull.class);
NotEmpty notEmptyAnnotation = tryGetAnnotation(NotEmpty.class);
if (minAndMaxLengthAnnotation != null) {
if (minAndMaxLengthAnnotation.min() != 0)
node.put("minLength", minAndMaxLengthAnnotation.min());
if (minAndMaxLengthAnnotation.max() != Integer.MAX_VALUE)
node.put("maxLength", minAndMaxLengthAnnotation.max());
}
else if (ctx.config.useMinLengthForNotNull && notNullAnnotation != null)
node.put("minLength", 1);
else if (notEmptyAnnotation != null || notBlankAnnotation != null)
node.put("minLength", 1);
return new MyJsonValueFormatVisitor();
}
@Override
public JsonArrayFormatVisitor expectArrayFormat(JavaType type) {
log.trace("expectArrayFormat {}", type);
node.put("type", "array");
if (ctx.config.uniqueItemClasses.stream().anyMatch(c -> type.getRawClass().isAssignableFrom(c))) {
// Adding '"uniqueItems": true' to be used with https://github.com/jdorn/json-editor
node.put("uniqueItems", true);
node.put("format", "checkbox");
} else {
// Try to set default format
if (ctx.config.defaultArrayFormat != null)
node.put("format", ctx.config.defaultArrayFormat);
}
Size sizeAnnotation = tryGetAnnotation(Size.class);
if (sizeAnnotation != null) {
node.put("minItems", sizeAnnotation.min());
node.put("maxItems", sizeAnnotation.max());
}
NotEmpty notEmptyAnnotation = tryGetAnnotation(NotEmpty.class);
if (notEmptyAnnotation != null)
node.put("minItems", 1);
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", defaultValue);
ObjectNode itemsNode = JsonNodeFactory.instance.objectNode();
node.set("items", itemsNode);
// We get improved result while processing scala-collections by getting elementType this way
// instead of using the one which we receive in JsonArrayFormatVisitor.itemsFormat
// This approach also works for Java
JavaType preferredElementType = type.getContentType();
class MyVisitor extends AbstractJsonFormatVisitorWithSerializerProvider implements JsonArrayFormatVisitor {
@Override
public void itemsFormat(JsonFormatVisitable handler, JavaType elementType) throws JsonMappingException {
log.trace("expectArrayFormat - handler: $handler - elementType: {} - preferredElementType: {}", elementType, preferredElementType);
JavaType type = ctx.tryToReMapType(preferredElementType);
JsonSchemaGeneratorVisitor visitor = createChildVisitor(itemsNode, null);
ctx.objectMapper.acceptJsonFormatVisitor(type, visitor);
}
@Override
public void itemsFormat(JsonFormatTypes format) {
log.trace("itemsFormat - format: {}", format);
itemsNode.put("type", format.value());
}
}
return new MyVisitor();
}
@Override
public JsonNumberFormatVisitor expectNumberFormat(JavaType type) {
log.trace("expectNumberFormat");
node.put("type", "number");
// Look for @Min, @Max, @DecimalMin, @DecimalMax => minimum, maximum
Min minAnnotation = tryGetAnnotation(Min.class);
if (minAnnotation != null)
node.put("minimum", minAnnotation.value());
Max maxAnnotation = tryGetAnnotation(Max.class);
if (maxAnnotation != null)
node.put("maximum", maxAnnotation.value());
DecimalMin decimalMinAnnotation = tryGetAnnotation(DecimalMin.class);
if (decimalMinAnnotation != null)
node.put("minimum", Double.valueOf(decimalMinAnnotation.value()));
DecimalMax decimalMaxAnnotation = tryGetAnnotation(DecimalMax.class);
if (decimalMaxAnnotation != null)
node.put("maximum", Double.valueOf(decimalMaxAnnotation.value()));
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", Double.valueOf(defaultValue));
if (currentProperty != null) {
JsonSchemaExamples examplesAnnotation = currentProperty.getAnnotation(JsonSchemaExamples.class);
if (examplesAnnotation != null) {
ArrayNode examples = JsonNodeFactory.instance.arrayNode();
for (String example : examplesAnnotation.value()) {
examples.add(example);
}
node.set("examples", examples);
}
}
return new MyJsonValueFormatVisitor();
}
@Override
public JsonAnyFormatVisitor expectAnyFormat(JavaType type) {
log.warn("Unable to process {} - "
+ "it is probably using custom serializer which does not override acceptJsonFormatVisitor", type);
return new JsonAnyFormatVisitor() {};
}
@Override
public JsonIntegerFormatVisitor expectIntegerFormat(JavaType type) {
log.trace("expectIntegerFormat");
node.put("type", "integer");
Min minAnnotation = tryGetAnnotation(Min.class);
if (minAnnotation != null)
node.put("minimum", minAnnotation.value());
Max maxAnnotation = tryGetAnnotation(Max.class);
if (maxAnnotation != null)
node.put("maximum", maxAnnotation.value());
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", Integer.valueOf(defaultValue));
if (currentProperty != null) {
JsonSchemaExamples examplesAnnotation = currentProperty.getAnnotation(JsonSchemaExamples.class);
if (examplesAnnotation != null) {
ArrayNode examples = JsonNodeFactory.instance.arrayNode();
for (String example : examplesAnnotation.value()) {
examples.add(example);
}
node.set("examples", examples);
}
}
return new MyJsonValueFormatVisitor();
}
@Override public JsonNullFormatVisitor expectNullFormat(JavaType type) {
log.trace("expectNullFormat {}", type);
node.put("type", "null");
return new JsonNullFormatVisitor.Base();
}
@Override public JsonBooleanFormatVisitor expectBooleanFormat(JavaType type) {
log.trace("expectBooleanFormat");
node.put("type", "boolean");
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", Boolean.valueOf(defaultValue));
return new MyJsonValueFormatVisitor();
}
@Override public JsonMapFormatVisitor expectMapFormat(JavaType type) throws JsonMappingException {
log.trace ("expectMapFormat {}", type);
// There is no way to specify map in jsonSchema,
// So we're going to treat it as type=object with additionalProperties = true,
// so that it can hold whatever the map can hold
node.put("type", "object");
// If we're annotated with @NotEmpty, make sure we add a minItems of 1 to our schema here.
NotEmpty notEmptyAnnotation = tryGetAnnotation(NotEmpty.class);
if (notEmptyAnnotation != null)
node.put("minProperties", 1);
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", defaultValue);
ObjectNode additionalPropsObject = JsonNodeFactory.instance.objectNode();
definitionsHandler.pushWorkInProgress();
JsonSchemaGeneratorVisitor childVisitor = createChildVisitor(additionalPropsObject, null);
ctx.objectMapper.acceptJsonFormatVisitor(ctx.tryToReMapType(type.getContentType()), childVisitor);
definitionsHandler.popworkInProgress();
node.set("additionalProperties", additionalPropsObject);
class MapVisitor extends AbstractJsonFormatVisitorWithSerializerProvider implements JsonMapFormatVisitor {
@Override public void keyFormat(JsonFormatVisitable handler, JavaType keyType) {
log.trace("JsonMapFormatVisitor.keyFormat handler: $handler - keyType: $keyType");
}
@Override public void valueFormat(JsonFormatVisitable handler, JavaType valueType) {
log.trace("JsonMapFormatVisitor.valueFormat handler: $handler - valueType: $valueType");
}
}
return new MapVisitor();
}
@Data @FieldDefaults(level = AccessLevel.PRIVATE, makeFinal = true) @Accessors(fluent = true)
static class PolymorphismInfo { String typePropertyName; String subTypeName; }
private PolymorphismInfo extractPolymorphismInfo(JavaType type) throws JsonMappingException {
JavaType baseType = Utils.getSuperClass(type);
if (baseType == null)
return null;
TypeSerializer serializer = ctx.getTypeSerializer(baseType);
if (serializer == null)
return null;
As inclusionMethod = serializer.getTypeInclusion();
if (inclusionMethod == JsonTypeInfo.As.PROPERTY
|| inclusionMethod == JsonTypeInfo.As.EXISTING_PROPERTY) {
TypeIdResolver idResolver = serializer.getTypeIdResolver();
assert idResolver != null;
String id;
if (idResolver instanceof MinimalClassNameIdResolver)
// use custom implementation, because default implementation needs instance and we don't have one
id = Utils.extractMinimalClassnameId(baseType, type);
else
id = idResolver.idFromValueAndType(null, type.getRawClass());
return new PolymorphismInfo(serializer.getPropertyName(), id);
}
else
throw new IllegalStateException("We do not support polymorphism using jsonTypeInfo.include() = " + inclusionMethod);
}
private List> extractSubTypes(JavaType type) {
return extractSubTypes(type.getRawClass());
}
private List> extractSubTypes(Class> type) {
AnnotatedClass ac = AnnotatedClassResolver.resolveWithoutSuperTypes(
ctx.objectMapper.getDeserializationConfig(),
type,
ctx.objectMapper.getDeserializationConfig());
JsonTypeInfo jsonTypeInfo = ac.getAnnotation(JsonTypeInfo.class);
if (jsonTypeInfo == null) {
return Collections.unmodifiableList( new ArrayList>() );
}
if (jsonTypeInfo.use() == JsonTypeInfo.Id.NAME) {
JsonSubTypes subTypeAnn = type.getDeclaredAnnotation(JsonSubTypes.class);
if (subTypeAnn == null) {
// We did not find it via @JsonSubTypes-annotation (Probably since it is using mixin's) => Must fallback to using collectAndResolveSubtypesByClass
Collection resolvedSubTypes
= ctx.objectMapper.getSubtypeResolver()
.collectAndResolveSubtypesByClass(ctx.objectMapper.getDeserializationConfig(), ac);
return resolvedSubTypes.stream()
.map(e -> e.getType())
.filter(c -> type.isAssignableFrom(c) && type != c)
// .toList(); // javac bug, lol (#9072339)
.collect(Collectors.toList());
}
Type[] subTypes = subTypeAnn.value();
return Stream.of(subTypes)
.map(subType -> subType.value())
.flatMap(subType -> {
List> subSubTypes = extractSubTypes(subType);
if (!subSubTypes.isEmpty())
return subSubTypes.stream();
else
return Stream.of(subType);
})
.collect(Collectors.toList());
}
else
return ctx.config.subclassesResolver.getSubclasses(type);
}
@Override
public JsonObjectFormatVisitor expectObjectFormat(JavaType type) throws JsonMappingException {
String defaultValue = extractDefaultValue();
if (defaultValue != null)
node.put("default", defaultValue);
List> subTypes = extractSubTypes(type);
// Check if we have subtypes
if (!subTypes.isEmpty()) {
// We have subtypes
//log.trace("polymorphism - subTypes: $subTypes")
ArrayNode anyOfArrayNode = JsonNodeFactory.instance.arrayNode();
node.set("oneOf", anyOfArrayNode);
for (Class> subType : subTypes) {
log.trace("polymorphism - subType: $subType");
DefinitionInfo definitionInfo = definitionsHandler.getOrCreateDefinition
(ctx.objectMapper.constructType(subType),
(t, objectNode) -> {
JsonSchemaGeneratorVisitor childVisitor = createChildVisitor(objectNode, null);
ctx.objectMapper.acceptJsonFormatVisitor(ctx.tryToReMapType(subType), childVisitor);
return null;
});
ObjectNode thisOneOfNode = JsonNodeFactory.instance.objectNode();
thisOneOfNode.put("$ref", definitionInfo.ref());
// If class is annotated with JsonSchemaTitle, we should add it
JsonSchemaTitle titleAnnotation = subType.getDeclaredAnnotation(JsonSchemaTitle.class);
if (titleAnnotation != null)
thisOneOfNode.put("title", titleAnnotation.value());
anyOfArrayNode.add(thisOneOfNode);
}
return null; // Returning null to stop jackson from visiting this object since we have done it manually
}
else {
// We do not have subtypes
if (level == 0) {
// This is the first level - we must not use definitions
return objectBuilder(type, node);
}
else {
DefinitionInfo definitionInfo = definitionsHandler.getOrCreateDefinition(type, this::objectBuilder);
if (definitionInfo.ref() != null)
node.put("$ref", definitionInfo.ref());
return definitionInfo.jsonObjectFormatVisitor();
}
}
}
private JsonObjectFormatVisitor objectBuilder(JavaType type, ObjectNode thisObjectNode) throws JsonMappingException {
thisObjectNode.put("type", "object");
thisObjectNode.put("additionalProperties", !ctx.config.failOnUnknownProperties);
AnnotatedClass ac = AnnotatedClassResolver.resolve(ctx.objectMapper.getDeserializationConfig(), type, ctx.objectMapper.getDeserializationConfig());
// If class is annotated with JsonSchemaFormat, we should add it
String format = ctx.resolvePropertyFormat(type);
if (format != null)
thisObjectNode.put("format", format);
// If class is annotated with JsonSchemaDescription, we should add it
JsonSchemaDescription descriptionAnnotation = ac.getAnnotations().get(JsonSchemaDescription.class);
if (descriptionAnnotation != null)
thisObjectNode.put("description", descriptionAnnotation.value());
else {
JsonPropertyDescription descriptionAnnotation2 = ac.getAnnotations().get(JsonPropertyDescription.class);
if (descriptionAnnotation2 != null)
thisObjectNode.put("description", descriptionAnnotation2.value());
}
// // Scala (just as long. wtf?)
// Option(ac.getAnnotations.get(classOf[JsonSchemaDescription])).map(_.value())
// .orElse(Option(ac.getAnnotations.get(classOf[JsonPropertyDescription])).map(_.value))
// .foreach {
// description: String =>
// thisObjectNode.put("description", description)
// }
// // Hypothetical syntax
// var description = (ac.getAnnotations().get(JsonSchemaDescription.class) ?| _.value())
// ?? (ac.getAnnotations().get(JsonPropertyDescription.class) ?| _.value());
// description ?| thisObjectNode.put("description", _);
// // Alt syntax
// var description = ac.getAnnotations().get(JsonSchemaDescription.class)?.value()
// description ??= ac.getAnnotations().get(JsonPropertyDescription.class)?.value();
// if (description) thisObjectNode.put("description", description);
// If class is annotated with JsonSchemaTitle, we should add it
JsonSchemaTitle titleAnnotation = ac.getAnnotations().get(JsonSchemaTitle.class);
if (titleAnnotation != null)
thisObjectNode.put("title", titleAnnotation.value());
// // alt syntax
// ac.getAnnotations().get(JsonSchemaTitle.class) ?| thisObjectNode.put("title", _.value());
// If class is annotated with JsonSchemaOptions, we should add it
JsonSchemaOptions optionsAnnotation = ac.getAnnotations().get(JsonSchemaOptions.class);
if (optionsAnnotation != null) {
ObjectNode optionsNode = Utils.getOptionsNode(thisObjectNode);
for (JsonSchemaOptions.Item item : optionsAnnotation.items())
optionsNode.put(item.name(), item.value());
}
// Add JsonSchemaInject to top-level, if exists.
// Possibly, it overrides further processing.
boolean injectOverridesAll;
JsonSchemaInject injectAnnotation = ctx.selectAnnotation(ac, JsonSchemaInject.class);
if (injectAnnotation != null) {
// Continue to render props if we merged injection
injectOverridesAll = ctx.injectFromAnnotation(thisObjectNode, injectAnnotation);
}
else
injectOverridesAll = false;
if (injectOverridesAll)
return null;
ObjectNode propertiesNode = Utils.getOrCreateObjectChild(thisObjectNode, "properties");
PolymorphismInfo polyInfo = extractPolymorphismInfo(type);
if (polyInfo != null) {
thisObjectNode.put("title", polyInfo.subTypeName);
// must inject the 'type'-param and value as enum with only one possible value
// This is done to make sure the json generated from the schema using this oneOf
// contains the correct "type info"
ArrayNode enumValuesNode = JsonNodeFactory.instance.arrayNode();
enumValuesNode.add(polyInfo.subTypeName);
ObjectNode enumObjectNode = Utils.getOrCreateObjectChild(propertiesNode, polyInfo.typePropertyName);
enumObjectNode.put("type", "string");
enumObjectNode.set("enum", enumValuesNode);
enumObjectNode.put("default", polyInfo.subTypeName);
if (ctx.config.hidePolymorphismTypeProperty) {
// Make sure the editor hides this polymorphism-specific property
ObjectNode optionsNode = JsonNodeFactory.instance.objectNode();
enumObjectNode.set("options", optionsNode);
optionsNode.put("hidden", true);
}
Utils.getRequiredArrayNode(thisObjectNode).add(polyInfo.typePropertyName);
if (ctx.config.useMultipleEditorSelectViaProperty) {
// https://github.com/jdorn/json-editor/issues/709
// Generate info to help generated editor to select correct oneOf-type
// when populating the gui/schema with existing data
ObjectNode objectOptionsNode = Utils.getOrCreateObjectChild( thisObjectNode, "options");
ObjectNode multipleEditorSelectViaPropertyNode = Utils.getOrCreateObjectChild( objectOptionsNode, "multiple_editor_select_via_property");
multipleEditorSelectViaPropertyNode.put("property", polyInfo.typePropertyName);
multipleEditorSelectViaPropertyNode.put("value", polyInfo.subTypeName);
}
}
class MyObjectVisitor extends AbstractJsonFormatVisitorWithSerializerProvider implements JsonObjectFormatVisitor {
@Override public void optionalProperty(BeanProperty prop) throws JsonMappingException {
log.trace("JsonObjectFormatVisitor.optionalProperty: prop: {}", prop);
handleProperty(prop.getName(), prop.getType(), prop, false);
}
@Override public void optionalProperty(String name, JsonFormatVisitable handler, JavaType propertyTypeHint) throws JsonMappingException {
log.trace("JsonObjectFormatVisitor.optionalProperty: name:{} handler:{} propertyTypeHint:{}", name, handler, propertyTypeHint);
handleProperty(name, propertyTypeHint, null, false);
}
@Override public void property(BeanProperty prop) throws JsonMappingException {
log.trace("JsonObjectFormatVisitor.property: prop:{}", prop);
handleProperty(prop.getName(), prop.getType(), prop, true);
}
@Override public void property(String name, JsonFormatVisitable handler, JavaType propertyTypeHint) throws JsonMappingException {
log.trace("JsonObjectFormatVisitor.property: name:{} handler:{} propertyTypeHint:{}", name, handler, propertyTypeHint);
handleProperty(name, propertyTypeHint, null, true);
}
// Used when rendering schema using propertyOrdering as specified here:
// https://github.com/jdorn/json-editor#property-ordering
int nextPropertyOrderIndex = 1;
void handleProperty(String propertyName, JavaType propertyType, BeanProperty prop, Boolean jsonPropertyRequired) throws JsonMappingException {
log.trace("JsonObjectFormatVisitor - {}: {}", propertyName, propertyType);
if (propertiesNode.get(propertyName) != null) {
log.debug("Ignoring property '{}' in $propertyType since it has already been added, probably as type-property using polymorphism", propertyName);
return;
}
// Need to check for Optional/Optional-special-case before we know what node to use here.
// record PropertyNode(ObjectNode main, ObjectNode meta) {}
@Data @FieldDefaults(level = AccessLevel.PRIVATE, makeFinal = true) @Accessors(fluent = true)
class PropertyNode { final ObjectNode main; final ObjectNode meta; }
// Check if we should set this property as required. Primitive types MUST have a value, as does anything
// with a @JsonProperty that has "required" set to true. Lastly, various javax.validation annotations also
// make this required.
boolean requiredProperty = propertyType.getRawClass().isPrimitive() || jsonPropertyRequired || ctx.validationAnnotationRequired(prop);
boolean optionalType = Optional.class.isAssignableFrom(propertyType.getRawClass())
|| propertyType.getRawClass().getName().equals("scala.Option");
boolean nullable = optionalType ||
!propertyType.getRawClass().isPrimitive() &&
(ctx.hasNullableAnnotation(prop)
|| ctx.config.nullableByDefault && !ctx.hasNotNullAnnotation(prop));
PropertyNode thisPropertyNode;
{
ObjectNode node = JsonNodeFactory.instance.objectNode();
propertiesNode.set(propertyName, node);
if (ctx.config.usePropertyOrdering) {
node.put("propertyOrder", nextPropertyOrderIndex);
nextPropertyOrderIndex += 1;
}
if (!requiredProperty && ((ctx.config.useOneOfForOption && optionalType) ||
(ctx.config.useOneOfForNullables && !optionalType))) {
// We support this type being null, insert a oneOf consisting of a sentinel "null" and the real type.
ArrayNode oneOfArray = JsonNodeFactory.instance.arrayNode();
node.set("oneOf", oneOfArray);
// Create our sentinel "null" value for the case no value is provided.
ObjectNode oneOfNull = JsonNodeFactory.instance.objectNode();
oneOfNull.put("type", "null");
oneOfNull.put("title", "Not included");
oneOfArray.add(oneOfNull);
// If our nullable/optional type has a value, it'll be this.
ObjectNode oneOfReal = JsonNodeFactory.instance.objectNode();
oneOfArray.add(oneOfReal);
// Return oneOfReal which, from now on, will be used as the node representing this property
thisPropertyNode = new PropertyNode(oneOfReal, node);
} else {
// Our type must not be null: primitives, @NotNull annotations, @JsonProperty annotations marked required etc.
thisPropertyNode = new PropertyNode(node, node);
}
}
// Continue processing this property
JsonSchemaGeneratorVisitor childVisitor = createChildVisitor(thisPropertyNode.main, prop);
// Push current work in progress since we're about to start working on a new class
definitionsHandler.pushWorkInProgress();
if ((Optional.class.isAssignableFrom(propertyType.getRawClass())
|| propertyType.getRawClass().getName().equals("scala.Option"))
&& propertyType.containedTypeCount() >= 1) {
// Property is scala Optional or Java Optional.
//
// Due to Java's Type Erasure, the type behind Optional is lost.
// To workaround this, we use the same workaround as jackson-scala-module described here:
// https://github.com/FasterXML/jackson-module-scala/wiki/FAQ#deserializing-optionint-and-other-primitive-challenges
JavaType optionType = Utils.resolveElementType(propertyType, prop, ctx.objectMapper);
ctx.objectMapper.acceptJsonFormatVisitor(ctx.tryToReMapType(optionType), childVisitor);
} else {
ctx.objectMapper.acceptJsonFormatVisitor(ctx.tryToReMapType(propertyType), childVisitor);
}
// Pop back the work we were working on..
definitionsHandler.popworkInProgress();
// If this property is required, add it to our array of required properties.
if (requiredProperty)
Utils.getRequiredArrayNode(thisObjectNode).add(propertyName);
if (prop == null)
return;
String format = ctx.resolvePropertyFormat(prop);
if (format != null)
thisPropertyNode.main.put("format", format);
// Optionally add description
JsonSchemaDescription descriptionAnn = prop.getAnnotation(JsonSchemaDescription.class);
JsonPropertyDescription descriptionAnn2 = prop.getAnnotation(JsonPropertyDescription.class);
if (descriptionAnn != null)
thisPropertyNode.meta.put("description", descriptionAnn.value());
else if (descriptionAnn2 != null)
thisPropertyNode.meta.put("description", descriptionAnn2.value());
// Optionally add title
JsonSchemaTitle titleAnn = prop.getAnnotation(JsonSchemaTitle.class);
if (titleAnn != null)
thisPropertyNode.meta.put("title", titleAnn.value());
else if (ctx.config.autoGenerateTitleForProperties) {
// We should generate 'pretty-name' based on propertyName
String title = Utils.camelCaseToSentenceCase(propertyName);
thisPropertyNode.meta.put("title", title);
}
// var title = prop.getAnnotation(JsonSchemaTitle.class) ?| _.value();
// if (ctx.config.autoGenerateTitleForProperties)
// title ??= Utils.generateTitleFromPropertyName(propertyName);
// title ?| thisPropertyNode.meta.put("title", _);
// Optionally add options
JsonSchemaOptions optionsAnn = prop.getAnnotation(JsonSchemaOptions.class);
if (optionsAnn != null) {
ObjectNode optionsNode = Utils.getOrCreateObjectChild(thisPropertyNode.meta, "options");
for (JsonSchemaOptions.Item option : optionsAnn.items())
optionsNode.put(option.name(), option.value());
}
// unpack operator and foreach-pipe
// prop.getAnnotation(JsonSchemaOptions.class) ?| *(_.items()) | option -> {
// ObjectNode optionsNode = Utils.getOrCreateObjectChild(thisPropertyNode.meta, "options");
// optionsNode.put(option.name(), option.value());
// };
// just null-safe (or null short-circuiting) pipe
// for (var option : (prop.getAnnotation(JsonSchemaOptions.class) ?| _.items()) ?? List.of()) {
// ObjectNode optionsNode = Utils.getOrCreateObjectChild(thisPropertyNode.meta, "options");
// optionsNode.put(_.name(), _.value());
// }
// null-safe foreach (possibly using :? operator)
// for (var option : prop.getAnnotation(JsonSchemaOptions.class) ?| _.items()) {
// ObjectNode optionsNode = Utils.getOrCreateObjectChild(thisPropertyNode.meta, "options");
// optionsNode.put(_.name(), _.value());
// }
JsonSchemaAdditional additionalAnn = prop.getAnnotation(JsonSchemaAdditional.class);
if (additionalAnn != null) {
ObjectNode optionsNode = Utils.getOrCreateObjectChild(thisPropertyNode.meta, "options");
optionsNode.put("additional", true);
}
// Optionally add JsonSchemaInject
JsonSchemaInject injectAnn = ctx.selectAnnotation(prop, JsonSchemaInject.class);
if (injectAnn == null) {
// Try to look at the class itself -- Looks like this is the only way to find it if the type is Enum
JsonSchemaInject injectAnn2 = prop.getType().getRawClass().getAnnotation(JsonSchemaInject.class);
if (injectAnn2 != null && ctx.annotationIsApplicable(injectAnn2))
injectAnn = injectAnn2;
}
if (injectAnn != null)
ctx.injectFromAnnotation(thisPropertyNode.meta, injectAnn);
}
}
return new MyObjectVisitor();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy