François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 1 | // Copyright 2020 The Chromium OS Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| 4 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 5 | #ifndef CROS_DISKS_RAR_MOUNTER_H_ |
| 6 | #define CROS_DISKS_RAR_MOUNTER_H_ |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 7 | |
| 8 | #include <memory> |
| 9 | #include <string> |
| 10 | #include <vector> |
| 11 | |
| 12 | #include <base/strings/string_piece.h> |
| 13 | #include <gtest/gtest_prod.h> |
| 14 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 15 | #include "cros-disks/archive_mounter.h" |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 16 | #include "cros-disks/fuse_mounter.h" |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 17 | |
| 18 | namespace cros_disks { |
| 19 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 20 | // An ArchiveMounter with handling of RAR multipart archives. |
| 21 | class RarMounter : public ArchiveMounter { |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 22 | public: |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 23 | RarMounter(const Platform* platform, |
| 24 | brillo::ProcessReaper* process_reaper, |
Sergei Datsenko | 1d2cbf8 | 2020-12-08 21:54:42 +1100 | [diff] [blame] | 25 | Metrics* metrics, |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 26 | std::unique_ptr<SandboxedProcessFactory> sandbox_factory); |
| 27 | RarMounter(const RarMounter&) = delete; |
| 28 | RarMounter& operator=(const RarMounter&) = delete; |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 29 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 30 | ~RarMounter() override; |
| 31 | |
| 32 | protected: |
| 33 | MountErrorType FormatInvocationCommand( |
| 34 | const base::FilePath& archive, |
| 35 | std::vector<std::string> params, |
| 36 | SandboxedProcess* sandbox) const override; |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 37 | |
| 38 | private: |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 39 | // Increments a sequence of digits or letters [begin, end). Returns true if |
| 40 | // success, and false in case of overflow. |
| 41 | static bool Increment(std::string::iterator begin, std::string::iterator end); |
| 42 | |
| 43 | // A semi-open index range [begin, end). |
| 44 | struct IndexRange { |
| 45 | size_t begin, end; |
| 46 | |
| 47 | bool empty() const { return begin == end; } |
| 48 | size_t size() const { return end - begin; } |
| 49 | |
| 50 | // Friend operators for testing, logging and debugging. |
| 51 | friend std::ostream& operator<<(std::ostream& out, const IndexRange& r) { |
| 52 | return out << "{ begin: " << r.begin << ", end: " << r.end << " }"; |
| 53 | } |
| 54 | |
| 55 | friend bool operator==(const IndexRange& a, const IndexRange& b) { |
| 56 | return a.begin == b.begin && a.end == b.end; |
| 57 | } |
| 58 | }; |
| 59 | |
| 60 | // Checks if the given path ends with ".partNNNN.rar", which is the new |
| 61 | // naming pattern for multipart archives. Returns the range of characters |
| 62 | // forming the numeric part NNNN if path matches the pattern, or an empty |
| 63 | // range otherwise. |
| 64 | static IndexRange ParseDigits(base::StringPiece path); |
| 65 | |
| 66 | // Adds bind paths using old naming scheme. |
Sergei Datsenko | 1d2cbf8 | 2020-12-08 21:54:42 +1100 | [diff] [blame] | 67 | void AddPathsWithOldNamingScheme(std::vector<std::string>* bind_paths, |
François Degros | a28315e | 2020-07-13 00:24:48 +1000 | [diff] [blame] | 68 | base::StringPiece original_path) const; |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 69 | |
| 70 | // Adds bind paths using new naming scheme. |
Sergei Datsenko | 1d2cbf8 | 2020-12-08 21:54:42 +1100 | [diff] [blame] | 71 | void AddPathsWithNewNamingScheme(std::vector<std::string>* bind_paths, |
François Degros | a28315e | 2020-07-13 00:24:48 +1000 | [diff] [blame] | 72 | base::StringPiece original_path, |
| 73 | const IndexRange& digits) const; |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 74 | |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 75 | // Prepares the bind paths for the given RAR file path. |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 76 | // |
| 77 | // If the given path is considered to be part of a multipart archive, this |
| 78 | // function tries to find all the related files. |
| 79 | // |
| 80 | // Two different naming schemes are supported. |
| 81 | // |
| 82 | // The old naming scheme is: |
| 83 | // |
| 84 | // basename.rar |
| 85 | // basename.r00 |
| 86 | // basename.r01 |
| 87 | // ... |
| 88 | // basename.r99 |
| 89 | // basename.s00 |
| 90 | // basename.s01 |
| 91 | // ... |
| 92 | // |
| 93 | // |
| 94 | // The new naming scheme is: |
| 95 | // |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 96 | // basename1.rar |
| 97 | // basename2.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 98 | // ... |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 99 | // basename9.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 100 | // |
| 101 | // or |
| 102 | // |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 103 | // basename01.rar |
| 104 | // basename02.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 105 | // ... |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 106 | // basename99.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 107 | // |
| 108 | // or |
| 109 | // |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 110 | // basename001.rar |
| 111 | // basename002.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 112 | // ... |
François Degros | 97f6a0d | 2020-08-24 15:58:30 +1000 | [diff] [blame] | 113 | // basename999.rar |
François Degros | 02e8149 | 2020-02-05 16:37:19 +1100 | [diff] [blame] | 114 | // etc. |
Sergei Datsenko | 1d2cbf8 | 2020-12-08 21:54:42 +1100 | [diff] [blame] | 115 | std::vector<std::string> GetBindPaths(base::StringPiece original_path) const; |
| 116 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 117 | FRIEND_TEST(RarMounterTest, Increment); |
| 118 | FRIEND_TEST(RarMounterTest, ParseDigits); |
| 119 | FRIEND_TEST(RarMounterTest, GetBindPathsWithOldNamingScheme); |
| 120 | FRIEND_TEST(RarMounterTest, GetBindPathsWithNewNamingScheme); |
| 121 | FRIEND_TEST(RarMounterTest, GetBindPathsStopsOnOverflow); |
François Degros | d6a775f | 2020-02-05 15:05:21 +1100 | [diff] [blame] | 122 | }; |
| 123 | |
| 124 | } // namespace cros_disks |
| 125 | |
Sergei Datsenko | c9904bb | 2020-12-11 12:46:02 +1100 | [diff] [blame] | 126 | #endif // CROS_DISKS_RAR_MOUNTER_H_ |