1 /*
2 * Copyright (c) 2021 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15 #include "pkg_pkgfile.h"
16 #include <ctime>
17 #include <limits>
18 #include <memory>
19 #include "dump.h"
20 #include "pkg_gzipfile.h"
21 #include "pkg_lz4file.h"
22 #include "pkg_stream.h"
23 #include "pkg_upgradefile.h"
24 #include "pkg_utils.h"
25 #include "pkg_zipfile.h"
26
27 namespace Hpackage {
~PkgFileImpl()28 PkgFileImpl::~PkgFileImpl()
29 {
30 auto iter = pkgEntryMapId_.begin();
31 while (iter != pkgEntryMapId_.end()) {
32 auto entry = iter->second;
33 delete entry;
34 iter = pkgEntryMapId_.erase(iter);
35 }
36 pkgEntryMapId_.clear();
37 pkgEntryMapFileName_.clear();
38 PkgManager::StreamPtr tmpStream = pkgStream_;
39 pkgManager_->ClosePkgStream(tmpStream);
40 }
41
AddPkgEntry(const std::string & fileName)42 PkgEntryPtr PkgFileImpl::AddPkgEntry(const std::string &fileName)
43 {
44 uint32_t nodeId = ++nodeId_;
45 PkgEntryPtr entry = nullptr;
46 switch (type_) {
47 case PKG_TYPE_UPGRADE:
48 entry = new UpgradeFileEntry(this, nodeId);
49 break;
50 case PKG_TYPE_ZIP:
51 entry = new ZipFileEntry(this, nodeId);
52 break;
53 case PKG_TYPE_LZ4: {
54 entry = new Lz4FileEntry(this, nodeId);
55 break;
56 }
57 case PKG_TYPE_GZIP: {
58 entry = new GZipFileEntry(this, nodeId);
59 break;
60 }
61 default:
62 return nullptr;
63 }
64 pkgEntryMapId_.insert(std::pair<uint32_t, PkgEntryPtr>(nodeId, entry));
65 pkgEntryMapFileName_.insert(std::pair<std::string, PkgEntryPtr>(fileName, entry));
66 return entry;
67 }
68
ExtractFile(const PkgEntryPtr node,PkgStreamPtr output)69 int32_t PkgFileImpl::ExtractFile(const PkgEntryPtr node, PkgStreamPtr output)
70 {
71 Updater::UPDATER_INIT_RECORD;
72 PKG_LOGI("ExtractFile %s", output->GetFileName().c_str());
73 if (!CheckState({PKG_FILE_STATE_WORKING}, PKG_FILE_STATE_WORKING)) {
74 PKG_LOGE("error state curr %d ", state_);
75 UPDATER_LAST_WORD(PKG_INVALID_STATE);
76 return PKG_INVALID_STATE;
77 }
78 auto entry = static_cast<PkgEntryPtr>(node);
79 if (entry == nullptr) {
80 PKG_LOGE("error get entry %s", pkgStream_->GetFileName().c_str());
81 UPDATER_LAST_WORD(PKG_INVALID_PARAM);
82 return PKG_INVALID_PARAM;
83 }
84 return entry->Unpack(output);
85 }
86
FindPkgEntry(const std::string & fileName)87 PkgEntryPtr PkgFileImpl::FindPkgEntry(const std::string &fileName)
88 {
89 if (!CheckState({PKG_FILE_STATE_WORKING}, PKG_FILE_STATE_WORKING)) {
90 PKG_LOGE("error state curr %d ", state_);
91 return nullptr;
92 }
93 std::multimap<std::string, PkgEntryPtr>::iterator iter = pkgEntryMapFileName_.find(fileName);
94 if (iter != pkgEntryMapFileName_.end()) {
95 return (*iter).second;
96 }
97 return nullptr;
98 }
99
CheckState(std::vector<uint32_t> states,uint32_t state)100 bool PkgFileImpl::CheckState(std::vector<uint32_t> states, uint32_t state)
101 {
102 bool ret = false;
103 for (auto s : states) {
104 if (state_ == s) {
105 state_ = state;
106 ret = true;
107 break;
108 }
109 }
110 return ret;
111 }
112
ConvertBufferToString(std::string & fileName,const PkgBuffer & buffer)113 int32_t PkgFileImpl::ConvertBufferToString(std::string &fileName, const PkgBuffer &buffer)
114 {
115 for (uint32_t i = 0; i < buffer.length; ++i) {
116 if (buffer.buffer[i] < 32 || buffer.buffer[i] >= 127) { // 32,127 : should be printable character
117 break;
118 }
119 fileName.push_back(buffer.buffer[i]);
120 }
121 return PKG_SUCCESS;
122 }
123
ConvertStringToBuffer(const std::string & fileName,const PkgBuffer & buffer,size_t & realLen)124 int32_t PkgFileImpl::ConvertStringToBuffer(const std::string &fileName, const PkgBuffer &buffer, size_t &realLen)
125 {
126 if (buffer.length < fileName.size()) {
127 PKG_LOGE("Invalid buffer");
128 return PKG_INVALID_PARAM;
129 }
130 for (uint32_t i = 0; i < fileName.size(); ++i) {
131 buffer.buffer[i] = static_cast<uint8_t>(fileName[i]);
132 (realLen)++;
133 }
134 return PKG_SUCCESS;
135 }
136
Init(PkgManager::FileInfoPtr localFileInfo,const PkgManager::FileInfoPtr fileInfo,PkgStreamPtr inStream)137 int32_t PkgEntry::Init(PkgManager::FileInfoPtr localFileInfo, const PkgManager::FileInfoPtr fileInfo,
138 PkgStreamPtr inStream)
139 {
140 if (localFileInfo == nullptr || fileInfo == nullptr || inStream == nullptr) {
141 PKG_LOGE("Failed to check input param");
142 return PKG_INVALID_PARAM;
143 }
144
145 fileName_.assign(inStream->GetFileName());
146 localFileInfo->identity.assign(fileInfo->identity);
147 localFileInfo->flags = fileInfo->flags;
148 localFileInfo->digestMethod = fileInfo->digestMethod;
149 localFileInfo->packMethod = fileInfo->packMethod;
150 localFileInfo->modifiedTime = fileInfo->modifiedTime;
151 localFileInfo->packedSize = fileInfo->packedSize;
152 localFileInfo->unpackedSize = fileInfo->unpackedSize;
153
154 // 填充file信息,默认值使用原始文件长度
155 if (localFileInfo->unpackedSize == 0) {
156 localFileInfo->unpackedSize = inStream->GetFileLength();
157 }
158 if (localFileInfo->packedSize == 0) {
159 localFileInfo->packedSize = inStream->GetFileLength();
160 }
161 if (localFileInfo->unpackedSize == 0) {
162 PKG_LOGE("Failed to check unpackedSize = 0");
163 return PKG_INVALID_PARAM;
164 }
165 if (localFileInfo->modifiedTime == 0) {
166 time(&localFileInfo->modifiedTime);
167 }
168 return PKG_SUCCESS;
169 }
170
AddSignData(uint8_t digestMethod,size_t currOffset,size_t & signOffset)171 void PkgFileImpl::AddSignData(uint8_t digestMethod, size_t currOffset, size_t &signOffset)
172 {
173 signOffset = currOffset;
174 if (digestMethod == PKG_DIGEST_TYPE_NONE) {
175 return;
176 }
177 std::vector<uint8_t> buffer(SIGN_SHA256_LEN + SIGN_SHA384_LEN, 0);
178 int32_t ret = pkgStream_->Write(buffer, buffer.size(), currOffset);
179 if (ret != PKG_SUCCESS) {
180 PKG_LOGE("Fail write sign for %s", pkgStream_->GetFileName().c_str());
181 return;
182 }
183 pkgStream_->Flush(currOffset + buffer.size());
184 PKG_LOGI("SavePackage success file length: %zu signOffset %zu", pkgStream_->GetFileLength(), signOffset);
185 }
186 } // namespace Hpackage
187