Package org.jboss.dna.connector.store.jpa.util

Source Code of org.jboss.dna.connector.store.jpa.util.Serializer$LargeValues

/*
* JBoss DNA (http://www.jboss.org/dna)
* See the COPYRIGHT.txt file distributed with this work for information
* regarding copyright ownership.  Some portions may be licensed
* to Red Hat, Inc. under one or more contributor license agreements.
* See the AUTHORS.txt file in the distribution for a full listing of
* individual contributors.
*
* JBoss DNA is free software. Unless otherwise indicated, all code in JBoss DNA
* is licensed to you under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* JBoss DNA is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.dna.connector.store.jpa.util;

import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.math.BigDecimal;
import java.net.URI;
import java.security.NoSuchAlgorithmException;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.UUID;
import org.jboss.dna.common.SystemFailureException;
import org.jboss.dna.common.util.SecureHash;
import org.jboss.dna.connector.store.jpa.model.basic.LargeValueEntity;
import org.jboss.dna.graph.DnaLexicon;
import org.jboss.dna.graph.ExecutionContext;
import org.jboss.dna.graph.property.Binary;
import org.jboss.dna.graph.property.BinaryFactory;
import org.jboss.dna.graph.property.DateTime;
import org.jboss.dna.graph.property.Name;
import org.jboss.dna.graph.property.Path;
import org.jboss.dna.graph.property.Property;
import org.jboss.dna.graph.property.PropertyFactory;
import org.jboss.dna.graph.property.PropertyType;
import org.jboss.dna.graph.property.Reference;
import org.jboss.dna.graph.property.UuidFactory;
import org.jboss.dna.graph.property.ValueFactories;
import org.jboss.dna.graph.property.ValueFactory;
import org.jboss.dna.graph.property.ValueFormatException;

/**
* @author Randall Hauch
*/
public class Serializer {

    public static final LargeValues NO_LARGE_VALUES = new NoLargeValues();
    public static final ReferenceValues NO_REFERENCES_VALUES = new NoReferenceValues();

    private final PropertyFactory propertyFactory;
    private final ValueFactories valueFactories;
    private final boolean excludeUuidProperty;

    public Serializer( ExecutionContext context,
                       boolean excludeUuidProperty ) {
        this.propertyFactory = context.getPropertyFactory();
        this.valueFactories = context.getValueFactories();
        this.excludeUuidProperty = excludeUuidProperty;
    }

    /**
     * Interface that represents the location where "large" objects are stored.
     *
     * @author Randall Hauch
     */
    public interface LargeValues {
        /**
         * Get the minimum size for large values, specified as {@link String#length() number of characters} for a {@link String}
         * or the {@link Binary#getSize() number of bytes for a binary value}
         *
         * @return the size at which a property value is considered to be <i>large</i>
         */
        long getMinimumSize();

        void write( byte[] hash,
                    long length,
                    PropertyType type,
                    Object value ) throws IOException;

        Object read( ValueFactories valueFactories,
                     byte[] hash,
                     long length ) throws IOException;
    }

    protected static class NoLargeValues implements LargeValues {
        public long getMinimumSize() {
            return Long.MAX_VALUE;
        }

        public void write( byte[] hash,
                           long length,
                           PropertyType type,
                           Object value ) {
        }

        public Object read( ValueFactories valueFactories,
                            byte[] hash,
                            long length ) {
            return null;
        }
    }

    /**
     * Interface used to record how Reference values are processed during serialization and deserialization.
     *
     * @author Randall Hauch
     */
    public interface ReferenceValues {
        void read( Reference reference );

        void write( Reference reference );

        void remove( Reference reference );
    }

    protected static class NoReferenceValues implements ReferenceValues {
        public void read( Reference arg0 ) {
        }

        public void remove( Reference arg0 ) {
        }

        public void write( Reference arg0 ) {
        }
    }

    /**
     * Serialize the properties' values to the object stream.
     * <p>
     * If any of the property values are considered {@link LargeValues#getMinimumSize() large}, the value's hash and length of the
     * property value will be written to the object stream, but the property value will be sent to the supplied
     * {@link LargeValueEntity} object.
     * </p>
     * <p>
     * This method does not automatically write each property value to the stream using
     * {@link ObjectOutputStream#writeObject(Object)}, but instead serializes the primitive values that make up the property value
     * object with a code that describes the {@link PropertyType property's type}. This is more efficient, since most of the
     * property values are really non-primitive objects, and writing to the stream using
     * {@link ObjectOutputStream#writeObject(Object)} would include larger class metadata.
     * </p>
     *
     * @param stream the stream where the properties' values are to be serialized; may not be null
     * @param number the number of properties exposed by the supplied <code>properties</code> iterator; must be 0 or positive
     * @param properties the iterator over the properties that are to be serialized; may not be null
     * @param largeValues the interface to use for writing large values; may not be null
     * @param references the interface to use for recording which {@link Reference} values were found during serialization, or
     *        null if the references do not need to be accumulated
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @see #deserializeAllProperties(ObjectInputStream, Collection, LargeValues)
     * @see #deserializeSomeProperties(ObjectInputStream, Collection, LargeValues, LargeValues, Name...)
     * @see #serializeProperty(ObjectOutputStream, Property, LargeValues, ReferenceValues)
     */
    public void serializeProperties( ObjectOutputStream stream,
                                     int number,
                                     Iterable<Property> properties,
                                     LargeValues largeValues,
                                     ReferenceValues references ) throws IOException {
        assert number >= 0;
        assert properties != null;
        assert largeValues != null;
        stream.writeInt(number);
        for (Property property : properties) {
            if (property == null) continue;
            serializeProperty(stream, property, largeValues, references);
        }
    }

    /**
     * Serialize the property's values to the object stream.
     * <p>
     * If any of the property values are considered {@link LargeValues#getMinimumSize() large}, the value's hash and length of the
     * property value will be written to the object stream, but the property value will be sent to the supplied
     * {@link LargeValueEntity} object.
     * </p>
     * <p>
     * This method does not automatically write each property value to the stream using
     * {@link ObjectOutputStream#writeObject(Object)}, but instead serializes the primitive values that make up the property value
     * object with a code that describes the {@link PropertyType property's type}. This is more efficient, since most of the
     * property values are really non-primitive objects, and writing to the stream using
     * {@link ObjectOutputStream#writeObject(Object)} would include larger class metadata.
     * </p>
     *
     * @param stream the stream where the property's values are to be serialized; may not be null
     * @param property the property to be serialized; may not be null
     * @param largeValues the interface to use for writing large values; may not be null
     * @param references the interface to use for recording which {@link Reference} values were found during serialization, or
     *        null if the references do not need to be accumulated
     * @return true if the property was serialized, or false if it was not
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @see #serializeProperties(ObjectOutputStream, int, Iterable, LargeValues, ReferenceValues)
     * @see #deserializePropertyValues(ObjectInputStream, Name, boolean, LargeValues, LargeValues, ReferenceValues)
     */
    public boolean serializeProperty( ObjectOutputStream stream,
                                      Property property,
                                      LargeValues largeValues,
                                      ReferenceValues references ) throws IOException {
        assert stream != null;
        assert property != null;
        assert largeValues != null;
        assert references != null;
        final Name name = property.getName();
        if (this.excludeUuidProperty && DnaLexicon.UUID.equals(name)) return false;
        // Write the name ...
        stream.writeObject(name.getString());
        // Write the number of values ...
        stream.writeInt(property.size());
        for (Object value : property) {
            if (value instanceof String) {
                String stringValue = (String)value;
                if (largeValues != null && stringValue.length() > largeValues.getMinimumSize()) {
                    // Store the value in the large values area, but record the hash and length here.
                    byte[] hash = computeHash(stringValue);
                    stream.writeChar('L');
                    stream.writeInt(hash.length);
                    stream.write(hash);
                    stream.writeLong(stringValue.length());
                    // Now write to the large objects ...
                    largeValues.write(computeHash(stringValue), stringValue.length(), PropertyType.STRING, stringValue);
                } else {
                    stream.writeChar('S');
                    stream.writeObject(stringValue);
                }
            } else if (value instanceof Boolean) {
                stream.writeChar('b');
                stream.writeBoolean(((Boolean)value).booleanValue());
            } else if (value instanceof Long) {
                stream.writeChar('l');
                stream.writeLong(((Long)value).longValue());
            } else if (value instanceof Double) {
                stream.writeChar('d');
                stream.writeDouble(((Double)value).doubleValue());
            } else if (value instanceof Integer) {
                stream.writeChar('i');
                stream.writeInt(((Integer)value).intValue());
            } else if (value instanceof Short) {
                stream.writeChar('s');
                stream.writeShort(((Short)value).shortValue());
            } else if (value instanceof Float) {
                stream.writeChar('f');
                stream.writeFloat(((Float)value).floatValue());
            } else if (value instanceof UUID) {
                stream.writeChar('U');
                UUID uuid = (UUID)value;
                stream.writeLong(uuid.getMostSignificantBits());
                stream.writeLong(uuid.getLeastSignificantBits());
            } else if (value instanceof URI) {
                URI uri = (URI)value;
                String stringValue = uri.toString();
                if (largeValues != null && stringValue.length() > largeValues.getMinimumSize()) {
                    // Store the URI in the large values area, but record the hash and length here.
                    byte[] hash = computeHash(stringValue);
                    stream.writeChar('L');
                    stream.writeInt(hash.length);
                    stream.write(hash);
                    stream.writeLong(stringValue.length());
                    // Now write to the large objects ...
                    largeValues.write(computeHash(stringValue), stringValue.length(), PropertyType.URI, stringValue);
                } else {
                    stream.writeChar('I');
                    stream.writeObject(stringValue);
                }
            } else if (value instanceof Name) {
                stream.writeChar('N');
                stream.writeObject(((Name)value).getString());
            } else if (value instanceof Path) {
                stream.writeChar('P');
                stream.writeObject(((Path)value).getString());
            } else if (value instanceof DateTime) {
                stream.writeChar('T');
                stream.writeObject(((DateTime)value).getString());
            } else if (value instanceof BigDecimal) {
                stream.writeChar('D');
                stream.writeObject(value);
            } else if (value instanceof Character) {
                stream.writeChar('c');
                char c = ((Character)value).charValue();
                stream.writeChar(c);
            } else if (value instanceof Reference) {
                stream.writeChar('R');
                Reference ref = (Reference)value;
                stream.writeObject(ref.getString());
                references.write(ref);
            } else if (value instanceof Binary) {
                Binary binary = (Binary)value;
                byte[] hash = null;
                long length = 0;
                try {
                    binary.acquire();
                    length = binary.getSize();
                    if (largeValues != null && length > largeValues.getMinimumSize()) {
                        // Store the value in the large values area, but record the hash and length here.
                        hash = binary.getHash();
                        stream.writeChar('L');
                        stream.writeInt(hash.length);
                        stream.write(hash);
                        stream.writeLong(length);
                        // Write to large objects after releasing the binary
                    } else {
                        // The value is small enough to store here ...
                        stream.writeChar('B');
                        stream.writeLong(length);
                        InputStream data = binary.getStream();
                        try {
                            byte[] buffer = new byte[1024];
                            int numRead = 0;
                            while ((numRead = data.read(buffer)) > -1) {
                                stream.write(buffer, 0, numRead);
                            }
                        } finally {
                            data.close();
                        }
                    }
                } finally {
                    binary.release();
                }
                // If this is a large value and the binary has been released, write it to the large objects ...
                if (largeValues != null && hash != null) {
                    largeValues.write(hash, length, PropertyType.BINARY, value);
                }
            } else {
                // Other kinds of values ...
                stream.writeChar('O');
                stream.writeObject(value);
            }
        }
        stream.flush();
        return true;
    }

    /**
     * Deserialize the existing properties from the supplied input stream, update the properties, and then serialize the updated
     * properties to the output stream.
     *
     * @param input the stream from which the existing properties are to be deserialized; may not be null
     * @param output the stream to which the updated properties are to be serialized; may not be null
     * @param updatedProperties the properties that are being updated (or removed, if there are no values); may not be null
     * @param largeValues the interface to use for writing large values; may not be null
     * @param removedLargeValues the interface to use for recording the large values that were removed; may not be null
     * @param references the interface to use for recording which {@link Reference} values were found during serialization, or
     *        null if the references do not need to be accumulated
     * @return the number of properties
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     */
    public int reserializeProperties( ObjectInputStream input,
                                      ObjectOutputStream output,
                                      Map<Name, Property> updatedProperties,
                                      LargeValues largeValues,
                                      LargeValues removedLargeValues,
                                      ReferenceValues references ) throws IOException, ClassNotFoundException {
        assert input != null;
        assert output != null;
        assert updatedProperties != null;
        assert largeValues != null;
        assert references != null;
        // Assemble a set of property names to skip deserializing
        Map<Name, Property> allProperties = new HashMap<Name, Property>();

        // Read the number of properties ...
        int count = input.readInt();
        // Deserialize all of the proeprties ...
        for (int i = 0; i != count; ++i) {
            // Read the property name ...
            String nameStr = (String)input.readObject();
            Name name = valueFactories.getNameFactory().create(nameStr);
            assert name != null;
            if (updatedProperties.containsKey(name)) {
                // Deserialized, but don't materialize ...
                deserializePropertyValues(input, name, true, largeValues, removedLargeValues, references);
            } else {
                // Now read the property values ...
                Object[] values = deserializePropertyValues(input, name, false, largeValues, removedLargeValues, references);
                // Add the property to the collection ...
                Property property = propertyFactory.create(name, values);
                assert property != null;
                allProperties.put(name, property);
            }
        }

        // Add all the updated properties ...
        for (Map.Entry<Name, Property> entry : updatedProperties.entrySet()) {
            Property updated = entry.getValue();
            if (updated == null) {
                allProperties.remove(entry.getKey());
            } else {
                allProperties.put(updated.getName(), updated);
            }
        }

        // Serialize properties ...
        int numProperties = allProperties.size();
        output.writeInt(numProperties);
        for (Property property : allProperties.values()) {
            if (property == null) continue;
            serializeProperty(output, property, largeValues, references);
        }
        return numProperties;
    }

    /**
     * Deserialize the properties, adjust all {@link Reference} values that point to an "old" UUID to point to the corresponding
     * "new" UUID, and reserialize the properties. If any reference is to a UUID not in the map, it is left untouched.
     * <p>
     * This is an efficient method that (for the most part) reads from the input stream and directly writes to the output stream.
     * The exception is when a Reference value is read, that Reference is attempted to be remapped to a new Reference and written
     * in place of the old reference. (Of course, if the Reference is to a UUID that is not in the "old" to "new" map, the old is
     * written directly.)
     * </p>
     *
     * @param input the stream from which the existing properties are to be deserialized; may not be null
     * @param output the stream to which the updated properties are to be serialized; may not be null
     * @param oldUuidToNewUuid the map of old-to-new UUIDs
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     */
    public void adjustReferenceProperties( ObjectInputStream input,
                                           ObjectOutputStream output,
                                           Map<String, String> oldUuidToNewUuid ) throws IOException, ClassNotFoundException {
        assert input != null;
        assert output != null;
        if (oldUuidToNewUuid == null) oldUuidToNewUuid = Collections.emptyMap();

        UuidFactory uuidFactory = valueFactories.getUuidFactory();
        ValueFactory<Reference> referenceFactory = valueFactories.getReferenceFactory();

        // Read the number of properties ...
        int count = input.readInt();
        output.writeInt(count);
        // Deserialize all of the proeprties ...
        for (int i = 0; i != count; ++i) {
            // Read and write the property name ...
            Object name = input.readObject();
            output.writeObject(name);
            // Read and write the number of values ...
            int numValues = input.readInt();
            output.writeInt(numValues);
            // Now read and write each property value ...
            for (int j = 0; j != numValues; ++j) {
                // Read and write the type of value ...
                char type = input.readChar();
                output.writeChar(type);
                switch (type) {
                    case 'S':
                        output.writeObject(input.readObject());
                        break;
                    case 'b':
                        output.writeBoolean(input.readBoolean());
                        break;
                    case 'i':
                        output.writeInt(input.readInt());
                        break;
                    case 'l':
                        output.writeLong(input.readLong());
                        break;
                    case 's':
                        output.writeShort(input.readShort());
                        break;
                    case 'f':
                        output.writeFloat(input.readFloat());
                        break;
                    case 'd':
                        output.writeDouble(input.readDouble());
                        break;
                    case 'c':
                        // char
                        output.writeChar(input.readChar());
                        break;
                    case 'U':
                        // UUID
                        output.writeLong(input.readLong());
                        output.writeLong(input.readLong());
                        break;
                    case 'I':
                        // URI
                        output.writeObject(input.readObject());
                        break;
                    case 'N':
                        // Name
                        output.writeObject(input.readObject());
                        break;
                    case 'P':
                        // Path
                        output.writeObject(input.readObject());
                        break;
                    case 'T':
                        // DateTime
                        output.writeObject(input.readObject());
                        break;
                    case 'D':
                        // BigDecimal
                        output.writeObject(input.readObject());
                        break;
                    case 'R':
                        // Reference
                        String refValue = (String)input.readObject();
                        Reference ref = referenceFactory.create(refValue);
                        try {
                            UUID toUuid = uuidFactory.create(ref);
                            String newUuid = oldUuidToNewUuid.get(toUuid.toString());
                            if (newUuid != null) {
                                // Create a new reference ...
                                ref = referenceFactory.create(newUuid);
                                refValue = ref.getString();
                            }
                        } catch (ValueFormatException e) {
                            // Unknown reference, so simply write it again ...
                        }
                        // Write the reference ...
                        output.writeObject(refValue);
                        break;
                    case 'B':
                        // Binary
                        // Read the length of the content ...
                        long binaryLength = input.readLong();
                        byte[] content = new byte[(int)binaryLength];
                        input.read(content);
                        // Now write out the value ...
                        output.writeLong(binaryLength);
                        output.write(content);
                        break;
                    case 'L':
                        // Large object ...
                        int hashLength = input.readInt();
                        byte[] hash = new byte[hashLength];
                        input.read(hash);
                        long length = input.readLong();
                        // write to the output ...
                        output.writeInt(hash.length);
                        output.write(hash);
                        output.writeLong(length);
                        break;
                    default:
                        // All other objects ...
                        output.writeObject(input.readObject());
                        break;
                }
            }
        }
    }

    /**
     * Deserialize the serialized properties on the supplied object stream.
     *
     * @param stream the stream that contains the serialized properties; may not be null
     * @param properties the collection into which each deserialized property is to be placed; may not be null
     * @param largeValues the interface to use for writing large values; may not be null
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     * @see #deserializePropertyValues(ObjectInputStream, Name, boolean, LargeValues, LargeValues, ReferenceValues)
     * @see #serializeProperties(ObjectOutputStream, int, Iterable, LargeValues, ReferenceValues)
     */
    public void deserializeAllProperties( ObjectInputStream stream,
                                          Collection<Property> properties,
                                          LargeValues largeValues ) throws IOException, ClassNotFoundException {
        assert stream != null;
        assert properties != null;
        // Read the number of properties ...
        int count = stream.readInt();
        for (int i = 0; i != count; ++i) {
            Property property = deserializeProperty(stream, largeValues);
            assert property != null;
            properties.add(property);
        }
    }

    /**
     * Deserialize the serialized properties on the supplied object stream.
     *
     * @param stream the stream that contains the serialized properties; may not be null
     * @param properties the collection into which each deserialized property is to be placed; may not be null
     * @param names the names of the properties that should be deserialized; should not be null or empty
     * @param largeValues the interface to use for writing large values; may not be null
     * @param skippedLargeValues the interface to use for recording the large values that were skipped; may not be null
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     * @see #deserializePropertyValues(ObjectInputStream, Name, boolean, LargeValues, LargeValues, ReferenceValues)
     * @see #serializeProperties(ObjectOutputStream, int, Iterable, LargeValues, ReferenceValues)
     */
    public void deserializeSomeProperties( ObjectInputStream stream,
                                           Collection<Property> properties,
                                           LargeValues largeValues,
                                           LargeValues skippedLargeValues,
                                           Name... names ) throws IOException, ClassNotFoundException {
        assert stream != null;
        assert properties != null;
        assert names != null;
        assert names.length > 0;
        Name nameToRead = null;
        Set<Name> namesToRead = null;
        if (names.length == 1) {
            nameToRead = names[0];
        } else {
            namesToRead = new HashSet<Name>();
            for (Name name : names) {
                if (name != null) namesToRead.add(name);
            }
        }

        // Read the number of properties ...
        boolean read = false;
        int count = stream.readInt();

        // Now, read the properties (or skip the ones that we're not supposed to read) ...
        for (int i = 0; i != count; ++i) {
            // Read the name ...
            String nameStr = (String)stream.readObject();
            Name name = valueFactories.getNameFactory().create(nameStr);
            assert name != null;
            read = name.equals(nameToRead) || (namesToRead != null && namesToRead.contains(namesToRead));
            if (read) {
                // Now read the property values ...
                Object[] values = deserializePropertyValues(stream, name, false, largeValues, skippedLargeValues, null);
                // Add the property to the collection ...
                Property property = propertyFactory.create(name, values);
                assert property != null;
                properties.add(property);
            } else {
                // Skip the property ...
                deserializePropertyValues(stream, name, true, largeValues, skippedLargeValues, null);
            }
        }
    }

    /**
     * Deserialize the serialized property on the supplied object stream.
     *
     * @param stream the stream that contains the serialized properties; may not be null
     * @param largeValues the interface to use for writing large values; may not be null
     * @return the deserialized property; never null
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     * @see #deserializeAllProperties(ObjectInputStream, Collection, LargeValues)
     * @see #serializeProperty(ObjectOutputStream, Property, LargeValues, ReferenceValues)
     */
    public Property deserializeProperty( ObjectInputStream stream,
                                         LargeValues largeValues ) throws IOException, ClassNotFoundException {
        // Read the name ...
        String nameStr = (String)stream.readObject();
        Name name = valueFactories.getNameFactory().create(nameStr);
        assert name != null;
        // Now read the property values ...
        Object[] values = deserializePropertyValues(stream, name, false, largeValues, largeValues, null);
        // Add the property to the collection ...
        return propertyFactory.create(name, values);
    }

    /**
     * Deserialize the serialized property on the supplied object stream.
     *
     * @param stream the stream that contains the serialized properties; may not be null
     * @param propertyName the name of the property being deserialized
     * @param skip true if the values don't need to be read, or false if they are to be read
     * @param largeValues the interface to use for writing large values; may not be null
     * @param skippedLargeValues the interface to use for recording the large values that were skipped; may not be null
     * @param references the interface to use for recording which {@link Reference} values were found (and/or removed) during
     *        deserialization; may not be null
     * @return the deserialized property values, or an empty list if there are no values
     * @throws IOException if there is an error writing to the <code>stream</code> or <code>largeValues</code>
     * @throws ClassNotFoundException if the class for the value's object could not be found
     * @see #deserializeAllProperties(ObjectInputStream, Collection, LargeValues)
     * @see #serializeProperty(ObjectOutputStream, Property, LargeValues, ReferenceValues)
     */
    public Object[] deserializePropertyValues( ObjectInputStream stream,
                                               Name propertyName,
                                               boolean skip,
                                               LargeValues largeValues,
                                               LargeValues skippedLargeValues,
                                               ReferenceValues references ) throws IOException, ClassNotFoundException {
        assert stream != null;
        assert propertyName != null;
        assert largeValues != null;
        assert skippedLargeValues != null;
        // Read the number of values ...
        int size = stream.readInt();
        Object[] values = skip ? null : new Object[size];
        for (int i = 0; i != size; ++i) {
            Object value = null;
            // Read the type of value ...
            char type = stream.readChar();
            switch (type) {
                case 'S':
                    String stringValue = (String)stream.readObject();
                    if (!skip) value = valueFactories.getStringFactory().create(stringValue);
                    break;
                case 'b':
                    boolean booleanValue = stream.readBoolean();
                    if (!skip) value = valueFactories.getBooleanFactory().create(booleanValue);
                    break;
                case 'i':
                    int intValue = stream.readInt();
                    if (!skip) value = valueFactories.getLongFactory().create(intValue);
                    break;
                case 'l':
                    long longValue = stream.readLong();
                    if (!skip) value = valueFactories.getLongFactory().create(longValue);
                    break;
                case 's':
                    short shortValue = stream.readShort();
                    if (!skip) value = valueFactories.getLongFactory().create(shortValue);
                    break;
                case 'f':
                    float floatValue = stream.readFloat();
                    if (!skip) value = valueFactories.getDoubleFactory().create(floatValue);
                    break;
                case 'd':
                    double doubleValue = stream.readDouble();
                    if (!skip) value = valueFactories.getDoubleFactory().create(doubleValue);
                    break;
                case 'c':
                    // char
                    String charValue = "" + stream.readChar();
                    if (!skip) value = valueFactories.getStringFactory().create(charValue);
                    break;
                case 'U':
                    // UUID
                    long msb = stream.readLong();
                    long lsb = stream.readLong();
                    if (!skip) {
                        UUID uuid = new UUID(msb, lsb);
                        value = valueFactories.getUuidFactory().create(uuid);
                    }
                    break;
                case 'I':
                    // URI
                    String uriStr = (String)stream.readObject();
                    if (!skip) value = valueFactories.getUriFactory().create(uriStr);
                    break;
                case 'N':
                    // Name
                    String nameValueStr = (String)stream.readObject();
                    if (!skip) value = valueFactories.getNameFactory().create(nameValueStr);
                    break;
                case 'P':
                    // Path
                    String pathStr = (String)stream.readObject();
                    if (!skip) value = valueFactories.getPathFactory().create(pathStr);
                    break;
                case 'T':
                    // DateTime
                    String dateTimeStr = (String)stream.readObject();
                    if (!skip) value = valueFactories.getDateFactory().create(dateTimeStr);
                    break;
                case 'D':
                    // BigDecimal
                    Object bigDecimal = stream.readObject();
                    if (!skip) value = valueFactories.getDecimalFactory().create(bigDecimal);
                    break;
                case 'R':
                    // Reference
                    String refValue = (String)stream.readObject();
                    Reference ref = valueFactories.getReferenceFactory().create(refValue);
                    if (skip) {
                        if (references != null) references.remove(ref);
                    } else {
                        value = ref;
                        if (references != null) references.read(ref);
                    }
                    break;
                case 'B':
                    // Binary
                    // Read the length of the content ...
                    long binaryLength = stream.readLong();
                    byte[] content = new byte[(int)binaryLength];
                    stream.read(content);
                    if (!skip) {
                        value = valueFactories.getBinaryFactory().create(content);
                    }
                    break;
                case 'L':
                    // Large object ...
                    // Read the hash ...
                    int hashLength = stream.readInt();
                    byte[] hash = new byte[hashLength];
                    stream.read(hash);
                    // Read the length of the content ...
                    long length = stream.readLong();
                    if (skip) {
                        skippedLargeValues.read(valueFactories, hash, length);
                    } else {
                        BinaryFactory factory = valueFactories.getBinaryFactory();
                        // Look for an already-loaded Binary value with the same hash ...
                        value = factory.find(hash);
                        if (value == null) {
                            // Didn't find an existing large value, so we have to read the large value ...
                            value = largeValues.read(valueFactories, hash, length);
                        }
                    }
                    break;
                default:
                    // All other objects ...
                    Object object = stream.readObject();
                    if (!skip) value = valueFactories.getObjectFactory().create(object);
                    break;
            }
            if (value != null) values[i] = value;
        }
        return values;
    }

    public byte[] computeHash( String value ) {
        try {
            return SecureHash.getHash(SecureHash.Algorithm.SHA_1, value.getBytes());
        } catch (NoSuchAlgorithmException e) {
            throw new SystemFailureException(e);
        }
    }
}
TOP

Related Classes of org.jboss.dna.connector.store.jpa.util.Serializer$LargeValues

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.
ga('send', 'pageview');