blob: 18dfadc5aebf0ef40da2bc0b6c88fe3987d5228a [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.bval.jsr.metadata;
import java.lang.annotation.ElementType;
import java.util.Collection;
import java.util.Collections;
import java.util.EnumSet;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import javax.validation.ConstraintDeclarationException;
import javax.validation.ElementKind;
import javax.validation.Valid;
import javax.validation.executable.ValidateOnExecution;
import org.apache.bval.jsr.metadata.HierarchyBuilder.ContainerDelegate;
import org.apache.bval.jsr.metadata.HierarchyBuilder.ElementDelegate;
import org.apache.bval.jsr.metadata.HierarchyBuilder.HierarchyDelegate;
import org.apache.bval.util.Exceptions;
import org.apache.bval.util.Validate;
class Liskov {
//@formatter:off
private enum ValidationElement {
constraints, cascades, groupConversions, validateOnExecution;
}
private enum StrengtheningIssue implements Predicate<Map<Meta<?>, Set<ValidationElement>>> {
overriddenHierarchy("overridden %s in inheritance hierarchy: %s") {
@Override
public boolean test(Map<Meta<?>, Set<ValidationElement>> detectedValidationElements) {
boolean lowestFound = false;
for (Set<ValidationElement> validated : detectedValidationElements.values()) {
if (lowestFound) {
return false;
}
lowestFound = !validated.isEmpty();
}
return true;
}
},
unrelatedInheritance("declared %s in unrelated inheritance hierarchies: %s") {
@Override
public boolean test(Map<Meta<?>, Set<ValidationElement>> detectedValidationElements) {
final Set<Class<?>> interfaces = detectedValidationElements.keySet().stream().map(Meta::getDeclaringClass)
.filter(Class::isInterface).collect(Collectors.toSet());
if (interfaces.isEmpty()) {
return true;
}
final boolean allRelated =
detectedValidationElements.keySet().stream().map(Meta::getDeclaringClass).allMatch(ifc -> interfaces
.stream().filter(Predicate.isEqual(ifc).negate()).allMatch(ifc2 -> related(ifc, ifc2)));
return allRelated;
}
};
//@formatter:on
final String format;
private StrengtheningIssue(String format) {
this.format = "Illegal strengthening: " + format;
}
Supplier<String> messageFor(Map<Meta<?>, Set<ValidationElement>> detectedValidationElements) {
return () -> {
final Set<ValidationElement> validationElements =
detectedValidationElements.values().stream().flatMap(Collection::stream)
.collect(Collectors.toCollection(() -> EnumSet.noneOf(ValidationElement.class)));
final String describeHierarchy = detectedValidationElements.keySet().stream().map(Meta::describeHost)
.collect(Collectors.joining(", ", "[", "]"));
return String.format(format, validationElements, describeHierarchy);
};
}
void check(Map<Meta<?>, Set<ValidationElement>> detectedValidationElements) {
Exceptions.raiseUnless(test(detectedValidationElements), ConstraintDeclarationException::new,
messageFor(detectedValidationElements));
}
}
static void validateContainerHierarchy(Collection<? extends ContainerDelegate<?>> delegates, ElementKind elementKind) {
if (Validate.notNull(delegates, "delegates").isEmpty()) {
return;
}
if (Validate.notNull(elementKind, "elementKind") == ElementKind.CONTAINER_ELEMENT) {
elementKind = getContainer(delegates.iterator().next().getHierarchyElement());
}
switch (elementKind) {
case RETURN_VALUE:
noRedeclarationOfReturnValueCascading(delegates);
final Map<Meta<?>, Set<ValidationElement>> detectedValidationElements =
detectValidationElements(delegates, ElementDelegate::getHierarchyElement, detectGroupConversion());
// pre-check return value overridden hierarchy:
Stream.of(StrengtheningIssue.values())
.filter((Predicate<? super StrengtheningIssue>) si -> !(si == StrengtheningIssue.overriddenHierarchy
&& detectedValidationElements.values().stream().filter(s -> !s.isEmpty()).count() < 2))
.forEach(si -> si.check(detectedValidationElements));
break;
case PARAMETER:
noStrengtheningOfPreconditions(delegates, detectConstraints(), detectCascading(), detectGroupConversion());
break;
default:
break;
}
}
static void validateCrossParameterHierarchy(Collection<? extends ElementDelegate<?, ?>> delegates) {
if (Validate.notNull(delegates, "delegates").isEmpty()) {
return;
}
noStrengtheningOfPreconditions(delegates, detectConstraints());
}
static void validateValidateOnExecution(Collection<? extends HierarchyDelegate<?, ?>> delegates) {
noStrengtheningOfPreconditions(delegates, detectValidateOnExecution());
}
private static ElementKind getContainer(Meta<?> meta) {
Meta<?> m = meta;
while (m.getElementType() == ElementType.TYPE_USE) {
m = m.getParent();
}
switch (m.getElementType()) {
case METHOD:
return ElementKind.RETURN_VALUE;
case PARAMETER:
return ElementKind.PARAMETER;
default:
return ElementKind.PROPERTY;
}
}
private static void noRedeclarationOfReturnValueCascading(Collection<? extends ContainerDelegate<?>> delegates) {
final Map<Class<?>, Meta<?>> cascadedReturnValues =
delegates.stream().filter(ContainerDelegate::isCascade).map(HierarchyDelegate::getHierarchyElement)
.collect(Collectors.toMap(Meta::getDeclaringClass, Function.identity()));
final boolean anyRelated = cascadedReturnValues.keySet().stream().anyMatch(t -> cascadedReturnValues.keySet()
.stream().filter(Predicate.isEqual(t).negate()).anyMatch(t2 -> related(t, t2)));
Exceptions.raiseIf(anyRelated, ConstraintDeclarationException::new,
"Multiple method return values marked @%s in hierarchy %s",
f -> f.args(Valid.class.getSimpleName(), cascadedReturnValues.values()));
}
@SafeVarargs
private static <D extends HierarchyDelegate<?, ?>> void noStrengtheningOfPreconditions(Collection<? extends D> delegates,
Function<? super D, ValidationElement>... detectors) {
final Map<Meta<?>, Set<ValidationElement>> detectedValidationElements =
detectValidationElements(delegates, HierarchyDelegate::getHierarchyElement, detectors);
if (detectedValidationElements.isEmpty()) {
return;
}
for (StrengtheningIssue s : StrengtheningIssue.values()) {
s.check(detectedValidationElements);
}
}
@SafeVarargs
private static <T> Map<Meta<?>, Set<ValidationElement>> detectValidationElements(Collection<? extends T> delegates,
Function<? super T, Meta<?>> toMeta, Function<? super T, ValidationElement>... detectors) {
final Map<Meta<?>, Set<ValidationElement>> detectedValidationElements = new LinkedHashMap<>();
delegates.forEach(d -> {
detectedValidationElements.put(toMeta.apply(d),
Stream.of(detectors).map(dt -> dt.apply(d)).filter(Objects::nonNull)
.collect(Collectors.toCollection(() -> EnumSet.noneOf(ValidationElement.class))));
});
if (detectedValidationElements.values().stream().allMatch(Collection::isEmpty)) {
// nothing declared
return Collections.emptyMap();
}
return detectedValidationElements;
}
private static boolean related(Class<?> c1, Class<?> c2) {
return c1.isAssignableFrom(c2) || c2.isAssignableFrom(c1);
}
private static Function<ElementDelegate<?, ?>, ValidationElement> detectConstraints() {
return d -> d.getDeclaredConstraints().length > 0 ? ValidationElement.constraints : null;
}
private static Function<ContainerDelegate<?>, ValidationElement> detectCascading() {
return d -> d.isCascade() ? ValidationElement.cascades : null;
}
private static Function<ContainerDelegate<?>, ValidationElement> detectGroupConversion() {
return d -> d.getGroupConversions().isEmpty() ? null : ValidationElement.groupConversions;
}
private static Function<HierarchyDelegate<?, ?>, ValidationElement> detectValidateOnExecution() {
return d -> d.getHierarchyElement().getHost().isAnnotationPresent(ValidateOnExecution.class)
? ValidationElement.validateOnExecution : null;
}
private Liskov() {
}
}