/* * StorageBackendFile.cpp * * Copyright (C) 2009 Matthias Schiffer * * 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 . */ #include "StorageBackendFile.h" #include #include #include namespace Mad { namespace Modules { namespace StorageBackendFile { bool StorageBackendFile::handleConfigEntry(const Core::ConfigEntry &entry, bool /*handled*/) { if(!entry[0].getKey().matches("Storage")) return false; boost::lock_guard lock(mutex); if(entry[1].getKey().matches("Root")) { if(!entry[2].empty()) return false; storageRoot = entry[1][0]; } else if(!entry[1].empty()) { return false; } return true; } void StorageBackendFile::configFinished() { boost::lock_guard lock(mutex); if(!boost::filesystem::exists(storageRoot)) { boost::filesystem::create_directories(storageRoot); } if(!boost::filesystem::is_directory(storageRoot)) { application->logf("StorageBackendFile: Can't create directory: '%s'.", storageRoot.string().c_str()); return; } configured = true; } boost::filesystem::path StorageBackendFile::getFileName(const std::string &type, const std::string &name) { boost::filesystem::path path(storageRoot); path /= type; path /= (name + ".xml"); return path; } std::set StorageBackendFile::listTypes() throw (Core::Exception) { boost::shared_lock lock(mutex); if(!configured) throw Core::Exception(Core::Exception::NOT_AVAILABLE); std::set ret; try { for(boost::filesystem::directory_iterator it(storageRoot); it != boost::filesystem::directory_iterator(); ++it) { boost::filesystem::path path = *it; if(boost::filesystem::is_directory(path)) ret.insert(path.filename()); } } catch(...) {} // Don't throw if we can't iterate over the dirs return ret; } std::set StorageBackendFile::list(const std::string &type) throw (Core::Exception) { boost::shared_lock lock(mutex); if(!configured) throw Core::Exception(Core::Exception::NOT_AVAILABLE); std::set ret; boost::filesystem::path path(storageRoot); path /= type; try { for(boost::filesystem::directory_iterator it(path); it != boost::filesystem::directory_iterator(); ++it) { boost::filesystem::path filePath = *it; if(boost::filesystem::is_regular_file(filePath) && filePath.extension() == ".xml") ret.insert(filePath.replace_extension().filename()); } } catch(...) {} // Don't throw if we can't iterate over the dirs return ret; } void StorageBackendFile::store(const std::string &type, const std::string &name, const Common::XmlData *data) throw (Core::Exception) { boost::lock_guard lock(mutex); if(!configured) throw Core::Exception(Core::Exception::NOT_AVAILABLE); boost::filesystem::path path = getFileName(type, name); boost::filesystem::create_directories(path.parent_path().directory_string()); data->toFile(path.file_string()); } boost::shared_ptr StorageBackendFile::load(const std::string &type, const std::string &name) throw (Core::Exception) { boost::shared_lock lock(mutex); if(!configured) throw Core::Exception(Core::Exception::NOT_AVAILABLE); boost::filesystem::path path = getFileName(type, name); return boost::shared_ptr(new Common::XmlData(path.file_string())); } } } }