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

com.aspectran.core.context.env.ProfilesParser Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2008-2025 The Aspectran Project
 *
 * 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
 *
 *     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 com.aspectran.core.context.env;

import com.aspectran.utils.Assert;
import com.aspectran.utils.annotation.jsr305.NonNull;
import com.aspectran.utils.annotation.jsr305.Nullable;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.StringTokenizer;
import java.util.function.Predicate;

/**
 * Internal parser used by {@link Profiles#of}.
 *
 * @since 7.5.0
 */
final class ProfilesParser {

    private enum Operator { NONE, AND, OR, NEGATE }

    private ProfilesParser() {
    }

    @NonNull
    static Profiles parse(String expression) {
        Profiles parsed = parseExpression(expression);
        return new ParsedProfiles(expression, parsed);
    }

    private static Profiles parseExpression(String expression) {
        Assert.hasText(expression, () -> "Invalid profile expression \"" + expression + "\": must contain text");
        StringTokenizer tokens = new StringTokenizer(expression, "()[]!, ", true);
        return parseTokens(expression, tokens);
    }

    private static Profiles parseTokens(String expression, StringTokenizer tokens) {
        return parseTokens(expression, tokens, Operator.NONE);
    }

    private static Profiles parseTokens(String expression, @NonNull StringTokenizer tokens, Operator operator) {
        List elements = new ArrayList<>();
        String token = null;
        int count = 0;
        while (tokens.hasMoreTokens()) {
            token = tokens.nextToken().trim();
            if (token.isEmpty()) {
                continue;
            }
            switch (token) {
                case "(":
                case "[":
                    Operator nested = "(".equals(token) ? Operator.AND : Operator.OR;
                    Profiles contents = parseTokens(expression, tokens, nested);
                    if (operator == Operator.NEGATE) {
                        return contents;
                    }
                    assertWellFormed(expression, elements, count);
                    elements.add(contents);
                    break;
                case ")":
                    assertWellFormed(expression, operator, Operator.AND, token);
                    return merge(expression, elements, operator);
                case "]":
                    assertWellFormed(expression, operator, Operator.OR, token);
                    return merge(expression, elements, operator);
                case "!":
                    elements.add(not(parseTokens(expression, tokens, Operator.NEGATE)));
                    break;
                case ",":
                    count++;
                    break;
                default:
                    Profiles value = equals(token);
                    if (operator == Operator.NEGATE) {
                        return value;
                    }
                    assertWellFormed(expression, elements, count);
                    elements.add(value);
            }
        }
        assertWellFormed(expression, Operator.NONE, operator, token);
        return merge(expression, elements, Operator.OR);
    }

    private static Profiles merge(String expression, @NonNull List elements, Operator operator) {
        assertWellFormed(expression, !elements.isEmpty(), "");
        if (elements.size() == 1) {
            return elements.get(0);
        } else {
            Profiles[] profiles = elements.toArray(new Profiles[0]);
            return (operator == Operator.AND ? and(profiles) : or(profiles));
        }
    }

    private static void assertWellFormed(String expression, Operator expected, Operator actual, String token) {
        Assert.isTrue(expected == actual, () -> {
            String message = null;
            if (actual == Operator.NEGATE) {
                message = "inappropriate use of negation operator; ‘!’ cannot be used alone";
            } else if (actual == Operator.AND && expected == Operator.OR) {
                message = "missing closing parenthesis of " + expected + " set; must be closed with ']', but ')'";
            } else if (actual == Operator.OR && expected == Operator.AND) {
                message = "missing closing parenthesis of " + expected + " set; must be closed with ')', but ']'";
            } else if (token != null) {
                message = "unnecessary operator '" + token + "'";
            }
            return "Malformed profile expression \"" + expression + "\"" + (message != null ? ": " + message : "");
        });
    }

    private static void assertWellFormed(String expression, @NonNull List elements, int size) {
        assertWellFormed(expression, elements.size() == size,
            "each profile or set of them must be separated by commas (',')");
    }

    private static void assertWellFormed(String expression, boolean wellFormed, String message) {
        Assert.isTrue(wellFormed, () -> "Malformed profile expression \"" + expression + "\": " + message);
    }

    @NonNull
    private static Profiles or(Profiles... profiles) {
        return activeProfile -> Arrays.stream(profiles).anyMatch(isMatch(activeProfile));
    }

    @NonNull
    private static Profiles and(Profiles... profiles) {
        return activeProfile -> Arrays.stream(profiles).allMatch(isMatch(activeProfile));
    }

    @NonNull
    private static Profiles not(Profiles profiles) {
        return activeProfile -> !profiles.matches(activeProfile);
    }

    @NonNull
    private static Profiles equals(String profile) {
        return activeProfile -> activeProfile.test(profile);
    }

    @NonNull
    private static Predicate isMatch(Predicate activeProfiles) {
        return profiles -> profiles.matches(activeProfiles);
    }

    private static class ParsedProfiles implements Profiles {

        private final String expression;

        private final Profiles parsed;

        ParsedProfiles(String expression, Profiles parsed) {
            this.expression = expression;
            this.parsed = parsed;
        }

        @Override
        public boolean matches(Predicate activeProfiles) {
            return parsed.matches(activeProfiles);
        }

        @Override
        public boolean equals(@Nullable Object other) {
            if (this == other) {
                return true;
            }
            if (other instanceof ParsedProfiles that) {
                return expression.equals(that.expression);
            }
            return false;
        }

        @Override
        public int hashCode() {
            return expression.hashCode();
        }

        @Override
        public String toString() {
            return expression;
        }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy