2014-08-01 11:09:40 -07:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
|
|
*
|
|
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
|
|
* accompanied this code).
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License version
|
|
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*
|
|
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
|
|
* questions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package crules;
|
|
|
|
|
2015-03-05 13:10:49 +00:00
|
|
|
import com.sun.source.tree.LambdaExpressionTree.BodyKind;
|
2014-08-01 11:09:40 -07:00
|
|
|
import com.sun.source.util.JavacTask;
|
|
|
|
import com.sun.source.util.TaskEvent.Kind;
|
2015-03-05 13:10:49 +00:00
|
|
|
import com.sun.tools.javac.code.Kinds;
|
2014-08-01 11:09:40 -07:00
|
|
|
import com.sun.tools.javac.code.Symbol;
|
2015-03-05 13:10:49 +00:00
|
|
|
import com.sun.tools.javac.code.Type;
|
2014-08-01 11:09:40 -07:00
|
|
|
import com.sun.tools.javac.tree.JCTree.JCExpression;
|
2015-03-05 13:10:49 +00:00
|
|
|
import com.sun.tools.javac.tree.JCTree.JCLambda;
|
2014-08-01 11:09:40 -07:00
|
|
|
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
|
|
|
|
import com.sun.tools.javac.tree.JCTree.Tag;
|
|
|
|
import com.sun.tools.javac.tree.TreeInfo;
|
|
|
|
import com.sun.tools.javac.tree.TreeScanner;
|
|
|
|
import com.sun.tools.javac.util.Assert;
|
|
|
|
|
2016-05-16 15:41:20 +02:00
|
|
|
/**This analyzer guards against complex messages (i.e. those that use string concatenation) passed
|
|
|
|
* to various Assert.check methods.
|
|
|
|
*/
|
2014-08-01 11:09:40 -07:00
|
|
|
public class AssertCheckAnalyzer extends AbstractCodingRulesAnalyzer {
|
|
|
|
|
2015-03-05 13:10:49 +00:00
|
|
|
enum AssertOverloadKind {
|
|
|
|
EAGER("crules.should.not.use.eager.string.evaluation"),
|
|
|
|
LAZY("crules.should.not.use.lazy.string.evaluation"),
|
|
|
|
NONE(null);
|
|
|
|
|
|
|
|
String errKey;
|
|
|
|
|
|
|
|
AssertOverloadKind(String errKey) {
|
|
|
|
this.errKey = errKey;
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean simpleArgExpected() {
|
|
|
|
return this == AssertOverloadKind.EAGER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-01 11:09:40 -07:00
|
|
|
public AssertCheckAnalyzer(JavacTask task) {
|
|
|
|
super(task);
|
|
|
|
treeVisitor = new AssertCheckVisitor();
|
|
|
|
eventKind = Kind.ANALYZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
class AssertCheckVisitor extends TreeScanner {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void visitApply(JCMethodInvocation tree) {
|
2015-03-05 13:10:49 +00:00
|
|
|
Symbol m = TreeInfo.symbolFor(tree);
|
|
|
|
AssertOverloadKind ak = assertOverloadKind(m);
|
|
|
|
if (ak != AssertOverloadKind.NONE &&
|
|
|
|
!m.name.contentEquals("error")) {
|
2014-08-01 11:09:40 -07:00
|
|
|
JCExpression lastParam = tree.args.last();
|
2015-03-05 13:10:49 +00:00
|
|
|
if (isSimpleStringArg(lastParam) != ak.simpleArgExpected()) {
|
|
|
|
messages.error(lastParam, ak.errKey);
|
2014-08-01 11:09:40 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
super.visitApply(tree);
|
|
|
|
}
|
|
|
|
|
2015-03-05 13:10:49 +00:00
|
|
|
AssertOverloadKind assertOverloadKind(Symbol method) {
|
|
|
|
if (method == null ||
|
|
|
|
!method.owner.getQualifiedName().contentEquals(Assert.class.getName()) ||
|
|
|
|
method.type.getParameterTypes().tail == null) {
|
|
|
|
return AssertOverloadKind.NONE;
|
|
|
|
}
|
|
|
|
Type formal = method.type.getParameterTypes().last();
|
|
|
|
if (types.isSameType(formal, syms.stringType)) {
|
|
|
|
return AssertOverloadKind.EAGER;
|
|
|
|
} else if (types.isSameType(types.erasure(formal), types.erasure(syms.supplierType))) {
|
|
|
|
return AssertOverloadKind.LAZY;
|
|
|
|
} else {
|
|
|
|
return AssertOverloadKind.NONE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean isSimpleStringArg(JCExpression e) {
|
|
|
|
switch (e.getTag()) {
|
|
|
|
case LAMBDA:
|
|
|
|
JCLambda lambda = (JCLambda)e;
|
|
|
|
return (lambda.getBodyKind() == BodyKind.EXPRESSION) &&
|
|
|
|
isSimpleStringArg((JCExpression)lambda.body);
|
|
|
|
default:
|
|
|
|
Symbol argSym = TreeInfo.symbolFor(e);
|
|
|
|
return (e.type.constValue() != null ||
|
|
|
|
(argSym != null && argSym.kind == Kinds.Kind.VAR));
|
|
|
|
}
|
|
|
|
}
|
2014-08-01 11:09:40 -07:00
|
|
|
}
|
|
|
|
}
|