// Copyright (c) 2013, Thomas Goyne <plorkyeran@aegisub.org> // // Permission to use, copy, modify, and distribute this software for any // purpose with or without fee is hereby granted, provided that the above // copyright notice and this permission notice appear in all copies. // // THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES // WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR // ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES // WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN // ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. // // Aegisub Project http://www.aegisub.org/ #include "config.h" #include "libaegisub/fs.h" #include "libaegisub/access.h" #include "libaegisub/exception.h" #include "libaegisub/log.h" #include <boost/algorithm/string/predicate.hpp> #define BOOST_NO_SCOPED_ENUMS #include <boost/filesystem.hpp> #undef BOOST_NO_SCOPED_ENUMS namespace bfs = boost::filesystem; namespace ec = boost::system::errc; // boost::filesystem functions throw a single exception type for all // errors, which isn't really what we want, so do some crazy wrapper // shit to map error codes to more useful exceptions. #define CHECKED_CALL(exp, src_path, dst_path) \ boost::system::error_code ec; \ exp; \ switch (ec.value()) {\ case ec::success: break; \ case ec::no_such_file_or_directory: throw FileNotFound(src_path); \ case ec::is_a_directory: throw NotAFile(src_path); \ case ec::not_a_directory: throw NotADirectory(src_path); \ case ec::no_space_on_device: throw DriveFull(dst_path); \ case ec::permission_denied: \ if (!src_path.empty()) \ acs::CheckFileRead(src_path); \ if (!dst_path.empty()) \ acs::CheckFileWrite(dst_path); \ throw AccessDenied(src_path); \ default: \ LOG_D("filesystem") << "Unknown error when calling '" << #exp << "': " << ec << ": " << ec.message(); \ throw FileSystemUnknownError(ec.message()); \ } #define CHECKED_CALL_RETURN(exp, src_path) \ CHECKED_CALL(auto ret = exp, src_path, agi::fs::path()); \ return ret #define WRAP_BFS(bfs_name, agi_name) \ auto agi_name(path const& p) -> decltype(bfs::bfs_name(p)) { \ CHECKED_CALL_RETURN(bfs::bfs_name(p, ec), p); \ } #define WRAP_BFS_IGNORE_ERROR(bfs_name, agi_name) \ auto agi_name(path const& p) -> decltype(bfs::bfs_name(p)) { \ boost::system::error_code ec; \ return bfs::bfs_name(p, ec); \ } // sasuga windows.h #undef CreateDirectory namespace agi { namespace fs { WRAP_BFS_IGNORE_ERROR(exists, Exists) WRAP_BFS_IGNORE_ERROR(is_regular_file, FileExists) WRAP_BFS_IGNORE_ERROR(is_directory, DirectoryExists) WRAP_BFS(file_size, SizeImpl) WRAP_BFS(last_write_time, ModifiedTime) WRAP_BFS(create_directories, CreateDirectory) WRAP_BFS(space, Space) WRAP_BFS(remove, Remove) WRAP_BFS(canonical, Canonicalize) uintmax_t Size(path const& p) { if (DirectoryExists(p)) throw NotAFile(p); return SizeImpl(p); } uintmax_t FreeSpace(path const& p) { return Space(p).available; } void Rename(const path& from, const path& to) { CHECKED_CALL(bfs::rename(from, to, ec), from, to); } bool HasExtension(path const& p, std::string const& ext) { auto filename = p.filename().string(); if (filename.size() < ext.size() + 1) return false; if (filename[filename.size() - ext.size() - 1] != '.') return false; return boost::iends_with(filename, ext); } } }