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

org.apache.jackrabbit.oak.run.FrozenNodeRefsUsingIndexCommand Maven / Gradle / Ivy

There is a newer version: 1.72.0
Show 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.run;

import java.io.IOException;
import java.util.LinkedList;
import java.util.List;

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.run.Utils.NodeStoreOptions;
import org.apache.jackrabbit.oak.run.commons.Command;
import org.apache.jackrabbit.oak.spi.state.ChildNodeEntry;
import org.apache.jackrabbit.oak.spi.state.NodeState;
import org.apache.jackrabbit.oak.spi.state.NodeStore;

import org.apache.jackrabbit.guava.common.io.Closer;

/**
 * Scans and lists all references to nt:frozenNode and returns an exit code of 1 if any are found (0 otherwise).
 * 

* This variant uses the /oak:index/reference by scanning through that list (only) * and checking if any reference points to an nt:frozenNode (under /jcr:system/jcr:versionStorage * at depth > 7). *

* Example: *

 * java -mx4g -jar oak-run-*.jar frozennoderefsusingindex mongodb://localhost/<dbname>
 * 
*/ public class FrozenNodeRefsUsingIndexCommand implements Command { public static final String NAME = "frozennoderefsusingindex"; @Override public void execute(String... args) throws Exception { String help = NAME + " {||} [options]"; Utils.NodeStoreOptions nopts = new Utils.NodeStoreOptions(help).parse(args); int count = countNtFrozenNodeReferences(nopts); if (count > 0) { System.err.println("FAILURE: " + count + " Reference(s) (in /oak:index/references) to nt:frozenNode found."); System.exit(1); } else { System.out.println("SUCCESS: No references (in /oak:index/references) to nt:frozenNode found."); } } /** * Browses through /oak:index/references and counts how many references therein are * to nt:frozenNode. */ private static int countNtFrozenNodeReferences(NodeStoreOptions nopts) throws IOException { Closer closer = Utils.createCloserWithShutdownHook(); try { // explicitly set readOnly mode System.out.println("Opening NodeStore in read-only mode."); NodeStore store = Utils.bootstrapNodeStore(nopts, closer, true); NodeState root = store.getRoot(); NodeState oakIndex = root.getChildNode("oak:index"); NodeState refIndex = oakIndex.getChildNode("reference"); NodeState uuidIndex = oakIndex.getChildNode("uuid"); NodeState uuids = uuidIndex.getChildNode(":index"); System.out.println("Scanning ... refindex = " + refIndex); NodeState references = refIndex.getChildNode(":references"); int count = 0; for (ChildNodeEntry child : references.getChildNodeEntries()) { String uuid = child.getName(); NodeState uuidRef = uuids.getChildNode(uuid); PropertyState entry = uuidRef.getProperty("entry"); Iterable uuidRefPaths = entry.getValue(Type.STRINGS); for (String uuidRefPath : uuidRefPaths) { if (!FrozenNodeRef.isFrozenNodeReferenceCandidate(uuidRefPath)) { continue; } NodeState p = getNodeByPath(root, uuidRefPath); PropertyState primaryType = p.getProperty("jcr:primaryType"); boolean isNtFrozenNode = "nt:frozenNode".equals(primaryType.getValue(Type.NAME)); if (!isNtFrozenNode) { // this is where we ultimately have to continue out in any case - as only an nt:frozenNode // is what we're interested in. continue; } count += countFrozenNodeReferences(root, child, uuidRefPath, p); } } System.out.println("Scanning done."); return count; } catch (Throwable e) { throw closer.rethrow(e); } finally { closer.close(); } } private static int countFrozenNodeReferences(NodeState root, ChildNodeEntry referenceChild, String uuidRefPath, NodeState resolvedNodeState) { String uuid = referenceChild.getName(); List paths = new LinkedList(); scanReferrerPaths(paths, "/", referenceChild.getNodeState()); int count = 0; for (String referrerPropertyPath : paths) { String referrerPath = PathUtils.getAncestorPath(referrerPropertyPath, 1); String referrerProperty = PathUtils.getName(referrerPropertyPath); FrozenNodeRef ref = new FrozenNodeRef(referrerPath, referrerProperty, "Reference", uuid, uuidRefPath); System.out.println(FrozenNodeRef.REFERENCE_TO_NT_FROZEN_NODE_FOUND_PREFIX + ref.toInfoString()); count++; } return count; } private static void scanReferrerPaths(List paths, String parentPath, NodeState referenceChildNodeState) { for (ChildNodeEntry e : referenceChildNodeState.getChildNodeEntries()) { String childName = e.getName(); NodeState childNode = e.getNodeState(); String childPath = PathUtils.concat(parentPath, childName); PropertyState p = childNode.getProperty("match"); if (p != null && p.getValue(Type.BOOLEAN)) { paths.add(childPath); } scanReferrerPaths(paths, childPath, childNode); } } private static NodeState getNodeByPath(NodeState base, String path) { Iterable elems = PathUtils.elements(path); NodeState n = base; for (String elem : elems) { n = n.getChildNode(elem); } return n; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy