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

spoon.refactoring.CtParameterRemoveRefactoring Maven / Gradle / Ivy

/*
 * SPDX-License-Identifier: (MIT OR CECILL-C)
 *
 * Copyright (C) 2006-2019 INRIA and contributors
 *
 * Spoon is available either under the terms of the MIT License (see LICENSE-MIT.txt) of the Cecill-C License (see LICENSE-CECILL-C.txt). You as the user are entitled to choose the terms under which to adopt Spoon.
 */
package spoon.refactoring;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

import spoon.SpoonException;
import spoon.reflect.code.CtAnnotationFieldAccess;
import spoon.reflect.code.CtArrayRead;
import spoon.reflect.code.CtExpression;
import spoon.reflect.code.CtFieldRead;
import spoon.reflect.code.CtInvocation;
import spoon.reflect.code.CtLiteral;
import spoon.reflect.code.CtNewArray;
import spoon.reflect.code.CtThisAccess;
import spoon.reflect.code.CtVariableRead;
import spoon.reflect.declaration.CtElement;
import spoon.reflect.declaration.CtExecutable;
import spoon.reflect.declaration.CtParameter;
import spoon.reflect.reference.CtExecutableReference;
import spoon.reflect.reference.CtParameterReference;
import spoon.reflect.visitor.CtAbstractVisitor;
import spoon.reflect.visitor.chain.CtConsumer;
import spoon.reflect.visitor.filter.AllMethodsSameSignatureFunction;
import spoon.reflect.visitor.filter.ExecutableReferenceFilter;
import spoon.reflect.visitor.filter.ParameterReferenceFunction;

/**
 * Removes target {@link CtParameter} from the parent target {@link CtExecutable}
 * and from all overriding/overridden methods of related type hierarchies
 * and from all lambda expressions (if any) implementing the modified interface.
 * It removes arguments from all invocations of refactored executables too.
* * Before the refactoring is started it checks that: *
    *
  • to be removed parameter is NOT used in any refactored implementation *
  • to be removed argument contains read only expression, which can be safely removed *
* If one of the validation constraints fails, then {@link RefactoringException} is thrown and nothing is changed. * You can override `#create*Issue(...)` methods to handle such exceptions individually. *
*/ public class CtParameterRemoveRefactoring implements CtRefactoring { private CtParameter target; private int parameterIndex; /** * List of all {@link CtExecutable}s whose parameter has to be removed */ private List> targetExecutables; /** * List of all {@link CtInvocation}s whose argument has to be removed */ private List> targetInvocations; public CtParameterRemoveRefactoring() { } /** * @return the {@link CtParameter} which has to be removed by this refactoring function */ public CtParameter getTarget() { return target; } /** * @param target the {@link CtParameter} which has to be removed by this refactoring function * @return this to support fluent API */ public CtParameterRemoveRefactoring setTarget(CtParameter target) { if (this.target == target) { return this; } this.target = target; this.parameterIndex = target.getParent().getParameters().indexOf(target); targetExecutables = null; targetInvocations = null; return this; } /** * @return computes and returns all executables, which will be modified by this refactoring */ public List> getTargetExecutables() { if (targetExecutables == null) { computeAllExecutables(); } return targetExecutables; } /** * @return computes and returns all invocations, which will be modified by this refactoring */ public List> getTargetInvocations() { if (targetInvocations == null) { computeAllInvocations(); } return targetInvocations; } @Override public void refactor() { if (getTarget() == null) { throw new SpoonException("The target of refactoring is not defined"); } detectIssues(); refactorNoCheck(); } /** * validates whether this refactoring can be done without changing behavior of the refactored code. */ protected void detectIssues() { checkAllExecutables(); checkAllInvocations(); } /** * search for all methods and lambdas which has to be refactored together with target method */ private void computeAllExecutables() { if (getTarget() == null) { throw new SpoonException("The target of refactoring is not defined"); } final List> executables = new ArrayList<>(); CtExecutable targetExecutable = target.getParent(); //all the executables, which belongs to same inheritance tree executables.add(targetExecutable); targetExecutable.map(new AllMethodsSameSignatureFunction()).forEach(new CtConsumer>() { @Override public void accept(CtExecutable executable) { executables.add(executable); } }); targetExecutables = Collections.unmodifiableList(executables); } /** * search for all methods and lambdas which has to be refactored together with target method */ private void computeAllInvocations() { ExecutableReferenceFilter execRefFilter = new ExecutableReferenceFilter(); for (CtExecutable exec : getTargetExecutables()) { execRefFilter.addExecutable(exec); } //all the invocations, which belongs to same inheritance tree final List> invocations = new ArrayList<>(); target.getFactory().getModel().filterChildren(execRefFilter).forEach(new CtConsumer>() { @Override public void accept(CtExecutableReference t) { CtElement parent = t.getParent(); if (parent instanceof CtInvocation) { invocations.add((CtInvocation) parent); } //else ignore other hits, which are not in context of invocation } }); targetInvocations = Collections.unmodifiableList(invocations); } private void checkAllExecutables() { for (CtExecutable executable : getTargetExecutables()) { checkExecutable(executable); } } private void checkExecutable(CtExecutable executable) { final CtParameter toBeRemovedParam = executable.getParameters().get(this.parameterIndex); toBeRemovedParam.map(new ParameterReferenceFunction()).forEach(new CtConsumer>() { @Override public void accept(CtParameterReference paramRef) { //some parameter uses are acceptable //e.g. parameter in invocation of super of method, which is going to be removed too. if (isAllowedParameterUsage(paramRef)) { return; } createParameterUsedIssue(toBeRemovedParam, paramRef); } }); } private void checkAllInvocations() { for (CtInvocation invocation : getTargetInvocations()) { checkInvocation(invocation); } } private void checkInvocation(CtInvocation invocation) { final CtExpression toBeRemovedExpression = invocation.getArguments().get(this.parameterIndex); if (!canRemoveExpression(toBeRemovedExpression)) { createExpressionCannotBeRemovedIssue(invocation, toBeRemovedExpression); } } /** * Detects whether found usage of removed parameter is acceptable * @param paramRef the found reference to * @return true if it is allowed parameter use */ protected boolean isAllowedParameterUsage(CtParameterReference paramRef) { return isRemovedParamOfRefactoredInvocation(paramRef); } /** * Detects whether `toBeRemovedExpression` can be safely removed during the refactoring * * @param toBeRemovedExpression the {@link CtExpression}, which will be removed by this refactoring * @return true if the expression used to deliver argument of removed parameter can be removed * false if cannot be removed and this refactoring has to be avoided. */ protected boolean canRemoveExpression(CtExpression toBeRemovedExpression) { class Context { boolean canBeRemoved = false; } final Context context = new Context(); toBeRemovedExpression.accept(new CtAbstractVisitor() { @Override public void visitCtVariableRead(CtVariableRead variableRead) { context.canBeRemoved = true; } @Override public void visitCtArrayRead(CtArrayRead arrayRead) { context.canBeRemoved = true; } @Override public void visitCtFieldRead(CtFieldRead fieldRead) { context.canBeRemoved = true; } @Override public void visitCtParameterReference(CtParameterReference reference) { context.canBeRemoved = true; } @Override public void visitCtLiteral(CtLiteral literal) { context.canBeRemoved = true; } @Override public void visitCtNewArray(CtNewArray newArray) { context.canBeRemoved = true; } @Override public void visitCtAnnotationFieldAccess(CtAnnotationFieldAccess annotationFieldAccess) { context.canBeRemoved = true; } @Override public void visitCtThisAccess(CtThisAccess thisAccess) { context.canBeRemoved = true; } //There are more expression which is save to remove. Including tree of unary/binary operators, conditional, etc. //It would be good to have a Filter, which matches read only expressions }); return context.canBeRemoved; } protected boolean isRemovedParamOfRefactoredInvocation(CtParameterReference paramRef) { CtInvocation invocation = paramRef.getParent(CtInvocation.class); if (invocation == null) { return false; } return getTargetInvocations().contains(invocation); } /** * Override this method to get access to details about this refactoring issue * @param usedParameter to be removed parameter, which is used by `parameterUsage` * @param parameterUsage the usage of parameter, which avoids it's remove */ protected void createParameterUsedIssue(CtParameter usedParameter, CtParameterReference parameterUsage) { throw new RefactoringException("The parameter " + usedParameter.getSimpleName() + " cannot be removed because it is used (" + parameterUsage.getPosition() + ")"); } /** * Override this method to get access to details about this refactoring issue. * @param toBeRemovedExpression is the expression which delivers value for the argument of the removed parameter, * where {@link #canRemoveExpression(CtExpression)} returned false. */ protected void createExpressionCannotBeRemovedIssue(CtInvocation invocation, CtExpression toBeRemovedExpression) { throw new RefactoringException("The expression " + toBeRemovedExpression + ", which creates argument of the to be removed parameter in invocation " + invocation + " cannot be removed." + " Override method `canRemoveExpression` to customize this behavior."); } protected void refactorNoCheck() { removeInvocationArguments(); removeMethodParameters(); } protected void removeInvocationArguments() { List> invocations = getTargetInvocations(); for (CtInvocation invocation : invocations) { removeInvocationArgument(invocation); } } protected void removeInvocationArgument(CtInvocation invocation) { invocation.removeArgument(invocation.getArguments().get(this.parameterIndex)); } protected void removeMethodParameters() { List> executables = getTargetExecutables(); for (CtExecutable executable : executables) { removeParameter(executable); } } protected void removeParameter(CtExecutable executable) { executable.removeParameter(executable.getParameters().get(this.parameterIndex)); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy