All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.micronaut.validation.validator.constraints.AbstractPatternValidator Maven / Gradle / Ivy

There is a newer version: 4.8.0
Show newest version
/*
 * 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 io.micronaut.core.annotation.NonNull;

import jakarta.validation.ValidationException;
import jakarta.validation.constraints.Pattern;
import java.lang.annotation.Annotation;
import java.util.Map;
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.
     *
     * @param pattern The pattern
     * @param flags   The flags
     */
    private record PatternKey(String pattern, int flags) {
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy