capital.scalable.restdocs.constraints.ConstraintReaderImpl 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.constraints;
import static capital.scalable.restdocs.constraints.ConstraintAndGroupDescriptionResolver.VALUE;
import static capital.scalable.restdocs.constraints.MethodParameterValidatorConstraintResolver.CONSTRAINT_CLASS;
import static capital.scalable.restdocs.i18n.SnippetTranslationResolver.translate;
import static capital.scalable.restdocs.util.FormatUtil.collectionToString;
import static java.util.Collections.emptyList;
import static java.util.Collections.singletonList;
import static java.util.Collections.singletonMap;
import static org.apache.commons.lang3.StringUtils.isBlank;
import static org.slf4j.LoggerFactory.getLogger;
import static org.springframework.util.ReflectionUtils.findField;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.springframework.core.MethodParameter;
import org.springframework.restdocs.constraints.Constraint;
import org.springframework.restdocs.constraints.ConstraintDescriptions;
import org.springframework.restdocs.constraints.ResourceBundleConstraintDescriptionResolver;
public class ConstraintReaderImpl implements ConstraintReader {
private static final Logger log = getLogger(ConstraintReaderImpl.class);
private final ConstraintAndGroupDescriptionResolver constraintDescriptionResolver;
private final SkippableConstraintResolver skippableConstraintResolver;
private final MethodParameterConstraintResolver constraintResolver;
private final ObjectMapper objectMapper;
private ConstraintReaderImpl(MethodParameterConstraintResolver actualResolver, ObjectMapper objectMapper) {
constraintDescriptionResolver = new ConstraintAndGroupDescriptionResolver(
new ResourceBundleConstraintDescriptionResolver());
skippableConstraintResolver = new SkippableConstraintResolver(
actualResolver, constraintDescriptionResolver);
constraintResolver = new HumanReadableConstraintResolver(skippableConstraintResolver);
this.objectMapper = objectMapper;
}
public static ConstraintReaderImpl create(ObjectMapper objectMapper) {
return CONSTRAINT_CLASS != null ? createWithValidation(objectMapper) : createWithoutValidation(objectMapper);
}
static ConstraintReaderImpl createWithoutValidation(ObjectMapper objectMapper) {
return new ConstraintReaderImpl(new NoOpMethodParameterConstraintResolver(), objectMapper);
}
static ConstraintReaderImpl createWithValidation(ObjectMapper objectMapper) {
return new ConstraintReaderImpl(new MethodParameterValidatorConstraintResolver(), objectMapper);
}
@Override
public List getOptionalMessages(Class> javaBaseClass, String javaFieldName) {
return skippableConstraintResolver.getOptionalMessages(javaFieldName, javaBaseClass);
}
@Override
public List getConstraintMessages(Class> javaBaseClass, String javaFieldName) {
ConstraintDescriptions constraints = new ConstraintDescriptions(javaBaseClass,
constraintResolver, constraintDescriptionResolver);
List constraintMessages = new ArrayList<>();
constraintMessages.addAll(constraints.descriptionsForProperty(javaFieldName));
constraintMessages.addAll(getEnumConstraintMessage(javaBaseClass, javaFieldName));
return constraintMessages;
}
@Override
public List getConstraintMessages(MethodParameter param) {
List constraints = constraintResolver.resolveForParameter(param);
List constraintMessages = new ArrayList<>();
for (Constraint constraint : constraints) {
constraintMessages.add(
constraintDescriptionResolver.resolveDescription(constraint));
}
constraintMessages.addAll(getEnumConstraintMessage(param));
Collections.sort(constraintMessages);
return constraintMessages;
}
private List getEnumConstraintMessage(Class> javaBaseClass, String javaFieldName) {
// could be getter actually
Field field = findField(javaBaseClass, javaFieldName);
if (field == null) {
return emptyList();
}
return getEnumConstraintMessage(field.getType());
}
private List getEnumConstraintMessage(MethodParameter param) {
return getEnumConstraintMessage(param.getParameterType());
}
private List getEnumConstraintMessage(Class> rawClass) {
if (!rawClass.isEnum()) {
return emptyList();
}
Class enumClass = (Class) rawClass;
List serializedEnumValues = new ArrayList<>();
for (Enum e : enumClass.getEnumConstants()) {
try {
String jsonValue = objectMapper.writeValueAsString(e);
// Result is wrapped in double quotes
jsonValue = StringUtils.removeStart(jsonValue, "\"");
jsonValue = StringUtils.removeEnd(jsonValue, "\"");
serializedEnumValues.add(jsonValue);
} catch (JsonProcessingException ex) {
log.error("Failed to convert enum {}", e, ex);
}
}
String value = collectionToString(serializedEnumValues);
String enumName = enumClass.getCanonicalName();
String message = constraintDescriptionResolver.resolveDescription(
new Constraint(enumName, singletonMap(VALUE, (Object) value)));
// fallback
if (isBlank(message) || message.equals(enumName)) {
message = translate("constraints-enum", value);
}
return singletonList(message);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy