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

org.apache.jackrabbit.oak.upgrade.RepositorySidegrade Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.jackrabbit.oak.upgrade;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;

import javax.jcr.RepositoryException;

import org.apache.jackrabbit.guava.common.collect.ImmutableSet;
import org.apache.commons.lang3.StringUtils;
import org.apache.jackrabbit.JcrConstants;
import org.apache.jackrabbit.oak.api.CommitFailedException;
import org.apache.jackrabbit.oak.api.PropertyState;
import org.apache.jackrabbit.oak.api.Type;
import org.apache.jackrabbit.oak.commons.PathUtils;
import org.apache.jackrabbit.oak.commons.collections.CollectionUtils;
import org.apache.jackrabbit.oak.plugins.document.DocumentNodeState;
import org.apache.jackrabbit.oak.plugins.migration.FilteringNodeState;
import org.apache.jackrabbit.oak.plugins.migration.NodeStateCopier;
import org.apache.jackrabbit.oak.plugins.migration.report.LoggingReporter;
import org.apache.jackrabbit.oak.plugins.migration.report.ReportingNodeState;
import org.apache.jackrabbit.oak.plugins.nodetype.TypePredicate;
import org.apache.jackrabbit.oak.plugins.version.Utils;
import org.apache.jackrabbit.oak.security.internal.SecurityProviderBuilder;
import org.apache.jackrabbit.oak.segment.SegmentNodeState;
import org.apache.jackrabbit.oak.segment.file.FileStore;
import org.apache.jackrabbit.oak.spi.commit.CommitHook;
import org.apache.jackrabbit.oak.spi.commit.CommitInfo;
import org.apache.jackrabbit.oak.spi.commit.CompositeEditorProvider;
import org.apache.jackrabbit.oak.spi.commit.EditorHook;
import org.apache.jackrabbit.oak.spi.commit.EmptyHook;
import org.apache.jackrabbit.oak.spi.security.SecurityConfiguration;
import org.apache.jackrabbit.oak.spi.security.SecurityProvider;
import org.apache.jackrabbit.oak.spi.state.ApplyDiff;
import org.apache.jackrabbit.oak.spi.state.ChildNodeEntry;
import org.apache.jackrabbit.oak.spi.state.NodeBuilder;
import org.apache.jackrabbit.oak.spi.state.NodeState;
import org.apache.jackrabbit.oak.spi.state.NodeStore;
import org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.LoggingCompositeHook;
import org.apache.jackrabbit.oak.upgrade.checkpoint.CheckpointRetriever;
import org.apache.jackrabbit.oak.upgrade.cli.node.FileStoreUtils;
import org.apache.jackrabbit.oak.upgrade.nodestate.MetadataExposingNodeState;
import org.apache.jackrabbit.oak.upgrade.nodestate.NameFilteringNodeState;
import org.apache.jackrabbit.oak.plugins.migration.version.VersionCopyConfiguration;
import org.apache.jackrabbit.oak.plugins.migration.version.VersionHistoryUtil;
import org.apache.jackrabbit.oak.plugins.migration.version.VersionableEditor;
import org.jetbrains.annotations.NotNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import static java.util.Objects.requireNonNull;
import static org.apache.jackrabbit.guava.common.collect.ImmutableSet.copyOf;
import static org.apache.jackrabbit.guava.common.collect.Sets.union;
import static org.apache.jackrabbit.JcrConstants.JCR_PRIMARYTYPE;
import static org.apache.jackrabbit.JcrConstants.JCR_SYSTEM;
import static org.apache.jackrabbit.oak.plugins.migration.NodeStateCopier.copyProperties;
import static org.apache.jackrabbit.oak.spi.security.authorization.permission.PermissionConstants.NT_REP_PERMISSION_STORE;
import static org.apache.jackrabbit.oak.spi.security.authorization.permission.PermissionConstants.REP_PERMISSION_STORE;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.DEFAULT_EXCLUDE_PATHS;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.DEFAULT_INCLUDE_PATHS;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.DEFAULT_MERGE_PATHS;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.calculateEffectiveIncludePaths;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.createIndexEditorProvider;
import static org.apache.jackrabbit.oak.upgrade.RepositoryUpgrade.createTypeEditorProvider;
import static org.apache.jackrabbit.oak.plugins.migration.version.VersionCopier.copyVersionStorage;
import static org.apache.jackrabbit.oak.plugins.migration.version.VersionHistoryUtil.getVersionStorage;

public class RepositorySidegrade {

    private static final Logger LOG = LoggerFactory.getLogger(RepositorySidegrade.class);

    static final int LOG_NODE_COPY = Integer.getInteger("oak.upgrade.logNodeCopy", 10000);

    private static final String WORKSPACE_NAME_PROP = "oak.upgrade.workspaceName";

    /**
     * Target node store.
     */
    private final NodeStore target;

    private final FileStore targetFileStore;

    private final boolean targetHasReferenceableFrozenNode;

    private final NodeStore source;

    /**
     * Paths to include during the copy process. Defaults to the root path "/".
     */
    private Set includePaths = DEFAULT_INCLUDE_PATHS;

    /**
     * Paths to exclude during the copy process. Empty by default.
     */
    private Set excludePaths = DEFAULT_EXCLUDE_PATHS;

    /**
     * Paths to merge during the copy process. Empty by default.
     */
    private Set mergePaths = DEFAULT_MERGE_PATHS;

    private boolean skipCheckpoints = false;

    private boolean forceCheckpoints = false;

    private boolean filterLongNames = true;

    private boolean verify = false;

    private boolean onlyVerify = false;

    private boolean migrateDocumentMetadata = false;

    private List customCommitHooks = null;

    VersionCopyConfiguration versionCopyConfiguration = new VersionCopyConfiguration();

    /**
     * Configures the version storage copy. Be default all versions are copied.
     * One may disable it completely by setting {@code null} here or limit it to
     * a selected date range: {@code }.
     * 
     * @param minDate
     *            minimum date of the versions to copy or {@code null} to
     *            disable the storage version copying completely. Default value:
     *            {@code 1970-01-01 00:00:00}.
     */
    public void setCopyVersions(Calendar minDate) {
        versionCopyConfiguration.setCopyVersions(minDate);
    }

    /**
     * Configures copying of the orphaned version histories (eg. ones that are
     * not referenced by the existing nodes). By default all orphaned version
     * histories are copied. One may disable it completely by setting
     * {@code null} here or limit it to a selected date range:
     * {@code }. 
*
* Please notice, that this option is overriden by the * {@link #setCopyVersions(Calendar)}. You can't copy orphaned versions * older than set in {@link #setCopyVersions(Calendar)} and if you set * {@code null} there, this option will be ignored. * * @param minDate * minimum date of the orphaned versions to copy or {@code null} * to not copy them at all. Default value: * {@code 1970-01-01 00:00:00}. */ public void setCopyOrphanedVersions(Calendar minDate) { versionCopyConfiguration.setCopyOrphanedVersions(minDate); } /** * Creates a tool for copying the full contents of the source repository * to the given target repository. Any existing content in the target * repository will be overwritten. * * @param source source node store * @param target target node store */ public RepositorySidegrade(NodeStore source, NodeStore target) { this.source = source; this.target = target; this.targetHasReferenceableFrozenNode = Utils.isFrozenNodeReferenceable(target.getRoot()); FileStore fs = null; if (target instanceof FileStoreUtils.NodeStoreWithFileStore) { fs = ((FileStoreUtils.NodeStoreWithFileStore) target).getFileStore(); } this.targetFileStore = fs; } /** * Returns the list of custom CommitHooks to be applied before the final * type validation, reference and indexing hooks. * * @return the list of custom CommitHooks */ public List getCustomCommitHooks() { return customCommitHooks; } /** * Sets the list of custom CommitHooks to be applied before the final * type validation, reference and indexing hooks. * * @param customCommitHooks the list of custom CommitHooks */ public void setCustomCommitHooks(List customCommitHooks) { this.customCommitHooks = customCommitHooks; } /** * Sets the paths that should be included when the source repository * is copied to the target repository. * * @param includes Paths to be included in the copy. */ public void setIncludes(@NotNull String... includes) { this.includePaths = copyOf(requireNonNull(includes)); } /** * Sets the paths that should be excluded when the source repository * is copied to the target repository. * * @param excludes Paths to be excluded from the copy. */ public void setExcludes(@NotNull String... excludes) { this.excludePaths = copyOf(requireNonNull(excludes)); } /** * Sets the paths that should be merged when the source repository * is copied to the target repository. * * @param merges Paths to be merged during copy. */ public void setMerges(@NotNull String... merges) { this.mergePaths = copyOf(requireNonNull(merges)); } public void setFilterLongNames(boolean filterLongNames) { this.filterLongNames = filterLongNames; } public void setVerify(boolean verify) { this.verify = verify; } public void setOnlyVerify(boolean onlyVerify) { this.onlyVerify = onlyVerify; } public void setSkipCheckpoints(boolean skipCheckpoints) { this.skipCheckpoints = skipCheckpoints; } public void setForceCheckpoints(boolean forceCheckpoints) { this.forceCheckpoints = forceCheckpoints; } public void setMigrateDocumentMetadata(boolean migrateDocumentMetadata) { this.migrateDocumentMetadata = migrateDocumentMetadata; } /** * Copies the full content from the source to the target repository. *

* The source repository must not be modified while * the copy operation is running to avoid an inconsistent copy. *

* Note that both the source and the target repository must be closed * during the copy operation as this method requires exclusive access * to the repositories. * * @throws RepositoryException if the copy operation fails */ public void copy() throws RepositoryException { try { if (!onlyVerify) { if (VersionHistoryUtil.getVersionStorage(target.getRoot()).exists() && !versionCopyConfiguration.skipOrphanedVersionsCopy()) { LOG.warn("The version storage on destination already exists. Orphaned version histories will be skipped."); versionCopyConfiguration.setCopyOrphanedVersions(null); } copyState(); } if (verify || onlyVerify) { verify(); } } catch (Exception e) { throw new RepositoryException("Failed to copy content", e); } } private void copyState() throws CommitFailedException, RepositoryException, IOException { boolean migrateCheckpoints = true; if (!isCompleteMigration() && !forceCheckpoints) { LOG.info("Checkpoints won't be migrated because of the specified paths"); migrateCheckpoints = false; } if (!versionCopyConfiguration.isCopyAll()) { LOG.info("Checkpoints won't be migrated because of the specified version settings"); migrateCheckpoints = false; } if (skipCheckpoints) { LOG.info("Checkpoints won't be migrated because of the --skip-checkpoints options"); migrateCheckpoints = false; } if (targetExists()) { LOG.info("Checkpoints won't be migrated because the destination repository exists"); migrateCheckpoints = false; } if (migrateCheckpoints) { migrateCheckpoints = migrateWithCheckpoints(); } if (!migrateCheckpoints) { migrateWithoutCheckpoints(); } } private boolean migrateWithCheckpoints() throws CommitFailedException, IOException { List checkpoints = CheckpointRetriever.getCheckpoints(source); if (checkpoints == null) { return false; } Map nameToRevision = new LinkedHashMap<>(); Map checkpointSegmentToDoc = new LinkedHashMap<>(); NodeState initialRoot = target.getRoot(); NodeState targetRoot = initialRoot; NodeState previousRoot = initialRoot; for (CheckpointRetriever.Checkpoint checkpoint : checkpoints) { NodeState checkpointRoot = source.retrieve(checkpoint.getName()); Map checkpointInfo = source.checkpointInfo(checkpoint.getName()); boolean tracePaths; boolean showDiff = false; if (previousRoot == initialRoot) { LOG.info("Migrating first checkpoint: {}", checkpoint.getName()); tracePaths = true; } else { LOG.info("Applying diff to {}", checkpoint.getName()); showDiff = true; tracePaths = false; } LOG.info("Checkpoint expiry time: {}, metadata: {}", checkpoint.getExpiryTime(), checkpointInfo); targetRoot = copyDiffToTarget(previousRoot, checkpointRoot, targetRoot, tracePaths, showDiff); previousRoot = checkpointRoot; String newCheckpointName = target.checkpoint(checkpoint.getExpiryTime() - System.currentTimeMillis(), checkpointInfo); if (checkpointInfo.containsKey("name")) { nameToRevision.put(checkpointInfo.get("name"), newCheckpointName); } checkpointSegmentToDoc.put(checkpoint.getName(), newCheckpointName); } NodeState sourceRoot = source.getRoot(); boolean tracePaths; boolean showDiff = false; if (previousRoot == initialRoot) { LOG.info("No checkpoints found; migrating head"); tracePaths = true; } else { LOG.info("Applying diff to head"); tracePaths = false; showDiff = true; } targetRoot = copyDiffToTarget(previousRoot, sourceRoot, targetRoot, tracePaths, showDiff); LOG.info("Rewriting checkpoint names in /:async {}", nameToRevision); NodeBuilder targetBuilder = targetRoot.builder(); NodeBuilder async = targetBuilder.getChildNode(":async"); for (Map.Entry e : nameToRevision.entrySet()) { async.setProperty(e.getKey(), e.getValue(), Type.STRING); PropertyState temp = async.getProperty(e.getKey() + "-temp"); if (temp == null) { continue; } List tempValues = CollectionUtils.toList(temp.getValue(Type.STRINGS)); for (Map.Entry sToD : checkpointSegmentToDoc.entrySet()) { if (tempValues.contains(sToD.getKey())) { tempValues.set(tempValues.indexOf(sToD.getKey()), sToD.getValue()); } } async.setProperty(e.getKey() + "-temp", tempValues, Type.STRINGS); } target.merge(targetBuilder, EmptyHook.INSTANCE, CommitInfo.EMPTY); return true; } private NodeState copyDiffToTarget(NodeState before, NodeState after, NodeState targetRoot, boolean tracePaths, boolean showDiff) throws IOException, CommitFailedException { NodeBuilder targetBuilder = targetRoot.builder(); if (targetFileStore == null) { NodeState currentRoot = wrapNodeState(after, tracePaths, true); NodeState baseRoot = wrapNodeState(before, false, true); currentRoot.compareAgainstBaseState(baseRoot, new ApplyDiff(targetBuilder)); } else { NodeState currentRoot = wrapNodeState(after, false, true); NodeState baseRoot = wrapNodeState(before, false, true); SegmentNodeState state = PersistingDiff.applyDiffOnNodeState(targetFileStore, baseRoot, currentRoot, targetRoot); state.compareAgainstBaseState(targetRoot, new ApplyDiff(targetBuilder)); } return target.merge(targetBuilder, EmptyHook.INSTANCE, CommitInfo.EMPTY); } private void migrateWithoutCheckpoints() throws CommitFailedException, RepositoryException { final List hooks = new ArrayList<>(); final String workspaceName = getWorkspaceName(); if (customCommitHooks != null) { hooks.addAll(customCommitHooks); } NodeState sourceRoot = wrapNodeState(source.getRoot(), true, false); NodeBuilder targetRoot = target.getRoot().builder(); copyWorkspace(sourceRoot, targetRoot); if (!versionCopyConfiguration.isCopyAll()) { NodeBuilder versionStorage = VersionHistoryUtil.getVersionStorage(targetRoot); if (!versionStorage.exists()) { // it's possible that this is a new repository and the version storage // hasn't been created/copied yet versionStorage = VersionHistoryUtil.createVersionStorage(targetRoot); } if (!versionCopyConfiguration.skipOrphanedVersionsCopy()) { copyVersionStorage(targetRoot, getVersionStorage(sourceRoot), versionStorage, versionCopyConfiguration); } hooks.add(new EditorHook(new VersionableEditor.Provider(sourceRoot, workspaceName, versionCopyConfiguration))); } SecurityProvider security = SecurityProviderBuilder.newBuilder().build(); for (SecurityConfiguration securityConfig : security.getConfigurations()) { hooks.addAll(securityConfig.getCommitHooks(workspaceName)); } // type validation, reference and indexing hooks hooks.add(new EditorHook(new CompositeEditorProvider( createTypeEditorProvider(), createIndexEditorProvider() ))); target.merge(targetRoot, new LoggingCompositeHook(hooks, null, false), CommitInfo.EMPTY); removeVersions(); } private void removeVersions() throws CommitFailedException { NodeState root = target.getRoot(); NodeState wrappedRoot = FilteringNodeState.wrap(PathUtils.ROOT_PATH, root, includePaths, excludePaths, FilteringNodeState.NONE, FilteringNodeState.NONE, targetHasReferenceableFrozenNode); NodeState versionStorage = getVersionStorage(root); List versionablesToStrip = VersionHistoryUtil.getVersionableNodes(wrappedRoot, versionStorage, new TypePredicate(root, JcrConstants.MIX_VERSIONABLE), versionCopyConfiguration.getVersionsMinDate()); if (!versionablesToStrip.isEmpty()) { LOG.info("Removing version histories for included paths"); NodeBuilder newRoot = VersionHistoryUtil.removeVersions(root, versionablesToStrip); target.merge(newRoot, EmptyHook.INSTANCE, CommitInfo.EMPTY); } } private boolean isCompleteMigration() { return includePaths.equals(DEFAULT_INCLUDE_PATHS) && excludePaths.equals(DEFAULT_EXCLUDE_PATHS) && mergePaths.equals(DEFAULT_MERGE_PATHS); } private void copyWorkspace(NodeState sourceRoot, NodeBuilder targetRoot) { final Set includes = calculateEffectiveIncludePaths(includePaths, sourceRoot); ImmutableSet.Builder excludes = new ImmutableSet.Builder<>(); excludes.addAll(excludePaths); if (!versionCopyConfiguration.isCopyAll()) { excludes.add("/jcr:system/jcr:versionStorage"); } excludes.add("/:async"); final Set merges = union(copyOf(this.mergePaths), Set.of("/jcr:system")); NodeStateCopier.builder() .include(includes) .exclude(excludes.build()) .merge(merges) .copy(sourceRoot, targetRoot); if (includePaths.contains("/")) { copyProperties(sourceRoot, targetRoot); } } private String getWorkspaceName() throws RepositoryException { String definedName = System.getProperty(WORKSPACE_NAME_PROP); String detectedName = deriveWorkspaceName(); if (StringUtils.isNotBlank(definedName)) { return definedName; } else if (StringUtils.isNotBlank(detectedName)) { return detectedName; } else { throw new RepositoryException("Can't detect the workspace name. Please use the system property " + WORKSPACE_NAME_PROP + " to set it manually."); } } /** * This method tries to derive the workspace name from the source repository. It uses the * fact that the /jcr:system/rep:permissionStore usually contains just one child * named after the workspace. * * @return the workspace name or null if it can't be derived */ private String deriveWorkspaceName() { NodeState permissionStore = source.getRoot().getChildNode(JCR_SYSTEM).getChildNode(REP_PERMISSION_STORE); List nameCandidates = new ArrayList(); for (ChildNodeEntry e : permissionStore.getChildNodeEntries()) { String primaryType = e.getNodeState().getName(JCR_PRIMARYTYPE); if (NT_REP_PERMISSION_STORE.equals(primaryType)) { nameCandidates.add(e.getName()); } } if (nameCandidates.size() == 1) { return nameCandidates.get(0); } else { return null; } } private void verify() { final NodeState sourceRoot = source.getRoot(); final NodeState targetRoot = target.getRoot(); final NodeState reportingSource = ReportingNodeState.wrap(sourceRoot, new LoggingReporter(LOG, "Verifying", LOG_NODE_COPY, -1)); LOG.info("Verifying whether repositories are identical"); if (targetRoot.compareAgainstBaseState(reportingSource, new LoggingEqualsDiff(LOG, "/"))) { LOG.info("Verification result: both repositories are identical"); } else { LOG.warn("Verification result: repositories are not identical"); } } private NodeState wrapNodeState(NodeState source, boolean tracePaths, boolean filterPaths) { NodeState wrapped = source; if (migrateDocumentMetadata && source instanceof DocumentNodeState) { wrapped = MetadataExposingNodeState.wrap(wrapped); } if (!isCompleteMigration() && filterPaths) { wrapped = FilteringNodeState.wrap("/", wrapped, includePaths, excludePaths, FilteringNodeState.NONE, FilteringNodeState.NONE, targetHasReferenceableFrozenNode); } if (tracePaths) { wrapped = ReportingNodeState.wrap(wrapped, new LoggingReporter(LOG, "Copying", LOG_NODE_COPY, -1)); } if (filterLongNames) { wrapped = NameFilteringNodeState.wrapRoot(wrapped); } return wrapped; } private boolean targetExists() { return target.getRoot().getChildNodeEntries().iterator().hasNext(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy