Package com.orientechnologies.orient.core.type.tree.provider

Source Code of com.orientechnologies.orient.core.type.tree.provider.OMVRBTreeRIDProvider

/*
  *
  *  *  Copyright 2014 Orient Technologies LTD (info(at)orientechnologies.com)
  *  *
  *  *  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.
  *  *
  *  * For more information: http://www.orientechnologies.com
  *
  */
package com.orientechnologies.orient.core.type.tree.provider;

import java.util.ArrayList;
import java.util.Collection;
import java.util.IdentityHashMap;
import java.util.StringTokenizer;

import com.orientechnologies.common.profiler.OProfilerMBean;
import com.orientechnologies.orient.core.Orient;
import com.orientechnologies.orient.core.config.OGlobalConfiguration;
import com.orientechnologies.orient.core.db.record.OIdentifiable;
import com.orientechnologies.orient.core.exception.OSerializationException;
import com.orientechnologies.orient.core.id.ORID;
import com.orientechnologies.orient.core.id.ORecordId;
import com.orientechnologies.orient.core.index.mvrbtree.OMVRBTree;
import com.orientechnologies.orient.core.metadata.schema.OType;
import com.orientechnologies.orient.core.record.ORecord;
import com.orientechnologies.orient.core.record.ORecordInternal;
import com.orientechnologies.orient.core.record.impl.ODocument;
import com.orientechnologies.orient.core.serialization.OSerializableStream;
import com.orientechnologies.orient.core.serialization.serializer.OStringSerializerHelper;
import com.orientechnologies.orient.core.serialization.serializer.string.OStringBuilderSerializable;
import com.orientechnologies.orient.core.storage.OStorage;
import com.orientechnologies.orient.core.type.tree.OMVRBTreeEntryPersistent;
import com.orientechnologies.orient.core.type.tree.OMVRBTreePersistent;
import com.orientechnologies.orient.core.type.tree.OMVRBTreeRID;

/**
* MVRB-Tree implementation to handle a set of RID. It's serialized as embedded or external binary. Once external cannot come back
* to the embedded mode.
*
* @author Luca Garulli (l.garulli--at--orientechnologies.com)
*/
public class OMVRBTreeRIDProvider extends OMVRBTreeProviderAbstract<OIdentifiable, OIdentifiable> implements
    OStringBuilderSerializable {
  public static final String            PERSISTENT_CLASS_NAME = "ORIDs";
  private static final long             serialVersionUID      = 1L;

  private OMVRBTreeRID                  tree;
  private boolean                       embeddedStreaming     = true;                           // KEEP THE STREAMING MODE
  private int                           binaryThreshold       = OGlobalConfiguration.MVRBTREE_RID_BINARY_THRESHOLD
                                                                  .getValueAsInteger();

  private final StringBuilder           buffer                = new StringBuilder(128);
  private boolean                       marshalling           = true;

  protected static final OProfilerMBean PROFILER              = Orient.instance().getProfiler();

  /**
   * Copy constructor
   */
  public OMVRBTreeRIDProvider(final OMVRBTreeRIDProvider iSource) {
    this(null, iSource.getClusterId(), iSource.getRoot());
    buffer.append(iSource.buffer);
    marshalling = false;
  }

  public OMVRBTreeRIDProvider(final OStorage iStorage, final int iClusterId, final ORID iRID) {
    this(iStorage, getDatabase().getClusterNameById(iClusterId));
    if (iRID != null)
      root = (ORecordId) iRID.copy();
    marshalling = false;
  }

  public OMVRBTreeRIDProvider(final ORID iRID) {
    this(null, getDatabase().getDefaultClusterId());
    if (iRID != null)
      root = (ORecordId) iRID.copy();
    marshalling = false;
  }

  public OMVRBTreeRIDProvider(final OStorage iStorage, final int iClusterId) {
    this(iStorage, getDatabase().getClusterNameById(iClusterId));
    marshalling = false;
    ORecordInternal.unsetDirty(record);
  }

  public OMVRBTreeRIDProvider(final OStorage iStorage, final int iClusterId, int binaryThreshold) {
    this(iStorage, getDatabase().getClusterNameById(iClusterId));
    marshalling = false;
    ORecordInternal.unsetDirty(record);
    this.binaryThreshold = binaryThreshold;
  }

  public OMVRBTreeRIDProvider(final String iClusterName) {
    this(null, iClusterName);
    marshalling = false;
  }

  protected OMVRBTreeRIDProvider(final OStorage iStorage, final String iClusterName) {
    super(new ODocument(), iStorage, iClusterName);
    ((ODocument) record).field("pageSize", pageSize);

    getDatabase().getMetadata().getSchema().getOrCreateClass(PERSISTENT_CLASS_NAME);
  }

  @Override
  public OMVRBTreeRIDProvider copy() {
    final OMVRBTreeRIDProvider copy = new OMVRBTreeRIDProvider(storage, clusterName);
    copy.setTree(tree);
    return copy;
  }

  public OMVRBTreeRIDEntryProvider getEntry(final ORID iRid) {
    return new OMVRBTreeRIDEntryProvider(this, iRid);
  }

  public OMVRBTreeRIDEntryProvider createEntry() {
    return new OMVRBTreeRIDEntryProvider(this);
  }

  public OStringBuilderSerializable toStream(final StringBuilder iBuffer) throws OSerializationException {
    final long timer = PROFILER.startChrono();

    if (buffer.length() > 0 && getDatabase().getTransaction().isActive() && buffer.indexOf("-") > -1) {
      // IN TRANSACTION: UNMARSHALL THE BUFFER TO AVOID TO STORE TEMP RIDS
      lazyUnmarshall();
      buffer.setLength(0);
    }

    tree.saveAllNewEntries();

    if (buffer.length() == 0)
      // MARSHALL IT
      try {
        if (isEmbeddedStreaming()) {
          marshalling = true;
          // SERIALIZE AS AN EMBEDDED STRING
          buffer.append(OStringSerializerHelper.SET_BEGIN);

          // PERSISTENT RIDS
          boolean first = true;
          for (OIdentifiable rid : tree.keySet()) {
            if (rid instanceof ORecord) {
              final ORecord record = (ORecord) rid;
              if (record.isDirty())
                record.save();
            }

            if (!first)
              buffer.append(OStringSerializerHelper.COLLECTION_SEPARATOR);
            else
              first = false;

            rid.getIdentity().toString(buffer);
          }

          // TEMPORARY RIDS
          final IdentityHashMap<ORecord, Object> tempRIDs = tree.getTemporaryEntries();
          if (tempRIDs != null && !tempRIDs.isEmpty())
            for (ORecord rec : tempRIDs.keySet()) {
              if (!first)
                buffer.append(OStringSerializerHelper.COLLECTION_SEPARATOR);
              else
                first = false;

              rec.getIdentity().toString(buffer);
            }

          buffer.append(OStringSerializerHelper.SET_END);
        } else {
          marshalling = true;
          buffer.append(OStringSerializerHelper.EMBEDDED_BEGIN);
          buffer.append(new String(toDocument().toStream()));
          buffer.append(OStringSerializerHelper.EMBEDDED_END);
        }

      } finally {
        marshalling = false;
        PROFILER.stopChrono(PROFILER.getProcessMetric("mvrbtree.toStream"), "Serialize a MVRBTreeRID", timer);
      }

    iBuffer.append(buffer);

    return this;
  }

  public OSerializableStream fromStream(final byte[] iStream) throws OSerializationException {
    record.fromStream(iStream);
    fromDocument((ODocument) record);
    return this;
  }

  public OStringBuilderSerializable fromStream(final StringBuilder iInput) throws OSerializationException {
    if (iInput != null) {
      // COPY THE BUFFER: IF THE TREE IS UNTOUCHED RETURN IT
      buffer.setLength(0);
      buffer.append(iInput);
    }

    return this;
  }

  public void lazyUnmarshall() {
    if (getSize() > 0 || marshalling || buffer.length() == 0)
      // ALREADY UNMARSHALLED
      return;

    marshalling = true;
    tree.setMarshalling(true);

    try {
      final char firstChar = buffer.charAt(0);

      String value = firstChar == OStringSerializerHelper.SET_BEGIN || firstChar == OStringSerializerHelper.LIST_BEGIN ? buffer
          .substring(1, buffer.length() - 1) : buffer.toString();

      if (firstChar == OStringSerializerHelper.SET_BEGIN || firstChar == OStringSerializerHelper.LIST_BEGIN
          || firstChar == OStringSerializerHelper.LINK) {
        setEmbeddedStreaming(true);
        final StringTokenizer tokenizer = new StringTokenizer(value, ",");
        while (tokenizer.hasMoreElements()) {
          final ORecordId rid = new ORecordId(tokenizer.nextToken());
          tree.put(rid, rid);
        }
      } else {
        setEmbeddedStreaming(false);
        value = firstChar == OStringSerializerHelper.EMBEDDED_BEGIN ? value.substring(1, value.length() - 1) : value.toString();
        fromStream(value.getBytes());
      }
    } finally {
      marshalling = false;
      tree.setMarshalling(false);
    }
  }

  public byte[] toStream() throws OSerializationException {
    return toDocument().toStream();
  }

  public OMVRBTree<OIdentifiable, OIdentifiable> getTree() {
    return tree;
  }

  public void setTree(final OMVRBTreePersistent<OIdentifiable, OIdentifiable> tree) {
    this.tree = (OMVRBTreeRID) tree;
  }

  @Override
  public int getSize() {
    if (embeddedStreaming)
      return size;
    else {
      final OMVRBTreeEntryPersistent<OIdentifiable, OIdentifiable> r = (OMVRBTreeEntryPersistent<OIdentifiable, OIdentifiable>) tree
          .getRoot();
      return r == null ? 0 : (((OMVRBTreeRIDEntryProvider) r.getProvider()).getTreeSize());
    }
  }

  @Override
  public boolean setSize(final int iSize) {
    if (embeddedStreaming)
      super.setSize(iSize);
    else {
      final OMVRBTreeEntryPersistent<OIdentifiable, OIdentifiable> r = (OMVRBTreeEntryPersistent<OIdentifiable, OIdentifiable>) tree
          .getRoot();
      if (r != null) {
        final OMVRBTreeRIDEntryProvider provider = (OMVRBTreeRIDEntryProvider) r.getProvider();
        if (provider != null && provider.setTreeSize(iSize))
          r.markDirty();
        return true;
      }
    }
    return false;
  }

  @Override
  public boolean setDirty() {
    if (!marshalling) {
      if (buffer != null)
        buffer.setLength(0);

      if (tree.getOwner() != null)
        tree.getOwner().setDirty();

      return super.setDirty();
    }
    return false;
  }

  public ODocument toDocument() {
    tree.saveAllNewEntries();

    // SERIALIZE AS LINK TO THE TREE STRUCTURE
    final ODocument doc = (ODocument) record;
    doc.setClassName(PERSISTENT_CLASS_NAME);
    doc.field("root", root != null ? root : null);
    doc.field("keySize", keySize);

    if (tree.getTemporaryEntries() != null && tree.getTemporaryEntries().size() > 0)
      doc.field("tempEntries", new ArrayList<ORecord>(tree.getTemporaryEntries().keySet()));

    return doc;
  }

  public void fromDocument(final ODocument iDocument) {
    pageSize = (Integer) iDocument.field("pageSize");
    iDocument.setLazyLoad(false);
    root = iDocument.field("root");
    if (iDocument.field("keySize") != null)
      keySize = iDocument.<Integer> field("keySize");

    tree.load();

    final Collection<OIdentifiable> tempEntries = iDocument.field("tempEntries");
    if (tempEntries != null && !tempEntries.isEmpty())
      for (OIdentifiable entry : tempEntries)
        if (entry != null)
          tree.put(entry, null);
  }

  public boolean isEmbeddedStreaming() {
    if (embeddedStreaming && !marshalling) {
      if (getDatabase().getTransaction().isActive())
        // FORCE STREAMING BECAUSE TX
        return true;

      if (binaryThreshold > 0 && getSize() > binaryThreshold && tree != null) {
        // CHANGE TO EXTERNAL BINARY
        tree.setDirtyOwner();
        setEmbeddedStreaming(false);
      }
    }
    return embeddedStreaming;
  }

  protected void setEmbeddedStreaming(final boolean iValue) {
    if (embeddedStreaming != iValue) {
      embeddedStreaming = iValue;

      if (!iValue)
        // ASSURE TO SAVE THE SIZE IN THE ROOT NODE
        setSize(size);
    }
  }

  @Override
  public boolean updateConfig() {
    pageSize = OGlobalConfiguration.MVRBTREE_RID_NODE_PAGE_SIZE.getValueAsInteger();
    return false;
  }
}
TOP

Related Classes of com.orientechnologies.orient.core.type.tree.provider.OMVRBTreeRIDProvider

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.