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

org.modeshape.jcr.index.local.MapDB Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 *
 * Licensed 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.modeshape.jcr.index.local;

import java.io.DataInput;
import java.io.DataOutput;
import java.io.IOException;
import java.io.Serializable;
import java.math.BigDecimal;
import java.net.URI;
import java.nio.charset.Charset;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Map;
import org.mapdb.BTreeKeySerializer;
import org.mapdb.DataInput2;
import org.mapdb.DataOutput2;
import org.mapdb.Fun;
import org.mapdb.Fun.Tuple2;
import org.mapdb.Serializer;
import org.modeshape.common.util.ObjectUtil;
import org.modeshape.jcr.api.value.DateTime;
import org.modeshape.jcr.cache.NodeKey;
import org.modeshape.jcr.value.DateTimeFactory;
import org.modeshape.jcr.value.Name;
import org.modeshape.jcr.value.NameFactory;
import org.modeshape.jcr.value.Path;
import org.modeshape.jcr.value.PathFactory;
import org.modeshape.jcr.value.Reference;
import org.modeshape.jcr.value.ReferenceFactory;
import org.modeshape.jcr.value.StringFactory;
import org.modeshape.jcr.value.UriFactory;
import org.modeshape.jcr.value.ValueFactories;
import org.modeshape.jcr.value.ValueFactory;

/**
 * @author Randall Hauch ([email protected])
 */
public class MapDB {

    public static interface Serializers {
        /**
         * Obtain a serializer for the given value type.
         *
         * @param type the type; may not be null
         * @return the serializer
         */
        Serializer serializerFor( Class type );

        /**
         * Obtain a serializer for the given key type.
         *
         * @param type the type; may not be null
         * @param comparator the comparator; may not be null
         * @param pack true if the serializer can/should pack keys together when possible, or false otherwise
         * @return the serializer
         */
        BTreeKeySerializer bTreeKeySerializerFor( Class type,
                                                     Comparator comparator,
                                                     boolean pack );

    }

    public static Serializers serializers( ValueFactories factories ) {
        return new SerializerSupplier(factories);
    }

    public final static Serializer NODE_KEY_SERIALIZER = new NodeKeySerializer();

    protected final static Serializer DEFAULT_SERIALIZER = Serializer.BASIC;
    protected final static BTreeKeySerializer DEFAULT_BTREE_KEY_SERIALIZER = BTreeKeySerializer.BASIC;

    public static final class SerializerSupplier implements Serializers {
        private final Map, Serializer> serializersByClass;
        private final Map, BTreeKeySerializer> bTreeKeySerializersByClass;
        private final Map, BTreeKeySerializer> packedBTreeKeySerializersByClass;

        protected SerializerSupplier( ValueFactories factories ) {
            // Create the serializers ...
            final PathFactory pathFactory = factories.getPathFactory();
            final NameFactory nameFactory = factories.getNameFactory();
            final StringFactory stringFactory = factories.getStringFactory();
            final ReferenceFactory refFactory = factories.getReferenceFactory();
            final UriFactory uriFactory = factories.getUriFactory();
            final DateTimeFactory dateFactory = factories.getDateFactory();
            final ValueFactory decimalFactory = factories.getDecimalFactory();
            serializersByClass = new HashMap, Serializer>();
            serializersByClass.put(String.class, Serializer.STRING);
            serializersByClass.put(Long.class, Serializer.LONG);
            serializersByClass.put(Boolean.class, Serializer.BOOLEAN);
            serializersByClass.put(Double.class, new DoubleSerializer());
            serializersByClass.put(BigDecimal.class, new ValueSerializer(stringFactory, decimalFactory));
            serializersByClass.put(URI.class, new ValueSerializer(stringFactory, uriFactory));
            serializersByClass.put(DateTime.class, new ValueSerializer(stringFactory, dateFactory));
            serializersByClass.put(Path.class, new ValueSerializer(stringFactory, pathFactory));
            serializersByClass.put(Name.class, new ValueSerializer(stringFactory, nameFactory));
            serializersByClass.put(Reference.class, new ValueSerializer(stringFactory, refFactory));
            serializersByClass.put(NodeKey.class, NODE_KEY_SERIALIZER);

            bTreeKeySerializersByClass = new HashMap, BTreeKeySerializer>();
            packedBTreeKeySerializersByClass = new HashMap, BTreeKeySerializer>();
            for (Map.Entry, Serializer> entry : serializersByClass.entrySet()) {
                Serializer serializer = entry.getValue();
                @SuppressWarnings( {"rawtypes", "unchecked"} )
                BTreeKeySerializer bTreeSerializer = new DelegatingKeySerializer(serializer);
                bTreeKeySerializersByClass.put(entry.getKey(), bTreeSerializer);
                packedBTreeKeySerializersByClass.put(entry.getKey(), bTreeSerializer);
            }

            // Override some of the types for string-based keys ...
            packedBTreeKeySerializersByClass.put(String.class,
                                                 new PackedStringKeySerializer(stringFactory, stringFactory));
            packedBTreeKeySerializersByClass.put(Name.class, new PackedStringKeySerializer(stringFactory, nameFactory));
            packedBTreeKeySerializersByClass.put(Path.class, new PackedStringKeySerializer(stringFactory, pathFactory));
            packedBTreeKeySerializersByClass.put(Name.class, new PackedStringKeySerializer(stringFactory, nameFactory));
            packedBTreeKeySerializersByClass.put(DateTime.class, new PackedStringKeySerializer(stringFactory,
                                                                                                         dateFactory));
            packedBTreeKeySerializersByClass.put(URI.class, new PackedStringKeySerializer(stringFactory, uriFactory));
            packedBTreeKeySerializersByClass.put(Reference.class, new PackedStringKeySerializer(stringFactory,
                                                                                                           refFactory));
            packedBTreeKeySerializersByClass.put(BigDecimal.class, new PackedStringKeySerializer(stringFactory,
                                                                                                             decimalFactory));
        }

        @Override
        public Serializer serializerFor( Class type ) {
            Serializer result = serializersByClass.get(type);
            if (result != null) return result;
            return DEFAULT_SERIALIZER;
        }

        @Override
        public BTreeKeySerializer bTreeKeySerializerFor( Class type,
                                                            final Comparator comparator,
                                                            boolean pack ) {
            Map, BTreeKeySerializer> byClass = pack ? packedBTreeKeySerializersByClass : bTreeKeySerializersByClass;
            final BTreeKeySerializer result = byClass.containsKey(type) ? byClass.get(type) : DEFAULT_BTREE_KEY_SERIALIZER;
            // Make sure the serializer uses the type's comparator ...
            if (result instanceof KeySerializerWithComparator) {
                KeySerializerWithComparator serializer = (KeySerializerWithComparator)result;
                assert comparator != null;
                return serializer.withComparator(comparator);
            }
            return bTreeKeySerializerWith(result, comparator);
        }

        private  BTreeKeySerializer bTreeKeySerializerWith( final BTreeKeySerializer original,
                                                                  final Comparator comparator ) {
            return new BTreeKeySerializerWitheComparator(original, comparator);
        }
    }

    /**
     * NodeKey serializer for MapDB (which must be in turn, Serializable)
     */
    private static class NodeKeySerializer implements Serializer, Serializable {
        private static final long serialVersionUID = 1L;

        protected NodeKeySerializer() {
        }

        @Override
        public void serialize( DataOutput out,
                               NodeKey value ) throws IOException {
            out.writeUTF(value.toString());
        }

        @Override
        public NodeKey deserialize( DataInput in,
                                    int available ) throws IOException {
            String keyStr = in.readUTF();
            return new NodeKey(keyStr);
        }

        @Override
        public int fixedSize() {
            return -1; // not fixed size
        }
    }

    private static class BTreeKeySerializerWitheComparator extends BTreeKeySerializer implements Serializable {
        private static final long serialVersionUID = 1L;
        private final BTreeKeySerializer original;
        private final Comparator comparator;

        protected BTreeKeySerializerWitheComparator( BTreeKeySerializer original,
                                                     Comparator comparator ) {
            this.original = original;
            this.comparator = comparator;
        }

        @Override
        public Object[] deserialize( DataInput in,
                                     int start,
                                     int end,
                                     int size ) throws IOException {
            return original.deserialize(in, start, end, size);
        }

        @Override
        public Comparator getComparator() {
            return comparator;
        }

        @Override
        public void serialize( DataOutput out,
                               int start,
                               int end,
                               Object[] keys ) throws IOException {
            original.serialize(out, start, end, keys);
        }
    }

    public static  BTreeKeySerializer> uniqueKeyBTreeSerializer( Serializer serializer,
                                                                                 Comparator comparator ) {
        return new UniqueKeyBTreeSerializer(serializer, uniqueKeyComparator(comparator));
    }

    public static  Serializer> uniqueKeySerializer( Serializer serializer,
                                                                    Comparator comparator ) {
        return new UniqueKeySerializer(serializer, uniqueKeyComparator(comparator));
    }

    public static  Comparator> uniqueKeyComparator( Comparator comparator ) {
        return new UniqueKeyComparator(comparator);
    }

    public static  Comparator> tupleComparator( Comparator aComparator,
                                                                       Comparator bComparator ) {
        return new TupleComparator(aComparator, bComparator);
    }

    public static  BTreeKeySerializer> tupleBTreeSerializer( Comparator aComparator,
                                                                                    Serializer aSerializer,
                                                                                    Serializer bSerializer,
                                                                                    Comparator> tupleComparator ) {
        return new LocalTuple2KeySerializer<>(aComparator, aSerializer, bSerializer, tupleComparator);
    }

    public static final class UniqueKey implements Serializable {
        private static final long serialVersionUID = 1L;

        protected final K actualKey;
        protected final long id;
        private final int hc;

        public UniqueKey( K actualKey,
                          long id ) {
            this.actualKey = actualKey;
            this.id = id;
            this.hc = actualKey != null ? actualKey.hashCode() : 0;
        }

        @Override
        public int hashCode() {
            return hc;
        }

        @Override
        public boolean equals( Object obj ) {
            if (this == obj) return true;
            if (obj instanceof UniqueKey) {
                @SuppressWarnings( "unchecked" )
                UniqueKey that = (UniqueKey)obj;
                if (this.actualKey == null && that.actualKey != null) return false;
                if (!this.actualKey.equals(that.actualKey)) return false;
                return this.id == that.id;
            }
            return false;
        }

        @Override
        public String toString() {
            return "[" + actualKey + "," + id + "]";
        }
    }

    public static final class UniqueKeyComparator implements Comparator>, Serializable {
        private static final long serialVersionUID = 1L;
        private final transient Comparator valueComparator;

        public UniqueKeyComparator( Comparator valueComparator ) {
            this.valueComparator = valueComparator;
        }

        @Override
        public int compare( UniqueKey o1,
                            UniqueKey o2 ) {
            if (o1 == o2) return 0;
            if (o1 == null) {
                return o2 == null ? 0 : 1;
            } else if (o2 == null) {
                return -1;
            }
            int diff = valueComparator.compare(o1.actualKey, o2.actualKey);
            if (diff != 0) return diff;
            long ldiff = o1.id - o2.id;
            return ldiff == 0L ? 0 : (ldiff <= 0L ? -1 : 1);
        }
    }

    public static final class ComparableUniqueKeyComparator implements Comparator>, Serializable {
        private static final long serialVersionUID = 1L;

        @Override
        public int compare( UniqueKey o1,
                            UniqueKey o2 ) {
            if (o1 == o2) return 0;
            int diff = ObjectUtil.compareWithNulls((Comparable)o1.actualKey, (Comparable)o2.actualKey);
            if (diff != 0) return diff;
            long ldiff = o1.id - o2.id;
            return ldiff == 0L ? 0 : (ldiff <= 0L ? -1 : 1);
        }
    }

    public static final class UniqueKeySerializer implements Serializer>, Serializable {
        private static final long serialVersionUID = 1L;
        protected final transient Serializer keySerializer;
        protected final transient Comparator> comparator;

        public UniqueKeySerializer( Serializer keySerializer,
                                    Comparator> comparator ) {
            this.keySerializer = keySerializer;
            this.comparator = comparator;
        }

        @Override
        public UniqueKey deserialize( DataInput in,
                                         int available ) throws IOException {
            K actualKey = keySerializer.deserialize(in, available);
            long id = in.readLong();
            return new UniqueKey(actualKey, id);
        }

        @Override
        public void serialize( DataOutput out,
                               UniqueKey value ) throws IOException {
            keySerializer.serialize(out, value.actualKey);
            out.writeLong(value.id);
        }

        @Override
        public int fixedSize() {
            return -1;
        }

        @Override
        public String toString() {
            return "UniqueKeySerializer<" + keySerializer + ">";
        }

    }

    public static final class UniqueKeyBTreeSerializer extends BTreeKeySerializer> implements Serializable {
        private static final long serialVersionUID = 1L;
        protected final transient Serializer keySerializer;
        protected final transient Comparator> comparator;

        public UniqueKeyBTreeSerializer( Serializer keySerializer,
                                         Comparator> comparator ) {
            this.keySerializer = keySerializer;
            this.comparator = comparator;
        }

        @Override
        public Comparator> getComparator() {
            return comparator;
        }

        @SuppressWarnings( "unchecked" )
        @Override
        public void serialize( DataOutput out,
                               int start,
                               int end,
                               Object[] keys ) throws IOException {
            for (int i = start; i < end; i++) {
                UniqueKey key = (UniqueKey)keys[i];
                keySerializer.serialize(out, key.actualKey);
                out.writeLong(key.id);
            }
        }

        @Override
        public Object[] deserialize( DataInput in,
                                     int start,
                                     int end,
                                     int size ) throws IOException {
            Object[] ret = new Object[size];
            for (int i = start; i < end; i++) {
                K key = keySerializer.deserialize(in, -1);
                long id = in.readLong();
                ret[i] = new UniqueKey(key, id);
            }
            return ret;
        }

        @Override
        public String toString() {
            return "UniqueKeyBTreeSerializer<" + keySerializer + ">";
        }

    }

    public static class NaturalComparator> implements Comparator, Serializable {
        private static final long serialVersionUID = 1L;

        @Override
        public int compare( K o1,
                            K o2 ) {
            return o1.compareTo(o2);
        }
    }

    /**
     * A key serializer that just writes data without applying any compression.
     *
     * @param  the type to be serialized
     */
    public static final class DelegatingKeySerializer> extends BTreeKeySerializer
        implements Serializable, KeySerializerWithComparator {
        private static final long serialVersionUID = 1L;
        protected final transient Serializer defaultSerializer;
        protected final transient Comparator comparator;

        public DelegatingKeySerializer( Serializer defaultSerializer ) {
            this(defaultSerializer, null);
        }

        public DelegatingKeySerializer( Serializer defaultSerializer,
                                        Comparator comparator ) {
            this.defaultSerializer = defaultSerializer;
            this.comparator = comparator != null ? comparator : new NaturalComparator();
        }

        @Override
        public Comparator getComparator() {
            return comparator;
        }

        @SuppressWarnings( "unchecked" )
        @Override
        public BTreeKeySerializer withComparator( Comparator comparator ) {
            if (comparator == null) return this;
            return new DelegatingKeySerializer(defaultSerializer, (Comparator)comparator);
        }

        @SuppressWarnings( "unchecked" )
        @Override
        public void serialize( DataOutput out,
                               int start,
                               int end,
                               Object[] keys ) throws IOException {
            for (int i = start; i < end; i++) {
                defaultSerializer.serialize(out, (K)keys[i]);
            }
        }

        @Override
        public Object[] deserialize( DataInput in,
                                     int start,
                                     int end,
                                     int size ) throws IOException {
            Object[] ret = new Object[size];
            for (int i = start; i < end; i++) {
                ret[i] = defaultSerializer.deserialize(in, -1);
            }
            return ret;
        }

        @Override
        public String toString() {
            return "DelegatingBTreeSerializer<" + defaultSerializer + ">";
        }
    }

    public static interface KeySerializerWithComparator {
        BTreeKeySerializer withComparator( Comparator comparator );
    }

    /**
     * Applies delta packing on {@code java.lang.String}. This serializer splits consequent strings to two parts: shared prefix
     * and different suffix. Only suffix is than stored.
     *
     * @param  the type to be serialized
     */
    public static class PackedStringKeySerializer> extends BTreeKeySerializer
        implements Serializable, KeySerializerWithComparator {
        private static final long serialVersionUID = 1L;
        private static final Charset UTF8_CHARSET = Charset.forName("UTF8");

        private final transient ValueFactory valueFactory;
        private final transient StringFactory stringFactory;
        protected final transient Comparator comparator;

        public PackedStringKeySerializer( StringFactory stringFactory,
                                          ValueFactory valueFactory ) {
            this(stringFactory, valueFactory, null);
        }

        protected PackedStringKeySerializer( StringFactory stringFactory,
                                             ValueFactory valueFactory,
                                             Comparator comparator ) {
            this.valueFactory = valueFactory;
            this.stringFactory = stringFactory;
            this.comparator = comparator != null ? comparator : new Comparator() {
                @Override
                public int compare( K o1,
                                    K o2 ) {
                    return o1.compareTo(o2);
                }
            };
        }

        @SuppressWarnings( "unchecked" )
        @Override
        public BTreeKeySerializer withComparator( Comparator comparator ) {
            if (comparator == null) return this;
            return new PackedStringKeySerializer<>(stringFactory, valueFactory, (Comparator)comparator);
        }

        @Override
        public Comparator getComparator() {
            return comparator;
        }

        @Override
        public void serialize( DataOutput out,
                               int start,
                               int end,
                               Object[] keys ) throws IOException {
            byte[] previous = null;
            for (int i = start; i < end; i++) {
                String key = stringFactory.create(keys[i]);
                byte[] b = key.getBytes(UTF8_CHARSET);
                leadingValuePackWrite(out, b, previous, 0);
                previous = b;
            }
        }

        @Override
        public Object[] deserialize( DataInput in,
                                     int start,
                                     int end,
                                     int size ) throws IOException {
            Object[] ret = new Object[size];
            byte[] previous = null;
            for (int i = start; i < end; i++) {
                byte[] b = leadingValuePackRead(in, previous, 0);
                if (b == null) continue;
                String str = new String(b, UTF8_CHARSET);
                ret[i] = valueFactory.create(str);
                previous = b;
            }
            return ret;
        }

        @Override
        public String toString() {
            return "ValueSerializer<" + valueFactory.getPropertyType() + ">";
        }
    }

    public static class DoubleSerializer implements Serializer, Serializable {
        private static final long serialVersionUID = 1L;

        @Override
        public void serialize( DataOutput out,
                               Double value ) throws IOException {
            out.writeDouble(value.doubleValue());
        }

        @Override
        public Double deserialize( DataInput in,
                                   int available ) throws IOException {
            if (available == 0) return null;
            return in.readDouble();
        }

        @Override
        public int fixedSize() {
            return -1;
        }
    }

    public static class ValueSerializer implements Serializer, Serializable {
        private static final long serialVersionUID = 1L;

        private final transient ValueFactory valueFactory;
        private final transient StringFactory stringFactory;

        public ValueSerializer( StringFactory stringFactory,
                                ValueFactory valueFactory ) {
            this.valueFactory = valueFactory;
            this.stringFactory = stringFactory;
        }

        @Override
        public int fixedSize() {
            return -1; // not fixed size
        }

        @Override
        public void serialize( DataOutput out,
                               V value ) throws IOException {
            out.writeUTF(stringFactory.create(value));
        }

        @Override
        public V deserialize( DataInput in,
                              int available ) throws IOException {
            String pathStr = in.readUTF();
            return valueFactory.create(pathStr);
        }

        @Override
        public String toString() {
            return "ValueSerializer<" + valueFactory.getPropertyType() + ">";
        }
    }

    /**
     * Applies delta compression on array of tuple. First tuple value may be shared between consequentive tuples, so only first
     * occurrence is serialized. An example:
     *
     * 
     *     Value            Serialized as
     *     -------------------------
     *     Tuple(1, 1)       1, 1
     *     Tuple(1, 2)          2
     *     Tuple(1, 3)          3
     *     Tuple(1, 4)          4
     * 
* * @param
first tuple value * @param second tuple value */ @SuppressWarnings( "unchecked" ) protected final static class LocalTuple2KeySerializer extends BTreeKeySerializer> implements Serializable { private static final long serialVersionUID = 0L; protected final Comparator aComparator; protected final Serializer aSerializer; protected final Serializer bSerializer; protected final Comparator> comparator; /** * Construct new Tuple2 Key Serializer. You may pass null for some value, In that case 'default' value will be used, * Comparable comparator and Default Serializer from DB. * * @param aComparator comparator used for first tuple value * @param aSerializer serializer used for first tuple value * @param bSerializer serializer used for second tuple value * @param comparator the comparator for the tuple */ public LocalTuple2KeySerializer( Comparator aComparator, Serializer aSerializer, Serializer bSerializer, Comparator> comparator ) { this.aComparator = aComparator; this.aSerializer = aSerializer; this.bSerializer = bSerializer; this.comparator = comparator; } @Override public void serialize( DataOutput out, int start, int end, Object[] keys ) throws IOException { int acount = 0; for (int i = start; i < end; i++) { Fun.Tuple2 t = (Fun.Tuple2)keys[i]; if (acount == 0) { // write new A aSerializer.serialize(out, t.a); // count how many A are following acount = 1; while (i + acount < end && aComparator.compare(t.a, ((Fun.Tuple2)keys[i + acount]).a) == 0) { acount++; } DataOutput2.packInt(out, acount); } bSerializer.serialize(out, t.b); acount--; } } @Override public Object[] deserialize( DataInput in, int start, int end, int size ) throws IOException { Object[] ret = new Object[size]; A a = null; int acount = 0; for (int i = start; i < end; i++) { if (acount == 0) { // read new A a = aSerializer.deserialize(in, -1); acount = DataInput2.unpackInt(in); } B b = bSerializer.deserialize(in, -1); ret[i] = Fun.t2(a, b); acount--; } assert (acount == 0); return ret; } @Override public Comparator> getComparator() { return comparator; } @Override public boolean equals( Object o ) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; LocalTuple2KeySerializer t = (LocalTuple2KeySerializer)o; return Fun.eq(aComparator, t.aComparator) && Fun.eq(aSerializer, t.aSerializer) && Fun.eq(bSerializer, t.bSerializer); } @Override public int hashCode() { int result = aComparator != null ? aComparator.hashCode() : 0; result = 31 * result + (aSerializer != null ? aSerializer.hashCode() : 0); result = 31 * result + (bSerializer != null ? bSerializer.hashCode() : 0); return result; } } protected final static class TupleComparator implements Comparator>, Serializable { private static final long serialVersionUID = 1L; private final Comparator aComparator; private final Comparator bComparator; protected TupleComparator( Comparator aComparator, Comparator bComparator ) { this.aComparator = aComparator; this.bComparator = bComparator; } @Override public int compare( Tuple2 o1, Tuple2 o2 ) { int i = aComparator.compare(o1.a, o2.a); if (i != 0) return i; // Before we check the second value in the tuples, check them against the "positive infinity" values ... if (o1.b == Fun.HI) { if (o2.b == Fun.HI) { // Both are positive infinity ... return 0; } // o1.b is positive infinity, and o2.b is not ... return 1; } else if (o2.b == Fun.HI) { // o1.b is not positive infinity, and o1.b is ... return -1; } // Neither is positive infinity, so use the actual comparator ... i = bComparator.compare(o1.b, o2.b); return i; } } private MapDB() { } }