#include "DirBlob.h" #include //TODO Remove and replace with exception hierarchy #include "messmer/fspp/fuse/FuseErrnoException.h" #include #include "MagicNumbers.h" using std::unique_ptr; using std::make_unique; using std::vector; using std::string; using std::pair; using std::make_pair; using blobstore::Blob; using blockstore::Key; using blockstore::Data; //TODO Refactor: Keep a parsed dir structure (list of entries and blob keys they're pointing to) in memory and serialize/deserialize it namespace cryfs { DirBlob::DirBlob(unique_ptr blob) : _blob(std::move(blob)) { assert(magicNumber() == MagicNumbers::DIR); } DirBlob::~DirBlob() { } unique_ptr DirBlob::InitializeEmptyDir(unique_ptr blob) { blob->resize(1); unsigned char magicNumber = MagicNumbers::DIR; blob->write(&magicNumber, 0, 1); return make_unique(std::move(blob)); } unsigned char DirBlob::magicNumber() const { unsigned char number; _blob->read(&number, 0, 1); return number; } unique_ptr> DirBlob::GetChildren() const { Data entries(_blob->size()-1); _blob->read(entries.data(), 1, _blob->size()-1); auto result = make_unique>(); const char *pos = (const char*)entries.data(); while(pos < (const char*)entries.data()+entries.size()) { pos = readAndAddNextChild(pos, result.get()); } return result; } const char *DirBlob::readAndAddNextChild(const char *pos, vector *result) const { // Read type magic number (whether it is a dir or a file) fspp::Dir::EntryType type = static_cast(*reinterpret_cast(pos)); pos += 1; size_t length = strlen(pos); std::string name(pos, length); result->emplace_back(fspp::Dir::Entry(type, name)); const char *posAfterName = pos + length + 1; const char *posAfterKey = posAfterName + strlen(posAfterName) + 1; return posAfterKey; } void DirBlob::AddChildDir(const std::string &name, const Key &blobKey) { AddChild(name, blobKey, fspp::Dir::EntryType::DIR); } void DirBlob::AddChildFile(const std::string &name, const Key &blobKey) { AddChild(name, blobKey, fspp::Dir::EntryType::FILE); } void DirBlob::AddChild(const std::string &name, const Key &blobKey, fspp::Dir::EntryType entryType) { //TODO blob.resize(blob.size()+X) has to traverse tree twice. Better would be blob.addSize(X) which returns old size uint64_t oldBlobSize = _blob->size(); string blobKeyStr = blobKey.ToString(); _blob->resize(oldBlobSize + name.size() + 1 + blobKeyStr.size() + 1); //Write entry type unsigned char entryTypeMagicNumber = static_cast(entryType); _blob->write(&entryTypeMagicNumber, oldBlobSize, 1); //Write entry name inclusive null terminator _blob->write(name.c_str(), oldBlobSize + 1, name.size()+1); //Write blob key inclusive null terminator _blob->write(blobKeyStr.c_str(), oldBlobSize + 1 + name.size() + 1, blobKeyStr.size()+1); } pair DirBlob::GetChild(const string &name) const { Data entries(_blob->size()-1); _blob->read(entries.data(), 1, _blob->size()-1); const char *pos = (const char*)entries.data(); while(pos < (const char*)entries.data()+entries.size()) { fspp::Dir::EntryType type = static_cast(*reinterpret_cast(pos)); pos += 1; // Skip entry type magic number (whether it is a dir or a file) size_t name_length = strlen(pos); if (name_length == name.size() && 0==std::memcmp(pos, name.c_str(), name_length)) { pos += strlen(pos) + 1; // Skip name return make_pair(type, Key::FromString(pos)); // Return key } pos += strlen(pos) + 1; // Skip name pos += strlen(pos) + 1; // Skip key } throw fspp::fuse::FuseErrnoException(ENOENT); } }