com.strategicgains.restexpress.plugin.swagger.ModelResolver Maven / Gradle / Ivy
/*
Copyright 2014, Strategic Gains, Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License 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 com.strategicgains.restexpress.plugin.swagger;
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.*;
import com.strategicgains.restexpress.plugin.swagger.domain.ApiModel;
import com.strategicgains.restexpress.plugin.swagger.domain.DataType;
import com.strategicgains.restexpress.plugin.swagger.domain.Items;
import com.strategicgains.restexpress.plugin.swagger.domain.Primitives;
import com.strategicgains.restexpress.plugin.swagger.annotations.ApiModelProperty;
/**
* This class implements a simplified least-common denominator reflection-based
* introspector in roughly the same behavior as the serializers supported in
* RestExpress (jackson/gson)
*
* Research notes: In developing this class, it was discovered that jackson has
* facilities to generate a variant of json-schema directly from an
* ObjectMapper. However, as of this writing, it didn't support handling the
* swagger-annotations. It also produced schema that while obviously
* syntactically correct, didn't leverage the "$ref" construct and produce the
* "flat" model structure that the Swagger spec prefers.
*
* Furthermore, the serializer is overridden with something non-jackson, I
* didn't want to tie the swagger model generation to APIs that get unhooked.
*
* So, with all that said, this class, much like gson and jackson:
*
* - any top-level class that is to be interpreted as a swagger "complex"
* type, MUST be annotated with ApiModel
* - non-static, non-transient private fields with the ApiModelProperty
* annotation are considered for properties of complex types, any
* accessors/getters methods are ignored
* - arrays and classes that implement java.util.Collection are treated as the
* "container" type. Additionally, classes that implement java.util.Set have the
* "uniqueItems" property set to true.
* - Properties from the class and all it's ancestors are scanned. If
* properties duplicate in the class hierarchy, the "lowest" subclass takes
* precedence over ancestors.
*
*
* @author russtrotter
*/
public class ModelResolver
{
private Map models;
public ModelResolver()
{
this(new HashMap());
}
public ModelResolver(Map models)
{
this.models = models;
}
public DataType resolve(Type cls)
{
return resolve(cls, null);
}
public DataType resolve(Type cls, String modelName)
{
return createNode(cls, modelName, null, null);
}
private ApiModel resolveClass(Class> target, String modelName)
{
String id = getModelId(target, modelName);
ApiModel model = models.get(id);
if (model != null)
{
return model;
}
if (target.isPrimitive())
{
System.err.println("Unable to resolve primitive class: " + target);
return model;
// throw new
// IllegalArgumentException("Unable to resolve primitive class: " +
// target);
}
if (target.isSynthetic())
{
System.err.println("Unable to resolve synthetic class: " + target);
return model;
// throw new
// IllegalArgumentException("Unable to resolve synthetic class: " +
// target);
}
if (target.isArray())
{
System.err.println("Unable to use arrays for models: " + target);
return model;
// throw new
// IllegalArgumentException("Unable to use arrays for models: " +
// target);
}
// com.wordnik.swagger.annotations.ApiModel apiModel =
// target.getAnnotation(com.wordnik.swagger.annotations.ApiModel.class);
// if (apiModel == null) {
// throw new IllegalArgumentException("Missing ApiModel annotation on: "
// + target);
// }
model = new ApiModel().id(id);
models.put(id, model);
Map properties = new HashMap();
for (Class> cls = target; !Object.class.equals(cls) && cls != null; cls = cls
.getSuperclass())
{
processFields(properties, cls, modelName);
}
List sorted = new ArrayList(properties.values());
// sort the properties based on property "position" attribute of
// annotation
Collections.sort(sorted, new Comparator()
{
@Override
public int compare(DataType o1, DataType o2)
{
return o1.getPosition() - o2.getPosition();
}
});
for (DataType property : sorted)
{
model.property(property);
if (property.isRequired())
{
model.required(property.getProperty());
}
}
return model;
}
private DataType createNode(Type target, String modelName, String dataType, String format)
{
DataType node = new DataType();
if (dataType != null && !dataType.isEmpty())
{
node.setType(dataType);
if(format != null && !format.isEmpty())
{
node.setFormat(format);
}
}
else if (target instanceof Class)
{
Class> targetClass = (Class>) target;
if (String.class.equals(target))
{
node.setType(Primitives.STRING);
}
else if (Integer.class.equals(target)
|| Integer.TYPE.equals(target))
{
node.setType(Primitives.INTEGER);
}
else if (Long.class.equals(target) || Long.TYPE.equals(target))
{
node.setType(Primitives.LONG);
}
else if (Boolean.class.equals(target)
|| Boolean.TYPE.equals(target))
{
node.setType(Primitives.BOOLEAN);
}
else if (Float.class.equals(target) || Float.TYPE.equals(target))
{
node.setType(Primitives.FLOAT);
}
else if (Double.class.equals(target) || Double.TYPE.equals(target))
{
node.setType(Primitives.DOUBLE);
}
else if (Byte.class.equals(target) || Byte.TYPE.equals(target))
{
node.setType(Primitives.BYTE);
}
else if (Date.class.equals(target))
{
node.setType(Primitives.DATE_TIME);
}
else if (targetClass.isArray())
{
node.setType("array");
DataType componentModel = createNode(targetClass
.getComponentType(), null, null, null);
node.setItems(new Items(componentModel));
}
else if (targetClass.isEnum())
{
node.setType(Primitives.STRING);
for (Object obj : targetClass.getEnumConstants())
{
node.addEnum(obj.toString());
}
}
else if (Void.class.equals(target) || Void.TYPE.equals(target))
{
node.setType(Primitives.VOID);
}
else
{
ApiModel subModel = resolveClass(targetClass, modelName);
node.setRef(subModel.getId());
}
}
else if (target instanceof ParameterizedType)
{
ParameterizedType parameterizedType = (ParameterizedType) target;
Class> rawType = (Class>) parameterizedType.getRawType();
if (Collection.class.isAssignableFrom(rawType))
{
node.setType("array");
if (Set.class.isAssignableFrom(rawType))
{
node.setUniqueItems(true);
}
DataType componentModel = createNode(parameterizedType
.getActualTypeArguments()[0], null, null, null);
node.setItems(new Items(componentModel));
}
else
{
System.err.println("Unhandled generic type: " + target);
return node;
// throw new IllegalArgumentException("Unhandled generic type: "
// + target);
}
}
else
{
System.err.println("Unhandled type: " + target);
return node;
// throw new UnsupportedOperationException("Unhandled type: " +
// target);
}
return node;
}
private void processFields(Map properties, Class> target, String modelName)
{
for (Field field : target.getDeclaredFields())
{
if ((field.getModifiers() & (Modifier.STATIC | Modifier.TRANSIENT)) == 0)
{
PropertyMetadata propertyMetadata = getPropertyMetadataFromField(field);
// Ignore all properties that are marked as hidden
if (propertyMetadata != null && propertyMetadata.hidden)
{
continue;
}
//Ignore properties that are marked to be excluded from this model
if(propertyMetadata != null && propertyMetadata.excludeFromModels != null)
{
String modelId = getModelId(target, modelName);
if(Arrays.asList(propertyMetadata.excludeFromModels).contains(modelId))
{
continue;
}
}
if (propertyMetadata == null) {
DataType property = createNode(field.getGenericType(), null, null, null)
.setProperty(field.getName());
properties.put(field.getName(), property);
}
else if(!properties.containsKey(field.getName()))
{
DataType property = createNode(field.getGenericType(), null, propertyMetadata.dataType, propertyMetadata.format)
.setDescription(propertyMetadata.notes)
.setRequired(propertyMetadata.required)
.setPosition(propertyMetadata.position)
.setProperty(field.getName());
properties.put(field.getName(), property);
}
else
{
DataType property = createNode(field.getGenericType(), null, propertyMetadata.dataType, propertyMetadata.format)
.setProperty(field.getName());
properties.put(field.getName(), property);
}
}
}
}
private String getModelId(Class> target, String modelName)
{
String id = null;
if(modelName != null && !modelName.isEmpty())
{
id = modelName;
}
else
{
id = target.getSimpleName();
}
return id;
}
//The logic of pulling the data out of the field annotation is encapsulated here to stay backward compatible
//with consumers using the com.wordnik.swagger.annotations.ApiModelProperty annotation.
private PropertyMetadata getPropertyMetadataFromField(Field field)
{
PropertyMetadata propertyMetadata = null;
if(field != null) {
for(Annotation a : field.getAnnotations())
{
if (a instanceof ApiModelProperty)
{
ApiModelProperty model = (ApiModelProperty) a;
propertyMetadata = new PropertyMetadata();
propertyMetadata.value = model.value();
propertyMetadata.allowableValues = model.allowableValues();
propertyMetadata.access = model.access();
propertyMetadata.notes = model.notes();
propertyMetadata.dataType = model.dataType();
propertyMetadata.format = model.format();
propertyMetadata.required = model.required();
propertyMetadata.position = model.position();
propertyMetadata.hidden = model.hidden();
propertyMetadata.excludeFromModels = model.excludeFromModels();
break;
}
else if (a instanceof com.wordnik.swagger.annotations.ApiModelProperty)
{
com.wordnik.swagger.annotations.ApiModelProperty model = (com.wordnik.swagger.annotations.ApiModelProperty) a;
propertyMetadata = new PropertyMetadata();
propertyMetadata.value = model.value();
propertyMetadata.allowableValues = model.allowableValues();
propertyMetadata.access = model.access();
propertyMetadata.notes = model.notes();
propertyMetadata.dataType = model.dataType();
propertyMetadata.required = model.required();
propertyMetadata.position = model.position();
propertyMetadata.hidden = model.hidden();
break;
}
}
}
return propertyMetadata;
}
private class PropertyMetadata {
String value = "";
String allowableValues = "";
String access = "";
String notes = "";
String dataType = "";
String format = "";
boolean required = false;
int position = 0;
boolean hidden = false;
String[] excludeFromModels = {};
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy