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

org.codenarc.rule.unnecessary.UnnecessaryTernaryExpressionRule.groovy Maven / Gradle / Ivy

There is a newer version: 3.5.0-groovy-4.0
Show newest version
/*
 * Copyright 2010 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
 *
 *      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.codenarc.rule.unnecessary

import org.codenarc.rule.AbstractAstVisitor
import org.codenarc.rule.AbstractAstVisitorRule
import org.codenarc.util.AstUtil
import org.codehaus.groovy.ast.expr.*

/**
 * Rule that checks for ternary expressions where the conditional expression always evaluates to
 * a boolean and the true and false expressions are merely returning true and
 * false constants. These cases can be replaced by a simple boolean expression.
 * Examples include:
 * 
    *
  • boolean result = x==99 ? true : false - can be replaced by boolean result = x==99
  • *
  • boolean result = x && y ? true : false - can be replaced by boolean result = x && y
  • *
  • def result = x||y ? false : true - can be replaced by def result = !(x||y)
  • *
  • boolean result = x >= 1 ? true: false - can be replaced by boolean result = x >= 1
  • *
  • boolean result = x < 99 ? Boolean.TRUE : Boolean.FALSE - can be replaced by boolean result = x < 99
  • *
  • def result = !x ? true : false - can be replaced by def result = !x
  • *
* * The rule also checks for ternary expressions where the true and false expressions are the same constant or * variable expression. Examples include: *
    *
  • def result = x ? '123' : '123' - can be replaced by def result = '123'
  • *
  • def result = x ? null : null - can be replaced by def result = null
  • *
  • def result = x ? 23 : 23 - can be replaced by def result = 23
  • *
  • def result = x ? MAX_VALUE : MAX_VALUE - can be replaced by def result = MAX_VALUE
  • *
* * @author Chris Mair */ class UnnecessaryTernaryExpressionRule extends AbstractAstVisitorRule { String name = 'UnnecessaryTernaryExpression' int priority = 3 Class astVisitorClass = UnnecessaryTernaryExpressionAstVisitor } class UnnecessaryTernaryExpressionAstVisitor extends AbstractAstVisitor { private static final List BOOLEAN_COMPARISON_OPERATIONS = ['<', '>', '>=', '<=', '==', '!=', '==~'] private static final List BOOLEAN_LOGIC_OPERATIONS = ['&&', '||'] @Override void visitTernaryExpression(TernaryExpression ternaryExpression) { if (isFirstVisit(ternaryExpression)) { def trueExpression = ternaryExpression.trueExpression def falseExpression = ternaryExpression.falseExpression def booleanExpression = ternaryExpression.booleanExpression if (areBothTheSame(trueExpression, falseExpression) || (isBooleanConditionalExpression(booleanExpression) && areTrueAndFalse(trueExpression, falseExpression))) { addViolation(ternaryExpression, 'The ternary expression is useless or nonsensical') } } super.visitTernaryExpression(ternaryExpression) } private boolean isBooleanConditionalExpression(BooleanExpression conditionalExpression) { def expression = conditionalExpression.expression if (expression instanceof NotExpression) { return true } if (expression instanceof BinaryExpression && isOperationThatReturnsABoolean(expression)) { return true } false } private boolean isOperationThatReturnsABoolean(BinaryExpression expression) { def operationName = expression.operation.text if (operationName in BOOLEAN_COMPARISON_OPERATIONS) { return true } if (operationName in BOOLEAN_LOGIC_OPERATIONS) { return true } false } private boolean areBothTheSame(Expression trueExpression, Expression falseExpression) { if (trueExpression instanceof ConstantExpression && falseExpression instanceof ConstantExpression && trueExpression.getValue() == falseExpression.getValue()) { return true } if (trueExpression instanceof VariableExpression && falseExpression instanceof VariableExpression && trueExpression.getName() == falseExpression.getName()) { return true } false } private boolean areTrueAndFalse(Expression trueExpression, Expression falseExpression) { (AstUtil.isTrue(trueExpression) && AstUtil.isFalse(falseExpression)) || (AstUtil.isFalse(trueExpression) && AstUtil.isTrue(falseExpression)) } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy