カスタムコード品質ルール custom-code-quality-rules
AEM Engineering のベストプラクティスに基づいて、Cloud Manager がコード品質テストの一環として実行するカスタムコード品質ルールについて詳しく説明します。
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
外部ソース(リクエストパラメーターやユーザー生成コンテンツなど)の書式指定文字列を使用すると、アプリケーションがサービス拒否(DoS)攻撃にさらされる可能性があります。書式指定文字列は外部で制御できる場合がありますが、信頼できるソースからのみ許可されます。
非準拠コード 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
ResourceResolverFactory
から取得された ResourceResolver
オブジェクトは、システムリソースを使用します。ResourceResolver
が使用されなくなった場合に、これらのリソースを再利用する指標がありますが、close()
メソッドを呼び出し、開いている ResourceResolver
オブジェクトを明示的に閉じるほうが効率的です。
既存の JCR セッションを使用して作成された ResourceResolver
オブジェクトは明示的に閉じることはできないという一般的な誤解があります。もう 1 つは、これらのオブジェクトを閉じると、基になる JCR セッションを閉じてしまうというの誤解があります。そのようなことはありません。どの方法で ResourceResolver
を開いても、使用しなくなったら閉じる必要があります。ResourceResolver
は Closeable
インターフェイスを実装するので、close()
を明示的に呼び出す代わりに、try-with-resources
構文を使用することもできます。
非準拠コード 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 サーブレットパスを使用しない do-not-use-sling-servlet-paths-to-register-servlets
- キー:CQRules:CQBP-75
- タイプ:コードスメル
- 深刻度:重大
- 最初の対象バージョン:バージョン 2018.4.0
Sling ドキュメントで説明されているように、パスによってサーブレットをバインドすることは推奨されません。パスバインドサーブレットでは、標準 JCR アクセス制御を使用できないので、追加のセキュリティをより厳格にする必要があります。パスバインドサーブレットを使用する代わりに、リポジトリにノードを作成し、リソースタイプによってサーブレットを登録することをお勧めします。
非準拠コード 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
一般に、例外は 1 回だけログに記録する必要があります。複数回ログに記録すると、例外が発生した回数がわからなくなるので、混乱が生じる可能性があります。この問題を引き起こす最も一般的なパターンは、キャッチされた例外をログに記録してスローすることです。
非準拠コード 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);
}
}
ログステートメントの直後にスローステートメントを使用しない 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 メソッドは読み取り専用操作に過ぎず、重要なアクションを構成しません。GET または HEAD 要求に応答して INFO レベルでログに記録すると、大量のログノイズが作成されるので、ログファイル内の有用な情報を特定するのが難しくなります。GET または HEAD リクエストを処理する際に、問題が発生した場合、ログには WARN または ERROR レベルで記録する必要があります。詳細なトラブルシューティング情報の場合、ログには DEBUG またはTRACE レベルで記録する必要があります。
非準拠コード 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
ベストプラクティスとして、ログメッセージは、アプリケーション内での問題の発生場所に関するコンテキスト情報を提供する必要があります。また、スタックトレースを使用してコンテキストを判断することもできます。これにより、一般的にログメッセージが読みやすく、わかりやすくなります。その結果、例外をログに記録する際に、例外のメッセージをログメッセージとして使用するのは適切ではありません。例外メッセージでは、問題が発生した詳細が示されます。これに対し、ログメッセージは、例外が発生した際にアプリケーションが何を実行していたかを読者に通知する必要があります。例外メッセージはログに記録されます。独自のメッセージを指定すると、ログがわかりやすくなります。
非準拠コード 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);
}
}
catch ブロックのログは、WARN または ERROR レベルにする必要がある 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 スケジュールジョブは実行を保証し、クラスター化環境と非クラスター化環境の両方に適しています。
Sling ジョブがクラスター環境で処理される方法について詳しくは、Apache Sling のイベントとジョブの取り扱いを参照してください。
AEM の非推奨 API は使用しない sonarqube-aem-deprecated
- キー:AMSCORE-553
- タイプ:コードスメル/Cloud Service との互換性
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2020.5.0
AEM API の表面は、使用が推奨されず非推奨と見なされる API を識別するために継続的に見直しされます。
多くの場合、これらの API は、標準の Java™ @Deprecated 注釈を使用して非推奨とされ、その結果、squid:CallToDeprecatedMethod
によって識別されます。
ただし、API が AEM のコンテキストで非推奨となるものの、他のコンテキストでは非推奨とならない場合があります。このルールは、この 2 番目のクラスを識別します。
OakPAL コンテンツルール oakpal-rules
以下の節では、Cloud Manager が実行する OakPAL チェックについて説明します。
お客様は、@ProviderType の注釈が付いた製品 API を実装または拡張しないでください product-apis-annotated-with-providertype-should-not-be-implemented-or-extended-by-customers
- キー:CQBP-84
- タイプ:バグ
- 深刻度:致命的
- 最初の対象バージョン:バージョン 2018.7.0
AEM API には、カスタムコードで使用されることを意図していますが実装できない Java™ インターフェイスとクラスが含まれています。例えば、インターフェイス com.day.cq.wcm.api.Page
を実装するのは AEM のみです。
これらのインターフェイスに新しいメソッドを追加しても、既存のコードには影響しないので、新しいメソッドの追加には後方互換性があります。ただし、カスタムコードがこれらのインターフェイスのいずれかを実装する場合、そのカスタムコードによってお客様に後方互換性のリスクがもたらされます。
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
への編集は、アドビの公式チャネルを通じてのみ行います。
パッケージには重複する 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 内でデフォルトのオーサリングモードを定義します。AEM 6.4 以降、クラシック UI は非推奨となったので、デフォルトのオーサリングモードがクラシック UI に設定されている場合、問題が発生するようになりました。
ダイアログボックスを持つコンポーネントはタッチ UI ダイアログボックスが必要 oakpal-components-dialogs
- キー:ComponentWithOnlyClassicUIDialog
- タイプ:コードスメル/Cloud Service との互換性
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2020.5.0
クラシック UI ダイアログを使用する AEM コンポーネントには、クラシック UI をサポートしていない Cloud Service デプロイメントモデルとの最適なオーサリングと互換性を確保するのに、タッチ UI ダイアログも必要です。このルールは、次のシナリオを検証します。
- クラシック UI ダイアログ(
dialog
子ノード)を持つコンポーネントには、対応するタッチ UI ダイアログ(cq:dialog
子ノード)が必要です。 - クラシック UI デザインダイアログ(
design_dialog
ノード)を使用しているコンポーネントには、対応するタッチ UI デザインダイアログ(cq:design_dialog
子ノード)が必要です。 - クラシック UI ダイアログとクラシック UI デザインダイアログの両方を持つコンポーネントには、対応するタッチ UI ダイアログと対応するタッチ UI デザインダイアログの両方が必要です。
AEM 最新化ツールのドキュメントには、コンポーネントをクラシック UI からタッチ UI に変換する方法に関する詳細とツールが記載されています。詳しくは、AEM Modernization Tools のドキュメントを参照してください。
リバースレプリケーションエージェントは使用しない oakpal-reverse-replication
- キー:ReverseReplication
- タイプ:コードスメル/Cloud Service との互換性
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2020.5.0
リバースレプリケーションのサポートは、リリースノート:レプリケーションエージェントの削除で説明しているように、Cloud Service のデプロイメントでは利用できません。
リバースレプリケーションを使用するお客様は、アドビに問い合わせて、代替ソリューションをご利用ください。
プロキシ対応のクライアントライブラリに含まれるリソースは 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 上でアセット処理を行う Assets マイクロサービスに移行すると、AEM のオンプレミスバージョンと AMS バージョンで使用されていたワークフロープロセスが、サポートされなくなる、または不要になります。
AEM Assets as a Cloud Service GitHub リポジトリの移行ツールを使用すると、AEM as a Cloud Service への移行中にワークフローモデルを更新できます。
静的なテンプレートより編集可能なテンプレートの使用を推奨 oakpal-static-template
- キー:StaticTemplateUsage
- タイプ:コードスメル
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2021.2.0
従来、AEM プロジェクトでは静的テンプレートを使用するのが一般的でしたが、最も柔軟性が高く、静的なテンプレートにはない追加機能をサポートしている編集可能なテンプレートを強くお勧めします。詳しくは、ページテンプレート - 編集可能なドキュメントを参照してください。
静的なテンプレートから編集可能なテンプレートへの移行は、AEM 最新化ツールを使用して、大幅に自動化できます。
従来の基盤コンポーネントの使用は推奨されない oakpal-usage-legacy
- キー:LegacyFoundationComponentUsage
- タイプ:コードスメル
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2021.2.0
一部の AEM リリースでは、従来の基盤コンポーネント(/libs/foundation
下のコンポーネントなど)は廃止され、コアコンポーネントに置き換わりました。使用する方法がオーバーレイであろうと継承であろうと、従来の基盤コンポーネントに基づいてカスタムコンポーネントを使用することは、お勧めしません。対応するコアコンポーネントに変換してください。
AEM Modernization Tools を使用すると、この変換が容易になります。
カスタム検索インデックス定義ノードは、/oak:index
の直接の子にする必要がある oakpal-custom-search
- キー: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
カスタム検索インデックス定義ノードに順序なしの子ノードがある場合、トラブルシューティングしにくい問題が発生するおそれがあります。このようなノードを避けるために、oak:QueryIndexDefinition
ノードのすべての子孫ノードは、タイプを nt:unstructured
にすることをお勧めします。
カスタム検索インデックス定義ノードには、子を持つ indexRules
という名前の子ノードを含める oakpal-custom-search-index
- キー:IndexRulesNode
- タイプ:コードスメル
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2021.2.0
適切に定義されたカスタム検索インデックス定義ノードには、indexRules
という名前の子ノードが含まれている必要があり、このノードに少なくとも 1 つの子が必要です。詳しくは、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
タイプのノード)に、値が lucene
に設定された type
プロパティが必要です。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
のノード)をデプロイすることは禁止されています。
damAssetLucene
タイプのカスタムフルテキストインデックス定義には、damAssetLucene
というプレフィックスを正しく付ける oakpal-dam-asset-lucene
- キー:CustomFulltextIndexesOfTheDamAssetCheck
- タイプ:改善点
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2024.6.0
AEM Cloud Service では、damAssetLucene
タイプのカスタムフルテキストインデックス定義に damAssetLucene
以外のプレフィックスを付けることが禁止されています。
インデックス定義ノードに同じ名前のプロパティを含めない oakpal-index-property-name
- キー:DuplicateNameProperty
- タイプ:改善点
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2024.6.0
AEM Cloud Service では、カスタム検索インデックス定義(つまり、タイプ oak:QueryIndexDefinition
のノード)に同じ名前のプロパティを含めることが禁止されています。
特定の標準インデックス定義のカスタマイズは禁止されている oakpal-customizing-ootb-index
- キー:RestrictIndexCustomization
- タイプ:改善点
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2024.6.0
AEM Cloud Service では、次の OOTB インデックスの許可されていない変更が禁止されています。
nodetypeLucene
slingResourceResolver
socialLucene
appsLibsLucene
authorizables
pathReference
アナライザーの tokenizer の設定は、「tokenizer
」という名前で作成する oakpal-tokenizer
- キー:AnalyzerTokenizerConfigCheck
- タイプ:改善点
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2024.6.0
AEM Cloud Service では、アナライザーで正しくない名前の tokenizer を作成することが禁止されています。Tokenizer は、常に tokenizer
として定義する必要があります。
インデックス作成定義の設定にスペースを含めることはできない oakpal-indexing-definitions-spaces
- キー:PathSpacesCheck
- タイプ:改善点
- 深刻度:軽度
- 最初の対象バージョン:バージョン 2024.7.0
AEM Cloud Service では、プロパティにスペースを使用したインデックス作成定義の作成が禁止されています。
Dispatcher 最適化ツール dispatcher-optimization-tool-rules
以下の節では、Cloud Manager で実行される Dispatcher 最適化ツール(DOT)チェックを示します。 各チェックの GitHub 定義と詳細については、リンクを参照してください。