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

org.grails.cli.compiler.AstUtils Maven / Gradle / Ivy

There is a newer version: 2023.1.0-RC1
Show newest version
/*
 * Copyright 2012-2023 the original author or 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 org.grails.cli.compiler;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;

import org.codehaus.groovy.ast.AnnotatedNode;
import org.codehaus.groovy.ast.AnnotationNode;
import org.codehaus.groovy.ast.ClassNode;
import org.codehaus.groovy.ast.FieldNode;
import org.codehaus.groovy.ast.MethodNode;
import org.codehaus.groovy.ast.expr.ArgumentListExpression;
import org.codehaus.groovy.ast.expr.ClosureExpression;
import org.codehaus.groovy.ast.expr.ConstantExpression;
import org.codehaus.groovy.ast.expr.Expression;
import org.codehaus.groovy.ast.expr.MethodCallExpression;
import org.codehaus.groovy.ast.stmt.BlockStatement;
import org.codehaus.groovy.ast.stmt.ExpressionStatement;
import org.codehaus.groovy.ast.stmt.Statement;
import org.springframework.util.PatternMatchUtils;

/**
 * General purpose AST utilities.
 *
 * @author Phillip Webb
 * @author Dave Syer
 * @author Greg Turnquist
 * @since 1.0.0
 */
public abstract class AstUtils {

    /**
     * Determine if a {@link ClassNode} has one or more of the specified annotations on
     * the class or any of its methods. N.B. the type names are not normally fully
     * qualified.
     *
     * @param node        the class to examine
     * @param annotations the annotations to look for
     * @return {@code true} if at least one of the annotations is found, otherwise
     * {@code false}
     */
    public static boolean hasAtLeastOneAnnotation(ClassNode node, String... annotations) {
        if (hasAtLeastOneAnnotation((AnnotatedNode) node, annotations)) {
            return true;
        }
        for (MethodNode method : node.getMethods()) {
            if (hasAtLeastOneAnnotation(method, annotations)) {
                return true;
            }
        }
        return false;
    }

    /**
     * Determine if an {@link AnnotatedNode} has one or more of the specified annotations.
     * N.B. the annotation type names are not normally fully qualified.
     *
     * @param node        the node to examine
     * @param annotations the annotations to look for
     * @return {@code true} if at least one of the annotations is found, otherwise
     * {@code false}
     */
    public static boolean hasAtLeastOneAnnotation(AnnotatedNode node, String... annotations) {
        for (AnnotationNode annotationNode : node.getAnnotations()) {
            for (String annotation : annotations) {
                if (PatternMatchUtils.simpleMatch(annotation, annotationNode.getClassNode().getName())) {
                    return true;
                }
            }
        }
        return false;
    }

    /**
     * Determine if a {@link ClassNode} has one or more fields of the specified types or
     * method returning one or more of the specified types. N.B. the type names are not
     * normally fully qualified.
     *
     * @param node  the class to examine
     * @param types the types to look for
     * @return {@code true} if at least one of the types is found, otherwise {@code false}
     */
    public static boolean hasAtLeastOneFieldOrMethod(ClassNode node, String... types) {
        Set typesSet = new HashSet<>(Arrays.asList(types));
        for (FieldNode field : node.getFields()) {
            if (typesSet.contains(field.getType().getName())) {
                return true;
            }
        }
        for (MethodNode method : node.getMethods()) {
            if (typesSet.contains(method.getReturnType().getName())) {
                return true;
            }
        }
        return false;
    }

    /**
     * Determine if a {@link ClassNode} subclasses any of the specified types N.B. the
     * type names are not normally fully qualified.
     *
     * @param node  the class to examine
     * @param types the types that may have been sub-classed
     * @return {@code true} if the class subclasses any of the specified types, otherwise
     * {@code false}
     */
    public static boolean subclasses(ClassNode node, String... types) {
        for (String type : types) {
            if (node.getSuperClass().getName().equals(type)) {
                return true;
            }
        }
        return false;
    }

    public static boolean hasAtLeastOneInterface(ClassNode classNode, String... types) {
        Set typesSet = new HashSet<>(Arrays.asList(types));
        for (ClassNode inter : classNode.getInterfaces()) {
            if (typesSet.contains(inter.getName())) {
                return true;
            }
        }
        return false;
    }

    /**
     * Extract a top-level {@code name} closure from inside this block if there is one,
     * optionally removing it from the block at the same time.
     *
     * @param block  a block statement (class definition)
     * @param name   the name to look for
     * @param remove whether the extracted closure should be removed
     * @return a beans Closure if one can be found, null otherwise
     */
    public static ClosureExpression getClosure(BlockStatement block, String name, boolean remove) {
        for (ExpressionStatement statement : getExpressionStatements(block)) {
            Expression expression = statement.getExpression();
            if (expression instanceof MethodCallExpression) {
                ClosureExpression closure = getClosure(name, (MethodCallExpression) expression);
                if (closure != null) {
                    if (remove) {
                        block.getStatements().remove(statement);
                    }
                    return closure;
                }
            }
        }
        return null;
    }

    private static List getExpressionStatements(BlockStatement block) {
        List statements = new ArrayList<>();
        for (Statement statement : block.getStatements()) {
            if (statement instanceof ExpressionStatement) {
                statements.add((ExpressionStatement) statement);
            }
        }
        return statements;
    }

    private static ClosureExpression getClosure(String name, MethodCallExpression expression) {
        Expression method = expression.getMethod();
        if (method instanceof ConstantExpression && name.equals(((ConstantExpression) method).getValue())) {
            return (ClosureExpression) ((ArgumentListExpression) expression.getArguments()).getExpression(0);
        }
        return null;
    }

    public static ClosureExpression getClosure(BlockStatement block, String name) {
        return getClosure(block, name, false);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy