/*
* Hibernate Validator, declare and validate application constraints
*
* License: Apache License, Version 2.0
* See the license.txt file in the root directory or <http://www.apache.org/licenses/LICENSE-2.0>.
*/
package org.hibernate.validator.internal.engine.resolver;
import java.lang.annotation.ElementType;
import java.util.HashMap;
import java.util.Map;
import javax.validation.Path;
import javax.validation.TraversableResolver;
/**
* Cache results of a delegated traversable resolver to optimize calls.
* It works only for a single validate* call and should not be used if
* {@code TraversableResolver} is accessed concurrently.
*
* @author Emmanuel Bernard
*/
public class CachingTraversableResolverForSingleValidation implements TraversableResolver {
private final TraversableResolver delegate;
private final Map<TraversableHolder, TraversableHolder> traversables = new HashMap<TraversableHolder, TraversableHolder>();
public CachingTraversableResolverForSingleValidation(TraversableResolver delegate) {
this.delegate = delegate;
}
@Override
public boolean isReachable(Object traversableObject, Path.Node traversableProperty, Class<?> rootBeanType, Path pathToTraversableObject, ElementType elementType) {
TraversableHolder currentLH = new TraversableHolder( traversableObject, traversableProperty );
TraversableHolder cachedLH = traversables.get( currentLH );
if ( cachedLH == null ) {
currentLH.isReachable = delegate.isReachable(
traversableObject,
traversableProperty,
rootBeanType,
pathToTraversableObject,
elementType
);
traversables.put( currentLH, currentLH );
cachedLH = currentLH;
}
else if ( cachedLH.isReachable == null ) {
cachedLH.isReachable = delegate.isReachable(
traversableObject,
traversableProperty,
rootBeanType,
pathToTraversableObject,
elementType
);
}
return cachedLH.isReachable;
}
@Override
public boolean isCascadable(Object traversableObject, Path.Node traversableProperty, Class<?> rootBeanType, Path pathToTraversableObject, ElementType elementType) {
TraversableHolder currentLH = new TraversableHolder( traversableObject, traversableProperty );
TraversableHolder cachedLH = traversables.get( currentLH );
if ( cachedLH == null ) {
currentLH.isCascadable = delegate.isCascadable(
traversableObject,
traversableProperty,
rootBeanType,
pathToTraversableObject,
elementType
);
traversables.put( currentLH, currentLH );
cachedLH = currentLH;
}
else if ( cachedLH.isCascadable == null ) {
cachedLH.isCascadable = delegate.isCascadable(
traversableObject,
traversableProperty,
rootBeanType,
pathToTraversableObject,
elementType
);
}
return cachedLH.isCascadable;
}
private static final class TraversableHolder {
private final Object traversableObject;
private final Path.Node traversableProperty;
private final int hashCode;
private Boolean isReachable;
private Boolean isCascadable;
private TraversableHolder(Object traversableObject, Path.Node traversableProperty) {
this.traversableObject = traversableObject;
this.traversableProperty = traversableProperty;
this.hashCode = buildHashCode();
}
@Override
public boolean equals(Object o) {
if ( this == o ) {
return true;
}
if ( o == null || getClass() != o.getClass() ) {
return false;
}
TraversableHolder that = (TraversableHolder) o;
if ( traversableObject != null ? !traversableObject.equals( that.traversableObject ) : that.traversableObject != null ) {
return false;
}
if ( !traversableProperty.equals( that.traversableProperty ) ) {
return false;
}
return true;
}
@Override
public int hashCode() {
return hashCode;
}
public int buildHashCode() {
// HV-1013 Using identity hash code in order to avoid calling hashCode() of objects which may
// be handling null properties not correctly
int result = traversableObject != null ? System.identityHashCode( traversableObject ) : 0;
result = 31 * result + traversableProperty.hashCode();
return result;
}
}
}