8073645: Add lambda-based lazy eval versions of Assert.check methods
Enhance Assert so that lazy string computation can occurr where needed; enhance static roding rule checkers to make sure the right version of the method is called. Reviewed-by: jlahoda
This commit is contained in:
parent
105275fb87
commit
1114c26925
@ -28,6 +28,10 @@
|
|||||||
</sequential>
|
</sequential>
|
||||||
</macrodef>
|
</macrodef>
|
||||||
|
|
||||||
|
<target name="crules" depends="build-all-tools,-def-jtreg">
|
||||||
|
<jtreg-tool name="all" tests="tools/all/RunCodingRules.java"/>
|
||||||
|
</target>
|
||||||
|
|
||||||
<target name="post-make" depends="clean, build-all-tools"/>
|
<target name="post-make" depends="clean, build-all-tools"/>
|
||||||
|
|
||||||
<target name="jtreg-debug" depends="build-all-tools,-def-jtreg">
|
<target name="jtreg-debug" depends="build-all-tools,-def-jtreg">
|
||||||
|
@ -156,7 +156,7 @@
|
|||||||
<filter targetName="clean" isVisible="true" />
|
<filter targetName="clean" isVisible="true" />
|
||||||
<filter targetName="jtreg" isVisible="true" />
|
<filter targetName="jtreg" isVisible="true" />
|
||||||
<filter targetName="jtreg-debug" isVisible="true" />
|
<filter targetName="jtreg-debug" isVisible="true" />
|
||||||
<filter targetName="checkstyle" isVisible="true" />
|
<filter targetName="crules" isVisible="true" />
|
||||||
</targetFilters>
|
</targetFilters>
|
||||||
<viewClosedWhenNoErrors value="true" />
|
<viewClosedWhenNoErrors value="true" />
|
||||||
<expanded value="false" />
|
<expanded value="false" />
|
||||||
|
@ -5,7 +5,31 @@
|
|||||||
import com.sun.tools.javac.util.Assert;
|
import com.sun.tools.javac.util.Assert;
|
||||||
|
|
||||||
public class Test {
|
public class Test {
|
||||||
public void check(String value) {
|
|
||||||
Assert.check(value.trim().length() > 0, "value=" + value);
|
String v;
|
||||||
|
|
||||||
|
public void check1(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, "value=" + value); //fail
|
||||||
|
}
|
||||||
|
public void check2(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, "value=" + "value"); //ok
|
||||||
|
}
|
||||||
|
public void check3(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, () -> "value=" + value); //ok
|
||||||
|
}
|
||||||
|
public void check4(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, value); //ok
|
||||||
|
}
|
||||||
|
public void check5(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, v); //ok
|
||||||
|
}
|
||||||
|
public void check6(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, () -> "value=" + "value"); //fail
|
||||||
|
}
|
||||||
|
public void check7(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, () -> value); //fail
|
||||||
|
}
|
||||||
|
public void check8(String value) {
|
||||||
|
Assert.check(value.trim().length() > 0, () -> v); //fail
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,2 +1,5 @@
|
|||||||
Test.java:9:21: compiler.err.proc.messager: compiler.misc.crules.should.not.use.string.concatenation
|
Test.java:12:58: compiler.err.proc.messager: compiler.misc.crules.should.not.use.eager.string.evaluation
|
||||||
1 error
|
Test.java:27:49: compiler.err.proc.messager: compiler.misc.crules.should.not.use.lazy.string.evaluation
|
||||||
|
Test.java:30:49: compiler.err.proc.messager: compiler.misc.crules.should.not.use.lazy.string.evaluation
|
||||||
|
Test.java:33:49: compiler.err.proc.messager: compiler.misc.crules.should.not.use.lazy.string.evaluation
|
||||||
|
4 errors
|
||||||
|
@ -23,10 +23,14 @@
|
|||||||
|
|
||||||
package crules;
|
package crules;
|
||||||
|
|
||||||
|
import com.sun.source.tree.LambdaExpressionTree.BodyKind;
|
||||||
import com.sun.source.util.JavacTask;
|
import com.sun.source.util.JavacTask;
|
||||||
import com.sun.source.util.TaskEvent.Kind;
|
import com.sun.source.util.TaskEvent.Kind;
|
||||||
|
import com.sun.tools.javac.code.Kinds;
|
||||||
import com.sun.tools.javac.code.Symbol;
|
import com.sun.tools.javac.code.Symbol;
|
||||||
|
import com.sun.tools.javac.code.Type;
|
||||||
import com.sun.tools.javac.tree.JCTree.JCExpression;
|
import com.sun.tools.javac.tree.JCTree.JCExpression;
|
||||||
|
import com.sun.tools.javac.tree.JCTree.JCLambda;
|
||||||
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
|
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
|
||||||
import com.sun.tools.javac.tree.JCTree.Tag;
|
import com.sun.tools.javac.tree.JCTree.Tag;
|
||||||
import com.sun.tools.javac.tree.TreeInfo;
|
import com.sun.tools.javac.tree.TreeInfo;
|
||||||
@ -35,6 +39,22 @@ import com.sun.tools.javac.util.Assert;
|
|||||||
|
|
||||||
public class AssertCheckAnalyzer extends AbstractCodingRulesAnalyzer {
|
public class AssertCheckAnalyzer extends AbstractCodingRulesAnalyzer {
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public AssertCheckAnalyzer(JavacTask task) {
|
public AssertCheckAnalyzer(JavacTask task) {
|
||||||
super(task);
|
super(task);
|
||||||
treeVisitor = new AssertCheckVisitor();
|
treeVisitor = new AssertCheckVisitor();
|
||||||
@ -45,20 +65,46 @@ public class AssertCheckAnalyzer extends AbstractCodingRulesAnalyzer {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visitApply(JCMethodInvocation tree) {
|
public void visitApply(JCMethodInvocation tree) {
|
||||||
Symbol method = TreeInfo.symbolFor(tree);
|
Symbol m = TreeInfo.symbolFor(tree);
|
||||||
if (method != null &&
|
AssertOverloadKind ak = assertOverloadKind(m);
|
||||||
method.owner.getQualifiedName().contentEquals(Assert.class.getName()) &&
|
if (ak != AssertOverloadKind.NONE &&
|
||||||
!method.name.contentEquals("error")) {
|
!m.name.contentEquals("error")) {
|
||||||
JCExpression lastParam = tree.args.last();
|
JCExpression lastParam = tree.args.last();
|
||||||
if (lastParam != null &&
|
if (isSimpleStringArg(lastParam) != ak.simpleArgExpected()) {
|
||||||
lastParam.type.tsym == syms.stringType.tsym &&
|
messages.error(lastParam, ak.errKey);
|
||||||
lastParam.hasTag(Tag.PLUS)) {
|
|
||||||
messages.error(tree, "crules.should.not.use.string.concatenation");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
super.visitApply(tree);
|
super.visitApply(tree);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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));
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -26,8 +26,10 @@
|
|||||||
# 0: symbol
|
# 0: symbol
|
||||||
crules.err.var.must.be.final=\
|
crules.err.var.must.be.final=\
|
||||||
Static variable {0} must be final
|
Static variable {0} must be final
|
||||||
crules.should.not.use.string.concatenation=\
|
crules.should.not.use.eager.string.evaluation=\
|
||||||
Should not use string concatenation.
|
Should not use eager string evaluation. Use lazy version instead.
|
||||||
|
crules.should.not.use.lazy.string.evaluation=\
|
||||||
|
Should not use eager lazy evaluation. Use eager version instead.
|
||||||
crules.no.defined.by=\
|
crules.no.defined.by=\
|
||||||
This method implements a public API method, and should be marked with @DefinedBy.
|
This method implements a public API method, and should be marked with @DefinedBy.
|
||||||
crules.wrong.defined.by=\
|
crules.wrong.defined.by=\
|
||||||
|
@ -184,6 +184,7 @@ public class Symtab {
|
|||||||
public final Type retentionType;
|
public final Type retentionType;
|
||||||
public final Type deprecatedType;
|
public final Type deprecatedType;
|
||||||
public final Type suppressWarningsType;
|
public final Type suppressWarningsType;
|
||||||
|
public final Type supplierType;
|
||||||
public final Type inheritedType;
|
public final Type inheritedType;
|
||||||
public final Type profileType;
|
public final Type profileType;
|
||||||
public final Type proprietaryType;
|
public final Type proprietaryType;
|
||||||
@ -449,6 +450,7 @@ public class Symtab {
|
|||||||
retentionType = enterClass("java.lang.annotation.Retention");
|
retentionType = enterClass("java.lang.annotation.Retention");
|
||||||
deprecatedType = enterClass("java.lang.Deprecated");
|
deprecatedType = enterClass("java.lang.Deprecated");
|
||||||
suppressWarningsType = enterClass("java.lang.SuppressWarnings");
|
suppressWarningsType = enterClass("java.lang.SuppressWarnings");
|
||||||
|
supplierType = enterClass("java.util.function.Supplier");
|
||||||
inheritedType = enterClass("java.lang.annotation.Inherited");
|
inheritedType = enterClass("java.lang.annotation.Inherited");
|
||||||
repeatableType = enterClass("java.lang.annotation.Repeatable");
|
repeatableType = enterClass("java.lang.annotation.Repeatable");
|
||||||
documentedType = enterClass("java.lang.annotation.Documented");
|
documentedType = enterClass("java.lang.annotation.Documented");
|
||||||
|
@ -129,7 +129,7 @@ public class TreeMaker implements JCTree.Factory {
|
|||||||
|| node instanceof JCErroneous
|
|| node instanceof JCErroneous
|
||||||
|| (node instanceof JCExpressionStatement
|
|| (node instanceof JCExpressionStatement
|
||||||
&& ((JCExpressionStatement)node).expr instanceof JCErroneous),
|
&& ((JCExpressionStatement)node).expr instanceof JCErroneous),
|
||||||
node.getClass().getSimpleName());
|
() -> node.getClass().getSimpleName());
|
||||||
JCCompilationUnit tree = new JCCompilationUnit(defs);
|
JCCompilationUnit tree = new JCCompilationUnit(defs);
|
||||||
tree.pos = pos;
|
tree.pos = pos;
|
||||||
return tree;
|
return tree;
|
||||||
|
@ -25,6 +25,8 @@
|
|||||||
|
|
||||||
package com.sun.tools.javac.util;
|
package com.sun.tools.javac.util;
|
||||||
|
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simple facility for unconditional assertions.
|
* Simple facility for unconditional assertions.
|
||||||
* The methods in this class are described in terms of equivalent assert
|
* The methods in this class are described in terms of equivalent assert
|
||||||
@ -93,6 +95,15 @@ public class Assert {
|
|||||||
error(msg);
|
error(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Equivalent to
|
||||||
|
* assert cond : msg.get();
|
||||||
|
* Note: message string is computed lazily.
|
||||||
|
*/
|
||||||
|
public static void check(boolean cond, Supplier<String> msg) {
|
||||||
|
if (!cond)
|
||||||
|
error(msg.get());
|
||||||
|
}
|
||||||
|
|
||||||
/** Equivalent to
|
/** Equivalent to
|
||||||
* assert (o == null) : value;
|
* assert (o == null) : value;
|
||||||
*/
|
*/
|
||||||
@ -109,6 +120,15 @@ public class Assert {
|
|||||||
error(msg);
|
error(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Equivalent to
|
||||||
|
* assert (o == null) : msg.get();
|
||||||
|
* Note: message string is computed lazily.
|
||||||
|
*/
|
||||||
|
public static void checkNull(Object o, Supplier<String> msg) {
|
||||||
|
if (o != null)
|
||||||
|
error(msg.get());
|
||||||
|
}
|
||||||
|
|
||||||
/** Equivalent to
|
/** Equivalent to
|
||||||
* assert (o != null) : msg;
|
* assert (o != null) : msg;
|
||||||
*/
|
*/
|
||||||
@ -118,6 +138,16 @@ public class Assert {
|
|||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Equivalent to
|
||||||
|
* assert (o != null) : msg.get();
|
||||||
|
* Note: message string is computed lazily.
|
||||||
|
*/
|
||||||
|
public static <T> T checkNonNull(T t, Supplier<String> msg) {
|
||||||
|
if (t == null)
|
||||||
|
error(msg.get());
|
||||||
|
return t;
|
||||||
|
}
|
||||||
|
|
||||||
/** Equivalent to
|
/** Equivalent to
|
||||||
* assert false;
|
* assert false;
|
||||||
*/
|
*/
|
||||||
|
Loading…
x
Reference in New Issue
Block a user