1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "format/Archive.h"
18 
19 #include <cstdio>
20 #include <memory>
21 #include <string>
22 #include <vector>
23 
24 #include "android-base/errors.h"
25 #include "android-base/macros.h"
26 #include "android-base/utf8.h"
27 #include "androidfw/StringPiece.h"
28 #include "util/Files.h"
29 #include "util/Util.h"
30 #include "ziparchive/zip_writer.h"
31 
32 using ::android::StringPiece;
33 using ::android::base::SystemErrorCodeToString;
34 
35 namespace aapt {
36 
37 namespace {
38 
39 class DirectoryWriter : public IArchiveWriter {
40  public:
41   DirectoryWriter() = default;
42 
Open(StringPiece out_dir)43   bool Open(StringPiece out_dir) {
44     dir_ = std::string(out_dir);
45     file::FileType type = file::GetFileType(dir_);
46     if (type == file::FileType::kNonExistant) {
47       error_ = "directory does not exist";
48       return false;
49     } else if (type != file::FileType::kDirectory) {
50       error_ = "not a directory";
51       return false;
52     }
53     return true;
54   }
55 
StartEntry(StringPiece path,uint32_t flags)56   bool StartEntry(StringPiece path, uint32_t flags) override {
57     if (file_) {
58       return false;
59     }
60 
61     std::string full_path = dir_;
62     file::AppendPath(&full_path, path);
63     file::mkdirs(std::string(file::GetStem(full_path)));
64 
65     file_ = {::android::base::utf8::fopen(full_path.c_str(), "wb"), fclose};
66     if (!file_) {
67       error_ = SystemErrorCodeToString(errno);
68       return false;
69     }
70     return true;
71   }
72 
Write(const void * data,int len)73   bool Write(const void* data, int len) override {
74     if (!file_) {
75       return false;
76     }
77 
78     if (fwrite(data, 1, len, file_.get()) != static_cast<size_t>(len)) {
79       error_ = SystemErrorCodeToString(errno);
80       file_.reset(nullptr);
81       return false;
82     }
83     return true;
84   }
85 
FinishEntry()86   bool FinishEntry() override {
87     if (!file_) {
88       return false;
89     }
90     file_.reset(nullptr);
91     return true;
92   }
93 
WriteFile(StringPiece path,uint32_t flags,android::InputStream * in)94   bool WriteFile(StringPiece path, uint32_t flags, android::InputStream* in) override {
95     if (!StartEntry(path, flags)) {
96       return false;
97     }
98 
99     const void* data = nullptr;
100     size_t len = 0;
101     while (in->Next(&data, &len)) {
102       if (!Write(data, static_cast<int>(len))) {
103         return false;
104       }
105     }
106 
107     if (in->HadError()) {
108       error_ = in->GetError();
109       return false;
110     }
111 
112     return FinishEntry();
113   }
114 
HadError() const115   bool HadError() const override {
116     return !error_.empty();
117   }
118 
GetError() const119   std::string GetError() const override {
120     return error_;
121   }
122 
123  private:
124   DISALLOW_COPY_AND_ASSIGN(DirectoryWriter);
125 
126   std::string dir_;
127   std::unique_ptr<FILE, decltype(fclose)*> file_ = {nullptr, fclose};
128   std::string error_;
129 };
130 
131 class ZipFileWriter : public IArchiveWriter {
132  public:
133   ZipFileWriter() = default;
134 
Open(StringPiece path)135   bool Open(StringPiece path) {
136     file_ = {::android::base::utf8::fopen(path.data(), "w+b"), fclose};
137     if (!file_) {
138       error_ = SystemErrorCodeToString(errno);
139       return false;
140     }
141     writer_ = util::make_unique<ZipWriter>(file_.get());
142     return true;
143   }
144 
StartEntry(StringPiece path,uint32_t flags)145   bool StartEntry(StringPiece path, uint32_t flags) override {
146     if (!writer_) {
147       return false;
148     }
149 
150     size_t zip_flags = 0;
151     if (flags & ArchiveEntry::kCompress) {
152       zip_flags |= ZipWriter::kCompress;
153     }
154 
155     if (flags & ArchiveEntry::kAlign) {
156       zip_flags |= ZipWriter::kAlign32;
157     }
158 
159     int32_t result = writer_->StartEntry(path.data(), zip_flags);
160     if (result != 0) {
161       error_ = ZipWriter::ErrorCodeString(result);
162       return false;
163     }
164     return true;
165   }
166 
Write(const void * data,int len)167   bool Write(const void* data, int len) override {
168     int32_t result = writer_->WriteBytes(data, len);
169     if (result != 0) {
170       error_ = ZipWriter::ErrorCodeString(result);
171       return false;
172     }
173     return true;
174   }
175 
FinishEntry()176   bool FinishEntry() override {
177     int32_t result = writer_->FinishEntry();
178     if (result != 0) {
179       error_ = ZipWriter::ErrorCodeString(result);
180       return false;
181     }
182     return true;
183   }
184 
WriteFile(StringPiece path,uint32_t flags,android::InputStream * in)185   bool WriteFile(StringPiece path, uint32_t flags, android::InputStream* in) override {
186     while (true) {
187       if (!StartEntry(path, flags)) {
188         return false;
189       }
190 
191       const void* data = nullptr;
192       size_t len = 0;
193       while (in->Next(&data, &len)) {
194         if (!Write(data, static_cast<int>(len))) {
195           return false;
196         }
197       }
198 
199       if (in->HadError()) {
200         error_ = in->GetError();
201         return false;
202       }
203 
204       if (!FinishEntry()) {
205         return false;
206       }
207 
208       // Check to see if the file was compressed enough. This is preserving behavior of AAPT.
209       if ((flags & ArchiveEntry::kCompress) != 0 && in->CanRewind()) {
210         ZipWriter::FileEntry last_entry;
211         int32_t result = writer_->GetLastEntry(&last_entry);
212         CHECK(result == 0);
213         if (last_entry.compressed_size + (last_entry.compressed_size / 10) >
214             last_entry.uncompressed_size) {
215           // The file was not compressed enough, rewind and store it uncompressed.
216           if (!in->Rewind()) {
217             // Well we tried, may as well keep what we had.
218             return true;
219           }
220 
221           int32_t result = writer_->DiscardLastEntry();
222           if (result != 0) {
223             error_ = ZipWriter::ErrorCodeString(result);
224             return false;
225           }
226           flags &= ~ArchiveEntry::kCompress;
227 
228           continue;
229         }
230       }
231       return true;
232     }
233   }
234 
HadError() const235   bool HadError() const override {
236     return !error_.empty();
237   }
238 
GetError() const239   std::string GetError() const override {
240     return error_;
241   }
242 
~ZipFileWriter()243   virtual ~ZipFileWriter() {
244     if (writer_) {
245       writer_->Finish();
246     }
247   }
248 
249  private:
250   DISALLOW_COPY_AND_ASSIGN(ZipFileWriter);
251 
252   std::unique_ptr<FILE, decltype(fclose)*> file_ = {nullptr, fclose};
253   std::unique_ptr<ZipWriter> writer_;
254   std::string error_;
255 };
256 
257 }  // namespace
258 
CreateDirectoryArchiveWriter(android::IDiagnostics * diag,StringPiece path)259 std::unique_ptr<IArchiveWriter> CreateDirectoryArchiveWriter(android::IDiagnostics* diag,
260                                                              StringPiece path) {
261   std::unique_ptr<DirectoryWriter> writer = util::make_unique<DirectoryWriter>();
262   if (!writer->Open(path)) {
263     diag->Error(android::DiagMessage(path) << writer->GetError());
264     return {};
265   }
266   return std::move(writer);
267 }
268 
CreateZipFileArchiveWriter(android::IDiagnostics * diag,StringPiece path)269 std::unique_ptr<IArchiveWriter> CreateZipFileArchiveWriter(android::IDiagnostics* diag,
270                                                            StringPiece path) {
271   std::unique_ptr<ZipFileWriter> writer = util::make_unique<ZipFileWriter>();
272   if (!writer->Open(path)) {
273     diag->Error(android::DiagMessage(path) << writer->GetError());
274     return {};
275   }
276   return std::move(writer);
277 }
278 
279 }  // namespace aapt
280