#pragma once #ifndef BLOBSTORE_IMPLEMENTATIONS_ONBLOCKS_DATATREE_H_ #define BLOBSTORE_IMPLEMENTATIONS_ONBLOCKS_DATATREE_H_ #include #include #include #include "../datanodestore/DataNodeView.h" namespace blockstore { class Key; } namespace blobstore { namespace onblocks { namespace datanodestore { class DataNodeStore; class DataInnerNode; class DataLeafNode; class DataNode; } namespace datatreestore { //TODO It is strange that DataLeafNode is still part in the public interface of DataTree. This should be separated somehow. class DataTree { public: DataTree(datanodestore::DataNodeStore *nodeStore, std::unique_ptr rootNode); virtual ~DataTree(); const blockstore::Key &key() const; uint32_t maxBytesPerLeaf() const; void traverseLeaves(uint32_t beginIndex, uint32_t endIndex, std::function func) const; void traverseLeaves(uint32_t beginIndex, uint32_t endIndex, std::function func); void resizeNumBytes(uint64_t newNumBytes); uint64_t numStoredBytes() const; private: datanodestore::DataNodeStore *_nodeStore; std::unique_ptr _rootNode; std::unique_ptr addDataLeaf(); void removeLastDataLeaf(); std::unique_ptr releaseRootNode(); friend class DataTreeStore; std::unique_ptr addDataLeafAt(datanodestore::DataInnerNode *insertPos); cpputils::optional_ownership_ptr createChainOfInnerNodes(unsigned int num, datanodestore::DataLeafNode *leaf); std::unique_ptr addDataLeafToFullTree(); void deleteLastChildSubtree(datanodestore::DataInnerNode *node); void ifRootHasOnlyOneChildReplaceRootWithItsChild(); void traverseLeaves(const datanodestore::DataNode *root, uint32_t leafOffset, uint32_t beginIndex, uint32_t endIndex, std::function func) const; uint32_t leavesPerFullChild(const datanodestore::DataInnerNode &root) const; uint64_t numStoredBytes(const datanodestore::DataNode &root) const; cpputils::optional_ownership_ptr LastLeaf(datanodestore::DataNode *root); std::unique_ptr LastLeaf(std::unique_ptr root); void flush() const; DISALLOW_COPY_AND_ASSIGN(DataTree); }; } } } #endif