/* * Copyright (c) 2013-2024, The PurpleI2P Project * * This file is part of Purple i2pd project and licensed under BSD3 * * See full license text in LICENSE file at top of project tree */ #ifndef FS_H__ #define FS_H__ #include #include #include #include #include #ifndef STD_FILESYSTEM #if __cplusplus >= 201703L // C++ 17 or higher #include #if (!defined(MAC_OSX) && !TARGET_OS_SIMULATOR && defined(__cpp_lib_filesystem)) // supports std::filesystem # define STD_FILESYSTEM 1 #endif #endif #endif namespace i2p { namespace fs { extern std::string dirSep; /** * @brief Class to work with NetDb & Router profiles * * Usage: * * const char alphabet[8] = {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h'}; * auto h = HashedStorage("name", "y", "z-", ".txt"); * h.SetPlace("/tmp/hs-test"); * h.GetName() -> gives "name" * h.GetRoot() -> gives "/tmp/hs-test/name" * h.Init(alphabet, 8); <- creates needed dirs, 8 is size of alphabet * h.Path("abcd"); <- returns /tmp/hs-test/name/ya/z-abcd.txt * h.Remove("abcd"); <- removes /tmp/hs-test/name/ya/z-abcd.txt, if it exists * std::vector files; * h.Traverse(files); <- finds all files in storage and saves in given vector */ class HashedStorage { protected: std::string root; /**< path to storage with it's name included */ std::string name; /**< name of the storage */ std::string prefix1; /**< hashed directory prefix */ std::string prefix2; /**< prefix of file in storage */ std::string suffix; /**< suffix of file in storage (extension) */ public: typedef std::function FilenameVisitor; HashedStorage(const char *n, const char *p1, const char *p2, const char *s): name(n), prefix1(p1), prefix2(p2), suffix(s) {}; /** create subdirs in storage */ bool Init(const char* chars, size_t cnt); const std::string & GetRoot() const { return root; } const std::string & GetName() const { return name; } /** set directory where to place storage directory */ void SetPlace(const std::string & path); /** path to file with given ident */ std::string Path(const std::string & ident) const; /** remove file by ident */ void Remove(const std::string & ident); /** find all files in storage and store list in provided vector */ void Traverse(std::vector & files); /** visit every file in this storage with a visitor */ void Iterate(FilenameVisitor v); }; /** @brief Returns current application name, default 'i2pd' */ const std::string & GetAppName (); /** @brief Set application name, affects autodetection of datadir */ void SetAppName (const std::string& name); /** @brief Returns datadir path */ const std::string & GetDataDir(); /** @brief Returns certsdir path */ const std::string & GetCertsDir(); /** @brief Returns datadir path in UTF-8 encoding */ const std::string GetUTF8DataDir(); /** * @brief Set datadir either from cmdline option or using autodetection * @param cmdline_param Value of cmdline parameter --datadir= * @param isService Value of cmdline parameter --service * * Examples of autodetected paths: * * Windows < Vista: C:\Documents and Settings\Username\Application Data\i2pd\ * Windows >= Vista: C:\Users\Username\AppData\Roaming\i2pd\ * Mac: /Library/Application Support/i2pd/ or ~/Library/Application Support/i2pd/ * Unix: /var/lib/i2pd/ (system=1) >> ~/.i2pd/ or /tmp/i2pd/ */ void DetectDataDir(const std::string & cmdline_datadir, bool isService = false); /** * @brief Set certsdir either from cmdline option or using autodetection * @param cmdline_param Value of cmdline parameter --certsdir= * * Examples of autodetected paths: * * Windows < Vista: C:\Documents and Settings\Username\Application Data\i2pd\certificates * Windows >= Vista: C:\Users\Username\AppData\Roaming\i2pd\certificates * Mac: /Library/Application Support/i2pd/ or ~/Library/Application Support/i2pd/certificates * Unix: /var/lib/i2pd/certificates (system=1) >> ~/.i2pd/ or /tmp/i2pd/certificates */ void SetCertsDir(const std::string & cmdline_certsdir); /** * @brief Create subdirectories inside datadir */ bool Init(); /** * @brief Get list of files in directory * @param path Path to directory * @param files Vector to store found files * @return true on success and false if directory not exists */ bool ReadDir(const std::string & path, std::vector & files); /** * @brief Remove file with given path * @param path Absolute path to file * @return true on success, false if file not exists, throws exception on error */ bool Remove(const std::string & path); /** * @brief Check existence of file * @param path Absolute path to file * @return true if file exists, false otherwise */ bool Exists(const std::string & path); uint32_t GetLastUpdateTime (const std::string & path); // seconds since epoch bool CreateDirectory (const std::string& path); template void _ExpandPath(std::stringstream & path, T c) { path << i2p::fs::dirSep << c; } template void _ExpandPath(std::stringstream & path, T c, Other ... other) { _ExpandPath(path, c); _ExpandPath(path, other ...); } /** * @brief Get path relative to datadir * * Examples (with datadir = "/tmp/i2pd"): * * i2p::fs::Path("test") -> '/tmp/i2pd/test' * i2p::fs::Path("test", "file.txt") -> '/tmp/i2pd/test/file.txt' */ template std::string DataDirPath(Other ... components) { std::stringstream s(""); s << i2p::fs::GetDataDir(); _ExpandPath(s, components ...); return s.str(); } template std::string StorageRootPath (const Storage& storage, Filename... filenames) { std::stringstream s(""); s << storage.GetRoot (); _ExpandPath(s, filenames...); return s.str(); } } // fs } // i2p #endif // /* FS_H__ */