org.apache.jackrabbit.core.persistence.mem.InMemBundlePersistenceManager.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.jackrabbit.core.persistence.mem.InMemBundlePersistenceManager.java

Source

/*
 * 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.core.persistence.mem;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

import javax.jcr.RepositoryException;

import org.apache.commons.io.IOUtils;
import org.apache.jackrabbit.core.fs.FileSystem;
import org.apache.jackrabbit.core.fs.FileSystemResource;
import org.apache.jackrabbit.core.fs.local.LocalFileSystem;
import org.apache.jackrabbit.core.id.NodeId;
import org.apache.jackrabbit.core.id.PropertyId;
import org.apache.jackrabbit.core.persistence.PMContext;
import org.apache.jackrabbit.core.persistence.bundle.AbstractBundlePersistenceManager;
import org.apache.jackrabbit.core.persistence.util.BLOBStore;
import org.apache.jackrabbit.core.persistence.util.BundleBinding;
import org.apache.jackrabbit.core.persistence.util.ErrorHandling;
import org.apache.jackrabbit.core.persistence.util.FileSystemBLOBStore;
import org.apache.jackrabbit.core.persistence.util.NodeInfo;
import org.apache.jackrabbit.core.persistence.util.NodePropBundle;
import org.apache.jackrabbit.core.persistence.util.Serializer;
import org.apache.jackrabbit.core.state.ItemStateException;
import org.apache.jackrabbit.core.state.NoSuchItemStateException;
import org.apache.jackrabbit.core.state.NodeReferences;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * <code>BundleInMemPersistenceManager</code> is a <code>HashMap</code>-based
 * <code>PersistenceManager</code> for Jackrabbit that keeps all data in memory
 * and that is capable of storing and loading its contents using a simple custom
 * binary serialization format (see {@link Serializer}).
 * <p/>
 * It is configured through the following properties:
 * <ul>
 * <li><code>initialCapacity</code>: initial capacity of the hash map used to store the data</li>
 * <li><code>loadFactor</code>: load factor of the hash map used to store the data</li>
 * <li><code>persistent</code>: if <code>true</code> the contents of the hash map
 * is loaded on startup and stored on shutdown;
 * if <code>false</code> nothing is persisted</li>
 * <li><code>useFileBlobStore</code>: if <code>true</code> the contents of the blobs
 * will be directly stored on the file system instead of in memory.</li>
 * <li><code>minBlobSize</code> use blob store for binaries properties larger 
 * than minBlobSite (bytes). Default is 4096.</li>
 * </ul>
 * <p/>
 * <b>Please note that this class should only be used for testing purposes.</b>
 *
 */
public class InMemBundlePersistenceManager extends AbstractBundlePersistenceManager {

    /** the default logger */
    private static Logger log = LoggerFactory.getLogger(InMemBundlePersistenceManager.class);

    /** flag indicating if this manager was initialized */
    protected boolean initialized;

    /**
     * Path where bundles are stored on shutdown.
     * (if <code>persistent==true</code>)
     */
    protected static final String BUNDLE_FILE_PATH = "/data/.bundle.bin";

    /**
     * Path where blobs are stored on shutdown.
     * (if <code>persistent==true</code> and <code>useFileBlobStore==false</code>)
     */
    protected static final String BLOBS_FILE_PATH = "/data/.blobs.bin";

    /**
     * Path where references are stored on shutdown.
     * (if <code>persistent==true</code>)
     */
    protected static final String REFS_FILE_PATH = "/data/.refs.bin";

    /**
     * File system where the content of the hash maps are read from/written to
     * (if <code>persistent==true</code>)
     */
    protected FileSystem wspFS;

    /**
     * File system where BLOB data is stored.
     * (if <code>useFileBlobStore==true</code>)
     */
    protected FileSystem blobFS;

    /** 
     * Initial size of buffer used to serialize objects. 
     */
    protected static final int INITIAL_BUFFER_SIZE = 1024;

    /**
     * the minimum size of a property until it gets written to the blob store
     * @see #setMinBlobSize(String)
     */
    private int minBlobSize = 0x1000;

    /**
     * Flag for error handling.
     */
    protected ErrorHandling errorHandling = new ErrorHandling();

    /**
     * the bundle binding
     */
    protected BundleBinding binding;

    /**
     * Bundle memory store.
     */
    protected Map<NodeId, byte[]> bundleStore;

    /**
     * References memory store.
     */
    protected Map<NodeId, byte[]> refsStore;

    /**
     * Blob store.
     */
    protected BLOBStore blobStore;

    /**
     * Blobs memory store used by the InMemBlobStore.
     * (if <code>useFileBlobStore==false</code>)
     */
    private Map<String, byte[]> blobs;

    /**
     *  initial capacity
     */
    protected int initialCapacity = 32768;

    /**
     *  load factor for the hash map
     */
    protected float loadFactor = 0.75f;

    /**
     *  Should hash map be persisted?
     */
    protected boolean persistent = true;

    /**
     * Store blobs on file system instead of memory.
     */
    protected boolean useFileBlobStore = false;

    /**
     * Creates a new <code>InMemBundlePersistenceManager</code> instance.
     */
    public InMemBundlePersistenceManager() {
        initialized = false;
    }

    public void setInitialCapacity(int initialCapacity) {
        this.initialCapacity = initialCapacity;
    }

    public void setInitialCapacity(String initialCapacity) {
        this.initialCapacity = Integer.parseInt(initialCapacity);
    }

    public String getInitialCapacity() {
        return Integer.toString(initialCapacity);
    }

    public void setLoadFactor(float loadFactor) {
        this.loadFactor = loadFactor;
    }

    public void setLoadFactor(String loadFactor) {
        this.loadFactor = Float.parseFloat(loadFactor);
    }

    public String getLoadFactor() {
        return Float.toString(loadFactor);
    }

    public boolean isPersistent() {
        return persistent;
    }

    public void setPersistent(boolean persistent) {
        this.persistent = persistent;
    }

    public void setPersistent(String persistent) {
        this.persistent = Boolean.valueOf(persistent).booleanValue();
    }

    public void setUseFileBlobStore(boolean useFileBlobStore) {
        this.useFileBlobStore = useFileBlobStore;
    }

    public void setUseFileBlobStore(String useFileBlobStore) {
        this.useFileBlobStore = Boolean.valueOf(useFileBlobStore).booleanValue();
    }

    public String getMinBlobSize() {
        return String.valueOf(minBlobSize);
    }

    public void setMinBlobSize(String minBlobSize) {
        this.minBlobSize = Integer.decode(minBlobSize).intValue();
    }

    /**
     * Reads the content of the hash maps from the file system
     *
     * @throws Exception if an error occurs
     */
    public synchronized void loadContents() throws Exception {
        // read item states
        FileSystemResource fsRes = new FileSystemResource(wspFS, BUNDLE_FILE_PATH);
        if (!fsRes.exists()) {
            return;
        }
        BufferedInputStream bis = new BufferedInputStream(fsRes.getInputStream());
        DataInputStream in = new DataInputStream(bis);

        try {
            int n = in.readInt(); // number of entries
            while (n-- > 0) {
                String s = in.readUTF(); // id
                NodeId id = NodeId.valueOf(s);
                int length = in.readInt(); // data length
                byte[] data = new byte[length];
                in.readFully(data); // data
                // store in map
                bundleStore.put(id, data);
            }
        } finally {
            in.close();
        }

        // read references
        fsRes = new FileSystemResource(wspFS, REFS_FILE_PATH);
        bis = new BufferedInputStream(fsRes.getInputStream());
        in = new DataInputStream(bis);

        try {
            int n = in.readInt(); // number of entries
            while (n-- > 0) {
                String s = in.readUTF(); // target id
                NodeId id = NodeId.valueOf(s);
                int length = in.readInt(); // data length
                byte[] data = new byte[length];
                in.readFully(data); // data
                // store in map
                refsStore.put(id, data);
            }
        } finally {
            in.close();
        }

        if (!useFileBlobStore) {
            // read blobs
            fsRes = new FileSystemResource(wspFS, BLOBS_FILE_PATH);
            bis = new BufferedInputStream(fsRes.getInputStream());
            in = new DataInputStream(bis);

            try {
                int n = in.readInt(); // number of entries
                while (n-- > 0) {
                    String id = in.readUTF(); // id
                    int length = in.readInt(); // data length
                    byte[] data = new byte[length];
                    in.readFully(data); // data
                    // store in map
                    blobs.put(id, data);
                }
            } finally {
                in.close();
            }
        }
    }

    /**
     * Writes the content of the hash maps stores to the file system.
     *
     * @throws Exception if an error occurs
     */
    public synchronized void storeContents() throws Exception {
        // write bundles
        FileSystemResource fsRes = new FileSystemResource(wspFS, BUNDLE_FILE_PATH);
        fsRes.makeParentDirs();
        BufferedOutputStream bos = new BufferedOutputStream(fsRes.getOutputStream());
        DataOutputStream out = new DataOutputStream(bos);

        try {
            out.writeInt(bundleStore.size()); // number of entries
            // entries
            for (NodeId id : bundleStore.keySet()) {
                out.writeUTF(id.toString()); // id

                byte[] data = bundleStore.get(id);
                out.writeInt(data.length); // data length
                out.write(data); // data
            }
        } finally {
            out.close();
        }

        // write references
        fsRes = new FileSystemResource(wspFS, REFS_FILE_PATH);
        fsRes.makeParentDirs();
        bos = new BufferedOutputStream(fsRes.getOutputStream());
        out = new DataOutputStream(bos);

        try {
            out.writeInt(refsStore.size()); // number of entries
            // entries
            for (NodeId id : refsStore.keySet()) {
                out.writeUTF(id.toString()); // target id

                byte[] data = refsStore.get(id);
                out.writeInt(data.length); // data length
                out.write(data); // data
            }
        } finally {
            out.close();
        }

        if (!useFileBlobStore) {
            // write blobs
            fsRes = new FileSystemResource(wspFS, BLOBS_FILE_PATH);
            fsRes.makeParentDirs();
            bos = new BufferedOutputStream(fsRes.getOutputStream());
            out = new DataOutputStream(bos);

            try {
                out.writeInt(blobs.size()); // number of entries
                // entries
                for (String id : blobs.keySet()) {
                    out.writeUTF(id); // id
                    byte[] data = blobs.get(id);
                    out.writeInt(data.length); // data length
                    out.write(data); // data
                }
            } finally {
                out.close();
            }
        }
    }

    //---------------------------------------------------< PersistenceManager >
    /**
     * {@inheritDoc}
     */
    public void init(PMContext context) throws Exception {
        if (initialized) {
            throw new IllegalStateException("already initialized");
        }
        super.init(context);
        // initialize mem stores
        bundleStore = new LinkedHashMap<NodeId, byte[]>(initialCapacity, loadFactor);
        refsStore = new HashMap<NodeId, byte[]>(initialCapacity, loadFactor);

        // Choose a FileSystem for the BlobStore based on whether data is persistent or not 
        if (useFileBlobStore) {
            blobFS = new LocalFileSystem();
            ((LocalFileSystem) blobFS).setRoot(new File(context.getHomeDir(), "blobs"));
            blobFS.init();
            blobStore = new FileSystemBLOBStore(blobFS);
        } else {
            blobStore = new InMemBLOBStore();
        }

        wspFS = context.getFileSystem();

        // load namespaces
        binding = new BundleBinding(errorHandling, blobStore, getNsIndex(), getNameIndex(), context.getDataStore());
        binding.setMinBlobSize(minBlobSize);

        if (persistent) {
            // deserialize contents of the stores
            loadContents();
        }
        initialized = true;
    }

    /**
     * {@inheritDoc}
     */
    public synchronized void close() throws Exception {
        if (!initialized) {
            throw new IllegalStateException("not initialized");
        }

        try {
            if (persistent) {
                // serialize contents of state and refs stores
                storeContents();
            } else if (useFileBlobStore) {
                blobFS.close();
                // not persistent, clear out blobs
                wspFS.deleteFolder("blobs");
            }
        } finally {
            initialized = false;
        }
    }

    /**
     * {@inheritDoc}
     */
    public NodeReferences loadReferencesTo(NodeId id) throws NoSuchItemStateException, ItemStateException {
        if (!initialized) {
            throw new IllegalStateException("not initialized");
        }
        if (!refsStore.containsKey(id)) {
            throw new NoSuchItemStateException(id.toString());
        }

        try {
            NodeReferences refs = new NodeReferences(id);
            Serializer.deserialize(refs, new ByteArrayInputStream(refsStore.get(id)));
            return refs;
        } catch (Exception e) {
            String msg = "failed to read references: " + id;
            log.error(msg, e);
            throw new ItemStateException(msg, e);
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected void store(NodeReferences refs) throws ItemStateException {
        try {
            ByteArrayOutputStream out = new ByteArrayOutputStream(INITIAL_BUFFER_SIZE);
            // serialize references
            Serializer.serialize(refs, out);
            // store in serialized format in map for better memory efficiency
            refsStore.put(refs.getTargetId(), out.toByteArray());
        } catch (Exception e) {
            String msg = "failed to store " + refs;
            log.debug(msg);
            throw new ItemStateException(msg, e);
        }
    }

    /**
     * {@inheritDoc}
     */
    public boolean existsReferencesTo(NodeId targetId) throws ItemStateException {
        if (!initialized) {
            throw new IllegalStateException("not initialized");
        }
        return refsStore.containsKey(targetId);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected void destroy(NodeReferences refs) throws ItemStateException {
        refsStore.remove(refs.getTargetId());
    }

    /**
     * {@inheritDoc}
     */
    public List<NodeId> getAllNodeIds(NodeId after, int maxCount) throws ItemStateException, RepositoryException {
        final List<NodeId> result = new ArrayList<NodeId>();
        boolean add = after == null;
        int count = 0;
        for (NodeId nodeId : bundleStore.keySet()) {
            if (add) {
                result.add(nodeId);
                if (++count == maxCount) {
                    break;
                }
            } else {
                add = nodeId.equals(after);
            }
        }
        return result;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected NodePropBundle loadBundle(NodeId id) throws ItemStateException {
        if (!bundleStore.containsKey(id)) {
            return null;
        }
        try {
            return binding.readBundle(new ByteArrayInputStream(bundleStore.get(id)), id);
        } catch (Exception e) {
            String msg = "failed to read bundle: " + id + ": " + e;
            log.error(msg);
            throw new ItemStateException(msg, e);
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected void storeBundle(NodePropBundle bundle) throws ItemStateException {
        ByteArrayOutputStream out = new ByteArrayOutputStream(INITIAL_BUFFER_SIZE);
        try {
            binding.writeBundle(out, bundle);
            bundleStore.put(bundle.getId(), out.toByteArray());
        } catch (IOException e) {
            String msg = "failed to write bundle: " + bundle.getId();
            log.error(msg, e);
            throw new ItemStateException(msg, e);
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected void destroyBundle(NodePropBundle bundle) throws ItemStateException {
        bundleStore.remove(bundle.getId());
    }

    /**
     * {@inheritDoc}
     */
    @Override
    protected BLOBStore getBlobStore() {
        return blobStore;
    }

    /**
     * Helper interface for closeable stores
     */
    protected static interface CloseableBLOBStore extends BLOBStore {
        void close();
    }

    /**
     * Trivial {@link BLOBStore} implementation that is backed by a {@link HashMap}.
     */
    protected class InMemBLOBStore implements CloseableBLOBStore {

        public InMemBLOBStore() {
            blobs = new HashMap<String, byte[]>();
        }

        /**
         * {@inheritDoc}
         */
        public String createId(PropertyId id, int index) {
            StringBuilder buf = new StringBuilder();
            buf.append(id.getParentId().toString());
            buf.append('.');
            buf.append(getNsIndex().stringToIndex(id.getName().getNamespaceURI()));
            buf.append('.');
            buf.append(getNameIndex().stringToIndex(id.getName().getLocalName()));
            buf.append('.');
            buf.append(index);
            return buf.toString();
        }

        /**
         * {@inheritDoc}
         */
        public void put(String blobId, InputStream in, long size) throws Exception {
            blobs.put(blobId, IOUtils.toByteArray(in));
        }

        /**
         * {@inheritDoc}
         */
        public InputStream get(String blobId) throws Exception {
            if (blobs.containsKey(blobId)) {
                return new ByteArrayInputStream(blobs.get(blobId));
            } else {
                return null;
            }
        }

        /**
         * {@inheritDoc}
         */
        public boolean remove(String blobId) throws Exception {
            return blobs.remove(blobId) != null;
        }

        /**
         * {@inheritDoc}
         */
        public void close() {
            blobs.clear();
        }
    }
}