自定义代码质量规则 custom-code-quality-rules

了解 Cloud Manager 在基于来自 AEM 工程的最佳实践的 代码质量测试过程中执行的自定义代码质量规则的详细信息。

NOTE
此处提供的代码示例仅用于说明目的。请参阅 SonarQube 的概念文档,了解其概念和质量规则。
NOTE
由于是 Adobe 专有信息,因此无法下载完整的 SonarQube 规则。可使用此链接下载规则的完整列表。有关规则的描述和示例,请继续阅读本文档。

SonarQube 规则 sonarqube-rules

以下部分详细介绍了 Cloud Manager 执行的 SonarQube 规则。

请勿使用有潜在危险的功能 do-not-use-potentially-dangerous-functions

  • 密钥:CQRules:CWE-676
  • 类型:漏洞
  • 严重性:主要
  • 开始版本:版本 2018.4.0

Thread.stop()Thread.interrupt() 方法可能会产生难以重现的问题,并且有时会产生安全漏洞。 应严格监控和验证其使用情况。 总的来说,传递信息是实现类似目标的一种更安全的方式。

不合规的代码 non-compliant-code

public class DontDoThis implements Runnable {
  private Thread thread;

  public void start() {
    thread = new Thread(this);
    thread.start();
  }

  public void stop() {
    thread.stop();  // UNSAFE!
  }

  public void run() {
    while (true) {
        somethingWhichTakesAWhileToDo();
    }
  }
}

合规的代码 compliant-code

public class DoThis implements Runnable {
  private Thread thread;
  private boolean keepGoing = true;

  public void start() {
    thread = new Thread(this);
    thread.start();
  }

  public void stop() {
    keepGoing = false;
  }

  public void run() {
    while (this.keepGoing) {
        somethingWhichTakesAWhileToDo();
    }
  }
}

请勿使用可能受到外部控制的格式字符串 do-not-use-format-strings-which-may-be-externally-controlled

  • 密钥:CQRules:CWE-134
  • 类型:漏洞
  • 严重性:主要
  • 开始版本:版本 2018.4.0

使用来自外部源的格式字符串(如请求参数或用户生成的内容)可能会使应用程序遭受拒绝服务攻击。在某些情况下,虽然格式字符串可能会受到外部控制,但仅允许来自受信任源。

不合规的代码 non-compliant-code-1

protected void doPost(SlingHttpServletRequest request, SlingHttpServletResponse response) {
  String messageFormat = request.getParameter("messageFormat");
  request.getResource().getValueMap().put("some property", String.format(messageFormat, "some text"));
  response.sendStatus(HttpServletResponse.SC_OK);
}

HTTP 请求应始终具有套接字和连接超时 http-requests-should-always-have-socket-and-connect-timeouts

  • 密钥:CQRules:ConnectionTimeoutMechanism
  • 类型:错误
  • 严重性:严重
  • 开始版本:版本 2018.6.0

从 AEM 应用程序内部执行 HTTP 请求时,配置适当的超时以避免不必要的线程消耗至关重要。不幸的是,Java™ 的默认 HTTP 客户端 java.net.HttpUrlConnection 和广泛使用的 Apache HTTP 组件客户端都没有默认超时。因此,必须明确配置超时。作为最佳实践,这些超时不应超过 60 秒。

不合规的代码 non-compliant-code-2

@Reference
private HttpClientBuilderFactory httpClientBuilderFactory;

public void dontDoThis() {
  HttpClientBuilder builder = httpClientBuilderFactory.newBuilder();
  HttpClient httpClient = builder.build();

  // do something with the client
}

public void dontDoThisEither() {
  URL url = new URL("http://www.google.com");
  URLConnection urlConnection = url.openConnection();

  BufferedReader in = new BufferedReader(new InputStreamReader(
    urlConnection.getInputStream()));

  String inputLine;
  while ((inputLine = in.readLine()) != null) {
    logger.info(inputLine);
  }

  in.close();
}

合规的代码 compliant-code-1

@Reference
private HttpClientBuilderFactory httpClientBuilderFactory;

public void doThis() {
  HttpClientBuilder builder = httpClientBuilderFactory.newBuilder();
  RequestConfig requestConfig = RequestConfig.custom()
    .setConnectTimeout(5000)
    .setSocketTimeout(5000)
    .build();
  builder.setDefaultRequestConfig(requestConfig);

  HttpClient httpClient = builder.build();

  // do something with the client
}

public void orDoThis() {
  URL url = new URL("http://www.google.com");
  URLConnection urlConnection = url.openConnection();
  urlConnection.setConnectTimeout(5000);
  urlConnection.setReadTimeout(5000);

  BufferedReader in = new BufferedReader(new InputStreamReader(
    urlConnection.getInputStream()));

  String inputLine;
  while ((inputLine = in.readLine()) != null) {
    logger.info(inputLine);
  }

  in.close();
}

对象 ResourceResolver 应该始终关闭 resourceresolver-objects-should-always-be-closed

  • 密钥:CQRules:CQBP-72
  • 类型:代码异味
  • 严重性:主要
  • 开始版本:版本 2018.4.0

ResourceResolverResourceResolverFactory 中获取的对象会消耗系统资源。虽然可以采取一些措施以在不再使用 ResourceResolver 时回收这些资源,但通过调用 close() 方法明确关闭任何打开的 ResourceResolver 对象会更有效。

一个常见的误解是 ResourceResolver 使用现有 JCR 会话创建的对象不应明确关闭。另一个误解是关闭这些对象会关闭底层的 JCR 会话。情况并非如此。无论 ResourceResolver 的打开方式如何,只要不再使用它,就应将它关闭。因为 ResourceResolver 实施 Closeable 接口,所以也可以使用 try-with-resources 语法,而不是显式调用 close()

不合规的代码 non-compliant-code-4

public void dontDoThis(Session session) throws Exception {
  ResourceResolver resolver = factory.getResourceResolver(Collections.singletonMap("user.jcr.session", (Object)session));
  // do some stuff with the resolver
}

合规的代码 compliant-code-2

public void doThis(Session session) throws Exception {
  ResourceResolver resolver = null;
  try {
    resolver = factory.getResourceResolver(Collections.singletonMap("user.jcr.session", (Object)session));
    // do something with the resolver
  } finally {
    if (resolver != null) {
      resolver.close();
    }
  }
}

public void orDoThis(Session session) throws Exception {
  try (ResourceResolver resolver = factory.getResourceResolver(Collections.singletonMap("user.jcr.session", (Object) session))){
    // do something with the resolver
  }
}

不要使用 Sling Servlet 路径来注册 Servlet do-not-use-sling-servlet-paths-to-register-servlets

  • 密钥:CQRules:CQBP-75
  • 类型:代码异味
  • 严重性:主要
  • 开始版本:版本 2018.4.0

Sling 文档所述,建议不要通过路径绑定 servlet。路径绑定的 servlet 不能使用标准 JCR 访问控制,因此,需要额外的安全严密性。 建议在存储库中创建节点并按资源类型注册 servlet,而不是使用路径绑定的 servlet。

不合规的代码 non-compliant-code-5

@Component(property = {
  "sling.servlet.paths=/apps/myco/endpoint"
})
public class DontDoThis extends SlingAllMethodsServlet {
 // implementation
}

捕获的异常应被记录或引发,而不是同时记录和引发 caught-exceptions-should-be-logged-or-thrown-but-not-both

  • 密钥:CQRules:CQBP-44—CatchAndEitherLogOrThrow
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2018.4.0

通常,一个异常应只记录一次。 多次记录异常可能会引起混乱,因为不清楚异常发生了多少次导致出现此问题的最常见模式是记录并抛出捕获的异常。

不合规的代码 non-compliant-code-6

public void dontDoThis() throws Exception {
  try {
    someOperation();
  } catch (Exception e) {
    logger.error("something went wrong", e);
    throw e;
  }
}

合规的代码 compliant-code-3

public void doThis() {
  try {
    someOperation();
  } catch (Exception e) {
    logger.error("something went wrong", e);
  }
}

public void orDoThis() throws MyCustomException {
  try {
    someOperation();
  } catch (Exception e) {
    throw new MyCustomException(e);
  }
}

避免在 Log 语句后面紧跟 Throw 语句 avoid-having-a-log-statement-immediately-followed-by-a-throw-statement

  • 密钥:CQRules:CQBP-44—ConsecutivelyLogAndThrow
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2018.4.0

另一个要避免的常见模式是记录一条消息,然后立即引发异常。 该问题通常表示异常消息最终会在日志文件中重复。

不合规的代码 non-compliant-code-7

public void dontDoThis() throws Exception {
  logger.error("something went wrong");
  throw new RuntimeException("something went wrong");
}

合规的代码 compliant-code-4

public void doThis() throws Exception {
  throw new RuntimeException("something went wrong");
}

在处理 GET 或 HEAD 请求时,避免在 INFO 级别进行记录 avoid-logging-at-info-when-handling-get-or-head-requests

  • 密钥:CQRules:CQBP-44—LogInfoInGetOrHeadRequests
  • 类型:代码异味
  • 严重性:轻微

通常,应使用 INFO 日志级别来划分重要操作,默认情况下,AEM 配置为在 INFO 级别或更高级别进行记录。 GET 和 HEAD 方法只能为只读操作,因此,不会构成重要操作。 在 INFO 级别进行记录来响应 GET 或 HEAD 请求可能会产生大量日志噪音,导致更难以识别日志文件中的有用信息。 处理 GET 或 HEAD 请求时,如果出现错误,日志记录应处于 WARN 或 ERROR 级别。为了获得更深入的故障排除信息,日志记录应处于 DEBUG 或 TRACE 级别。

NOTE
此工作流程不适用于每个请求的 access.log-type 日志记录。

不合规的代码 non-compliant-code-8

public void doGet() throws Exception {
  logger.info("handling a request from the user");
}

合规的代码 compliant-code-5

public void doGet() throws Exception {
  logger.debug("handling a request from the user.");
}

不要使用 Exception.getMessage() 作为记录语句的第一个参数 do-not-use-exception-getmessage-as-the-first-parameter-of-a-logging-statement

  • 密钥:CQRules:CQBP-44—ExceptionGetMessageIsFirstLogParam
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2018.4.0

作为最佳实践,日志消息应提供有关应用程序中发生异常的位置的上下文信息。 虽然也可以使用堆栈跟踪来确定上下文,但通常日志消息将更易于阅读和理解。 因此,在记录异常时,将异常消息用作日志消息是一种不好的做法。 异常消息应该详细说明出了什么问题。相反,日志消息应该告知 Reader 异常发生时应用程序正在做什么。仍会记录异常消息。 通过指定您自己的消息,可使日志更易于理解。

不合规的代码 non-compliant-code-9

public void dontDoThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.error(e.getMessage(), e);
  }
}

合规的代码 compliant-code-6

public void doThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.error("Unable to do something", e);
  }
}

应在 WARN 或 ERROR 级别记录 Catch 块 logging-in-catch-blocks-should-be-at-the-warn-or-error-level

  • 密钥:CQRules:CQBP-44—WrongLogLevelInCatchBlock
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2018.4.0

顾名思义,Java™ 异常应始终在异常情况下使用。 因此,当捕获到异常时,请务必确保在适当的级别记录日志消息:WARN 或 ERROR。这样做可确保这些消息正确显示在日志中。

不合规的代码 non-compliant-code-10

public void dontDoThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.debug(e.getMessage(), e);
  }
}

合规的代码 compliant-code-7

public void doThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.error("Unable to do something", e);
  }
}

请勿将堆栈跟踪输出到控制台 do-not-print-stack-traces-to-the-console

  • 密钥:CQRules:CQBP-44—ExceptionPrintStackTrace
  • 类型:代码异味
  • 严重性:轻微
  • 从以下版本开始:版本 2018.4.0

上下文对于理解日志消息是至关重要的。使用 Exception.printStackTrace() 会导致仅将堆栈跟踪输出到标准错误流,从而丢失所有上下文。此外,在像 AEM 这样的多线程应用程序中,如果使用此方法并行打印多个异常,则其堆栈跟踪可能会发生重叠,从而导致产生严重的混淆。应仅通过记录框架来记录异常。

不合规的代码 non-compliant-code-11

public void dontDoThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    e.printStackTrace();
  }
}

合规的代码 compliant-code-8

public void doThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.error("Unable to do something", e);
  }
}

请勿输出到标准输出或标准错误 do-not-output-to-standard-output-or-standard-error

  • 密钥:CQRules:CQBP-44—LogLevelConsolePrinters
  • 类型:代码异味
  • 严重性:轻微
  • 从以下版本开始:版本 2018.4.0

AEM 中的记录应始终通过记录框架 SLF4J 完成。直接输出到标准输出或标准错误流将丢失记录框架提供的结构和上下文信息,并且有时可能会导致出现性能问题。

不合规的代码 non-compliant-code-12

public void dontDoThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    System.err.println("Unable to do something");
  }
}

合规的代码 compliant-code-9

public void doThis() {
  try {
    someMethodThrowingAnException();
  } catch (Exception e) {
    logger.error("Unable to do something", e);
  }
}

避免硬编码 /apps/libs 路径 avoid-hardcoded-apps-and-libs-paths

  • 密钥:CQRules:CQBP-71
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2018.4.0

/libs/apps 开头的路径通常不应进行硬编码。这些路径通常相对 Sling 搜索路径进行存储,默认为 /libs,/apps。使用绝对路径可能会引入细微的缺陷,这些缺陷只会在项目生命周期的后期才会出现。

不合规的代码 non-compliant-code-13

public boolean dontDoThis(Resource resource) {
  return resource.isResourceType("/libs/foundation/components/text");
}

合规的代码 compliant-code-10

public void doThis(Resource resource) {
  return resource.isResourceType("foundation/components/text");
}

不应使用 Sling 调度程序 sonarqube-sling-scheduler

  • 密钥:CQRules:AMSCORE-554
  • 类型:代码异味 / Cloud Service 兼容性
  • 严重性:轻微
  • 开始版本:版本 2020.5.0

Sling 调度程序不得用于需要保证执行的任务。 Sling 计划作业可保证执行,并且更适合集群和非集群环境。

请参阅 Apache Sling 事件和作业处理文档,详细了解如何在集群环境中处理 Sling 作业。

不应使用 AEM 弃用的 API sonarqube-aem-deprecated

  • 密钥:AMSCORE-553
  • 类型:代码异味 / Cloud Service 兼容性
  • 严重性:轻微
  • 开始版本:版本 2020.5.0

AEM API 接口正在不断修订,以识别建议不要使用的 API,从而将其弃用。

通常情况下,使用标准 Java™ @Deprecated 注释来弃用这些 API,如 squid:CallToDeprecatedMethod 所标识。

但在某些情况下,API 会在 AEM 上下文中被弃用,但在其他上下文中可能不会被弃用。此规则标识第二个类。

OakPAL 内容规则 oakpal-rules

以下部分详细介绍了 Cloud Manager 执行的 OakPAL 检查。

NOTE
OakPAL 是一个框架,它使用独立的 Oak 存储库来验证内容包。 它由一个 AEM 合作伙伴和 2019 年 AEM Rock Star 北美奖获奖者开发。

客户不应实施或扩展使用 @ProviderType 注释的产品 API product-apis-annotated-with-providertype-should-not-be-implemented-or-extended-by-customers

  • 密钥:CQBP-84
  • 类型:错误
  • 严重性:严重
  • 开始版本:版本 2018.7.0

AEM API 包含 Java™ 接口和类,仅供自定义代码使用,但不能由自定义代码实现。例如,只有 AEM 实施了该接口 com.day.cq.wcm.api.Page

向这些接口添加新方法不会影响现有代码,从而使新方法的添加具有向后兼容。但是,如果自定义代码实现了其中一个接口,则该自定义代码会给客户带来向后兼容性风险。

AEM 注释使用 org.osgi.annotation.versioning.ProviderType 或偶尔使用旧版注释 aQute.bnd.annotation.ProviderType来注释仅用于其实现的接口和类。此规则检测自定义代码实现此类接口或扩展类的实例。

不合规的代码 non-compliant-code-3

import com.day.cq.wcm.api.Page;

public class DontDoThis implements Page {
// implementation here
}

客户包不应该在/libs以下位置创建或编辑节点 oakpal-customer-package

  • 密钥:BannedPath
  • 类型:错误
  • 严重性:阻断
  • 从以下版本开始:版本 2019.6.0

客户应将 AEM 内容存储库中的 /libs 内容树视为只读,这是一个长期存在的最佳实践。修改 /libs 下的节点和属性会给主要和次要更新带来重大风险。 对 /libs 的编辑只能由 Adobe 通过正式渠道进行。

包不应包含重复的 OSGi 配置 oakpal-package-osgi

  • 密钥:DuplicateOsgiConfigurations
  • 类型:错误
  • 严重性:主要
  • 开始版本:版本 2019.6.0

复杂项目中的一个常见问题是将同一个 OSGi 组件配置多次。这个问题会导致无法明确哪种配置将是可操作的。此规则是“运行模式感知型”的,因为它只会识别在同一运行模式或运行模式组合中多次配置同一组件的问题。

不合规的代码 non-compliant-code-osgi

+ apps
  + projectA
    + config
      + com.day.cq.commons.impl.ExternalizerImpl
  + projectB
    + config
      + com.day.cq.commons.impl.ExternalizerImpl

合规的代码 compliant-code-osgi

+ apps
  + shared-config
    + config
      + com.day.cq.commons.impl.ExternalizerImpl

Config 和 Install 文件夹应只包含 OSGi 节点 oakpal-config-install

  • 密钥:ConfigAndInstallShouldOnlyContainOsgiNodes
  • 类型:错误
  • 严重性:主要
  • 开始版本:版本 2019.6.0

出于安全原因,包含 /config//install/ 的路径只能由 AEM 中的管理用户读取,并且只能用于 OSGi 配置和 OSGi 捆绑包。 将其他类型的内容放在包含这些区段的路径下,则会导致应用程序行为在管理用户和非管理用户之间无意中发生变化。

一个常见的问题是,在组件对话框中或在为内联编辑指定富文本编辑器配置时使用名为 config 的节点。要解决此问题,应将违规节点重命名为合规的名称。 对于富文本编辑器配置,使用 cq:inplaceEditing 节点上的 configPath 属性指定新位置。

不合规的代码 non-compliant-code-config-install

+ cq:editConfig [cq:EditConfig]
  + cq:inplaceEditing [cq:InplaceEditConfig]
    + config [nt:unstructured]
      + rtePlugins [nt:unstructured]

合规的代码 compliant-code-config-install

+ cq:editConfig [cq:EditConfig]
  + cq:inplaceEditing [cq:InplaceEditConfig]
    ./configPath = inplaceEditingConfig (String)
    + inplaceEditingConfig [nt:unstructured]
      + rtePlugins [nt:unstructured]

包不应重叠 oakpal-no-overlap

  • 密钥:PackageOverlaps
  • 类型:错误
  • 严重性:主要
  • 开始版本:版本 2019.6.0

“包不应包含重复的 OSGi 配置”规则类似,此问题在复杂项目中是一个常见问题,其中相同的节点路径被多个单独的内容包写入。虽然可使用内容包依赖项来确保获得一致的结果,但最好是完全避免重叠。

默认创作模式不应是经典 UI oakpal-default-authoring

  • 密钥:ClassicUIAuthoringMode
  • 类型:代码异味 / Cloud Service 兼容性
  • 严重性:轻微
  • 开始版本:版本 2020.5.0

OSGi 配置 com.day.cq.wcm.core.impl.AuthoringUIModeServiceImpl 定义 AEM 中的默认创作模式。 由于经典 UI 自 AEM 6.4 之后已被弃用,因此,在将默认创作模式配置为经典 UI 时,现在将引发问题。

带有对话框的组件应该有触摸 UI 对话框 oakpal-components-dialogs

  • 密钥:ComponentWithOnlyClassicUIDialog
  • 类型:代码异味 / Cloud Service 兼容性
  • 严重性:轻微
  • 开始版本:版本 2020.5.0

具有经典 UI 对话框的 AEM 组件还应具有触摸 UI 对话框,以实现最佳创作并与不支持经典 UI 的 Cloud Service 部署模型兼容。此规则验证以下场景:

  • 具有经典 UI 对话框(即 dialog 子节点)的组件必须具有相应的 Touch UI 对话框(即 cq:dialog 子节点)。
  • 具有经典 UI 设计对话框(即 design_dialog 节点)的组件必须具有相应的 Touch UI 设计对话框(即 cq:design_dialog 子节点)。
  • 具有经典 UI 对话框和经典 UI 设计对话框的组件必须具有相应的 Touch UI 对话框和Touch UI 设计对话框。

“AEM 现代化工具”文档提供了有关如何将组件从经典 UI 转换为 Touch UI 的详细信息以及使用的工具。请参阅“AEM 现代化工具”文档,了解更多详细信息。

不应使用反向复制代理 oakpal-reverse-replication

  • 密钥:ReverseReplication
  • 类型:代码异味 / Cloud Service 兼容性
  • 严重性:轻微
  • 开始版本:版本 2020.5.0

Cloud Service 部署中不支持反向复制,如发行说明:移除复制代理中所述。

使用反向复制的客户应联系 Adobe 以获取替代解决方案。

启用代理的客户端库中包含的资源应位于名为 resources 的文件夹中 oakpal-resources-proxy

  • 密钥:ClientlibProxyResource
  • 类型:错误
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

AEM 客户端库可能包含静态资源,如图像和字体。如“使用客户端库”文档中所述,在使用代理客户端库时,这些静态资源必须包含在名为 resources 的子文件夹中,才能在发布实例上有效引用这些资源。

不合规的代码 non-compliant-proxy-enabled

+ apps
  + projectA
    + clientlib
      - allowProxy=true
      + images
        + myimage.jpg

合规的代码 compliant-proxy-enabled

+ apps
  + projectA
    + clientlib
      - allowProxy=true
      + resources
        + myimage.jpg

与 Cloud Service 不兼容的工作流的使用 oakpal-usage-cloud-service

  • 密钥:CloudServiceIncompatibleWorkflowProcess
  • 类型:代码异味
  • 严重性:阻断
  • 从以下版本开始:版本 2021.2.0

随着迁移到 AEM Cloud Service 上使用资产微服务进行资产处理,在 AEM 的内部部署和 AMS 版本中使用的功能已不再受支持或不再需要。

AEM Assets as a Cloud Service GitHub 存储库中的迁移工具可在迁移到 AEM as a Cloud Service 的过程中用于更新工作流模型。

建议不要使用静态模板来支持可编辑的模板 oakpal-static-template

  • 密钥:StaticTemplateUsage
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

虽然静态模板的使用历来在 AEM 项目中很普遍,但强烈建议使用可编辑的模板,因为它们提供了最大的灵活性,并支持静态模板中不存在的附加功能。要获取更多信息,请参阅“页面模板 - 可编辑”文档

可以使用 AEM Modernization Tools在很大程度上实现从静态模板到可编辑模板的迁移的自动化。

建议不要使用旧的基础组件 oakpal-usage-legacy

  • 密钥:LegacyFoundationComponentUsage
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

旧的基础组件(即 /libs/foundation 下的组件)已在多个 AEM 版本中被弃用以便支持核心组件。建议不要使用旧的基础组件作为自定义组件的基础(无论是通过叠加还是继承),并且应将这些基础组件转换为对应的核心组件。

AEM 现代化工具可以促进此转换。

  • 密钥:OakIndexLocation
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

AEM Cloud Service 要求自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)是 /oak:index 的直接子节点。必须移动其他位置的索引才能与 AEM Cloud Service 兼容。有关搜索索引的更多信息,请参阅“内容搜索和索引编制”文档

自定义搜索索引定义节点的 compatVersion 必须为 2 oakpal-custom-search-compatVersion

  • 密钥:IndexCompatVersion
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

AEM Cloud Service 要求自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)必须将 compatVersion 属性设置为 2。AEM Cloud Service 不支持任何其他值。有关搜索索引的更多信息,请参阅“内容搜索和索引编制”文档

自定义搜索索引定义节点的后代节点的类型必须是nt:unstructured oakpal-descendent-nodes

  • 密钥:IndexDescendantNodeType
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

当自定义搜索索引定义节点具有无序子节点时,可能会出现难以解决的问题。为了避免出现此类节点,Adobe 建议 oak:QueryIndexDefinition 节点的所有后代节点的类型为 nt:unstructured

自定义搜索索引定义节点必须包含一个名为 indexRules 的具有子节点的子节点 oakpal-custom-search-index

  • 密钥:IndexRulesNode
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

正确定义的自定义搜索索引定义节点必须包含一个名为 indexRules 的子节点,并且该节点必须至少有一个子节点。有关更多信息,请参阅 Oak 文档

自定义搜索索引定义节点必须遵循命名惯例 oakpal-custom-search-definitions

  • 密钥:IndexName
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

AEM Cloud Service 要求自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)必须按照内容搜索和索引中所述的特定模式进行命名。

自定义搜索索引定义节点必须使用索引类型 lucene oakpal-index-type-lucene

  • 密钥:IndexType
  • 类型:代码异味
  • 严重性:轻微
  • 开始版本:版本 2021.2.0

AEM Cloud Service 要求自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)必须具有 type 属性的值,设置为 lucene。在迁移到 AEM Cloud Service 之前,必须更新使用旧索引类型的索引编制。有关更多信息,请参阅“内容搜索和索引编制”文档

自定义搜索索引定义节点不得包含名为seed的属性 oakpal-property-name-seed

  • 密钥:IndexSeedProperty
  • 类型:代码异味
  • 严重性:轻微
  • 从以下版本开始:版本 2021.2.0

AEM Cloud Service 禁止自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)包含名为 seed 的属性。在迁移到 AEM Cloud Service 之前,必须更新使用此属性的索引编制。有关更多信息,请参阅“内容搜索和索引编制”文档

自定义搜索索引定义节点不得包含名为reindex的属性 oakpal-reindex-property

  • 密钥:IndexReindexProperty
  • 类型:代码异味
  • 严重性:轻微
  • 从以下版本开始:版本 2021.2.0

AEM Cloud Service 禁止自定义搜索索引定义(即 oak:QueryIndexDefinition 类型的节点)包含名为 reindex 的属性。在迁移到 AEM Cloud Service 之前,必须更新使用此属性的索引编制。有关更多信息,请参阅“内容搜索和索引编制”文档

索引定义节点不得部署在 UI 内容包中 oakpal-ui-content-package

  • :IndexNotUnderUIContent
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.6.0

AEM Cloud Service 禁止在 UI 内容包中部署自定义搜索索引定义(类型为 oak:QueryIndexDefinition 的节点)。

WARNING
建议您尽快解决此问题,因为从 Cloud Manager 2024 年 8 月版本开始,它将导致管道失败

类型damAssetLucene的自定义全文索引定义必须正确添加前缀damAssetLucene oakpal-dam-asset-lucene

  • :CustomFulltextIndexesOfTheDamAssetCheck
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.6.0

AEM Cloud Service 禁止类型 damAssetLucene 的自定义全文索引定义使用除 damAssetLucene 之外的任何内容作为前缀。

WARNING
建议您尽快解决此问题,因为从 Cloud Manager 2024 年 8 月版本开始,它将导致管道失败

索引定义节点不得包含同名的属性 oakpal-index-property-name

  • :DuplicateNameProperty
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.6.0

AEM Cloud Service 禁止自定义搜索索引定义(即,类型为 oak:QueryIndexDefinition 的节点)包含同名的属性。

WARNING
建议您尽快解决此问题,因为从 Cloud Manager 2024 年 8 月版本开始,它将导致管道失败

禁止自定义某些现成的索引定义 oakpal-customizing-ootb-index

  • :RestrictIndexCustomization
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.6.0

AEM Cloud Service 禁止对以下 OOTB 索引进行未经授权的修改:

  • nodetypeLucene
  • slingResourceResolver
  • socialLucene
  • appsLibsLucene
  • authorizables
  • pathReference
WARNING
建议您尽快解决此问题,因为从 Cloud Manager 2024 年 8 月版本开始,它将导致管道失败

分析器中的标记器配置应使用tokenizer名称创建 oakpal-tokenizer

  • :AnalyzerTokenizerConfigCheck
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.6.0

AEM Cloud Service 禁止在分析器中创建名称错误的标记器。标记器应始终定义为 tokenizer

索引定义的配置不应包含空格 oakpal-indexing-definitions-spaces

  • Key: PathSpacesCheckKey
  • 类型:改进
  • 严重性:轻微
  • 开始版本:版本 2024.7.0

AEM Cloud Service 禁止创建包含带空格属性的索引定义。

Dispatcher 优化工具 dispatcher-optimization-tool-rules

以下部分列出了 Cloud Manager 执行的 Dispatcher 优化工具 (DOT) 检查。访问每个检查的链接以查看其 GitHub 定义和详细信息。

recommendation-more-help
c6cdc82b-cee9-48e0-a6ee-48149d5e72c3