/*
 * 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.
Author:graemerocher
Type parameters:
  • <A> – The annotation type.
Since:1.2
/** * Abstract pattern validator. * * @param <A> The annotation type. * * @author graemerocher * @since 1.2 */
abstract class AbstractPatternValidator<A extends Annotation> implements ConstraintValidator<A, CharSequence> { private static final Pattern.Flag[] ZERO_FLAGS = new Pattern.Flag[0]; private static final Map<PatternKey, java.util.regex.Pattern> COMPUTED_PATTERNS = new ConcurrentHashMap<>(10);
Gets the pattern for the given annotation metadata.
Params:
  • annotationMetadata – The metadata
  • isOptional – Whether the pattern is required to be returned
Returns:The pattern
/** * 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<String> 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.
/** * 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); } } }