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

capital.scalable.restdocs.jackson.FieldDocumentationVisitorContext Maven / Gradle / Ivy

/*-
 * #%L
 * Spring Auto REST Docs Core
 * %%
 * Copyright (C) 2015 - 2018 Scalable Capital GmbH
 * %%
 * 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.
 * #L%
 */
package capital.scalable.restdocs.jackson;

import static capital.scalable.restdocs.constraints.ConstraintReader.CONSTRAINTS_ATTRIBUTE;
import static capital.scalable.restdocs.constraints.ConstraintReader.DEPRECATED_ATTRIBUTE;
import static capital.scalable.restdocs.constraints.ConstraintReader.OPTIONAL_ATTRIBUTE;
import static capital.scalable.restdocs.i18n.SnippetTranslationResolver.translate;
import static capital.scalable.restdocs.util.FieldUtil.fromGetter;
import static capital.scalable.restdocs.util.FieldUtil.isGetter;
import static capital.scalable.restdocs.util.FormatUtil.addDot;
import static capital.scalable.restdocs.util.FormatUtil.join;
import static capital.scalable.restdocs.util.TypeUtil.isPrimitive;
import static capital.scalable.restdocs.util.TypeUtil.resolveAnnotation;
import static org.apache.commons.lang3.StringUtils.isBlank;
import static org.apache.commons.lang3.StringUtils.isNotBlank;
import static org.apache.commons.lang3.StringUtils.trimToEmpty;
import static org.slf4j.LoggerFactory.getLogger;
import static org.springframework.restdocs.payload.PayloadDocumentation.fieldWithPath;

import java.util.ArrayList;
import java.util.List;

import capital.scalable.restdocs.constraints.ConstraintReader;
import capital.scalable.restdocs.javadoc.JavadocReader;
import com.fasterxml.jackson.databind.DeserializationConfig;
import com.fasterxml.jackson.databind.DeserializationFeature;
import org.slf4j.Logger;
import org.springframework.restdocs.payload.FieldDescriptor;
import org.springframework.restdocs.snippet.Attributes.Attribute;

class FieldDocumentationVisitorContext {
    private static final Logger log = getLogger(FieldDocumentationVisitorContext.class);

    private final List fields = new ArrayList<>();
    private final JavadocReader javadocReader;
    private final ConstraintReader constraintReader;
    private final DeserializationConfig deserializationConfig;

    public FieldDocumentationVisitorContext(JavadocReader javadocReader,
            ConstraintReader constraintReader,
            DeserializationConfig deserializationConfig) {
        this.javadocReader = javadocReader;
        this.constraintReader = constraintReader;
        this.deserializationConfig = deserializationConfig;
    }

    public List getFields() {
        return fields;
    }

    public void addField(InternalFieldInfo info, String jsonType) {
        String jsonFieldPath = info.getJsonFieldPath();
        String javaFieldTypeName = info.getJavaFieldType().getRawClass().getSimpleName();

        if (isPresent(jsonFieldPath, javaFieldTypeName)) {
            return; // do not add duplicates
        }

        Class javaBaseClass = info.getJavaBaseClass();
        String javaFieldName = info.getJavaFieldName();
        String comment = resolveComment(javaBaseClass, javaFieldName);
        comment = join("
", comment, resolveSeeTag(javaBaseClass, javaFieldName)); FieldDescriptor fieldDescriptor = fieldWithPath(jsonFieldPath) .type(jsonType) .description(comment); Attribute constraints = constraintAttribute(javaBaseClass, javaFieldName); Attribute optionals = optionalAttribute(javaBaseClass, javaFieldName); Attribute deprecated = deprecatedAttribute(javaBaseClass, javaFieldName); fieldDescriptor.attributes(constraints, optionals, deprecated); fields.add(fieldDescriptor); log.debug("({}) {} added", jsonFieldPath, javaFieldTypeName); } private boolean isPresent(String jsonFieldPath, String javaFieldTypeName) { log.trace(" = WAS ADDED? {}", jsonFieldPath); for (FieldDescriptor descriptor : fields) { if (descriptor.getPath().equals(jsonFieldPath)) { log.trace(" = YES {}", descriptor.getPath()); log.debug("({}) {} NOT added", jsonFieldPath, javaFieldTypeName); return true; } log.trace(" = NO {}", descriptor.getPath()); } return false; } private Attribute constraintAttribute(Class javaBaseClass, String javaFieldName) { return new Attribute(CONSTRAINTS_ATTRIBUTE, resolveConstraintDescriptions(javaBaseClass, javaFieldName)); } private Attribute optionalAttribute(Class javaBaseClass, String javaFieldName) { return new Attribute(OPTIONAL_ATTRIBUTE, resolveOptionalMessages(javaBaseClass, javaFieldName)); } private Attribute deprecatedAttribute(Class javaBaseClass, String javaFieldName) { return new Attribute(DEPRECATED_ATTRIBUTE, resolveDeprecatedMessage(javaBaseClass, javaFieldName)); } private List resolveOptionalMessages(Class javaBaseClass, String javaFieldName) { List optionalMessages = new ArrayList<>(); if (isPrimitive(javaBaseClass, javaFieldName)) { boolean failOnNullForPrimitives = deserializationConfig.hasDeserializationFeatures( DeserializationFeature.FAIL_ON_NULL_FOR_PRIMITIVES.getMask()); optionalMessages.add("" + !failOnNullForPrimitives); return optionalMessages; } optionalMessages.addAll(constraintReader.getOptionalMessages(javaBaseClass, javaFieldName)); // fallback to field itself if we got a getter and no annotation on it if (optionalMessages.isEmpty() && isGetter(javaFieldName)) { optionalMessages.addAll( constraintReader.getOptionalMessages(javaBaseClass, fromGetter(javaFieldName))); } // if there was no default constraint resolved at all, default to optional=true if (!optionalMessages.contains("false") && !optionalMessages.contains("true")) { optionalMessages.add(0, "true"); } return optionalMessages; } private List resolveConstraintDescriptions(Class javaBaseClass, String javaFieldName) { List descriptions = new ArrayList<>(); descriptions.addAll(constraintReader.getConstraintMessages(javaBaseClass, javaFieldName)); // fallback to field itself if we got a getter and no annotation on it if (descriptions.isEmpty() && isGetter(javaFieldName)) { descriptions.addAll(constraintReader .getConstraintMessages(javaBaseClass, fromGetter(javaFieldName))); } return descriptions; } private String resolveDeprecatedMessage(Class javaBaseClass, String javaFieldName) { boolean isDeprecated = resolveAnnotation(javaBaseClass, javaFieldName, Deprecated.class) != null; String comment = resolveTag(javaBaseClass, javaFieldName, "deprecated"); if (isDeprecated || isNotBlank(comment)) { return trimToEmpty(comment); } else { return null; } } private String resolveSeeTag(Class javaBaseClass, String javaFieldName) { String comment = resolveTag(javaBaseClass, javaFieldName, "see"); if (isNotBlank(comment)) { return addDot(translate("tags-see", comment)); } else { return ""; } } private String resolveComment(Class javaBaseClass, String javaFieldOrMethodName) { // prefer method comments first String comment = javadocReader.resolveMethodComment(javaBaseClass, javaFieldOrMethodName); if (isBlank(comment)) { // fallback to field comment = javadocReader.resolveFieldComment(javaBaseClass, javaFieldOrMethodName); } if (isBlank(comment) && isGetter(javaFieldOrMethodName)) { // fallback if name is getter method but comment is on field itself comment = javadocReader.resolveFieldComment(javaBaseClass, fromGetter(javaFieldOrMethodName)); } return comment; } private String resolveTag(Class javaBaseClass, String javaFieldOrMethodName, String tagName) { // prefer method comments first String comment = javadocReader.resolveMethodTag(javaBaseClass, javaFieldOrMethodName, tagName); if (isBlank(comment)) { // fallback to field comment = javadocReader.resolveFieldTag(javaBaseClass, javaFieldOrMethodName, tagName); } if (isBlank(comment) && isGetter(javaFieldOrMethodName)) { // fallback if name is getter method but comment is on field itself comment = javadocReader.resolveFieldTag(javaBaseClass, fromGetter(javaFieldOrMethodName), tagName); } return comment; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy