lombok.javac.handlers.ast.JavacMethod Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of lombok-pg Show documentation
Show all versions of lombok-pg Show documentation
lombok-pg is a collection of extensions to Project Lombok
/*
* Copyright © 2011 Philipp Eichhorn
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package lombok.javac.handlers.ast;
import static com.sun.tools.javac.code.Flags.*;
import static lombok.ast.AST.*;
import static lombok.ast.IMethod.ArgumentStyle.BOXED_TYPES;
import static lombok.ast.IMethod.ArgumentStyle.INCLUDE_ANNOTATIONS;
import static lombok.javac.handlers.Javac.*;
import static lombok.javac.handlers.ast.JavacASTUtil.boxedType;
import static lombok.javac.handlers.ast.JavacResolver.METHOD;
import java.util.ArrayList;
import java.util.List;
import lombok.AccessLevel;
import lombok.core.AST.Kind;
import lombok.core.util.As;
import lombok.core.util.Is;
import lombok.javac.JavacNode;
import lombok.javac.handlers.Javac;
import lombok.javac.handlers.replace.*;
import com.sun.tools.javac.code.Type;
import com.sun.tools.javac.tree.JCTree;
import com.sun.tools.javac.tree.JCTree.JCAssign;
import com.sun.tools.javac.tree.JCTree.JCAnnotation;
import com.sun.tools.javac.tree.JCTree.JCExpression;
import com.sun.tools.javac.tree.JCTree.JCExpressionStatement;
import com.sun.tools.javac.tree.JCTree.JCMethodDecl;
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
import com.sun.tools.javac.tree.JCTree.JCModifiers;
import com.sun.tools.javac.tree.JCTree.JCStatement;
import com.sun.tools.javac.tree.JCTree.JCTypeApply;
import com.sun.tools.javac.tree.JCTree.JCTypeParameter;
import com.sun.tools.javac.tree.JCTree.JCVariableDecl;
public final class JavacMethod implements lombok.ast.IMethod {
private final JavacNode methodNode;
private final JCTree source;
private final JavacASTMaker builder;
private JavacMethod(final JavacNode methodNode, final JCTree source) {
if (!(methodNode.get() instanceof JCMethodDecl)) {
throw new IllegalArgumentException();
}
this.methodNode = methodNode;
this.source = source;
builder = new JavacASTMaker(methodNode, source);
}
public T build(final lombok.ast.Node> node) {
return builder. build(node);
}
public T build(final lombok.ast.Node> node, final Class extectedType) {
return builder.build(node, extectedType);
}
public List build(final List extends lombok.ast.Node>> nodes) {
return builder.build(nodes);
}
public List build(final List extends lombok.ast.Node>> nodes, final Class extectedType) {
return builder.build(nodes, extectedType);
}
public lombok.ast.TypeRef returns() {
return isConstructor() ? null : Type(returnType());
}
public lombok.ast.TypeRef boxedReturns() {
return boxedType(returnType());
}
public boolean returns(final Class> clazz) {
return returns(clazz.getSimpleName());
}
public boolean returns(final String typeName) {
final JCExpression returnType = returnType();
if (returnType == null) return false;
final String type;
if (returnType instanceof JCTypeApply) {
type = ((JCTypeApply) returnType).clazz.toString();
} else {
type = returnType.toString();
}
return type.endsWith(typeName);
}
private JCExpression returnType() {
return isConstructor() ? null : get().restype;
}
public void replaceReturnType(final lombok.ast.TypeRef returnType) {
if (isConstructor()) return;
get().restype = build(returnType);
}
public void replaceReturns(final lombok.ast.Statement> replacement) {
new ReturnStatementReplaceVisitor(this, replacement).visit(get());
}
public void replaceVariableName(final String oldName, final String newName) {
new VariableNameReplaceVisitor(this, oldName, newName).visit(get());
}
public void forceQualifiedThis() {
new ThisReferenceReplaceVisitor(this, This(Type(surroundingType().name()))).visit(get());
}
public AccessLevel accessLevel() {
if ((get().mods.flags & PUBLIC) != 0) return AccessLevel.PUBLIC;
if ((get().mods.flags & PROTECTED) != 0) return AccessLevel.PROTECTED;
if ((get().mods.flags & PRIVATE) != 0) return AccessLevel.PRIVATE;
return AccessLevel.PACKAGE;
}
public boolean isSynchronized() {
return (get().mods.flags & SYNCHRONIZED) != 0;
}
public boolean isStatic() {
return (get().mods.flags & STATIC) != 0;
}
public boolean isConstructor() {
return "".equals(methodNode.getName());
}
public boolean isAbstract() {
return (get().mods.flags & ABSTRACT) != 0;
}
public boolean isEmpty() {
return (get().body == null) || get().body.stats.isEmpty();
}
public JCMethodDecl get() {
return (JCMethodDecl) methodNode.get();
}
public JavacNode node() {
return methodNode;
}
public JavacNode getAnnotation(final Class extends java.lang.annotation.Annotation> expectedType) {
return getAnnotation(expectedType.getName());
}
public JavacNode getAnnotation(final String typeName) {
JavacNode annotationNode = null;
for (JavacNode child : node().down()) {
if (child.getKind() != Kind.ANNOTATION) continue;
if (Javac.matchesType((JCAnnotation) child.get(), typeName)) {
annotationNode = child;
}
}
return annotationNode;
}
public boolean hasNonFinalArgument() {
for (JCVariableDecl param : get().params) {
if ((param.mods == null) || (param.mods.flags & FINAL) == 0) {
return true;
}
}
return false;
}
public boolean hasArguments() {
return !get().params.isEmpty();
}
public String name() {
return node().getName();
}
public void makePrivate() {
makePackagePrivate();
get().mods.flags |= PRIVATE;
}
public void makePackagePrivate() {
get().mods.flags &= ~(PRIVATE | PROTECTED | PUBLIC);
}
public void makeProtected() {
makePackagePrivate();
get().mods.flags |= PROTECTED;
}
public void makePublic() {
makePackagePrivate();
get().mods.flags |= PUBLIC;
}
public void replaceBody(final lombok.ast.Statement>... statements) {
replaceBody(As.list(statements));
}
public void replaceBody(final List> statements) {
replaceBody(Block().withStatements(statements));
}
public void replaceBody(final lombok.ast.Block body) {
final lombok.ast.Block bodyWithConstructorCall = new lombok.ast.Block();
if (!isEmpty()) {
final JCStatement suspect = get().body.stats.get(0);
if (isConstructorCall(suspect)) bodyWithConstructorCall.withStatement(Stat(suspect));
}
bodyWithConstructorCall.withStatements(body.getStatements());
get().body = builder.build(bodyWithConstructorCall);
addSuppressWarningsAll(get().mods, node(), get().pos);
}
public void rebuild() {
node().rebuild();
}
public JavacType surroundingType() {
return JavacType.typeOf(node(), source);
}
public List> statements() {
final List> methodStatements = new ArrayList>();
for (JCStatement statement : get().body.stats) {
if (isConstructorCall(statement)) continue;
methodStatements.add(Stat(statement));
}
return methodStatements;
}
private boolean isConstructorCall(final JCStatement supect) {
if (!(supect instanceof JCExpressionStatement)) return false;
final JCExpression supectExpression = ((JCExpressionStatement) supect).expr;
if (!(supectExpression instanceof JCMethodInvocation)) return false;
return Is.oneOf(((JCMethodInvocation) supectExpression).meth.toString(), "super", "this");
}
public List annotations() {
return annotations(get().mods);
}
private List annotations(final JCModifiers mods) {
final List annotations = new ArrayList();
for (JCAnnotation annotation : mods.annotations) {
Type type = METHOD.resolveMember(node(), annotation);
if (type.toString().startsWith("lombok.")) continue;
lombok.ast.Annotation ann = Annotation(Type(annotation.annotationType));
for (JCExpression arg : annotation.args) {
if (arg instanceof JCAssign) {
JCAssign assign = (JCAssign) arg;
ann.withValue(assign.lhs.toString(), Expr(assign.rhs));
} else {
ann.withValue(Expr(arg));
}
}
annotations.add(ann);
}
return annotations;
}
public List arguments(final ArgumentStyle... style) {
final List styles = As.list(style);
final List methodArguments = new ArrayList();
for (JCVariableDecl param : get().params) {
lombok.ast.TypeRef argType = styles.contains(BOXED_TYPES) ? boxedType(param.vartype) : Type(param.vartype);
lombok.ast.Argument arg = Arg(argType, As.string(param.name));
if (styles.contains(INCLUDE_ANNOTATIONS)) arg.withAnnotations(annotations(param.mods));
methodArguments.add(arg);
}
return methodArguments;
}
public List typeParameters() {
final List typeParameters = new ArrayList();
if (isConstructor()) return typeParameters;
for (JCTypeParameter typaram : get().typarams) {
final lombok.ast.TypeParam typeParam = TypeParam(As.string(typaram.name));
for (JCExpression expr : typaram.bounds) {
typeParam.withBound(Type(expr));
}
typeParameters.add(typeParam);
}
return typeParameters;
}
public List thrownExceptions() {
final List thrownExceptions = new ArrayList();
for (Object thrownException : get().thrown) {
thrownExceptions.add(Type(thrownException));
}
return thrownExceptions;
}
@Override
public String toString() {
return get().toString();
}
public static JavacMethod methodOf(final JavacNode node, final JCTree source) {
JavacNode methodNode = Javac.methodNodeOf(node);
return methodNode == null ? null : new JavacMethod(methodNode, source);
}
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy