30 KiB
title | permalink | keywords |
---|---|---|
PMD Release Notes | pmd_release_notes.html | changelog, release notes |
????? - 6.0.0-SNAPSHOT
The PMD team is pleased to announce PMD 6.0.0.
This is a major release.
Table Of Contents
New and noteworthy
Java 9 support
The Java grammar has been updated to support analyzing Java 9 projects:
- private methods in interfaces are possible
- The underscore "_" is considered an invalid identifier
- Diamond operator for anonymous classes
- The module declarations in
module-info.java
can be parsed - Concise try-with-resources statements are supported
Java 9 support is enabled by default. You can switch back to an older java version
via the command line, e.g. -language java -version 1.8
.
Revamped Apex CPD
We are now using the Apex Jorje Lexer to tokenize Apex code for CPD. This change means:
- All comments are now ignored for CPD. This is consistent with how other languages such as Java and Groovy work.
- Tokenization honors the language specification, which improves accuracy.
CPD will therefore have less false positives and false negatives.
Java Type Resolution
As part of Google Summer of Code 2017, Bendegúz Nagy's work on type resolution for Java continues. For this release he has extended support for method calls for both instance and static methods.
Method shadowing and overloading are supported, as are varargs. However, the selection of the target method upon the presence of generics and type inference is still work in progress. Expect it in forecoming releases.
As for fields, the basic support was in place for release 5.8.0, but has now been expanded to support static fields.
Metrics Framework
As part of Google Summer of Code 2017, Clément Fournier is continuing his work on the new metrics framework for object-oriented metrics.
There are already a couple of metrics (e.g. ATFD, WMC, Cyclo, LoC) implemented. More metrics are planned. Based on those metrics, rules like "GodClass" detection can be implemented more easily.
The Metrics framework has been abstracted and is available in pmd-core
for other languages. With this
PMD release, the metrics framework is supported for both Java and Apex.
Error Reporting
A number of improvements on error reporting have taken place, meaning changes to some of the report formats.
Also of note, the xml report now provides a XML Schema definition, allowing easier parsing and validation.
Processing Errors
Processing errors can now provide not only the message previously included on some reports, but also a full stacktrace. This will allow better error reports when providing feedback to the PMD team and help in debugging issues.
The report formats providing full stacktrace of errors are:
- html
- summaryhtml
- textcolor
- vbhtml
- xml
Configuration Errors
For a long time reports have been notified of configuration errors on rules, but they have remained hidden. On a push to make these more evident to users, and help them get the best results out of PMD, we have started to include them on the reports.
So far, only reports that include processing errors are showing configuration errors. In other words, the report formats providing configuration error reporting are:
- csv
- html
- summaryhtml
- text
- textcolor
- vbhtml
- xml
As we move forward we will be able to detect and report more configuration errors (ie: incomplete auxclasspath
)
and include them to such reports.
Apex Rule Suppression
Apex violations can now be suppressed very similarly to how it's done in Java, by making use of a
@SuppressWarnings
annotation.
Supported syntax includes:
@SupressWarnings('PMD') // to supress all Apex rules
@SupressWarnings('all') // to supress all Apex rules
@SupressWarnings('PMD.ARuleName') // to supress only the rule named ARuleName
@SupressWarnings('PMD.ARuleName, PMD.AnotherRuleName') // to supress only the rule named ARuleName or AnotherRuleName
Notice this last scenario is slightly different to the Java syntax. This is due to differences in the Apex grammar for annotations.
New Rules
-
The rule
NcssCount
(rulesetjava-codesize
) replaces the three rules "NcssConstructorCount", "NcssMethodCount", and "NcssTypeCount". The new rule uses the metrics framework to achieve the same. It has two properties, to define the report level for method and class sizes separately. Constructors and methods are considered the same. -
The new rule
DoNotExtendJavaLangThrowable
(rulesetjava-strictexception
) is a companion for thejava-strictexception.xml/DoNotExtendJavaLangError
, detecting direct extensions ofjava.lang.Throwable
. -
The new rule
ForLoopCanBeForeach
(rulesetjava-migration
) helps to identify those for-loops that can be safely refactored into for-each-loops available since java 1.5. -
The new rule
AvoidDirectAccessTriggerMap
(rulesetapex-style
) helps to identify direct array access to triggers, which can produce bugs by iether accessing non-existing indexes, or them leaving out. You should use for-each-loops instead. -
The new rule
AvoidHardcodingId
(rulesetapex-style
) detects hardcoded strings that look like identifiers and flags them. Record IDs change between environments, meaning hardcoded ids are bound to fail under a different setup. -
A whole new ruleset has been added to Apex,
apex-empty
. It currently migrates 5 rules from the equivalentjava-empty
ruleset for Apex. The ruleset includes:EmptyCatchBlock
to detect catch blocks completely ignoring exceptions.EmptyIfStmt
for if blocks with no content, that can be safely removed.EmptyTryOrFinallyBlock
for empty try / finally blocks that can be safely removed.EmptyWhileStmt
for empty while loops that can be safely removed.EmptyStatementBlock
for empty code blocks that can be safely removed.
-
The new rule
AvoidSoslInLoops
(rulesetapex-performance
) is the companion of the oldapex-performance/AvoidSoqlInLoops
rule, flagging SOSL (Salesforce Object Search Language) queries when within loops, to avoid governor issues, and hitting the database too often.
Modified Rules
-
The rule
UnnecessaryFinalModifier
(rulesetjava-unnecessarycode
) has been merged into the ruleUnnecessaryModifier
of the same ruleset. As part of this, the rule has been revamped to detect more cases. It will now flag anonymous class' methods marked as final (can't be overridden, so it's pointless), along with final methods overridden / defined within enum instances. It will also flagfinal
modifiers on try-with-resources. -
The rule
UnnecessaryParentheses
(rulesetjava-controversial
) has been merged intoUselessParentheses
(rulesetjava-unnecessary
). The rule covers all scenarios previously covered by either rule. -
The rule
UncommentedEmptyConstructor
(rulesetjava-design
) will now ignore empty constructors annotated withjavax.inject.Inject
. -
The rule
AbstractClassWithoutAnyMethod
(rulesetjava-design
) will now ignore classes annotated withcom.google.auto.value.AutoValue
. -
The rule
GodClass
(rulesetjava-design
) has been revamped to use the new metrics framework. -
The rule
LooseCoupling
(rulesetjava-coupling
) has been replaced by the typeresolution-based implementation. -
The rule
CloneMethodMustImplementCloneable
(rulesetjava-clone
) has been replaced by the typeresolution-based implementation and is now able to detect cases if a class implements or extends a Cloneable class/interface. -
The rule
UnusedImports
(rulesetjava-imports
) has been replaced by the typeresolution-based implementation and is now able to detect unused on-demand imports. -
The rule
SignatureDeclareThrowsException
(ruleset 'java-strictexception') has been replaced by the typeresolution-based implementation. It has a new propertyIgnoreJUnitCompletely
, which allows all methods in a JUnit testcase to throws exceptions. -
The rule
NPathComplexity
(rulesetjava-codesize
) has been revamped to use the new metrics framework. Its report threshold can be configured via the propertyreportLevel
, which replaces the now deprecated propertyminimum
.
Deprecated Rules
-
The rules
NcssConstructorCount
,NcssMethodCount
, andNcssTypeCount
(rulesetjava-codesize
) have been deprecated. They will be replaced by the new ruleNcssCount
in the same ruleset. -
The rule
LooseCoupling
in rulesetjava-typeresolution
is deprecated. Use the rule with the same name from rulesetjava-coupling
instead. -
The rule
CloneMethodMustImplementCloneable
in rulesetjava-typeresolution
is deprecated. Use the rule with the same name from rulesetjava-clone
instead. -
The rule
UnusedImports
in rulesetjava-typeresolution
is deprecated. Use the rule with the same name from rulesetjava-imports
instead. -
The rule
SignatureDeclareThrowsException
in rulesetjava-typeresolution
is deprecated. Use the rule with the same name from rulesetjava-strictexception
instead.
Removed Rules
- The deprecated rule
UseSingleton
has been removed from the rulesetjava-design
. The rule has been renamed long time ago toUseUtilityClass
.
Java Symbol Table
A bug in symbol table prevented
the symbol table analysis to properly match primitive arrays types. The issue affected the java-unsedcode/UnusedPrivateMethod
rule, but other rules may now produce improved results as consequence of this fix.
Apex Parser Update
The Apex parser version was bumped, from 1.0-sfdc-187
to 1.0-sfdc-224
. This update let us take full advantage
of the latest improvements from Salesforce, but introduces some breaking changes:
BlockStatements
are now created for all control structures, even if no brace is used. We have therefore added ahasCurlyBrace
method to differentiate between both scenarios.- New AST node types are available. In particular
CastExpression
,ConstructorPreamble
,IllegalStoreExpression
,MethodBlockStatement
,Modifier
,MultiStatement
,NestedExpression
,NestedStoreExpression
,NewKeyValueObjectExpression
andStatementExecuted
- Some nodes have been removed. Such is the case of
TestNode
,DottedExpression
andNewNameValueObjectExpression
(replaced byNewKeyValueObjectExpression
)
All existing rules have been updated to reflect these changes. If you have custom rules, be sure to update them.
Incremental Analysis
The incremental analysis feature first introduced in PMD 5.6.0 has been enhanced. A few minor issues have been fixed, and several improvements have been performed to make it more accurate.
The cache will now detect changes to the JARs referenced in the auxclasspath
instead of simply looking at their paths
and order. This means that if you are referencing a JAR you are overwriting in some way, the incremental analysis can
now detect it and invalidate it's cache to avoid false reports.
Similarly, any changes to the execution classpath of PMD will invalidate the cache. This means that if you have custom rules packaged in a jar, any changes to it will invalidate the cache automatically.
We have also improved logging on the analysis code, allowing better insight into how the cache is performing, under debug / verbose builds you can even see individual hits / misses to the cache (and the reason for any miss!)
Finally, as this feature keeps maturing, we are gently pushing this forward. If not using incremental analysis, a warning will now be produced suggesting users to adopt it for better performance.
Fixed Issues
- all
- #532: [core] security concerns on URL-based rulesets
- #538: [core] Provide an XML Schema for XML reports
- #600: [core] Nullpointer while creating cache File
- #604: [core] Incremental analysis should detect changes to jars in classpath
- #608: [core] Add DEBUG log when applying incremental analysis
- #618: [core] Incremental Analysis doesn't close file correctly on Windows upon a cache hit
- #643: [core] PMD Properties (dev-properties) breaks markup on CodeClimateRenderer
- #680: [core] Isolate classloaders for runtime and auxclasspath
- apex
- cpp
- #448: [cpp] Write custom CharStream to handle continuation characters
- java
- #1513: [java] Remove deprecated rule UseSingleton
- #328: [java] java.lang.ClassFormatError: Absent Code attribute in method that is not native or abstract in class file javax/servlet/jsp/PageContext
- #487: [java] Fix typeresolution for anonymous extending object
- #496: [java] processing error on generics inherited from enclosing class
- #510: [java] Typeresolution fails on a simple primary when the source is loaded from a class literal
- #527: [java] Lombok getter annotation on enum is not recognized correctly
- #534: [java] NPE in MethodTypeResolution for static methods
- #603: [core] incremental analysis should invalidate upon Java rule plugin changes
- #650: [java] ProcesingError analyzing code under 5.8.1
- java-basic
- #565: [java] False negative on DontCallThreadRun when extending Thread
- java-comments
- #536: [java] CommentDefaultAccessModifierRule ignores constructors
- java-controversial
- java-design
- java-sunsecure
- #468: [java] ArrayIsStoredDirectly false positive
- java-unusedcode
- #521: [java] UnusedPrivateMethod returns false positives with primitive data type in map argument
- java-unnecessarycode
API Changes
-
The class
net.sourceforge.pmd.lang.dfa.NodeType
has been converted to an enum. All node types are enum members now instead of int constants. The names for node types are retained. -
The properties API (rule and report properties) have been revamped to be fully typesafe. This is everything around
net.sourceforge.pmd.PropertyDescriptor
. -
The rule classes
net.sourceforge.pmd.lang.apex.rule.apexunit.ApexUnitTestClassShouldHaveAsserts
andnet.sourceforge.pmd.lang.apex.rule.apexunit.ApexUnitTestShouldNotUseSeeAllDataTrue
have been renamed toApexUnitTestClassShouldHaveAssertsRule
andApexUnitTestShouldNotUseSeeAllDataTrueRule
, respectively. This is to comply with the naming convention, that each rule class should be suffixed with "Rule". This change has no impact on custom rulesets, since the rule names themselves didn't change. -
The never implemented method
PMD.processFiles(PMDConfiguration, RuleSetFactory, Collection<File>, RuleContext, ProgressMonitor)
along with the interfaceProgressMonitor
has been removed. -
The method
PMD.setupReport(RuleSets, RuleContext, String)
is gone. It was used to report dysfunctional rules. But PMD does this now automatically before processing the files, so there is no need for this method anymore. -
All APIs deprecated in older versions are now removed. This includes:
Renderer.getPropertyDefinitions
AbstractRenderer.defineProperty(String, String)
AbstractRenderer.propertyDefinitions
ReportListener
Report.addListener(ReportListener)
SynchronizedReportListener
CPDConfiguration.CPDConfiguration(int, Language, String)
CPDConfiguration.getRendererFromString(String)
StreamUtil
StringUtil.appendXmlEscaped(StringBuilder, String)
StringUtil.htmlEncode(String)
-
Several methods in
net.sourceforge.pmd.util.CollectionUtil
have been deprecated and will be removed in PMD 7.0.0. In particular:CollectionUtil.addWithoutDuplicates(T[], T)
CollectionUtil.addWithoutDuplicates(T[], T[])
CollectionUtil.areSemanticEquals(T[], T[])
CollectionUtil.areEqual(Object, Object)
CollectionUtil.arraysAreEqual(Object, Object)
CollectionUtil.valuesAreTransitivelyEqual(Object[], Object[])
-
Several methods in
net.sourceforge.pmd.util.StringUtil
have been deprecated and will be removed in PMD 7.0.0. In particular:StringUtil.startsWithAny(String, String[])
StringUtil.isNotEmpty(String)
StringUtil.isEmpty(String)
StringUtil.isMissing(String)
StringUtil.areSemanticEquals(String, String)
StringUtil.replaceString(String, String, String)
StringUtil.replaceString(String, char, String)
StringUtil.substringsOf(String, char)
StringUtil.substringsOf(String, String)
StringUtil.asStringOn(StringBuffer, Iterator, String)
StringUtil.asStringOn(StringBuilder, Object[], String)
StringUtil.lpad(String, int)
-
The class
net.sourceforge.pmd.lang.java.typeresolution.typedefinition.JavaTypeDefinition
is now abstract, and has been enhanced to provide several new methods.
External Contributions
- #287: [apex] Make Rule suppression work - Robert Sösemann
- #420: [java] Fix UR anomaly in assert statements - Clément Fournier
- #482: [java] Metrics testing framework + improved capabilities for metrics - Clément Fournier
- #484: [core] Changed linux usage to a more unix like path - patriksevallius
- #486: [java] Add basic method typeresolution - Bendegúz Nagy
- #492: [java] Typeresolution for overloaded methods - Bendegúz Nagy
- #495: [core] Custom rule reinitialization code - Clément Fournier
- #479: [core] Typesafe and immutable properties - Clément Fournier
- #499: [java] Metrics memoization tests - Clément Fournier
- #501: [java] Add support for most specific vararg method type resolution - Bendegúz Nagy
- #502: [java] Add support for static field type resolution - Bendegúz Nagy
- #505: [java] Followup on metrics - Clément Fournier
- #506: [java] Add reduction rules to type inference - Bendegúz Nagy
- #511: [core] Prepare abstraction of the metrics framework - Clément Fournier
- #512: [java] Add incorporation to type inference - Bendegúz Nagy
- #513: [java] Fix for maximally specific method selection - Bendegúz Nagy
- #514: [java] Add static method type resolution - Bendegúz Nagy
- #517: [doc] Metrics documentation - Clément Fournier
- #518: [core] Properties refactoring: factorized enumerated property - Clément Fournier
- #523: [java] Npath complexity metric and rule - Clément Fournier
- #524: [java] Add support for explicit type arguments with method invocation - Bendegúz Nagy
- #525: [core] Fix line ending and not ignored files issues - Matias Comercio
- #528: [core] Fix typo - Ayoub Kaanich
- #529: [java] Abstracted the Java metrics framework - Clément Fournier
- #530: [java] Fix issue #527: Lombok getter annotation on enum is not recognized correctly - Clément Fournier
- #533: [core] improve error message - Dennis Kieselhorst
- #535: [apex] Fix broken Apex visitor adapter - Clément Fournier
- #542: [java] Metrics abstraction - Clément Fournier
- #545: [apex] Apex metrics framework - Clément Fournier
- #548: [java] Metrics documentation - Clément Fournier
- #550: [java] Add basic resolution to type inference - Bendegúz Nagy
- #553: [java] Refactored ParserTst into a static utility class + add getSourceFromClass - Clément Fournier
- #554: [java] Fix #537: UnnecessaryParentheses fails to detect obvious scenario - Clément Fournier
- #555: [java] Changed metrics/CyclomaticComplexityRule to use WMC when reporting classes - Clément Fournier
- #556: [java] Fix #357: UncommentedEmptyConstructor consider annotations on Constructor - Clément Fournier
- #557: [java] Fix NPath metric not counting ternaries correctly - Clément Fournier
- #563: [java] Add support for basic method type inference for strict invocation - Bendegúz Nagy
- #566: [java] New rule in migrating ruleset: ForLoopCanBeForeach - Clément Fournier
- #567: [java] Last API change for metrics (metric options) - Clément Fournier
- #570: [java] Model lower, upper and intersection types - Bendegúz Nagy
- #573: [java] Data class rule - Clément Fournier
- #576: [doc][java] Add hint for Guava users in InefficientEmptyStringCheck - mmoehring
- #578: [java] Refactored god class rule - Clément Fournier
- #579: [java] Update parsing to produce upper and lower bounds - Bendegúz Nagy
- #580: [core] Add AbstractMetric to topple the class hierarchy of metrics - Clément Fournier
- #581: [java] Relax AbstractClassWithoutAnyMethod when class is annotated by @AutoValue - Niklas Baudy
- #583: [java] Documentation about writing metrics - Clément Fournier
- #585: [java] Moved NcssCountRule to codesize.xml - Clément Fournier
- #587: [core] Properties refactoring: Move static constants of ValueParser to class ValueParserConstants - Clément Fournier
- #588: [java] XPath function to compute metrics - Clément Fournier
- #598: [java] Fix #388: controversial.AvoidLiteralsInIfCondition 0.0 false positive - Clément Fournier
- #602: [java] [apex] Separate multifile analysis from metrics - Clément Fournier
- #620: [core] Moved properties to n.s.pmd.properties - Clément Fournier
- #625: [apex] empty code ruleset for apex - Jan Aertgeerts
- #632: [apex] Add AvoidDirectAccessTriggerMap rule to the style set - Jan Aertgeerts
- #644: [core] Prevent internal dev-properties from being displayed on CodeClimate renderer - Filipe Esperandio
- #660: [apex] avoid sosl in loops - Jan Aertgeerts
- #661: [apex] avoid hardcoding id's - Jan Aertgeerts
- #666: [java] Add DoNotExtendJavaLangThrowable rule - Robert Painsi
- #668: [core] Fix javadoc warnings on pmd-core - Clément Fournier
- #669: [core] Builder pattern for properties - Clément Fournier
- #675: [java] Fix in Java grammar: Try with final resource node error - Gonzalo Ibars Ingman
- #679: [core] Token scheme generalization - Gonzalo Ibars Ingman
- #694: [core] Add minor fixes to root pom - Matias Comercio
- #696: [core] Add remove operation over nodes - Matias Comercio
- #722: [java] Move NPathComplexity from metrics to design - Clément Fournier