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

org.infinispan.schematic.internal.SchematicEntryLiteral Maven / Gradle / Ivy

Go to download

Module for storing JSON/BSON documents and JSON Schemas in Infinispan

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.infinispan.schematic.internal;

import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Set;
import java.util.concurrent.atomic.AtomicReference;
import org.infinispan.Cache;
import org.infinispan.atomic.Delta;
import org.infinispan.atomic.DeltaAware;
import org.infinispan.commons.marshall.SerializeWith;
import org.infinispan.commons.util.Util;
import org.infinispan.schematic.SchematicEntry;
import org.infinispan.schematic.document.Binary;
import org.infinispan.schematic.document.Document;
import org.infinispan.schematic.document.Document.Field;
import org.infinispan.schematic.document.EditableDocument;
import org.infinispan.schematic.document.Path;
import org.infinispan.schematic.internal.delta.Operation;
import org.infinispan.schematic.internal.delta.PutOperation;
import org.infinispan.schematic.internal.delta.RemoveOperation;
import org.infinispan.schematic.internal.document.BasicDocument;
import org.infinispan.schematic.internal.document.MutableDocument;
import org.infinispan.schematic.internal.document.Paths;
import org.infinispan.schematic.internal.marshall.Ids;
import org.infinispan.util.logging.Log;
import org.infinispan.util.logging.LogFactory;

/**
 * The primary implementation of {@link SchematicEntry}.
 * 
 * @author Randall Hauch  (C) 2011 Red Hat Inc.
 * @since 5.1
 * @see org.infinispan.atomic.AtomicHashMap
 */
@SerializeWith( SchematicEntryLiteral.Externalizer.class )
public class SchematicEntryLiteral implements SchematicEntry, DeltaAware {

    private static final Log LOGGER = LogFactory.getLog(SchematicEntryLiteral.class);
    private static final boolean TRACE = LOGGER.isTraceEnabled();

    protected static class FieldPath {
        protected static final Path ROOT = Paths.path();
        protected static final Path METADATA = Paths.path(FieldName.METADATA);
        protected static final Path CONTENT = Paths.path(FieldName.CONTENT);
        protected static final Path ID = Paths.path(FieldName.METADATA, FieldName.ID);
        protected static final Path CONTENT_TYPE = Paths.path(FieldName.METADATA, FieldName.CONTENT_TYPE);
    }

    /**
     * Construction only allowed through this factory method. This factory is intended for use internally by the CacheDelegate.
     * User code should use {@link SchematicEntryLookup#getSchematicValue(CacheContext, String)}.
     * 
     * @param cache underlying cache
     * @param key key under which the schematic value exists
     * @return the schematic entry
     */
    public static SchematicEntry newInstance( Cache cache,
                                              String key ) {
        SchematicEntry value = new SchematicEntryLiteral(key);
        SchematicEntry oldValue = cache.putIfAbsent(key, value);
        if (oldValue != null) value = oldValue;
        return value;
    }

    private volatile MutableDocument value;
    private final AtomicReference delta = new AtomicReference(null);
    private volatile SchematicEntryProxy proxy;
    volatile boolean copied = false;
    volatile boolean removed = false;

    public SchematicEntryLiteral() {
        value = new BasicDocument(FieldName.METADATA, new BasicDocument(), FieldName.CONTENT, new BasicDocument());
    }

    public SchematicEntryLiteral( String key ) {
        Document metadata = new BasicDocument(FieldName.ID, key);
        value = new BasicDocument(FieldName.METADATA, metadata, FieldName.CONTENT, new BasicDocument());
    }

    protected SchematicEntryLiteral( MutableDocument document ) {
        this.value = document;
        assert this.value != null;
    }

    protected SchematicEntryLiteral( String key,
                                     Document content,
                                     Document metadata,
                                     String defaultContentType ) {
        this(key);
        internalSetContent(content, metadata, defaultContentType);
    }

    protected SchematicEntryLiteral( String key,
                                     Binary content,
                                     Document metadata,
                                     String defaultContentType ) {
        this(key);
        internalSetContent(content, metadata, defaultContentType);
    }

    protected final String key() {
        return getMetadata().getString(FieldName.ID);
    }

    public SchematicEntryLiteral copyForWrite() {
        SchematicEntryLiteral clone = new SchematicEntryLiteral((MutableDocument)value.clone());
        clone.proxy = proxy;
        clone.copied = true;
        return clone;
    }

    protected final MutableDocument data() {
        return value;
    }

    protected void setDocument( Document document ) {
        assert this.value != null;
        this.value = (MutableDocument)document;
    }

    @Override
    public String toString() {
        return "SchematicEntryLiteral" + value;
    }

    /**
     * Builds a thread-safe proxy for this instance so that concurrent reads are isolated from writes.
     * 
     *
     * @param context the cache context
     * @param mapKey the key
     * @return an instance of {@link SchematicEntryProxy}
     */
    public SchematicEntry getProxy( CacheContext context,
                                    String mapKey ) {
        // construct the proxy lazily
        if (proxy == null) { // DCL is OK here since proxy is volatile (and we live in a post-JDK 5 world)
            synchronized (this) {
                if (proxy == null) proxy = new SchematicEntryProxy(context, mapKey);
            }
        }
        return proxy;
    }

    public void markRemoved( boolean b ) {
        removed = b;
    }

    @Override
    public Delta delta() {
        SchematicDelta delta = this.delta.getAndSet(null);
        return delta != null ? delta : new SchematicEntryWholeDelta(value);
    }

    protected final SchematicDelta getDelta() {
        return delta.get();
    }

    protected void createDelta( CacheContext context ) {
        assert delta.get() == null;
        if (context.isDeltaContainingChangesEnabled()) {
            delta.set(new SchematicEntryDelta(key()));
        } else {
            delta.set(new SchematicEntryWholeDelta(value));
        }
    }

    @Override
    public void commit() {
        if (TRACE) {
            LOGGER.trace("Committed " + getKey() + ": " + data());
        }
        copied = false;
        delta.set(null);
    }

    @Override
    public Document getMetadata() {
        return value.getDocument(FieldName.METADATA);
    }

    protected String getKey() {
        return getMetadata().getString(FieldName.ID);
    }

    @Override
    public String getContentType() {
        return getMetadata().getString(FieldName.CONTENT_TYPE);
    }

    @Override
    public Object getContent() {
        return value.get(FieldName.CONTENT);
    }

    @Override
    public Document getContentAsDocument() {
        return value.getDocument(FieldName.CONTENT);
    }

    @Override
    public Binary getContentAsBinary() {
        return value.getBinary(FieldName.CONTENT);
    }

    @Override
    public boolean hasDocumentContent() {
        return getContentAsDocument() != null;
    }

    @Override
    public boolean hasBinaryContent() {
        return getContentAsBinary() != null;
    }

    protected Object setContent( Object content ) {
        assert content != null;
        if (content instanceof EditableDocument) {
            content = ((EditableDocument)content).unwrap();
        }

        Object existing = this.value.put(FieldName.CONTENT, content);
        SchematicDelta delta = this.delta.get();
        if (delta != null && delta.isRecordingOperations()) {
            if (existing != null) {
                delta.addOperation(new PutOperation(FieldPath.ROOT, FieldName.CONTENT, existing, content));
            } else {
                delta.addOperation(new RemoveOperation(FieldPath.ROOT, FieldName.CONTENT, content));
            }
        }
        return existing;
    }

    protected void setMetadata( Document metadata,
                                String defaultContentType ) {
        if (metadata != null) {
            if (metadata instanceof EditableDocument) metadata = ((EditableDocument)metadata).unwrap();

            // Copy all the metadata into the entry's metadata ...
            Document existingMetadata = getMetadata();
            MutableDocument newMetadata = new BasicDocument(metadata.size() + 1);
            newMetadata.put(FieldName.ID, existingMetadata.get(FieldName.ID));
            for (Field field : metadata.fields()) {
                String fieldName = field.getName();
                if (fieldName.equals(FieldName.ID)) continue;
                newMetadata.put(fieldName, field.getValue());
            }

            // Make sure the metadata has the content type
            if (newMetadata.getString(FieldName.CONTENT_TYPE) == null) {
                newMetadata.put(FieldName.CONTENT_TYPE, defaultContentType);
            }

            // Now record the change ...
            value.put(FieldName.METADATA, newMetadata);
            SchematicDelta delta = this.delta.get();
            if (delta != null && delta.isRecordingOperations()) {
                PutOperation op = new PutOperation(FieldPath.ROOT, FieldName.METADATA, existingMetadata, newMetadata);
                delta.addOperation(op);
            }
        }
    }

    protected void internalSetContent( Document content,
                                       Document metadata,
                                       String defaultContentType ) {
        setContent(content);
        setMetadata(metadata, defaultContentType);
    }

    protected void internalSetContent( Binary content,
                                       Document metadata,
                                       String defaultContentType ) {
        setContent(content);
        setMetadata(metadata, defaultContentType);
    }

    @Override
    public void setContent( Document content,
                            Document metadata,
                            String defaultContentType ) {
        // Should always go through the proxy instead
        throw new UnsupportedOperationException();
    }

    @Override
    public void setContent( Binary content,
                            Document metadata,
                            String defaultContentType ) {
        // Should always go through the proxy instead
        throw new UnsupportedOperationException();
    }

    @Override
    public EditableDocument editDocumentContent() {
        // Should always go through the proxy instead
        throw new UnsupportedOperationException();
    }

    @Override
    public EditableDocument editMetadata() {
        // Should always go through the proxy instead
        throw new UnsupportedOperationException();
    }

    @Override
    public Document asDocument() {
        return value;
    }

    boolean apply( Iterable changes ) {
        try {
            for (Operation o : changes) {
                o.replay(value);
            }
        } catch (AssertionError e) {
            LOGGER.debug("Assertion while applying changes to " + value + " --> " + changes);
            throw e;
        }
        return true;
    }

    /**
     * The {@link org.infinispan.marshall.Externalizer Externalizer} for {@link SchematicEntryLiteral} instances.
     */
    public static final class Externalizer extends SchematicExternalizer {
        private static final long serialVersionUID = 1L;

        @Override
        public void writeObject( ObjectOutput output,
                                 SchematicEntryLiteral literal ) throws IOException {
            output.writeObject(literal.data());
        }

        @Override
        public SchematicEntryLiteral readObject( ObjectInput input ) throws IOException, ClassNotFoundException {
            MutableDocument doc = (MutableDocument)input.readObject();
            return new SchematicEntryLiteral(doc);
        }

        @Override
        public Integer getId() {
            return Ids.SCHEMATIC_VALUE_LITERAL;
        }

        @SuppressWarnings( "unchecked" )
        @Override
        public Set> getTypeClasses() {
            return Util.>asSet(SchematicEntryLiteral.class);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy