capital.scalable.restdocs.snippet.StandardTableSnippet Maven / Gradle / Ivy
Show all versions of spring-auto-restdocs-core Show documentation
/*-
* #%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.snippet;
import static capital.scalable.restdocs.OperationAttributeHelper.determineTemplateFormatting;
import static capital.scalable.restdocs.OperationAttributeHelper.getHandlerMethod;
import static capital.scalable.restdocs.constraints.ConstraintReader.CONSTRAINTS_ATTRIBUTE;
import static capital.scalable.restdocs.constraints.ConstraintReader.DEFAULT_VALUE_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.javadoc.JavadocUtil.convertFromJavadoc;
import static capital.scalable.restdocs.util.FormatUtil.addDot;
import static capital.scalable.restdocs.util.FormatUtil.join;
import static java.util.stream.Collectors.toList;
import static org.apache.commons.lang3.StringUtils.capitalize;
import static org.apache.commons.lang3.StringUtils.trimToEmpty;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import capital.scalable.restdocs.util.TemplateFormatting;
import org.apache.commons.lang3.StringUtils;
import org.springframework.restdocs.operation.Operation;
import org.springframework.restdocs.payload.FieldDescriptor;
import org.springframework.restdocs.snippet.TemplatedSnippet;
import org.springframework.web.method.HandlerMethod;
public abstract class StandardTableSnippet extends TemplatedSnippet {
protected StandardTableSnippet(String snippetName, Map attributes) {
super(snippetName, attributes);
}
@Override
protected Map createModel(Operation operation) {
HandlerMethod handlerMethod = getHandlerMethod(operation);
Map model = defaultModel();
if (handlerMethod == null) {
return model;
}
Collection fieldDescriptors =
createFieldDescriptors(operation, handlerMethod);
TemplateFormatting templateFormatting = determineTemplateFormatting(operation);
return createModel(handlerMethod, model, fieldDescriptors, templateFormatting);
}
protected abstract Collection createFieldDescriptors(Operation operation,
HandlerMethod handlerMethod);
protected abstract String[] getTranslationKeys();
protected void enrichModel(Map model, HandlerMethod handlerMethod) {
// can be used to add additional fields
}
private Map createModel(HandlerMethod handlerMethod, Map model,
Collection fieldDescriptors, TemplateFormatting templateFormatting) {
model.put("content", fieldDescriptors.stream()
.map(descriptor -> createModelForDescriptor(descriptor, templateFormatting))
.collect(toList()));
model.put("hasContent", !fieldDescriptors.isEmpty());
model.put("noContent", fieldDescriptors.isEmpty());
enrichModel(model, handlerMethod);
return model;
}
private Map defaultModel() {
Map model = new HashMap<>();
for (String key : getTranslationKeys()) {
model.put(key, translate(key));
}
model.put("content", "");
model.put("hasContent", false);
model.put("noContent", true);
return model;
}
protected Map createModelForDescriptor(FieldDescriptor descriptor,
TemplateFormatting templateFormatting) {
String path = descriptor.getPath();
String type = toString(descriptor.getType());
String methodComment = resolveComment(descriptor);
String deprecatedComment = resolveDeprecated(descriptor);
String completeComment = join("", deprecatedComment, methodComment);
String description = convertFromJavadoc(completeComment, templateFormatting);
String optional = resolveOptional(descriptor, templateFormatting);
List constraints = resolveConstraints(descriptor);
final String defaultValue = resolveDefaultValue(descriptor);
description = joinAndFormat(description, constraints, defaultValue, templateFormatting);
Map model = new HashMap<>();
model.put("path", path);
model.put("type", type);
model.put("optional", optional);
model.put("description", description);
return model;
}
private String defaultValuePrefix(final String description) {
// if we have no description, we don't want to add new lines
return StringUtils.isEmpty(description) ? StringUtils.EMPTY : "\n\n";
}
private List resolveConstraints(FieldDescriptor descriptor) {
return (List) descriptor.getAttributes()
.get(CONSTRAINTS_ATTRIBUTE);
}
private String resolveOptional(FieldDescriptor descriptor,
TemplateFormatting templateFormatting) {
List optionalMessages = (List) descriptor.getAttributes()
.get(OPTIONAL_ATTRIBUTE);
return "" + join(templateFormatting.getLineBreak(), optionalMessages.toArray());
}
private String resolveComment(FieldDescriptor descriptor) {
return capitalize(addDot(toString(descriptor.getDescription())));
}
private String resolveDeprecated(FieldDescriptor descriptor) {
Object deprecated = descriptor.getAttributes().get(DEPRECATED_ATTRIBUTE);
if (deprecated != null) {
return addDot(translate("tags-deprecated", capitalize(toString(deprecated))));
} else {
return "";
}
}
private String resolveDefaultValue(FieldDescriptor descriptor) {
Object defaultValue = descriptor.getAttributes().get(DEFAULT_VALUE_ATTRIBUTE);
if (defaultValue != null) {
return addDot(translate("default-value", toString(defaultValue)));
} else {
return "";
}
}
private String toString(Object value) {
if (value != null) {
return trimToEmpty(value.toString());
} else {
return "";
}
}
private String joinAndFormat(String description, List constraints,
final String defaultValue, TemplateFormatting templateFormatting) {
StringBuilder res = new StringBuilder(description);
if (!description.isEmpty() && !description.endsWith(".")) {
res.append('.');
}
StringBuilder constr = formatConstraints(constraints, templateFormatting.getLineBreak());
if (res.length() > 0 && constr.length() > 0) {
res.append("\n\n");
}
res.append(constr.toString());
if (StringUtils.isNotEmpty(defaultValue)) {
res.append(defaultValuePrefix(description)).append(defaultValue);
}
return res.toString().replace("|", "\\|");
}
private StringBuilder formatConstraints(List constraints, String forcedLineBreak) {
StringBuilder res = new StringBuilder();
for (String constraint : constraints) {
if (constraint.trim().isEmpty()) {
continue;
}
if (res.length() > 0) {
res.append(forcedLineBreak);
}
res.append(constraint.trim());
if (!constraint.endsWith(".")) {
res.append('.');
}
}
return res;
}
}