blob: dbb4e491fd333a122e75acdd80e3ffd0a73c8901 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2021 Fabrice TIERCELIN and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Fabrice TIERCELIN - initial API and implementation
*******************************************************************************/
package org.eclipse.jdt.internal.ui.fix;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.text.edits.TextEditGroup;
import org.eclipse.jdt.core.ICompilationUnit;
import org.eclipse.jdt.core.dom.ASTVisitor;
import org.eclipse.jdt.core.dom.CompilationUnit;
import org.eclipse.jdt.core.dom.IfStatement;
import org.eclipse.jdt.core.dom.rewrite.ASTRewrite;
import org.eclipse.jdt.internal.corext.dom.ASTNodes;
import org.eclipse.jdt.internal.corext.fix.CleanUpConstants;
import org.eclipse.jdt.internal.corext.fix.CompilationUnitRewriteOperationsFix;
import org.eclipse.jdt.internal.corext.fix.CompilationUnitRewriteOperationsFix.CompilationUnitRewriteOperation;
import org.eclipse.jdt.internal.corext.fix.LinkedProposalModel;
import org.eclipse.jdt.internal.corext.refactoring.structure.CompilationUnitRewrite;
import org.eclipse.jdt.ui.cleanup.CleanUpRequirements;
import org.eclipse.jdt.ui.cleanup.ICleanUpFix;
import org.eclipse.jdt.ui.text.java.IProblemLocation;
/**
* A fix that detects two <code>if</code> conditions that are identical and removes the second one:
* <ul>
* <li>The conditions should be passive,</li>
* <li>No exceptions should be awaited,</li>
* <li>It should not create unreachable code below the <code>if</code> statement which would create a compile error, that is to say the case where
* only the removed block doesn't fall through,
* all the other cases fall through,
* there is an <code>else</code> clause (not only <code>if</code>/<code>else</code> clauses)
* and a statement after the control workflow.</li>
* </ul>
*/
public class UnreachableBlockCleanUp extends AbstractMultiFix {
public UnreachableBlockCleanUp() {
this(Collections.emptyMap());
}
public UnreachableBlockCleanUp(final Map<String, String> options) {
super(options);
}
@Override
public CleanUpRequirements getRequirements() {
boolean requireAST= isEnabled(CleanUpConstants.UNREACHABLE_BLOCK);
return new CleanUpRequirements(requireAST, false, false, null);
}
@Override
public String[] getStepDescriptions() {
if (isEnabled(CleanUpConstants.UNREACHABLE_BLOCK)) {
return new String[] { MultiFixMessages.UnreachableBlockCleanUp_description };
}
return new String[0];
}
@Override
public String getPreview() {
StringBuilder bld= new StringBuilder();
bld.append("if (isValid && isFound) {\n"); //$NON-NLS-1$
bld.append(" return 0;\n"); //$NON-NLS-1$
if (!isEnabled(CleanUpConstants.UNREACHABLE_BLOCK)) {
bld.append("} else if (isFound && isValid) {\n"); //$NON-NLS-1$
bld.append(" return 1;\n"); //$NON-NLS-1$
}
bld.append("}\n"); //$NON-NLS-1$
if (isEnabled(CleanUpConstants.UNREACHABLE_BLOCK)) {
bld.append("\n\n"); //$NON-NLS-1$
}
return bld.toString();
}
@Override
protected ICleanUpFix createFix(final CompilationUnit unit) throws CoreException {
if (!isEnabled(CleanUpConstants.UNREACHABLE_BLOCK)) {
return null;
}
final List<CompilationUnitRewriteOperation> rewriteOperations= new ArrayList<>();
unit.accept(new ASTVisitor() {
@Override
public boolean visit(final IfStatement visited) {
if (!ASTNodes.isExceptionExpected(visited)
&& ASTNodes.isPassive(visited.getExpression())) {
boolean hasNoUnreachableCodeBelow= ASTNodes.getNextStatement(visited) == null
|| !ASTNodes.fallsThrough(visited.getThenStatement());
IfStatement duplicateIf= visited;
do {
duplicateIf= ASTNodes.as(duplicateIf.getElseStatement(), IfStatement.class);
if (duplicateIf == null
|| !ASTNodes.isPassive(duplicateIf.getExpression())) {
return true;
}
if ((hasNoUnreachableCodeBelow
|| duplicateIf.getElseStatement() == null
|| ASTNodes.fallsThrough(duplicateIf.getThenStatement())
|| !ASTNodes.fallsThrough(duplicateIf.getElseStatement()))
&& ASTNodes.isPassiveWithoutFallingThrough(duplicateIf.getExpression())
&& ASTNodes.match(visited.getExpression(), duplicateIf.getExpression())) {
rewriteOperations.add(new UnreachableBlockOperation(duplicateIf));
return false;
}
hasNoUnreachableCodeBelow|= !ASTNodes.fallsThrough(duplicateIf.getThenStatement());
} while (true);
}
return true;
}
});
if (rewriteOperations.isEmpty()) {
return null;
}
return new CompilationUnitRewriteOperationsFix(MultiFixMessages.UnreachableBlockCleanUp_description, unit,
rewriteOperations.toArray(new CompilationUnitRewriteOperation[0]));
}
@Override
public boolean canFix(final ICompilationUnit compilationUnit, final IProblemLocation problem) {
return false;
}
@Override
protected ICleanUpFix createFix(final CompilationUnit unit, final IProblemLocation[] problems) throws CoreException {
return null;
}
private static class UnreachableBlockOperation extends CompilationUnitRewriteOperation {
private final IfStatement duplicateIf;
public UnreachableBlockOperation(final IfStatement duplicateIf) {
this.duplicateIf= duplicateIf;
}
@Override
public void rewriteAST(final CompilationUnitRewrite cuRewrite, final LinkedProposalModel linkedModel) throws CoreException {
ASTRewrite rewrite= cuRewrite.getASTRewrite();
TextEditGroup group= createTextEditGroup(MultiFixMessages.UnreachableBlockCleanUp_description, cuRewrite);
if (duplicateIf.getElseStatement() == null) {
rewrite.remove(duplicateIf, group);
} else {
ASTNodes.replaceButKeepComment(rewrite, duplicateIf, ASTNodes.createMoveTarget(rewrite, duplicateIf.getElseStatement()), group);
}
}
}
}