org.apache.groovy.ast.tools.ConstructorNodeUtils Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of groovy Show documentation
Show all versions of groovy Show documentation
Groovy: A powerful multi-faceted language for the JVM
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.groovy.ast.tools;
import org.codehaus.groovy.ast.ClassHelper;
import org.codehaus.groovy.ast.ClassNode;
import org.codehaus.groovy.ast.Parameter;
import org.codehaus.groovy.ast.PropertyNode;
import org.codehaus.groovy.ast.expr.ConstructorCallExpression;
import org.codehaus.groovy.ast.expr.Expression;
import org.codehaus.groovy.ast.expr.MethodCallExpression;
import org.codehaus.groovy.ast.expr.VariableExpression;
import org.codehaus.groovy.ast.stmt.BlockStatement;
import org.codehaus.groovy.ast.stmt.ExpressionStatement;
import org.codehaus.groovy.ast.stmt.Statement;
import org.codehaus.groovy.runtime.InvokerHelper;
import org.codehaus.groovy.transform.ImmutableASTTransformation;
import java.util.List;
import static java.util.stream.Collectors.toList;
import static org.apache.groovy.ast.tools.MethodCallUtils.appendS;
import static org.apache.groovy.ast.tools.MethodCallUtils.toStringX;
import static org.codehaus.groovy.ast.ClassHelper.make;
import static org.codehaus.groovy.ast.tools.GeneralUtils.args;
import static org.codehaus.groovy.ast.tools.GeneralUtils.block;
import static org.codehaus.groovy.ast.tools.GeneralUtils.callX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.constX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.ctorX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.declS;
import static org.codehaus.groovy.ast.tools.GeneralUtils.forS;
import static org.codehaus.groovy.ast.tools.GeneralUtils.ifS;
import static org.codehaus.groovy.ast.tools.GeneralUtils.listX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.localVarX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.notX;
import static org.codehaus.groovy.ast.tools.GeneralUtils.param;
import static org.codehaus.groovy.ast.tools.GeneralUtils.stmt;
import static org.codehaus.groovy.ast.tools.GeneralUtils.throwS;
import static org.codehaus.groovy.ast.tools.GeneralUtils.varX;
/**
* Utility class for working with ConstructorNodes
*/
public class ConstructorNodeUtils {
private static final ClassNode EXCEPTION = make(IllegalArgumentException.class);
private static final ClassNode IMMUTABLE_TYPE = make(ImmutableASTTransformation.class);
private static final ClassNode STRINGBUILDER_TYPE = make(StringBuilder.class);
private static final ClassNode INVOKER_TYPE = make(InvokerHelper.class);
private ConstructorNodeUtils() { }
/**
* Return the first statement from the constructor code if it is a call to super or this, otherwise null.
*
* @param code the code to check
* @return the first statement if a special call or null
*/
public static ConstructorCallExpression getFirstIfSpecialConstructorCall(final Statement code) {
if (code == null) return null;
if (code instanceof BlockStatement) {
final BlockStatement block = (BlockStatement) code;
final List statementList = block.getStatements();
if (statementList.isEmpty()) return null;
// handle blocks of blocks
return getFirstIfSpecialConstructorCall(statementList.get(0));
}
if (!(code instanceof ExpressionStatement)) return null;
Expression expression = ((ExpressionStatement) code).getExpression();
if (!(expression instanceof ConstructorCallExpression)) return null;
ConstructorCallExpression cce = (ConstructorCallExpression) expression;
if (cce.isSpecialCall()) return cce;
return null;
}
public static Statement checkPropNamesS(final VariableExpression namedArgs, final boolean pojo, final List props) {
if (!pojo) {
return stmt(callX(IMMUTABLE_TYPE, "checkPropNames", args(varX("this"), namedArgs)));
}
Expression validNames = localVarX("validNames", ClassHelper.LIST_TYPE);
Parameter name = param(ClassHelper.STRING_TYPE, "arg");
MethodCallExpression names = callX(namedArgs, "keySet");
names.setImplicitThis(false);
MethodCallExpression isNameValid = callX(validNames, "contains", varX(name));
isNameValid.setImplicitThis(false);
Expression sb = localVarX("sb");
Expression toString = pojo ? toStringX(sb) : callX(INVOKER_TYPE, "toString", sb);
Statement errorBlock = block(
declS(sb, ctorX(STRINGBUILDER_TYPE)),
appendS(sb, constX("Unknown named argument: ")),
appendS(sb, varX(name)),
throwS(ctorX(EXCEPTION, toString))
);
return block(
declS(validNames, listX(props.stream().map(p -> constX(p.getName())).collect(toList()))),
forS(name, names, ifS(notX(isNameValid), errorBlock))
);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy