io.micronaut.validation.validator.constraints.AbstractPatternValidator Maven / Gradle / Ivy
/*
* Copyright 2017-2020 original authors
*
* 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
*
* https://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 io.micronaut.validation.validator.constraints;
import io.micronaut.core.annotation.AnnotationValue;
import edu.umd.cs.findbugs.annotations.NonNull;
import javax.validation.ValidationException;
import javax.validation.constraints.Pattern;
import java.lang.annotation.Annotation;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.regex.PatternSyntaxException;
/**
* Abstract pattern validator.
*
* @param The annotation type.
*
* @author graemerocher
* @since 1.2
*/
abstract class AbstractPatternValidator implements ConstraintValidator {
private static final Pattern.Flag[] ZERO_FLAGS = new Pattern.Flag[0];
private static final Map COMPUTED_PATTERNS = new ConcurrentHashMap<>(10);
/**
* Gets the pattern for the given annotation metadata.
* @param annotationMetadata The metadata
* @param isOptional Whether the pattern is required to be returned
* @return The pattern
*/
java.util.regex.Pattern getPattern(
@NonNull AnnotationValue> annotationMetadata,
boolean isOptional) {
final Optional regexp = annotationMetadata.get("regexp", String.class);
final String pattern;
if (isOptional) {
pattern = regexp.orElse(".*");
} else {
pattern = regexp
.orElseThrow(() -> new ValidationException("No pattern specified"));
}
final Pattern.Flag[] flags = annotationMetadata.get("flags", Pattern.Flag[].class).orElse(ZERO_FLAGS);
if (isOptional && pattern.equals(".*") && flags.length == 0) {
return null;
}
int computedFlag = 0;
for (Pattern.Flag flag : flags) {
computedFlag = computedFlag | flag.getValue();
}
final PatternKey key = new PatternKey(pattern, computedFlag);
java.util.regex.Pattern regex = COMPUTED_PATTERNS.get(key);
if (regex == null) {
try {
if (computedFlag != 0) {
regex = java.util.regex.Pattern.compile(pattern, computedFlag);
} else {
regex = java.util.regex.Pattern.compile(pattern);
}
} catch (PatternSyntaxException e) {
throw new IllegalArgumentException("Invalid regular expression", e);
}
COMPUTED_PATTERNS.put(key, regex);
}
return regex;
}
/**
* Key used to cache patterns.
*/
private static final class PatternKey {
final String pattern;
final int flags;
PatternKey(String pattern, int flags) {
this.pattern = pattern;
this.flags = flags;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
PatternKey that = (PatternKey) o;
return flags == that.flags &&
pattern.equals(that.pattern);
}
@Override
public int hashCode() {
return Objects.hash(pattern, flags);
}
}
}