| // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef THIRD_PARTY_ZLIB_GOOGLE_ZIP_H_ |
| #define THIRD_PARTY_ZLIB_GOOGLE_ZIP_H_ |
| |
| #include <vector> |
| |
| #include "base/callback.h" |
| #include "base/files/file_path.h" |
| #include "base/files/platform_file.h" |
| #include "base/time/time.h" |
| #include "build/build_config.h" |
| |
| namespace base { |
| class File; |
| } |
| |
| namespace zip { |
| |
| class WriterDelegate; |
| |
| // Abstraction for file access operation required by Zip(). |
| // Can be passed to the ZipParams for providing custom access to the files, |
| // for example over IPC. |
| // If none is provided, the files are accessed directly. |
| // All parameters paths are expected to be absolute. |
| class FileAccessor { |
| public: |
| virtual ~FileAccessor() = default; |
| |
| struct DirectoryContentEntry { |
| DirectoryContentEntry(const base::FilePath& path, bool is_directory) |
| : path(path), is_directory(is_directory) {} |
| base::FilePath path; |
| bool is_directory = false; |
| }; |
| |
| // Opens files specified in |paths|. |
| // Directories should be mapped to invalid files. |
| virtual std::vector<base::File> OpenFilesForReading( |
| const std::vector<base::FilePath>& paths) = 0; |
| |
| virtual bool DirectoryExists(const base::FilePath& path) = 0; |
| virtual std::vector<DirectoryContentEntry> ListDirectoryContent( |
| const base::FilePath& dir_path) = 0; |
| virtual base::Time GetLastModifiedTime(const base::FilePath& path) = 0; |
| }; |
| |
| class ZipParams { |
| public: |
| ZipParams(const base::FilePath& src_dir, const base::FilePath& dest_file); |
| #if defined(OS_POSIX) |
| // Does not take ownership of |dest_fd|. |
| ZipParams(const base::FilePath& src_dir, int dest_fd); |
| |
| int dest_fd() const { return dest_fd_; } |
| #endif |
| |
| const base::FilePath& src_dir() const { return src_dir_; } |
| |
| const base::FilePath& dest_file() const { return dest_file_; } |
| |
| // Restricts the files actually zipped to the paths listed in |
| // |src_relative_paths|. They must be relative to the |src_dir| passed in the |
| // constructor and will be used as the file names in the created zip file. All |
| // source paths must be under |src_dir| in the file system hierarchy. |
| void set_files_to_zip(const std::vector<base::FilePath>& src_relative_paths) { |
| src_files_ = src_relative_paths; |
| } |
| const std::vector<base::FilePath>& files_to_zip() const { return src_files_; } |
| |
| using FilterCallback = base::Callback<bool(const base::FilePath&)>; |
| void set_filter_callback(FilterCallback filter_callback) { |
| filter_callback_ = filter_callback; |
| } |
| const FilterCallback& filter_callback() const { return filter_callback_; } |
| |
| void set_include_hidden_files(bool include_hidden_files) { |
| include_hidden_files_ = include_hidden_files; |
| } |
| bool include_hidden_files() const { return include_hidden_files_; } |
| |
| // Sets a custom file accessor for file operations. Default is to directly |
| // access the files (with fopen and the rest). |
| // Useful in cases where running in a sandbox process and file access has to |
| // go through IPC, for example. |
| void set_file_accessor(std::unique_ptr<FileAccessor> file_accessor) { |
| file_accessor_ = std::move(file_accessor); |
| } |
| FileAccessor* file_accessor() const { return file_accessor_.get(); } |
| |
| private: |
| base::FilePath src_dir_; |
| |
| base::FilePath dest_file_; |
| #if defined(OS_POSIX) |
| int dest_fd_ = base::kInvalidPlatformFile; |
| #endif |
| |
| // The relative paths to the files that should be included in the zip file. If |
| // this is empty, all files in |src_dir_| are included. |
| std::vector<base::FilePath> src_files_; |
| |
| // Filter used to exclude files from the ZIP file. Only effective when |
| // |src_files_| is empty. |
| FilterCallback filter_callback_; |
| |
| // Whether hidden files should be included in the ZIP file. Only effective |
| // when |src_files_| is empty. |
| bool include_hidden_files_ = true; |
| |
| // Abstraction around file system access used to read files. An implementation |
| // that accesses files directly is provided by default. |
| std::unique_ptr<FileAccessor> file_accessor_; |
| }; |
| |
| // Zip files specified into a ZIP archives. The source files and ZIP destination |
| // files (as well as other settings) are specified in |params|. |
| bool Zip(const ZipParams& params); |
| |
| // Zip the contents of src_dir into dest_file. src_path must be a directory. |
| // An entry will *not* be created in the zip for the root folder -- children |
| // of src_dir will be at the root level of the created zip. For each file in |
| // src_dir, include it only if the callback |filter_cb| returns true. Otherwise |
| // omit it. |
| typedef base::Callback<bool(const base::FilePath&)> FilterCallback; |
| bool ZipWithFilterCallback(const base::FilePath& src_dir, |
| const base::FilePath& dest_file, |
| const FilterCallback& filter_cb); |
| |
| // Convenience method for callers who don't need to set up the filter callback. |
| // If |include_hidden_files| is true, files starting with "." are included. |
| // Otherwise they are omitted. |
| bool Zip(const base::FilePath& src_dir, const base::FilePath& dest_file, |
| bool include_hidden_files); |
| |
| #if defined(OS_POSIX) |
| // Zips files listed in |src_relative_paths| to destination specified by file |
| // descriptor |dest_fd|, without taking ownership of |dest_fd|. The paths listed |
| // in |src_relative_paths| are relative to the |src_dir| and will be used as the |
| // file names in the created zip file. All source paths must be under |src_dir| |
| // in the file system hierarchy. |
| bool ZipFiles(const base::FilePath& src_dir, |
| const std::vector<base::FilePath>& src_relative_paths, |
| int dest_fd); |
| #endif // defined(OS_POSIX) |
| |
| // Unzip the contents of zip_file into dest_dir. |
| // For each file in zip_file, include it only if the callback |filter_cb| |
| // returns true. Otherwise omit it. |
| // If |log_skipped_files| is true, files skipped during extraction are printed |
| // to debug log. |
| typedef base::Callback<bool(const base::FilePath&)> FilterCallback; |
| bool UnzipWithFilterCallback(const base::FilePath& zip_file, |
| const base::FilePath& dest_dir, |
| const FilterCallback& filter_cb, |
| bool log_skipped_files); |
| |
| // Unzip the contents of zip_file, using the writers provided by writer_factory. |
| // For each file in zip_file, include it only if the callback |filter_cb| |
| // returns true. Otherwise omit it. |
| // If |log_skipped_files| is true, files skipped during extraction are printed |
| // to debug log. |
| typedef base::RepeatingCallback<std::unique_ptr<WriterDelegate>( |
| const base::FilePath&)> |
| WriterFactory; |
| typedef base::RepeatingCallback<bool(const base::FilePath&)> DirectoryCreator; |
| bool UnzipWithFilterAndWriters(const base::PlatformFile& zip_file, |
| const WriterFactory& writer_factory, |
| const DirectoryCreator& directory_creator, |
| const FilterCallback& filter_cb, |
| bool log_skipped_files); |
| |
| // Unzip the contents of zip_file into dest_dir. |
| bool Unzip(const base::FilePath& zip_file, const base::FilePath& dest_dir); |
| |
| } // namespace zip |
| |
| #endif // THIRD_PARTY_ZLIB_GOOGLE_ZIP_H_ |