From ef39b74d4d8d23e7439ae0557d87bfc70f1733d8 Mon Sep 17 00:00:00 2001 From: Konrad Windszus Date: Tue, 10 Dec 2024 17:11:31 +0100 Subject: [PATCH] Update index definition naming validation Adobe updated the policy in https://experienceleague.adobe.com/en/docs/experience-manager-cloud-service/content/operations/indexing#preparing-the-new-index-definition Add unit tests This closes #33 --- README.md | 6 +- pom.xml | 19 +++--- .../aem/cloud/AemCloudValidator.java | 29 ++++++---- .../aem/cloud/AemCloudValidatorTest.java | 58 ++++++++++++++++++- 4 files changed, 90 insertions(+), 22 deletions(-) diff --git a/README.md b/README.md index cea8e00..d2e96db 100644 --- a/README.md +++ b/README.md @@ -73,13 +73,13 @@ Currently only Oak index definitions of type `lucene` are supported in AEMaaCS. ## Follow naming policy for Oak index definition node names -There is a mandatory naming policy for Oak index definition node names which enforces them to end with `-custom-`. The format is used in [`IndexName`](https://github.com/apache/jackrabbit-oak/blob/08c7b20e0676739d9c445b5249c3f71004b6b894/oak-search/src/main/java/org/apache/jackrabbit/oak/plugins/index/search/spi/query/IndexName.java#L36) and allows for upgrades of existing index definitions in blue/green deployments. +There is a mandatory naming policy for Oak index definition node names which enforces them to either comply with pattern `--custom-` (customized OOTB index) or `.--custom-` (fully customized index). The format is used in [`IndexName`](https://github.com/apache/jackrabbit-oak/blob/92e9020246a5099d22cd7929a67a03efb49615d3/oak-core/src/main/java/org/apache/jackrabbit/oak/plugins/index/IndexName.java#L35-L44) and allows for upgrades of existing index definitions in blue/green deployments. -Further details in . +Further details in . # Usage with Maven -You can use this validator with the [FileVault Package Maven Plugin][3] in version 1.1.0 or higher like this +You can use this validator with the [FileVault Package Maven Plugin][3] in version 1.4.0 or higher like this ``` diff --git a/pom.xml b/pom.xml index 7ee93d6..c3f7887 100644 --- a/pom.xml +++ b/pom.xml @@ -51,6 +51,7 @@ 3.5.4 + 2.20.8 8 ${java.target.version} UTF-8 @@ -61,7 +62,7 @@ org.apache.jackrabbit.vault vault-validation - 3.4.2 + 3.8.2 org.slf4j @@ -101,18 +102,22 @@ test + + org.apache.jackrabbit + jackrabbit-spi-commons + ${jackrabbit.version} + + + org.apache.jackrabbit + jackrabbit-jcr-commons + ${jackrabbit.version} + javax.jcr jcr 2.0 test - - org.apache.jackrabbit - jackrabbit-jcr-commons - 2.20.0 - test - org.apache.jackrabbit oak-jackrabbit-api diff --git a/src/main/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidator.java b/src/main/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidator.java index 1f4cd49..cac817e 100644 --- a/src/main/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidator.java +++ b/src/main/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidator.java @@ -22,12 +22,16 @@ import java.util.Collections; import java.util.regex.Pattern; +import org.apache.jackrabbit.spi.Name; +import org.apache.jackrabbit.spi.commons.name.NameFactoryImpl; +import org.apache.jackrabbit.util.Text; import org.apache.jackrabbit.vault.packaging.PackageProperties; import org.apache.jackrabbit.vault.packaging.PackageType; import org.apache.jackrabbit.vault.util.Constants; -import org.apache.jackrabbit.vault.util.DocViewNode; +import org.apache.jackrabbit.vault.util.DocViewNode2; import org.apache.jackrabbit.vault.validation.spi.DocumentViewXmlValidator; import org.apache.jackrabbit.vault.validation.spi.MetaInfPathValidator; +import org.apache.jackrabbit.vault.validation.spi.NodeContext; import org.apache.jackrabbit.vault.validation.spi.NodePathValidator; import org.apache.jackrabbit.vault.validation.spi.PropertiesValidator; import org.apache.jackrabbit.vault.validation.spi.ValidationContext; @@ -40,7 +44,7 @@ public class AemCloudValidator implements NodePathValidator, MetaInfPathValidato static final String VIOLATION_MESSAGE_READONLY_MUTABLE_PATH = "Using mutable nodes in this repository location is only allowed in author-specific packages as it is not writable by the underlying service user on a publish instance. Consider to use repoinit scripts instead or move that content to another location. Further details at https://experienceleague.adobe.com/docs/experience-manager-learn/cloud-service/debugging/debugging-aem-as-a-cloud-service/build-and-deployment.html?lang=en#including-%2Fvar-in-content-package"; static final String VIOLATION_MESSAGE_INSTALL_HOOK_IN_MUTABLE_PACKAGE = "Using install hooks in mutable content packages leads to deployment failures as the underlying service user on the publish does not have the right to execute those."; - static final String VIOLATION_MESSAGE_INVALID_INDEX_DEFINITION_NODE_NAME = "All Oak index definition node names must end with '-custom-' but found name '%s'. Further details at https://experienceleague.adobe.com/docs/experience-manager-cloud-service/operations/indexing.html?lang=en#how-to-use"; + static final String VIOLATION_MESSAGE_INVALID_INDEX_DEFINITION_NODE_NAME = "All Oak index definition node names must follow scheme '--custom-' (for a customized OOTB index) or '.--custom-' (for a fully customized index) but found name '%s'. Further details at https://experienceleague.adobe.com/en/docs/experience-manager-cloud-service/content/operations/indexing#preparing-the-new-index-definition"; static final String VIOLATION_MESSAGE_LIBS_NODES = "Nodes below '/libs' may be overwritten by future product upgrades. Rather use '/apps'. Further details at https://experienceleague.adobe.com/docs/experience-manager-cloud-service/implementing/developing/full-stack/overlays.html?lang=en#developing"; static final String VIOLATION_MESSAGE_MUTABLE_NODES_IN_MIXED_PACKAGE = "Mutable nodes in mixed package types are not installed!"; static final String VIOLATION_MESSAGE_MUTABLE_NODES_AND_IMMUTABLE_NODES_IN_SAME_PACKAGE = "Mutable and immutable nodes must not be mixed in the same package. You must separate those into two packages and give them both a dedicated package type!"; @@ -48,7 +52,11 @@ public class AemCloudValidator implements NodePathValidator, MetaInfPathValidato // this path is relative to META-INF private static final Path INSTALL_HOOK_PATH = Paths.get(Constants.VAULT_DIR, Constants.HOOKS_DIR); - private static final Pattern INDEX_DEFINITION_NAME_PATTERN = Pattern.compile(".*-custom-\\d++"); + /** + * The allowed patterns are defined in https://experienceleague.adobe.com/en/docs/experience-manager-cloud-service/content/operations/indexing#preparing-the-new-index-definition + */ + private static final Pattern INDEX_DEFINITION_NAME_PATTERN = Pattern.compile(".*-\\d++-custom-\\d++"); + private static final Collection IMMUTABLE_PATH_PREFIXES = Arrays.asList("/apps", "/libs", "/oak:index"); private static final Collection WRITABLE_PATHS_BY_DISTRIBUTION_IMPORTER = Arrays.asList( "/content", // access provided by system user content-writer-service and sling-distribution-importer @@ -69,6 +77,7 @@ public class AemCloudValidator implements NodePathValidator, MetaInfPathValidato private boolean hasImmutableNodes; private static final int MAX_NUM_VIOLATIONS_PER_TYPE = 5; + private static final Name PN_TYPE = NameFactoryImpl.getInstance().create(Name.NS_DEFAULT_URI, "type"); private int numVarNodeViolations = 0; private int numLibNodeViolations = 0; private int numMutableNodeViolations = 0; @@ -196,19 +205,19 @@ static boolean isPackagePathInstalledConditionally(String runMode, Path packageR } @Override - public @Nullable Collection validate(@NotNull DocViewNode node, @NotNull String nodePath, - @NotNull Path filePath, boolean isRoot) { - if ("oak:QueryIndexDefinition".equals(node.primary)) { + public @Nullable Collection validate(@NotNull DocViewNode2 node, @NotNull NodeContext nodeContext, boolean isRoot) { + if ("oak:QueryIndexDefinition".equals(node.getPrimaryType().orElse(""))) { Collection messages = new ArrayList<>(); - String indexType = node.getValue("{}type"); + String indexType = node.getPropertyValue(PN_TYPE).orElse(""); if (!"lucene".equals(indexType)) { messages.add(new ValidationMessage(defaultSeverity, String.format(VIOLATION_MESSAGE_NON_LUCENE_TYPE_INDEX_DEFINITION, indexType))); } - // check node name - if (!INDEX_DEFINITION_NAME_PATTERN.matcher(node.name).matches()) { + // check node name (jcr qualified name as contained in the path) + String qualifiedName = Text.getName(nodeContext.getNodePath()); + if (!INDEX_DEFINITION_NAME_PATTERN.matcher(qualifiedName).matches()) { messages.add(new ValidationMessage(defaultSeverity, - String.format(VIOLATION_MESSAGE_INVALID_INDEX_DEFINITION_NODE_NAME, node.name))); + String.format(VIOLATION_MESSAGE_INVALID_INDEX_DEFINITION_NODE_NAME, qualifiedName))); } return messages; } diff --git a/src/test/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidatorTest.java b/src/test/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidatorTest.java index 3ea6eb6..eb46a54 100644 --- a/src/test/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidatorTest.java +++ b/src/test/java/biz/netcentric/filevault/validator/aem/cloud/AemCloudValidatorTest.java @@ -15,16 +15,25 @@ import java.nio.file.Paths; import java.util.ArrayList; +import java.util.Arrays; import java.util.Collection; +import java.util.List; import java.util.Optional; +import org.apache.jackrabbit.spi.Name; +import org.apache.jackrabbit.spi.commons.name.NameConstants; +import org.apache.jackrabbit.spi.commons.name.NameFactoryImpl; import org.apache.jackrabbit.vault.packaging.PackageType; +import org.apache.jackrabbit.vault.util.DocViewNode2; +import org.apache.jackrabbit.vault.util.DocViewProperty2; +import org.apache.jackrabbit.vault.validation.spi.NodeContext; import org.apache.jackrabbit.vault.validation.spi.ValidationMessage; import org.apache.jackrabbit.vault.validation.spi.ValidationMessageSeverity; +import org.apache.jackrabbit.vault.validation.spi.util.NodeContextImpl; import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.Test; -import static biz.netcentric.filevault.validator.aem.cloud.AemCloudValidator.VIOLATION_MESSAGE_INSTALL_HOOK_IN_MUTABLE_PACKAGE; +import static org.junit.jupiter.api.Assertions.assertEquals; class AemCloudValidatorTest { @@ -75,7 +84,7 @@ void testAllowHooksInMutableContent() { Collection messages = new ArrayList<>(); Optional.ofNullable(validator.validateMetaInfPath(Paths.get("vault/hooks/install-hook.jar"))).ifPresent(messages::addAll); Optional.ofNullable(validator.done()).ifPresent(messages::addAll); - Assertions.assertTrue(messages.stream().anyMatch(message -> message.getMessage().equals(VIOLATION_MESSAGE_INSTALL_HOOK_IN_MUTABLE_PACKAGE))); + Assertions.assertTrue(messages.stream().anyMatch(message -> message.getMessage().equals(AemCloudValidator.VIOLATION_MESSAGE_INSTALL_HOOK_IN_MUTABLE_PACKAGE))); validator = new AemCloudValidator(true, false, true, PackageType.CONTENT, null, ValidationMessageSeverity.ERROR); messages = new ArrayList<>(); @@ -84,4 +93,49 @@ void testAllowHooksInMutableContent() { Assertions.assertTrue(messages.isEmpty()); } + @Test + void testValidIndexDefinitions() { + AemCloudValidator validator = new AemCloudValidator(true, false, false, PackageType.CONTENT, null, ValidationMessageSeverity.ERROR); + Collection messages = new ArrayList<>(); + List properties = Arrays.asList( + new DocViewProperty2(NameConstants.JCR_PRIMARYTYPE, "oak:QueryIndexDefinition"), + new DocViewProperty2(NameFactoryImpl.getInstance().create(Name.NS_DEFAULT_URI, "type"), "lucene")); + // valid lucene index definition + NodeContext context = new NodeContextImpl("/oak:index/prefix.myindex-1-custom-1", Paths.get("_oak_index/test"),Paths.get("./jcr_root")); + DocViewNode2 node = new DocViewNode2(NameConstants.JCR_ROOT, properties); + Optional.ofNullable(validator.validate(node, context, true)).ifPresent(messages::addAll); + Assertions.assertTrue(messages.isEmpty()); + context = new NodeContextImpl("/oak:index/productindex-1-custom-1", Paths.get("_oak_index/test"),Paths.get("./jcr_root")); + Optional.ofNullable(validator.validate(node, context, true)).ifPresent(messages::addAll); + Assertions.assertTrue(messages.isEmpty()); + } + + @Test + void testInvalidLuceneIndexDefinitions() { + AemCloudValidator validator = new AemCloudValidator(true, false, false, PackageType.CONTENT, null, ValidationMessageSeverity.ERROR); + Collection messages = new ArrayList<>(); + List properties = Arrays.asList( + new DocViewProperty2(NameConstants.JCR_PRIMARYTYPE, "oak:QueryIndexDefinition"), + new DocViewProperty2(NameFactoryImpl.getInstance().create(Name.NS_DEFAULT_URI, "type"), "lucene")); + NodeContext context = new NodeContextImpl("/oak:index/myindex", Paths.get("_oak_index/test"),Paths.get("./jcr_root")); + DocViewNode2 node = new DocViewNode2(NameConstants.JCR_ROOT, properties); + Optional.ofNullable(validator.validate(node, context, true)).ifPresent(messages::addAll); + assertEquals(1, messages.size()); + assertEquals(String.format(AemCloudValidator.VIOLATION_MESSAGE_INVALID_INDEX_DEFINITION_NODE_NAME, "myindex"), messages.iterator().next().getMessage()); + } + + @Test + void testInvalidPropertyIndexDefinition() { + AemCloudValidator validator = new AemCloudValidator(true, false, false, PackageType.CONTENT, null, ValidationMessageSeverity.ERROR); + Collection messages = new ArrayList<>(); + List properties = Arrays.asList( + new DocViewProperty2(NameConstants.JCR_PRIMARYTYPE, "oak:QueryIndexDefinition"), + new DocViewProperty2(NameFactoryImpl.getInstance().create(Name.NS_DEFAULT_URI, "type"), "property")); + // invalid property index definition + DocViewNode2 node = new DocViewNode2(NameConstants.JCR_ROOT, properties); + NodeContext context = new NodeContextImpl("/oak:index/prefix.myindex-1-custom-1", Paths.get("_oak_index/test"),Paths.get("./jcr_root")); + Optional.ofNullable(validator.validate(node, context, true)).ifPresent(messages::addAll); + assertEquals(1, messages.size()); + assertEquals(String.format(AemCloudValidator.VIOLATION_MESSAGE_NON_LUCENE_TYPE_INDEX_DEFINITION, "property"), messages.iterator().next().getMessage()); + } }