Mercurial > hg > orthanc
view OrthancFramework/Sources/HttpServer/WebDavStorage.cpp @ 5559:462b8f8a619c
todo private tags in ExtraMainDicomTags
author | Alain Mazy <am@orthanc.team> |
---|---|
date | Mon, 22 Apr 2024 10:39:33 +0200 |
parents | 48b8dae6dc77 |
children | f7adfb22e20e |
line wrap: on
line source
/** * Orthanc - A Lightweight, RESTful DICOM Store * Copyright (C) 2012-2016 Sebastien Jodogne, Medical Physics * Department, University Hospital of Liege, Belgium * Copyright (C) 2017-2024 Osimis S.A., Belgium * Copyright (C) 2021-2024 Sebastien Jodogne, ICTEAM UCLouvain, Belgium * * This program is free software: you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * This program 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 program. If not, see * <http://www.gnu.org/licenses/>. **/ #include "../PrecompiledHeaders.h" #include "WebDavStorage.h" #include "../Logging.h" #include "../OrthancException.h" #include "../SystemToolbox.h" #include "../TemporaryFile.h" #include "../Toolbox.h" namespace Orthanc { class WebDavStorage::StorageFile : public boost::noncopyable { private: std::unique_ptr<TemporaryFile> file_; std::string content_; MimeType mime_; boost::posix_time::ptime time_; void Touch() { time_ = boost::posix_time::second_clock::universal_time(); } public: StorageFile() : mime_(MimeType_Binary) { Touch(); } void SetContent(const std::string& content, MimeType mime, bool isMemory) { if (isMemory) { content_ = content; file_.reset(); } else { content_.clear(); file_.reset(new TemporaryFile); file_->Write(content); } mime_ = mime; Touch(); } MimeType GetMimeType() const { return mime_; } void GetContent(std::string& target) const { if (file_.get() == NULL) { target = content_; } else { file_->Read(target); } } const boost::posix_time::ptime& GetTime() const { return time_; } uint64_t GetContentLength() const { if (file_.get() == NULL) { return content_.size(); } else { return file_->GetFileSize(); } } }; class WebDavStorage::StorageFolder : public boost::noncopyable { private: typedef std::map<std::string, StorageFile*> Files; typedef std::map<std::string, StorageFolder*> Subfolders; Files files_; Subfolders subfolders_; boost::posix_time::ptime time_; void Touch() { time_ = boost::posix_time::second_clock::universal_time(); } void CheckName(const std::string& name) { if (name.empty() || name.find('/') != std::string::npos || name.find('\\') != std::string::npos || name.find('\0') != std::string::npos) { throw OrthancException(ErrorCode_ParameterOutOfRange, "Bad resource name for WebDAV: " + name); } } bool IsExisting(const std::string& name) const { return (files_.find(name) != files_.end() || subfolders_.find(name) != subfolders_.end()); } public: StorageFolder() { Touch(); } ~StorageFolder() { for (Files::iterator it = files_.begin(); it != files_.end(); ++it) { assert(it->second != NULL); delete it->second; } for (Subfolders::iterator it = subfolders_.begin(); it != subfolders_.end(); ++it) { assert(it->second != NULL); delete it->second; } } size_t GetSize() const { return files_.size() + subfolders_.size(); } const boost::posix_time::ptime& GetModificationTime() const { return time_; } const StorageFile* LookupFile(const std::string& name) const { Files::const_iterator found = files_.find(name); if (found == files_.end()) { return NULL; } else { assert(found->second != NULL); return found->second; } } bool CreateSubfolder(const std::string& name) { CheckName(name); if (IsExisting(name)) { LOG(ERROR) << "WebDAV folder already existing: " << name; return false; } else { subfolders_[name] = new StorageFolder; Touch(); return true; } } bool StoreFile(const std::string& name, const std::string& content, MimeType mime, bool isMemory) { CheckName(name); if (subfolders_.find(name) != subfolders_.end()) { LOG(ERROR) << "WebDAV folder already existing: " << name; return false; } Files::iterator found = files_.find(name); if (found == files_.end()) { std::unique_ptr<StorageFile> f(new StorageFile); f->SetContent(content, mime, isMemory); files_[name] = f.release(); } else { assert(found->second != NULL); found->second->SetContent(content, mime, isMemory); } Touch(); return true; } StorageFolder* LookupFolder(const std::vector<std::string>& path) { if (path.empty()) { return this; } else { Subfolders::const_iterator found = subfolders_.find(path[0]); if (found == subfolders_.end()) { return NULL; } else { assert(found->second != NULL); std::vector<std::string> p(path.begin() + 1, path.end()); return found->second->LookupFolder(p); } } } void ListCollection(Collection& collection) const { for (Files::const_iterator it = files_.begin(); it != files_.end(); ++it) { assert(it->second != NULL); std::unique_ptr<File> f(new File(it->first)); f->SetContentLength(it->second->GetContentLength()); f->SetCreationTime(it->second->GetTime()); collection.AddResource(f.release()); } for (Subfolders::const_iterator it = subfolders_.begin(); it != subfolders_.end(); ++it) { std::unique_ptr<Folder> f(new Folder(it->first)); f->SetModificationTime(it->second->GetModificationTime()); collection.AddResource(f.release()); } } bool DeleteItem(const std::vector<std::string>& path) { if (path.size() == 0) { throw OrthancException(ErrorCode_InternalError); } else if (path.size() == 1) { { Files::iterator f = files_.find(path[0]); if (f != files_.end()) { assert(f->second != NULL); delete f->second; files_.erase(f); Touch(); return true; } } { Subfolders::iterator f = subfolders_.find(path[0]); if (f != subfolders_.end()) { assert(f->second != NULL); delete f->second; subfolders_.erase(f); Touch(); return true; } } return false; } else { Subfolders::iterator f = subfolders_.find(path[0]); if (f != subfolders_.end()) { assert(f->second != NULL); std::vector<std::string> p(path.begin() + 1, path.end()); if (f->second->DeleteItem(p)) { Touch(); return true; } else { return false; } } else { return false; } } } void RemoveEmptyFolders() { std::list<std::string> emptyFolders; for (Subfolders::const_iterator it = subfolders_.begin(); it != subfolders_.end(); ++it) { assert(it->second != NULL); it->second->RemoveEmptyFolders(); if (it->second->GetSize() == 0) { assert(it->second != NULL); delete it->second; emptyFolders.push_back(it->first); } } for (std::list<std::string>::const_iterator it = emptyFolders.begin(); it != emptyFolders.end(); ++it) { assert(subfolders_.find(*it) != subfolders_.end()); subfolders_.erase(*it); } } }; WebDavStorage::StorageFolder* WebDavStorage::LookupParentFolder(const std::vector<std::string>& path) { if (path.empty()) { throw OrthancException(ErrorCode_ParameterOutOfRange); } std::vector<std::string> p(path.begin(), path.end() - 1); return root_->LookupFolder(p); } WebDavStorage::WebDavStorage(bool isMemory) : root_(new StorageFolder), isMemory_(isMemory) { } bool WebDavStorage::IsExistingFolder(const std::vector<std::string>& path) { boost::recursive_mutex::scoped_lock lock(mutex_); return (root_->LookupFolder(path) != NULL); } bool WebDavStorage::ListCollection(Collection& collection, const std::vector<std::string>& path) { boost::recursive_mutex::scoped_lock lock(mutex_); const StorageFolder* folder = root_->LookupFolder(path); if (folder == NULL) { return false; } else { folder->ListCollection(collection); return true; } } bool WebDavStorage::GetFileContent(MimeType& mime, std::string& content, boost::posix_time::ptime& modificationTime, const std::vector<std::string>& path) { boost::recursive_mutex::scoped_lock lock(mutex_); const StorageFolder* folder = LookupParentFolder(path); if (folder == NULL) { return false; } else { const StorageFile* file = folder->LookupFile(path.back()); if (file == NULL) { return false; } else { mime = file->GetMimeType(); file->GetContent(content); modificationTime = file->GetTime(); return true; } } } bool WebDavStorage::StoreFile(const std::string& content, const std::vector<std::string>& path) { boost::recursive_mutex::scoped_lock lock(mutex_); StorageFolder* folder = LookupParentFolder(path); if (folder == NULL) { LOG(WARNING) << "Inexisting folder in WebDAV: " << Toolbox::FlattenUri(path); return false; } else { LOG(INFO) << "Storing " << content.size() << " bytes in WebDAV bucket: " << Toolbox::FlattenUri(path);; MimeType mime = SystemToolbox::AutodetectMimeType(path.back()); return folder->StoreFile(path.back(), content, mime, isMemory_); } } bool WebDavStorage::CreateFolder(const std::vector<std::string>& path) { boost::recursive_mutex::scoped_lock lock(mutex_); StorageFolder* folder = LookupParentFolder(path); if (folder == NULL) { LOG(WARNING) << "Inexisting folder in WebDAV: " << Toolbox::FlattenUri(path); return false; } else { LOG(INFO) << "Creating folder in WebDAV bucket: " << Toolbox::FlattenUri(path); return folder->CreateSubfolder(path.back()); } } bool WebDavStorage::DeleteItem(const std::vector<std::string>& path) { if (path.empty()) { return false; // Cannot delete the root } else { boost::recursive_mutex::scoped_lock lock(mutex_); LOG(INFO) << "Deleting from WebDAV bucket: " << Toolbox::FlattenUri(path); return root_->DeleteItem(path); } } void WebDavStorage::RemoveEmptyFolders() { boost::recursive_mutex::scoped_lock lock(mutex_); root_->RemoveEmptyFolders(); } }