PMD - 用 Java 开发代码规则

在 PMD 的使用过程中,可以用 Java 或 XPath 来开发代码规则。

本文讲述如何使用 Java 来开发。

AST 对应的 Java 类库

PMD 使用抽象语法树(AST)来分析代码。它本身已经创建了一系列的 Java 类库来操作不同的语法元素。

针对 Apex 语言的 6.22.0 API 的官方文档

在使用 Java 开发规则时,我们主要就是利用这些类来分析代码,从而找出违反规则的情况。

比如:

  • ASTAnnotation:对应的是注解,比如 @IsTest
  • ASTBreakStatement:对应的是 break 语句
  • ASTWhileLoopStatement:对应的是 while 循环
  • ASTLiteralExpression:对应的是字符串赋值语句

实现步骤

  1. 建立新的 Java 类
  2. 重写 visit() 函数,在函数中分析代码,找到违反规则的部分
  3. 将规则注册到规则集文件中

GitHub 中可以参考现有规则的 Java 实现。

举例

规则:代码中不能出现 System.debug() 函数。

建立新的 Java 类

建立一个新的类,名叫 SystemDebugShouldBeAvoidedRule,并且扩展 AbstractApexRule 类。

import java.util.List;

import net.sourceforge.pmd.lang.apex.ast.*;
import net.sourceforge.pmd.lang.apex.rule.*;

public class SystemDebugShouldBeAvoidedRule extends AbstractApexRule {
}

重写 visit() 函数

在新的类中重写 visit() 函数。

因为规则检测的入口是函数,所以参数是 ASTMethod 类型。

AST 类库中定义了很多方法,比如 findDescendantsOfType() 可以得到某个代码块下面的所有特定类型的元素。

import java.util.List;

import net.sourceforge.pmd.lang.apex.ast.*;
import net.sourceforge.pmd.lang.apex.rule.*;

public class SystemDebugShouldBeAvoidedRule extends AbstractApexRule {
    @Override
    public Object visit(ASTMethod node, Object data) {
        boolean hasSystemDebug = false;

        // 得到所有的调用函数的代码
        List<ASTMethodCallExpression> methodCallList = node.findDescendantsOfType(ASTMethodCallExpression.class);
        for (ASTMethodCallExpression methodCall : methodCallList) {
            // 得到函数名
            String assertMethodName = methodCall.getFullMethodName();

            // 和 "system.debug" 比较
            if ("system.debug".equalsIgnoreCase(assertMethodName)) {
                hasSystemDebug = true;
                break;
            }
        }

        if (hasSystemDebug) {
            // addViolationWithMessage() 函数会生成警告
            addViolationWithMessage(data, node, "''{0}'' method should not have System.Debug function.", new Object[] { node.getImage() });
        }
        
        // 返回 data,其中会包含警告信息
        return data;
    }
}

将规则注册到规则集文件中

每个规则集都是以 XML 形式出现的。我们将新的规则注册进去:

<rule name="SystemDebugShouldBeAvoided"
          since="6.23.0"
          language="apex"
          message="System Debug should be avoided"
          class="<Java类的路径>.SystemDebugShouldBeAvoidedRule">
        <description>
            System Debug should be avoided
        </description>
        <priority>4</priority> <!-- 优先级从 1 到 5,表示从最严重到最轻微的规则-->
        <example>
            <![CDATA[
				public void doSomething() {
					System.debug('test');
				}
			]]>
        </example>
    </rule>

这样,这条规则就可以使用了。

posted @ 2020-07-28 02:45  程程哥  阅读(902)  评论(0编辑  收藏  举报