All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.adobe.acs.commons.oak.impl.EnsureOakIndexJobHandler Maven / Gradle / Ivy

The newest version!
/*
 * ACS AEM Commons
 *
 * Copyright (C) 2013 - 2023 Adobe
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.adobe.acs.commons.oak.impl;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import javax.jcr.Node;
import javax.jcr.Property;
import javax.jcr.PropertyType;
import javax.jcr.RepositoryException;
import javax.jcr.Session;

import org.apache.commons.lang3.ArrayUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.sling.api.resource.LoginException;
import org.apache.sling.api.resource.ModifiableValueMap;
import org.apache.sling.api.resource.PersistenceException;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ResourceResolverFactory;
import org.apache.sling.api.resource.ValueMap;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.adobe.acs.commons.analysis.jcrchecksum.ChecksumGenerator;
import com.adobe.acs.commons.analysis.jcrchecksum.impl.options.CustomChecksumGeneratorOptions;
import com.adobe.acs.commons.oak.impl.EnsureOakIndex.OakIndexDefinitionException;
import com.day.cq.commons.jcr.JcrConstants;
import com.day.cq.commons.jcr.JcrUtil;

public class EnsureOakIndexJobHandler implements Runnable {
    //@formatter:off
    static final Logger log = LoggerFactory.getLogger(EnsureOakIndexJobHandler.class);
    static final String PN_FORCE_REINDEX = "forceReindex";

    static final String PN_DELETE = "delete";

    static final String PN_IGNORE = "ignore";

    static final String PN_DISABLE = "disable";

    static final String NT_OAK_QUERY_INDEX_DEFINITION = "oak:QueryIndexDefinition";

    static final String NT_OAK_UNSTRUCTURED = "oak:Unstructured";

    static final String PN_TYPE = "type";

    static final String DISABLED = "disabled";

    static final String PN_RECREATE_ON_UPDATE = "recreateOnUpdate";

    static final String PN_REINDEX_COUNT = "reindexCount";

    static final String PN_SEED = "seed";

    static final String PN_REINDEX = "reindex";

    static final String NN_FACETS = "facets";

    static final String ENSURE_OAK_INDEX_USER_NAME = "Ensure Oak Index";

    static final String[] MANDATORY_IGNORE_PROPERTIES = {
            // JCR Properties
            JcrConstants.JCR_PRIMARYTYPE,
            JcrConstants.JCR_LASTMODIFIED,
            JcrConstants.JCR_LAST_MODIFIED_BY,
            JcrConstants.JCR_MIXINTYPES,
            JcrConstants.JCR_CREATED,
            JcrConstants.JCR_CREATED_BY,
            // Ensure Oak Index Properties
            PN_RECREATE_ON_UPDATE,
            PN_FORCE_REINDEX,
            PN_DELETE,
            PN_IGNORE,
            PN_DISABLE,
            // Oak Index Properties
            PN_REINDEX,
            PN_REINDEX_COUNT,
            PN_SEED
    };

    static final String[] MANDATORY_EXCLUDE_SUB_TREES = {
            // For the real index definition node
            "[" + NT_OAK_QUERY_INDEX_DEFINITION + "]/" + NN_FACETS + "/" + JcrConstants.JCR_CONTENT,
            // For the ensure oak index definition node
            "[" + NT_OAK_UNSTRUCTURED + "]/" + NN_FACETS + "/" + JcrConstants.JCR_CONTENT
    };

    static final String[] MANDATORY_EXCLUDE_NODE_NAMES = new String[]{ };

    private static final String[] NAME_PROPERTIES = new String[] {"propertyNames", "declaringNodeTypes"} ;

    static final String SERVICE_NAME = "ensure-oak-index";

    private final EnsureOakIndex ensureOakIndex;

    private final List ignoreProperties = new ArrayList<>();
    private final List excludeSubTrees = new ArrayList<>();
    private final List excludeNodeNames = new ArrayList<>();

    private String oakIndexesPath;

    private String ensureDefinitionsPath;
    //@formatter:on

    EnsureOakIndexJobHandler(EnsureOakIndex ensureOakIndex, String oakIndexPath, String ensureDefinitionsPath) {
        this.ensureOakIndex = ensureOakIndex;
        this.oakIndexesPath = oakIndexPath;
        this.ensureDefinitionsPath = ensureDefinitionsPath;

        this.ignoreProperties.addAll(Arrays.asList(MANDATORY_IGNORE_PROPERTIES));
        this.excludeSubTrees.addAll(Arrays.asList(MANDATORY_EXCLUDE_SUB_TREES));
        this.excludeNodeNames.addAll(Arrays.asList(MANDATORY_EXCLUDE_NODE_NAMES));

        if (ensureOakIndex != null) {
            this.ignoreProperties.addAll(ensureOakIndex.getIgnoreProperties());
        }
    }

    @Override
    @SuppressWarnings("squid:S1141")
    public void run() {
        Map authInfo = Collections.singletonMap(ResourceResolverFactory.SUBSERVICE, (Object) SERVICE_NAME);
        try (ResourceResolver resourceResolver = this.ensureOakIndex.getResourceResolverFactory().getServiceResourceResolver(authInfo)) {

            // we should rethink this nested try here ...
            try {
                this.ensure(resourceResolver, ensureDefinitionsPath, oakIndexesPath);
            } catch (IOException e) {
                log.error("Could not ensure management of Oak Index [ {} ]", oakIndexesPath, e);
            }
        } catch (IllegalArgumentException e) {
            log.error("Could not ensure oak indexes due to illegal arguments.",e);
        } catch (LoginException e) {
            log.error("Could not get an admin resource resolver to ensure Oak Indexes", e);
        } catch (Exception e) {
            log.error("Unknown error occurred while ensuring indexes", e);
        }
    }

    /**
     * Main work method. Responsible for ensuring the ensure definitions under srcPath are reflected in the real oak
     * index under oakIndexesPath.
     * 

* The handling is split, so that all re-indexings can be combined into a single commit; this * ensures, that a single repository traversal can be used to reindex all affected indexes. * * @param resourceResolver the resource resolver (must have permissions to read definitions and change indexes) * @param ensureDefinitionsPath the path containing the ensure definitions * @param oakIndexesPath the path of the real oak index * @throws RepositoryException * @throws IOException */ private void ensure(final ResourceResolver resourceResolver, final String ensureDefinitionsPath, final String oakIndexesPath) throws RepositoryException, IOException { final Resource ensureDefinitions = resourceResolver.getResource(ensureDefinitionsPath); final Resource oakIndexes = resourceResolver.getResource(oakIndexesPath); if (ensureDefinitions == null) { throw new IllegalArgumentException("Unable to find Ensure Definitions resource at [ " + ensureDefinitionsPath + " ]"); } else if (oakIndexes == null) { throw new IllegalArgumentException("Unable to find Oak Indexes resource at [ " + oakIndexesPath + " ]"); } final Iterator ensureDefinitionsIterator = ensureDefinitions.listChildren(); if (!ensureDefinitionsIterator.hasNext()) { log.info("Ensure Definitions path [ {} ] does NOT have children to process", ensureDefinitions.getPath()); } final List delayedProcessing = new ArrayList<>(); // First, handle all things that may not result in a a collective re-indexing // Includes: IGNORES, DELETES, DISABLED ensure definitions while (ensureDefinitionsIterator.hasNext()) { final Resource ensureDefinition = ensureDefinitionsIterator.next(); final Resource oakIndex = oakIndexes.getChild(ensureDefinition.getName()); log.debug("Ensuring Oak Index [ {} ] ~> [ {}/{} ]", ensureDefinition.getPath(), oakIndexesPath, ensureDefinition.getName()); if (!handleLightWeightIndexOperations( ensureDefinition, oakIndex)) { delayedProcessing.add(ensureDefinition); } } if (resourceResolver.hasChanges()) { log.info("Saving all DELETES, IGNORES, and DISABLES to [ {} ]", oakIndexesPath); resourceResolver.commit(); log.debug("Commit succeeded"); } // Combine the index updates which will potentially result in a repository traversal into a single commit. // second iteration: handle CREATE, UPDATE and REINDEXING Iterator delayedProcessingEnsureDefinitions = delayedProcessing.iterator(); while (delayedProcessingEnsureDefinitions.hasNext()) { final Resource ensureDefinition = delayedProcessingEnsureDefinitions.next(); final Resource oakIndex = oakIndexes.getChild(ensureDefinition.getName()); handleHeavyWeightIndexOperations(oakIndexes, ensureDefinition, oakIndex); } if (resourceResolver.hasChanges()) { log.info("Saving all CREATE, UPDATES, and RE-INDEXES, re-indexing may start now."); resourceResolver.commit(); log.debug("Commit succeeded"); } } /** * Handle CREATE and UPDATE operations. * * @param oakIndexes * @param ensureDefinition * @param oakIndex * @throws RepositoryException * @throws PersistenceException * @throws IOException */ void handleHeavyWeightIndexOperations(final Resource oakIndexes, final @NotNull Resource ensureDefinition, final @Nullable Resource oakIndex) throws RepositoryException, IOException { final ValueMap ensureDefinitionProperties = ensureDefinition.getValueMap(); try { Resource ensuredOakIndex = null; validateEnsureDefinition(ensureDefinition); if (oakIndex == null) { // CREATE ensuredOakIndex = this.create(ensureDefinition, oakIndexes); // Force re-index if (ensureDefinitionProperties.get(PN_FORCE_REINDEX, false)) { this.forceRefresh(ensuredOakIndex); } } else { // UPDATE boolean forceReindex = ensureDefinitionProperties.get(PN_FORCE_REINDEX, false); if (ensureDefinitionProperties.get(PN_RECREATE_ON_UPDATE, false)) { // Recreate on Update, refresh not required (is implicit) this.delete(oakIndex); this.create(ensureDefinition, oakIndexes); } else { // Normal Update this.update(ensureDefinition, oakIndexes, forceReindex); } } } catch (OakIndexDefinitionException e) { log.error("Skipping processing of {}", ensureDefinition.getPath(), e); } } /** * handle the operations IGNORE, DELETE and DISABLE * * @param ensureDefinition * @param oakIndex * @return true if the definition has been handled; if true the definition needs further processing * @throws RepositoryException * @throws PersistenceException */ boolean handleLightWeightIndexOperations( final @NotNull Resource ensureDefinition, final @Nullable Resource oakIndex) throws RepositoryException, PersistenceException { final ValueMap ensureDefinitionProperties = ensureDefinition.getValueMap(); boolean result = true; if (ensureDefinitionProperties.get(PN_IGNORE, false)) { // IGNORE log.debug("Ignoring index definition at [ {} ]", ensureDefinition.getPath()); } else if (ensureDefinitionProperties.get(PN_DELETE, false)) { // DELETE if (oakIndex != null) { this.delete(oakIndex); } else { // Oak index does not exist log.info("Requesting deletion of a non-existent Oak Index at [ {}/{} ].\nConsider removing the Ensure Definition at [ {} ] if it is no longer needed.", oakIndexesPath, ensureDefinition.getName(), ensureDefinition.getPath()); } } else if (ensureDefinitionProperties.get(PN_DISABLE, false)) { // DISABLE index if (oakIndex != null) { this.disableIndex(oakIndex); } else { // Oak index does not exist log.info("Requesting disable of a non-existent Oak Index at [ {}/{} ].\nConsider removing the Ensure Definition at [ {} ] if it is no longer needed.", oakIndexesPath, ensureDefinition.getName(), ensureDefinition.getPath()); } } else { // handle updates, creates and all reindexing stuff in the second round result = false; } return result; } /** * Forces index refresh for create or updates (that require updating). * * @param oakIndex the index representing the oak index * @throws PersistenceException */ public void forceRefresh(final @NotNull Resource oakIndex) throws PersistenceException { final ModifiableValueMap mvm = oakIndex.adaptTo(ModifiableValueMap.class); if (mvm == null ) { String msg = String.format("Cannot adapt %s to a ModifiableValueMap (permissions?)", oakIndex.getPath()); throw new PersistenceException(msg); } mvm.put(PN_REINDEX, true); log.info("Forcing re-index of [ {} ]", oakIndex.getPath()); } /** * Create the oak index based on the ensure definition. * * @param ensuredDefinition the ensure definition * @param oakIndexes the parent oak index folder * @return the updated oak index resource * @throws PersistenceException * @throws RepositoryException */ public Resource create(final @NotNull Resource ensuredDefinition, final @NotNull Resource oakIndexes) throws RepositoryException { final Node oakIndex = JcrUtil.copy( ensuredDefinition.adaptTo(Node.class), oakIndexes.adaptTo(Node.class), ensuredDefinition.getName()); oakIndex.setPrimaryType(NT_OAK_QUERY_INDEX_DEFINITION); oakIndex.setProperty(JcrConstants.JCR_CREATED, Calendar.getInstance()); oakIndex.setProperty(JcrConstants.JCR_CREATED_BY, ENSURE_OAK_INDEX_USER_NAME); log.info("Created Oak Index at [ {} ] with Ensure Definition [ {} ]", oakIndex.getPath(), ensuredDefinition.getPath()); return ensuredDefinition.getResourceResolver().getResource(oakIndex.getPath()); } /** * Update the oak index with the ensure definition. * * @param ensureDefinition the ensure definition * @param oakIndexes the parent oak index folder * @param forceReindex indicates if a recreate of the index is requested * @return the updated oak index resource * @throws RepositoryException * @throws IOException */ @SuppressWarnings("squid:S3776") public Resource update(final @NotNull Resource ensureDefinition, final @NotNull Resource oakIndexes, boolean forceReindex) throws RepositoryException, IOException { final ValueMap ensureDefinitionProperties = ensureDefinition.getValueMap(); final Resource oakIndex = oakIndexes.getChild(ensureDefinition.getName()); final Node oakIndexNode = oakIndex.adaptTo(Node.class); final Node ensureDefinitionNode = ensureDefinition.adaptTo(Node.class); if (!this.needsUpdate(ensureDefinition, oakIndex)) { if (ensureDefinitionProperties.get(PN_FORCE_REINDEX, false)) { log.info("Skipping update... Oak Index at [ {} ] is the same as [ {} ] and forceIndex flag is ignored", oakIndex.getPath(), ensureDefinition.getPath()); } else { log.info("Skipping update... Oak Index at [ {} ] is the same as [ {} ]", oakIndex.getPath(), ensureDefinition.getPath()); } return null; } // Handle oak:QueryIndexDefinition node // Do NOT delete it as this will delete the existing index below it // Clear out existing properties final Iterator existingOakIndexProperties = copyIterator(oakIndexNode.getProperties()); while(existingOakIndexProperties.hasNext()) { final Property property = existingOakIndexProperties.next(); final String propertyName = property.getName(); if (this.ignoreProperties.contains(propertyName)) { continue; } JcrUtil.setProperty(oakIndexNode, propertyName, null); } // Add new properties final Iterator addProperties = copyIterator(ensureDefinitionNode.getProperties()); while (addProperties.hasNext()) { final Property property = addProperties.next(); if (this.ignoreProperties.contains(property.getName())) { // Skip ignored properties continue; } if (ArrayUtils.contains(NAME_PROPERTIES, property.getName()) && property.getType() != PropertyType.NAME) { log.warn("{}@{} property should be of type: Name[]", oakIndex.getPath(), property.getName()); } JcrUtil.copy(property, oakIndexNode, property.getName()); } JcrUtil.setProperty(oakIndexNode, JcrConstants.JCR_LAST_MODIFIED_BY, ENSURE_OAK_INDEX_USER_NAME); JcrUtil.setProperty(oakIndexNode, JcrConstants.JCR_LASTMODIFIED, Calendar.getInstance()); // Handle all sub-nodes (ex. Lucene Property Indexes) // Delete child nodes Iterator children = oakIndex.listChildren(); while (children.hasNext()) { children.next().adaptTo(Node.class).remove(); } // Deep copy over child nodes children = ensureDefinition.listChildren(); while (children.hasNext()) { final Resource child = children.next(); JcrUtil.copy(child.adaptTo(Node.class), oakIndex.adaptTo(Node.class), child.getName()); } if (forceReindex) { log.info("Updated Oak Index at [ {} ] with configuration [ {} ], triggering reindex", oakIndex.getPath(), ensureDefinition.getPath()); forceRefresh(oakIndex); } else { // A reindexing should be required to make this change effective, so WARN if not present log.warn("Updated Oak Index at [ {} ] with configuration [ {} ], but no reindex requested!", oakIndex.getPath(), ensureDefinition.getPath()); } return oakIndex; } /** * Disables an index, so it's no longer updated by Oak. * * @param oakIndex the index * @throws PersistenceException */ public void disableIndex(@NotNull Resource oakIndex) throws PersistenceException { final ModifiableValueMap oakIndexProperties = oakIndex.adaptTo(ModifiableValueMap.class); oakIndexProperties.put(PN_TYPE, DISABLED); log.info("Disabled index at {}", oakIndex.getPath()); } /** * Determines if the ensure definition is the same as the the same-named oak:index definition. * * @param ensureDefinition the ensure index definition * @param oakIndex the oak index definition * @return true if the ensure definition and the oak index definition are different * @throws IOException * @throws RepositoryException */ boolean needsUpdate(@NotNull Resource ensureDefinition, @NotNull Resource oakIndex) throws IOException, RepositoryException { final Session session = ensureDefinition.getResourceResolver().adaptTo(Session.class); final ChecksumGenerator checksumGenerator = this.ensureOakIndex.getChecksumGenerator(); // Compile checksum for the ensureDefinition node system final CustomChecksumGeneratorOptions ensureDefinitionOptions = new CustomChecksumGeneratorOptions(); ensureDefinitionOptions.addIncludedNodeTypes(new String[]{NT_OAK_UNSTRUCTURED}); ensureDefinitionOptions.addExcludedProperties(this.ignoreProperties); ensureDefinitionOptions.addExcludedSubTrees(this.excludeSubTrees); ensureDefinitionOptions.addExcludedNodeNames(this.excludeNodeNames); final Map srcChecksum = checksumGenerator.generateChecksums(session, ensureDefinition.getPath(), ensureDefinitionOptions); // Compile checksum for the oakIndex node system final CustomChecksumGeneratorOptions oakIndexOptions = new CustomChecksumGeneratorOptions(); oakIndexOptions.addIncludedNodeTypes(new String[]{NT_OAK_QUERY_INDEX_DEFINITION}); oakIndexOptions.addExcludedProperties(this.ignoreProperties); oakIndexOptions.addExcludedSubTrees(this.excludeSubTrees); oakIndexOptions.addExcludedNodeNames(this.excludeNodeNames); final Map destChecksum = checksumGenerator.generateChecksums(session, oakIndex.getPath(), oakIndexOptions); // Compare checksums return !StringUtils.equals(srcChecksum.get(ensureDefinition.getPath()), destChecksum.get(oakIndex.getPath())); } /** * Delete the oak index node. * * @param oakIndex the oak index node to delete * @throws RepositoryException * @throws PersistenceException */ public void delete(final @NotNull Resource oakIndex) throws RepositoryException { if (oakIndex.adaptTo(Node.class) != null) { // Remove the node and its descendants oakIndex.adaptTo(Node.class).remove(); } else { log.warn("Oak Index at [ {} ] could not be adapted to a Node for removal.", oakIndex.getPath()); } } /** * Validate that the ensure definition is in a valid format; uses for create and updates. * * @param ensureDefinition the ensure definition ensureDefinition * @throws RepositoryException * @throws OakIndexDefinitionException */ public void validateEnsureDefinition(@NotNull Resource ensureDefinition) throws RepositoryException, OakIndexDefinitionException { Node node = ensureDefinition.adaptTo(Node.class); if (node == null) { throw new EnsureOakIndex.OakIndexDefinitionException("Resource " + ensureDefinition.getPath() + " cannot be adapted to a Node"); } else if (!node.isNodeType(NT_OAK_UNSTRUCTURED)) { throw new EnsureOakIndex.OakIndexDefinitionException("Resource " + ensureDefinition.getPath() + " is not of jcr:primaryType " + NT_OAK_UNSTRUCTURED); } final ValueMap properties = ensureDefinition.getValueMap(); if (StringUtils.isBlank(properties.get(PN_TYPE, String.class))) { throw new EnsureOakIndex.OakIndexDefinitionException( "Ensure Definition at " + ensureDefinition.getPath() + " missing required property 'type'"); } } /** * Creates a copy of an iterator. This allows us to safely change the underlying structure of the src iterator, without disturbing the wrapping iteration; * @param src the src iterator to copy. * @return a copy of the src iterator. */ private Iterator copyIterator(Iterator src) { List dest = new ArrayList(); while (src.hasNext()) { dest.add(src.next()); } return dest.iterator(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy