// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #ifndef ELECTRON_SHELL_COMMON_ASAR_ARCHIVE_H_ #define ELECTRON_SHELL_COMMON_ASAR_ARCHIVE_H_ #include #include #include #include #include #include #include "base/files/file.h" #include "base/files/file_path.h" #include "base/synchronization/lock.h" #include "base/values.h" namespace asar { class ScopedTemporaryFile; enum class HashAlgorithm { kSHA256, kNone, }; struct IntegrityPayload { IntegrityPayload(); ~IntegrityPayload(); IntegrityPayload(const IntegrityPayload& other); HashAlgorithm algorithm = HashAlgorithm::kNone; std::string hash; uint32_t block_size = 0U; std::vector blocks; }; // This class represents an asar package, and provides methods to read // information from it. It is thread-safe after |Init| has been called. class Archive { public: struct FileInfo { FileInfo(); ~FileInfo(); bool unpacked = false; bool executable = false; uint32_t size = 0U; uint64_t offset = 0U; std::optional integrity; }; enum class FileType { kFile = UV_DIRENT_FILE, kDirectory = UV_DIRENT_DIR, kLink = UV_DIRENT_LINK, }; struct Stats : public FileInfo { FileType type = FileType::kFile; }; explicit Archive(const base::FilePath& path); virtual ~Archive(); // disable copy Archive(const Archive&) = delete; Archive& operator=(const Archive&) = delete; // Read and parse the header. bool Init(); std::optional HeaderIntegrity() const; std::optional RelativePath() const; // Get the info of a file. bool GetFileInfo(const base::FilePath& path, FileInfo* info) const; // Fs.stat(path). bool Stat(const base::FilePath& path, Stats* stats) const; // Fs.readdir(path). bool Readdir(const base::FilePath& path, std::vector* files) const; // Fs.realpath(path). bool Realpath(const base::FilePath& path, base::FilePath* realpath) const; // Copy the file into a temporary file, and return the new path. // For unpacked file, this method will return its real path. bool CopyFileOut(const base::FilePath& path, base::FilePath* out); // Returns the file's fd. // Using this fd will not validate the integrity of any files // you read out of the ASAR manually. Callers are responsible // for integrity validation after this fd is handed over. int GetUnsafeFD() const; base::FilePath path() const { return path_; } private: bool initialized_ = false; bool header_validated_ = false; const base::FilePath path_; base::File file_{base::File::FILE_OK}; int fd_ = -1; uint32_t header_size_ = 0; std::optional header_; // Cached external temporary files. base::Lock external_files_lock_; std::unordered_map> external_files_; }; } // namespace asar #endif // ELECTRON_SHELL_COMMON_ASAR_ARCHIVE_H_