org.apache.jackrabbit.oak.run.FrozenNodeRefsByScanningCommand Maven / Gradle / Ivy
/*
* 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.lang.management.ManagementFactory;
import java.util.Collections;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import javax.jcr.ItemNotFoundException;
import javax.jcr.Node;
import javax.jcr.NodeIterator;
import javax.jcr.Property;
import javax.jcr.PropertyIterator;
import javax.jcr.PropertyType;
import javax.jcr.Repository;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.jcr.SimpleCredentials;
import javax.jcr.Value;
import javax.jcr.query.Query;
import javax.jcr.query.QueryManager;
import javax.jcr.query.QueryResult;
import org.apache.jackrabbit.oak.Oak;
import org.apache.jackrabbit.oak.jcr.Jcr;
import org.apache.jackrabbit.oak.plugins.index.lucene.IndexTracker;
import org.apache.jackrabbit.oak.plugins.index.lucene.LuceneIndexEditorProvider;
import org.apache.jackrabbit.oak.plugins.index.lucene.LuceneIndexProvider;
import org.apache.jackrabbit.oak.plugins.index.lucene.hybrid.DocumentQueue;
import org.apache.jackrabbit.oak.run.cli.CommonOptions;
import org.apache.jackrabbit.oak.run.cli.NodeStoreFixture;
import org.apache.jackrabbit.oak.run.cli.NodeStoreFixtureProvider;
import org.apache.jackrabbit.oak.run.cli.Options;
import org.apache.jackrabbit.oak.run.commons.Command;
import org.apache.jackrabbit.oak.spi.commit.Observer;
import org.apache.jackrabbit.oak.spi.query.QueryIndexProvider;
import org.apache.jackrabbit.oak.spi.state.NodeStore;
import org.apache.jackrabbit.oak.stats.StatisticsProvider;
import org.apache.jackrabbit.guava.common.io.Closer;
import org.apache.jackrabbit.guava.common.util.concurrent.MoreExecutors;
import joptsimple.OptionParser;
import joptsimple.OptionSet;
import joptsimple.OptionSpec;
/**
* Scans and lists all references to nt:frozenNode and returns an exit code of 1 if any are found (0 otherwise).
*
* This variant does a *very expensive repository scan* for all properties formatted as uuid
* ( LIKE \"________-____-____-____-____________\" )
* and checking if any reference points to an nt:frozenNode (under /jcr:system/jcr:versionStorage
* at depth > 7).
*
* Note that any property with uuid that cannot be resolved will *not be reported*, as that
* is a legitimate use case of uuid property use. Only uuids that resolve will be analysed.
*
* Example:
*
* java -mx4g -jar oak-run-*.jar frozennoderefsbyscanning mongodb://localhost/<dbname> -user=admin -password=admin
*
*/
public class FrozenNodeRefsByScanningCommand implements Command {
static {
// disable any query limits as our query is going to be a full scan, and we are aware of it
System.setProperty("oak.queryLimitReads", String.valueOf(Long.MAX_VALUE));
// // disable the WARN of the TraversingCursor
// LoggerContext c = (LoggerContext) LoggerFactory.getILoggerFactory();
// Logger logger = c.getLogger("org.apache.jackrabbit.oak.plugins.index.Cursors$TraversingCursor");
// logger.setLevel(Level.ERROR);
}
public static final String NAME = "frozennoderefsbyscanning";
private final String summary = "Scans repository and lists all references to nt:frozenNode";
@Override
public void execute(String... args) throws Exception {
OptionParser parser = new OptionParser();
Options opts = new Options();
opts.setCommandName(NAME);
opts.setSummary(summary);
opts.setConnectionString(CommonOptions.DEFAULT_CONNECTION_STRING);
OptionSpec userOption = parser.accepts("user", "User name").withOptionalArg().defaultsTo("admin");
OptionSpec passwordOption = parser.accepts("password", "Password").withOptionalArg().defaultsTo("admin");
OptionSet options = opts.parseAndConfigure(parser, args);
System.out.println("Opening nodestore (readOnly=true)...");
// explicitly set readOnly mode (overwriting any possibly set -read-write= option)
NodeStoreFixture nodeStoreFixture = NodeStoreFixtureProvider.create(opts, true);
System.out.println("Nodestore opened.");
int count = uuidscan(userOption, passwordOption, options, nodeStoreFixture);
if (count > 0) {
System.err.println("FAILURE: " + count + " Reference(s) (in any uuid formatted property value) to nt:frozenNode found.");
System.exit(1);
} else {
System.out.println("SUCCESS: No references (in any uuid formatted property value) to nt:frozenNode found.");
}
}
/**
* Scans the repository (via an expensive traversing query, ouch, hence it needs username/password)
* and returns the number of references found. For this, any value formatted like a UUID is considered,
* then verified if it points to an nt:frozenNode.
*/
private int uuidscan(OptionSpec userOption, OptionSpec passwordOption,
OptionSet options, NodeStoreFixture nodeStoreFixture) throws IOException {
NodeStore nodeStore = nodeStoreFixture.getStore();
String user = userOption.value(options);
String password = passwordOption.value(options);
Closer closer = Utils.createCloserWithShutdownHook();
closer.register(nodeStoreFixture);
int count = 0;
try {
System.out.println("Logging in...");
Session session = openSession(nodeStore, "crx.default", user, password);
System.out.println("Logged in, querying...");
QueryManager qm = session.getWorkspace().getQueryManager();
// query for only getting 'Reference' and 'WeakReference' would be :
// SELECT * FROM [nt:base] AS p WHERE PROPERTY(*, 'Reference') IS NOT NULL OR PROPERTY(*, 'WeakReference') IS NOT NULL
Query q = qm.createQuery("SELECT * FROM [nt:base] AS p WHERE PROPERTY(*, '*') LIKE \"________-____-____-____-____________\"", "JCR-SQL2");
QueryResult qr = q.execute();
NodeIterator it = qr.getNodes();
while (it.hasNext()) {
Node n = it.nextNode();
PropertyIterator pit = n.getProperties();
while (pit.hasNext()) {
Property p = pit.nextProperty();
if ("jcr:uuid".equals(p.getName())) {
// jcr:uuid should be skipped as that's the identifier of a node, not a reference
continue;
}
if (!p.isMultiple()) {
String propValue = p.getValue().getString();
if (propValue.matches("........-....-....-....-............")) {
if (verify(session, n, p, propValue)) {
count++;
}
}
} else {
for (Value v : p.getValues()) {
String propValue = v.getString();
if (propValue.matches("........-....-....-....-............")) {
if (verify(session, n, p, propValue)) {
count++;
}
}
}
}
}
}
System.out.println("logout...");
session.logout();
System.out.println("done.");
return count;
} catch (Throwable e) {
throw closer.rethrow(e);
} finally {
closer.close();
}
}
private boolean verify(Session session, Node n, Property p, String propValue) throws RepositoryException {
try {
Node node = session.getNodeByIdentifier(propValue);
String path = node.getPath();
boolean candidate = FrozenNodeRef.isFrozenNodeReferenceCandidate(path);
if (!candidate) {
return false;
}
Property primaryType = node.getProperty("jcr:primaryType");
String primaryTypeValue = primaryType.getString();
boolean isNtFrozenNode = "nt:frozenNode".equals(primaryTypeValue);
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.
return false;
}
String uuid = propValue;
String referrerPath = n.getPath();
String referrerProperty = p.getName();
FrozenNodeRef ref = new FrozenNodeRef(referrerPath, referrerProperty, PropertyType.nameFromValue(p.getType()), uuid, path);
System.out.println(FrozenNodeRef.REFERENCE_TO_NT_FROZEN_NODE_FOUND_PREFIX + ref.toInfoString());
return true;
} catch (ItemNotFoundException notFound) {
// then ignore
return false;
}
}
// from JsonIndexCommand, slightly modified to be able to set the workspace name
public static Session openSession(NodeStore nodeStore, String workspaceName, String user, String password) throws RepositoryException {
if (nodeStore == null) {
return null;
}
StatisticsProvider statisticsProvider = StatisticsProvider.NOOP;
Oak oak = new Oak(nodeStore).with(ManagementFactory.getPlatformMBeanServer());
oak.getWhiteboard().register(StatisticsProvider.class, statisticsProvider, Collections.emptyMap());
LuceneIndexProvider provider = /*JsonIndexCommand.*/createLuceneIndexProvider();
oak.with((QueryIndexProvider) provider).with((Observer) provider).with(/*JsonIndexCommand.*/createLuceneIndexEditorProvider());
Jcr jcr = new Jcr(oak);
jcr.with(workspaceName);
Repository repository = jcr.createRepository();
return repository.login(new SimpleCredentials(user, password.toCharArray()));
}
// from JsonIndexCommand, unmodified
private static LuceneIndexEditorProvider createLuceneIndexEditorProvider() {
LuceneIndexEditorProvider ep = new LuceneIndexEditorProvider();
ScheduledExecutorService executorService = MoreExecutors
.getExitingScheduledExecutorService((ScheduledThreadPoolExecutor) Executors.newScheduledThreadPool(5));
StatisticsProvider statsProvider = StatisticsProvider.NOOP;
int queueSize = Integer.getInteger("queueSize", 1000);
long queueTimeout = Long.getLong("queueTimeoutMillis", 100);
IndexTracker tracker = new IndexTracker();
DocumentQueue queue = new DocumentQueue(queueSize, queueTimeout, tracker, executorService, statsProvider);
ep.setIndexingQueue(queue);
return ep;
}
// from JsonIndexCommand, unmodified
private static LuceneIndexProvider createLuceneIndexProvider() {
return new LuceneIndexProvider();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy