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 "daemon_app.h"
16
17 namespace Hdc {
HdcDaemonApp(HTaskInfo hTaskInfo)18 HdcDaemonApp::HdcDaemonApp(HTaskInfo hTaskInfo)
19 : HdcTransferBase(hTaskInfo)
20 {
21 commandBegin = CMD_APP_BEGIN;
22 commandData = CMD_APP_DATA;
23 funcAppModFinish = nullptr;
24 }
25
~HdcDaemonApp()26 HdcDaemonApp::~HdcDaemonApp()
27 {
28 WRITE_LOG(LOG_DEBUG, "~HdcDaemonApp");
29 }
30
ReadyForRelease()31 bool HdcDaemonApp::ReadyForRelease()
32 {
33 if (!HdcTaskBase::ReadyForRelease()) {
34 return false;
35 }
36 if (!asyncCommand.ReadyForRelease()) {
37 return false;
38 }
39 WRITE_LOG(LOG_DEBUG, "HdcDaemonApp ready for release");
40 return true;
41 }
42
CommandDispatch(const uint16_t command,uint8_t * payload,const int payloadSize)43 bool HdcDaemonApp::CommandDispatch(const uint16_t command, uint8_t *payload, const int payloadSize)
44 {
45 if (!HdcTransferBase::CommandDispatch(command, payload, payloadSize)) {
46 return false;
47 }
48 bool ret = true;
49 switch (command) {
50 case CMD_APP_CHECK: {
51 string dstPath = "/data/local/tmp/";
52 string bufString(reinterpret_cast<char *>(payload), payloadSize);
53 SerialStruct::ParseFromString(ctxNow.transferConfig, bufString);
54 // update transferconfig to main context
55 ctxNow.master = false;
56 ctxNow.fsOpenReq.data = &ctxNow;
57 #ifdef HDC_PCDEBUG
58 char tmpPath[256] = "";
59 size_t size = 256;
60 uv_os_tmpdir(tmpPath, &size);
61 dstPath = tmpPath;
62 dstPath += Base::GetPathSep();
63 #endif
64 dstPath += ctxNow.transferConfig.optionalName;
65 ctxNow.localPath = dstPath;
66 ctxNow.transferBegin = Base::GetRuntimeMSec();
67 ctxNow.fileSize = ctxNow.transferConfig.fileSize;
68 ++refCount;
69 uv_fs_open(loopTask, &ctxNow.fsOpenReq, ctxNow.localPath.c_str(),
70 UV_FS_O_TRUNC | UV_FS_O_CREAT | UV_FS_O_WRONLY, S_IWUSR | S_IRUSR | S_IRGRP | S_IROTH,
71 OnFileOpen);
72 break;
73 }
74 case CMD_APP_UNINSTALL: {
75 // This maybe has a command implanting risk, since it is a controllable device, it can be ignored
76 string bufString(reinterpret_cast<char *>(payload), payloadSize);
77 string options = "";
78 string packages = "";
79 vector<string> segments;
80 Base::SplitString(bufString, " ", segments);
81 for (auto seg: segments) {
82 if (seg[0] == '-') {
83 options += " " + seg;
84 } else {
85 packages += " " + seg;
86 }
87 }
88 PackageShell(false, options.c_str(), packages);
89 break;
90 }
91 default:
92 break;
93 }
94 return ret;
95 };
96
AsyncInstallFinish(bool finish,int64_t exitStatus,const string result)97 bool HdcDaemonApp::AsyncInstallFinish(bool finish, int64_t exitStatus, const string result)
98 {
99 if (mode == APPMOD_INSTALL) {
100 unlink(ctxNow.localPath.c_str());
101 }
102 asyncCommand.DoRelease();
103 string echo = result;
104 echo = Base::ReplaceAll(echo, "\n", " ");
105 vector<uint8_t> vecBuf;
106 vecBuf.push_back(mode);
107 vecBuf.push_back(exitStatus == 0);
108 vecBuf.insert(vecBuf.end(), (uint8_t *)echo.c_str(), (uint8_t *)echo.c_str() + echo.size());
109 SendToAnother(CMD_APP_FINISH, vecBuf.data(), vecBuf.size());
110 --refCount;
111 #ifdef UNIT_TEST
112 Base::WriteBinFile((UT_TMP_PATH + "/appinstall.result").c_str(), (uint8_t *)MESSAGE_SUCCESS.c_str(),
113 MESSAGE_SUCCESS.size(), true);
114 #endif
115 return true;
116 }
117
PackageShell(bool installOrUninstall,const char * options,const string package)118 void HdcDaemonApp::PackageShell(bool installOrUninstall, const char *options, const string package)
119 {
120 ++refCount;
121 // asynccmd Other processes, no RunningProtect protection
122 chmod(package.c_str(), 0644); // 0644 : permission
123 string doBuf;
124 string opts = string(options);
125 if (installOrUninstall) { // either -p or -s is always required in install
126 if (opts.find("p") == string::npos && opts.find("s") == string::npos) {
127 // basic mode: blank options or both "-s" / "-p" are omitted
128 // eg. hdc install x.hap --> bm install -p x.hap
129 // eg. hdc install -r x.hap --> bm install -r -p x.hap
130 doBuf = Base::StringFormat("bm install %s -p %s", options, package.c_str());
131 } else {
132 // advansed mode for -p/-r/-s and some other options in the future
133 doBuf = Base::StringFormat("bm install %s %s", options, package.c_str());
134 }
135 } else { // -n is always required in uninstall
136 if (opts.find("n") == string::npos) {
137 // basic mode: blank options or "-n" is omitted
138 // eg. hdc uninstall com.xx.xx --> bm uninstall -n com.xx.xx
139 // eg. hdc uninstall -s com.xx.xx --> bm uninstall -s -n com.xx.xx
140 doBuf = Base::StringFormat("bm uninstall %s -n %s", options, package.c_str());
141 } else {
142 // advansed mode for -s/-n and some other options in the future
143 doBuf = Base::StringFormat("bm uninstall %s %s", options, package.c_str());
144 }
145 }
146
147 funcAppModFinish = std::bind(&HdcDaemonApp::AsyncInstallFinish, this, std::placeholders::_1, std::placeholders::_2,
148 std::placeholders::_3);
149 if (installOrUninstall) {
150 mode = APPMOD_INSTALL;
151 } else {
152 mode = APPMOD_UNINSTALL;
153 }
154 asyncCommand.Initial(loopTask, funcAppModFinish, AsyncCmd::OPTION_COMMAND_ONETIME);
155 asyncCommand.ExecuteCommand(doBuf);
156 }
157
Sideload(const char * pathOTA)158 void HdcDaemonApp::Sideload(const char *pathOTA)
159 {
160 mode = APPMOD_SIDELOAD;
161 LogMsg(MSG_OK, "[placeholders] sideload %s", pathOTA);
162 TaskFinish();
163 unlink(pathOTA);
164 }
165
WhenTransferFinish(CtxFile * context)166 void HdcDaemonApp::WhenTransferFinish(CtxFile *context)
167 {
168 if (context->lastErrno > 0) {
169 constexpr int bufSize = 1024;
170 char buf[bufSize] = { 0 };
171 uv_strerror_r(static_cast<int>(-context->lastErrno), buf, bufSize);
172 WRITE_LOG(LOG_DEBUG, "HdcDaemonApp WhenTransferFinish with errno:%d", context->lastErrno);
173 LogMsg(MSG_FAIL, "Transfer App at:%lld/%lld(Bytes), Reason: %s",
174 context->indexIO, context->fileSize, buf);
175 return;
176 }
177 if (ctxNow.transferConfig.functionName == CMDSTR_APP_SIDELOAD) {
178 Sideload(context->localPath.c_str());
179 } else if (ctxNow.transferConfig.functionName == CMDSTR_APP_INSTALL) {
180 PackageShell(true, context->transferConfig.options.c_str(), context->localPath.c_str());
181 } else {
182 }
183 };
184 }
185