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

org.apache.jackrabbit.oak.index.merge.IndexMerge 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.index.merge;

import java.io.File;
import java.io.IOException;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

import javax.jcr.PropertyType;

import org.apache.felix.inventory.Format;
import org.apache.jackrabbit.oak.api.CommitFailedException;
import org.apache.jackrabbit.oak.api.Type;
import org.apache.jackrabbit.oak.commons.PathUtils;
import org.apache.jackrabbit.oak.commons.json.JsonObject;
import org.apache.jackrabbit.oak.commons.json.JsopReader;
import org.apache.jackrabbit.oak.commons.json.JsopTokenizer;
import org.apache.jackrabbit.oak.json.TypeCodes;
import org.apache.jackrabbit.oak.plugins.index.CompositeIndexEditorProvider;
import org.apache.jackrabbit.oak.plugins.index.IndexPathService;
import org.apache.jackrabbit.oak.plugins.index.IndexPathServiceImpl;
import org.apache.jackrabbit.oak.plugins.index.IndexUpdateProvider;
import org.apache.jackrabbit.oak.plugins.index.counter.NodeCounterEditorProvider;
import org.apache.jackrabbit.oak.plugins.index.inventory.IndexDefinitionPrinter;
import org.apache.jackrabbit.oak.plugins.index.lucene.IndexCopier;
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.property.PropertyIndexEditorProvider;
import org.apache.jackrabbit.oak.plugins.index.reference.ReferenceEditorProvider;
import org.apache.jackrabbit.oak.plugins.memory.PropertyStates;
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.spi.blob.BlobStore;
import org.apache.jackrabbit.oak.spi.blob.GarbageCollectableBlobStore;
import org.apache.jackrabbit.oak.spi.commit.CommitInfo;
import org.apache.jackrabbit.oak.spi.commit.EditorHook;
import org.apache.jackrabbit.oak.spi.mount.MountInfoProvider;
import org.apache.jackrabbit.oak.spi.mount.Mounts;
import org.apache.jackrabbit.oak.spi.state.NodeBuilder;
import org.apache.jackrabbit.oak.spi.state.NodeStore;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import joptsimple.OptionParser;
import joptsimple.OptionSet;
import joptsimple.OptionSpec;

/**
 * Merge custom index definitions with out-of-the-box index definitions.
 */
public class IndexMerge {

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

    public static final String OAK_CHILD_ORDER = ":childOrder";

    private EditorHook hook;
    private ExecutorService executorService;

    private boolean quiet;

    public static void main(String... args) throws Exception {
        new IndexMerge().execute(args);
    }

    /**
     * Execute the command.
     *
     * @param args the command line arguments
     */
    public void execute(String... args) throws Exception {
        OptionParser parser = new OptionParser();
        OptionSpec quietOption = parser.accepts("quiet", "be less chatty");
        OptionSpec indexDirectory = parser.accepts("indexDir", "Index directory").
                withRequiredArg();
        Options opts = new Options();
        OptionSet options = opts.parseAndConfigure(parser, args);
        quiet = options.has(quietOption);
        boolean isReadWrite = opts.getCommonOpts().isReadWrite();
        boolean success = true;
        String indexRootDir = indexDirectory.value(options);
        if (indexRootDir == null) {
            throw new IllegalArgumentException("Required argument indexDir missing");
        }
        if (!isReadWrite) {
            log("Repository connected in read-only mode. Use '--read-write' for write operations");
        }
        try (NodeStoreFixture fixture = NodeStoreFixtureProvider.create(opts)) {
            NodeStore nodeStore = fixture.getStore();
            BlobStore blobStore = fixture.getBlobStore();
            if (isReadWrite) {
                if (blobStore == null) {
                    throw new IllegalArgumentException("No blob store specified");
                }
                if (!(blobStore instanceof GarbageCollectableBlobStore)) {
                    throw new IllegalArgumentException("Not a garbage collectable blob store: " + blobStore);
                }
            }
            initHook(indexRootDir, (GarbageCollectableBlobStore) blobStore);

            JsonObject indexes = getIndexDefinitions(nodeStore);
            // the superseded indexes of the old repository
            List supersededKeys = new ArrayList<>(getSupersededIndexDefs(indexes));
            Collections.sort(supersededKeys);

            // keep only new indexes that are not superseded
            Map indexMap = indexes.getChildren();
            for (String superseded : supersededKeys) {
                if (indexMap.containsKey(superseded)) {
                    log("Ignoring superseded index " + superseded);
                    indexMap.remove(superseded);
                }
            }
            Set indexKeys = indexes.getChildren().keySet();

            IndexDefMergerUtils.merge(indexes, indexes);

            Set newIndexKeys =  new HashSet<>(indexes.getChildren().keySet());
            newIndexKeys.removeAll(indexKeys);
            if (newIndexKeys.isEmpty()) {
                log("No indexes to merge");
            }
            for (String newIndexKey : newIndexKeys) {
                log("New index: " + newIndexKey);
                JsonObject merged = indexMap.get(newIndexKey);
                String def = merged.toString();
                log("Merged definition: " + def);
                if (isReadWrite) {
                    storeIndex(nodeStore, newIndexKey, merged);
                }
            }
        }
        if (executorService != null) {
            executorService.shutdown();
        }
        if (!success) {
            System.exit(1);
        }
    }

    private void storeIndex(NodeStore ns, String newIndexName, JsonObject indexDef) {
        NodeBuilder rootBuilder = ns.getRoot().builder();
        NodeBuilder b = rootBuilder;
        for(String p : PathUtils.elements(newIndexName)) {
            b = b.child(p);
        }
        build("  ", b, indexDef);
        try {
            ns.merge(rootBuilder, hook, CommitInfo.EMPTY);
            log("Added index " + newIndexName);
        } catch (CommitFailedException e) {
            LOG.error("Failed to add index " + newIndexName, e);
        }
    }

    private void build(String linePrefix, NodeBuilder builder, JsonObject json) {
        for(Entry e : json.getProperties().entrySet()) {
            String k = e.getKey();
            String value = e.getValue();
            JsopTokenizer tokenizer = new JsopTokenizer(value);
            if (tokenizer.matches('[')) {
                ArrayList list = new ArrayList<>();
                while (!tokenizer.matches(']')) {
                   String jsonString = tokenizer.getToken();
                   list.add(jsonString);
                   tokenizer.matches(',');
                }
                log(linePrefix + "array " + k + " = " + list + " (String[])");
                builder.setProperty(k, list, Type.STRINGS);
            } else if (tokenizer.matches(JsopReader.TRUE)) {
                log(linePrefix + "property " + k + " = true (Boolean)");
                builder.setProperty(k, true);
            } else if (tokenizer.matches(JsopReader.FALSE)) {
                log(linePrefix + "property " + k + " = false (Boolean)");
                builder.setProperty(k, false);
            } else if (tokenizer.matches(JsopReader.STRING)) {
                String jsonString = tokenizer.getToken();
                int split = TypeCodes.split(jsonString);
                if (split != -1) {
                    int type = TypeCodes.decodeType(split, jsonString);
                    String v = TypeCodes.decodeName(split, jsonString);
                    if (type == PropertyType.BINARY) {
                        throw new UnsupportedOperationException();
                    } else {
                        builder.setProperty(PropertyStates.createProperty(k, v, type));
                    }
                }
            } else if (tokenizer.matches(JsopReader.NUMBER)) {
                String num = tokenizer.getToken();
                boolean isDouble = num.indexOf('.') >= 0;
                if (isDouble) {
                    double d = Double.parseDouble(num);
                    log(linePrefix + "property " + k + " = " + d + " (Double)");
                    builder.setProperty(k, d);
                } else {
                    long x = Long.parseLong(num);
                    log(linePrefix + "property " + k + " = " + x + " (Long)");
                    builder.setProperty(k, x);
                }
            }
        }
        ArrayList childOrder = new ArrayList<>();
        for(Entry e : json.getChildren().entrySet()) {
            String k = e.getKey();
            JsonObject el = e.getValue();
            log(linePrefix + "child " + k);
            build(linePrefix + "  ", builder.child(k), (JsonObject) el);
            childOrder.add(k);
        }
        if (!childOrder.isEmpty()) {
            builder.setProperty(OAK_CHILD_ORDER, childOrder, Type.NAMES);
        }
    }

    /**
     * Get the names of the index definitions that are superseded in one of the
     * indexes.
     *
     * @param indexDefs all index definitions
     * @return the superseded indexes
     */
    public static Set getSupersededIndexDefs(JsonObject indexDefs) {
        HashSet supersededIndexes = new HashSet<>();
        for(JsonObject d : indexDefs.getChildren().values()) {
            String supersedes = d.getProperties().get("supersedes");
            if (supersedes != null) {
                JsopTokenizer tokenizer = new JsopTokenizer(supersedes);
                if (tokenizer.matches('[')) {
                    while (!tokenizer.matches(']')) {
                        if (tokenizer.matches(JsopReader.STRING)) {
                            String s = tokenizer.getToken();
                            if (!s.contains("/@")) {
                                supersededIndexes.add(s);
                            }
                        } else {
                            throw new IllegalArgumentException("Unexpected token: " + tokenizer.getToken());
                        }
                        tokenizer.matches(',');
                    }
                }
            }
        }
        return supersededIndexes;
    }

    /**
     * Get the the index definitions from a node store. It uses the index path
     * service and index definition printer from Oak.
     *
     * @param nodeStore the source node store
     * @return a JSON object with all index definitions
     */
    private static JsonObject getIndexDefinitions(NodeStore nodeStore) throws IOException {
        IndexPathService imageIndexPathService = new IndexPathServiceImpl(nodeStore);
        IndexDefinitionPrinter indexDefinitionPrinter = new IndexDefinitionPrinter(nodeStore, imageIndexPathService);
        StringWriter writer = new StringWriter();
        PrintWriter printWriter = new PrintWriter(writer);
        indexDefinitionPrinter.print(printWriter, Format.JSON, false);
        printWriter.flush();
        writer.flush();
        String json = writer.toString();
        return JsonObject.fromJson(json, true);
    }

    private void log(String message) {
        if(!quiet) {
            System.out.println(message);
        }
    }

    private void initHook(String indexRootDir, GarbageCollectableBlobStore blobStore) throws IOException {
        IndexTracker tracker = new IndexTracker();
        executorService = Executors.newFixedThreadPool(2);
        IndexCopier indexCopier = new IndexCopier(executorService, new File(indexRootDir));
        MountInfoProvider mip = createMountInfoProvider();
        LuceneIndexEditorProvider luceneEditor = new LuceneIndexEditorProvider(indexCopier, tracker, null, null, mip);
        luceneEditor.setBlobStore(blobStore);
        CompositeIndexEditorProvider indexEditor = new CompositeIndexEditorProvider(
                luceneEditor,
                new PropertyIndexEditorProvider().with(mip),
                new ReferenceEditorProvider().with(mip),
                new NodeCounterEditorProvider().with(mip)
        );
        IndexUpdateProvider updateProvider = new IndexUpdateProvider(
                indexEditor, "async", false);
        hook = new EditorHook(updateProvider);
    }

    private static MountInfoProvider createMountInfoProvider() {
        // TODO probably need the ability to configure mounts
        return Mounts.newBuilder()
                .mount("libs", true, Arrays.asList(
                        // pathsSupportingFragments
                        "/oak:index/*$"
                ), Arrays.asList(
                        // mountedPaths
                        "/libs",
                        "/apps",
                        "/jcr:system/rep:permissionStore/oak:mount-libs-crx.default"))
                .build();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy