/* * 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 org.apache.jackrabbit.core.id.ItemId; import org.apache.jackrabbit.core.id.NodeId; import org.apache.jackrabbit.core.id.PropertyId; import org.apache.jackrabbit.core.fs.FileSystem; import org.apache.jackrabbit.core.fs.FileSystemPathUtil; import org.apache.jackrabbit.core.fs.FileSystemResource; import org.apache.jackrabbit.core.fs.local.LocalFileSystem; import org.apache.jackrabbit.core.fs.mem.MemoryFileSystem; import org.apache.jackrabbit.core.persistence.AbstractPersistenceManager; import org.apache.jackrabbit.core.persistence.PMContext; import org.apache.jackrabbit.core.persistence.util.BLOBStore; import org.apache.jackrabbit.core.state.ItemStateException; import org.apache.jackrabbit.core.state.NoSuchItemStateException; import org.apache.jackrabbit.core.state.NodeReferences; import org.apache.jackrabbit.core.state.NodeState; import org.apache.jackrabbit.core.state.PropertyState; import org.apache.jackrabbit.core.persistence.util.FileSystemBLOBStore; import org.apache.jackrabbit.core.persistence.util.Serializer; import org.apache.jackrabbit.core.value.InternalValue; import org.apache.jackrabbit.spi.Name; import org.slf4j.Logger; import org.slf4j.LoggerFactory; 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.util.HashMap; import java.util.Map; /** * <code>InMemPersistenceManager</code> is a very simple <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> * </ul> * <b>Please note that this class should only be used for testing purposes.</b> * * @deprecated Please migrate to a bundle persistence manager * (<a href="https://issues.apache.org/jira/browse/JCR-2802">JCR-2802</a>) */ @Deprecated public class InMemPersistenceManager extends AbstractPersistenceManager { private static Logger log = LoggerFactory.getLogger(InMemPersistenceManager.class); protected boolean initialized; protected Map<ItemId, byte[]> stateStore; protected Map<NodeId, byte[]> refsStore; // initial size of buffer used to serialize objects protected static final int INITIAL_BUFFER_SIZE = 1024; // some constants used in serialization protected static final String STATE_FILE_PATH = "/data/.state.bin"; protected static final String REFS_FILE_PATH = "/data/.refs.bin"; protected static final byte NODE_ENTRY = 0; protected static final byte PROP_ENTRY = 1; // file system where BLOB data is stored protected FileSystem blobFS; // BLOBStore that manages BLOB data in the file system protected BLOBStore blobStore; /** * file system where the content of the hash maps are read from/written to * (if <code>persistent==true</code>) */ protected FileSystem wspFS; // 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; /** * Creates a new <code>InMemPersistenceManager</code> instance. */ public InMemPersistenceManager() { 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(); } protected static String buildBlobFilePath(String parentUUID, Name propName, int index) { StringBuilder sb = new StringBuilder(); char[] chars = parentUUID.toCharArray(); int cnt = 0; for (char ch : chars) { if (ch == '-') { continue; } //if (cnt > 0 && cnt % 4 == 0) { if (cnt == 2 || cnt == 4) { sb.append(FileSystem.SEPARATOR_CHAR); } sb.append(ch); cnt++; } sb.append(FileSystem.SEPARATOR_CHAR); sb.append(FileSystemPathUtil.escapeName(propName.toString())); sb.append('.'); sb.append(index); sb.append(".bin"); return sb.toString(); } /** * 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, STATE_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) { byte type = in.readByte(); // entry type ItemId id; if (type == NODE_ENTRY) { // entry type: node String s = in.readUTF(); // id id = NodeId.valueOf(s); } else { // entry type: property String s = in.readUTF(); // id id = PropertyId.valueOf(s); } int length = in.readInt(); // data length byte[] data = new byte[length]; in.readFully(data); // data // store in map stateStore.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(); } } /** * Writes the content of the hash maps to the file system * * @throws Exception if an error occurs */ public synchronized void storeContents() throws Exception { // write item states FileSystemResource fsRes = new FileSystemResource(wspFS, STATE_FILE_PATH); fsRes.makeParentDirs(); BufferedOutputStream bos = new BufferedOutputStream(fsRes.getOutputStream()); DataOutputStream out = new DataOutputStream(bos); try { out.writeInt(stateStore.size()); // number of entries // entries for (ItemId id : stateStore.keySet()) { if (id.denotesNode()) { out.writeByte(NODE_ENTRY); // entry type } else { out.writeByte(PROP_ENTRY); // entry type } out.writeUTF(id.toString()); // id byte[] data = stateStore.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(); } } //---------------------------------------------------< PersistenceManager > /** * {@inheritDoc} */ public void init(PMContext context) throws Exception { if (initialized) { throw new IllegalStateException("already initialized"); } stateStore = new HashMap<ItemId, byte[]>(initialCapacity, loadFactor); refsStore = new HashMap<NodeId, byte[]>(initialCapacity, loadFactor); wspFS = context.getFileSystem(); // Choose a FileSystem for the BlobStore based on whether data is persistent or not if (persistent) { blobFS = new LocalFileSystem(); ((LocalFileSystem) blobFS).setRoot(new File(context.getHomeDir(), "blobs")); } else { blobFS = new MemoryFileSystem(); } blobFS.init(); blobStore = new FileSystemBLOBStore(blobFS); if (persistent) { // deserialize contents of state and refs 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 { // clear out blob store try { String[] folders = blobFS.listFolders("/"); for (String folder: folders) { blobFS.deleteFolder(folder); } String[] files = blobFS.listFiles("/"); for (String file : files) { blobFS.deleteFile(file); } } catch (Exception e) { // ignore } } // close BLOB file system blobFS.close(); blobFS = null; blobStore = null; stateStore.clear(); stateStore = null; refsStore.clear(); refsStore = null; } finally { initialized = false; } } /** * {@inheritDoc} */ public synchronized NodeState load(NodeId id) throws NoSuchItemStateException, ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } byte[] data = stateStore.get(id); if (data == null) { throw new NoSuchItemStateException(id.toString()); } ByteArrayInputStream in = new ByteArrayInputStream(data); try { NodeState state = createNew(id); Serializer.deserialize(state, in); return state; } catch (Exception e) { String msg = "failed to read node state: " + id; log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ public synchronized PropertyState load(PropertyId id) throws NoSuchItemStateException, ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } byte[] data = stateStore.get(id); if (data == null) { throw new NoSuchItemStateException(id.toString()); } ByteArrayInputStream in = new ByteArrayInputStream(data); try { PropertyState state = createNew(id); Serializer.deserialize(state, in, blobStore); return state; } catch (Exception e) { String msg = "failed to read property state: " + id; log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ protected void store(NodeState state) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } try { ByteArrayOutputStream out = new ByteArrayOutputStream(INITIAL_BUFFER_SIZE); // serialize node state Serializer.serialize(state, out); // store in serialized format in map for better memory efficiency stateStore.put(state.getNodeId(), out.toByteArray()); // there's no need to close a ByteArrayOutputStream //out.close(); } catch (Exception e) { String msg = "failed to write node state: " + state.getNodeId(); log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ protected void store(PropertyState state) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } try { ByteArrayOutputStream out = new ByteArrayOutputStream(INITIAL_BUFFER_SIZE); // serialize property state Serializer.serialize(state, out, blobStore); // store in serialized format in map for better memory efficiency stateStore.put(state.getPropertyId(), out.toByteArray()); // there's no need to close a ByteArrayOutputStream //out.close(); } catch (Exception e) { String msg = "failed to store property state: " + state.getPropertyId(); log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ protected void destroy(NodeState state) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } // remove node state stateStore.remove(state.getNodeId()); } /** * {@inheritDoc} */ protected void destroy(PropertyState state) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } // delete binary values (stored as files) InternalValue[] values = state.getValues(); if (values != null) { for (InternalValue val : values) { if (val != null) { val.deleteBinaryResource(); } } } // remove property state stateStore.remove(state.getPropertyId()); } /** * {@inheritDoc} */ public synchronized NodeReferences loadReferencesTo(NodeId id) throws NoSuchItemStateException, ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } byte[] data = refsStore.get(id); if (data == null) { throw new NoSuchItemStateException(id.toString()); } ByteArrayInputStream in = new ByteArrayInputStream(data); try { NodeReferences refs = new NodeReferences(id); Serializer.deserialize(refs, in); return refs; } catch (Exception e) { String msg = "failed to load references: " + id; log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ protected void store(NodeReferences refs) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } 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()); // there's no need to close a ByteArrayOutputStream //out.close(); } catch (Exception e) { String msg = "failed to store " + refs; log.debug(msg); throw new ItemStateException(msg, e); } } /** * {@inheritDoc} */ protected void destroy(NodeReferences refs) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } // remove node references refsStore.remove(refs.getTargetId()); } /** * {@inheritDoc} */ public boolean exists(PropertyId id) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } return stateStore.containsKey(id); } /** * {@inheritDoc} */ public boolean exists(NodeId id) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } return stateStore.containsKey(id); } /** * {@inheritDoc} */ public boolean existsReferencesTo(NodeId id) throws ItemStateException { if (!initialized) { throw new IllegalStateException("not initialized"); } return refsStore.containsKey(id); } }