1 /*
2  * Copyright (C) 2019 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 #pragma once
18 
19 #include <memory>
20 #include <string>
21 #include <vector>
22 
23 #include <adb_unique_fd.h>
24 
25 #include "fastdeploy/proto/ApkEntry.pb.h"
26 
27 class ApkArchiveTester;
28 
29 // Manipulates an APK archive. Process it by mmaping it in order to minimize
30 // I/Os.
31 class ApkArchive {
32   public:
33     friend ApkArchiveTester;
34 
35     // A convenience struct to store the result of search operation when
36     // locating the EoCDr, CDr, and Signature Block.
37     struct Location {
38         off_t offset = 0;
39         off_t size = 0;
40         bool valid = false;
41     };
42 
43     ApkArchive(const std::string& path);
44     ~ApkArchive();
45 
46     com::android::fastdeploy::APKDump ExtractMetadata();
47 
48     // Parses the CDr starting from |input| and returns number of bytes consumed.
49     // Extracts local file header offset, data size and calculates MD5 hash of the record.
50     // 0 indicates invalid CDr.
51     static size_t ParseCentralDirectoryRecord(const char* input, size_t size, std::string* md5Hash,
52                                               int64_t* localFileHeaderOffset, int64_t* dataSize);
53     // Calculates Local File Entry size including header using offset and data size from CDr.
54     // 0 indicates invalid Local File Entry.
55     size_t CalculateLocalFileEntrySize(int64_t localFileHeaderOffset, int64_t dataSize) const;
56 
57   private:
58     std::string ReadMetadata(Location loc) const;
59 
60     // Retrieve the location of the Central Directory Record.
61     Location GetCDLocation();
62 
63     // Retrieve the location of the signature block starting from Central
64     // Directory Record
65     Location GetSignatureLocation(off_t cdRecordOffset);
66 
67     // Find the End of Central Directory Record, starting from the end of the
68     // file.
69     off_t FindEndOfCDRecord() const;
70 
71     // Find Central Directory Record, starting from the end of the file.
72     Location FindCDRecord(const char* cursor);
73 
74     // Checks if the archive can be used.
75     bool ready() const { return fd_ >= 0; }
76 
77     std::string path_;
78     off_t size_;
79     unique_fd fd_;
80 };
81