1 /*
2 * Copyright (c) 2023 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
16 #include "cloud_file_download_napi.h"
17
18 #include <sys/types.h>
19
20 #include "cloud_sync_manager.h"
21 #include "dfs_error.h"
22 #include "utils_log.h"
23 #include "uv.h"
24
25 namespace OHOS::FileManagement::CloudSync {
26 using namespace FileManagement::LibN;
27 using namespace std;
28 const int32_t ARGS_ONE = 1;
29
CloudFileDownloadNapi(napi_env env,napi_value exports)30 CloudFileDownloadNapi::CloudFileDownloadNapi(napi_env env, napi_value exports) : NExporter(env, exports) {}
31
~CloudFileDownloadNapi()32 CloudFileDownloadNapi::~CloudFileDownloadNapi() {}
33
Constructor(napi_env env,napi_callback_info info)34 napi_value CloudFileDownloadNapi::Constructor(napi_env env, napi_callback_info info)
35 {
36 LOGI("CloudFileDownloadNapi::Constructor begin");
37 NFuncArg funcArg(env, info);
38 if (!funcArg.InitArgs(NARG_CNT::ZERO)) {
39 LOGE("Start Number of arguments unmatched");
40 NError(E_PARAMS).ThrowErr(env);
41 return nullptr;
42 }
43
44 LOGI("CloudFileDownloadNapi::Constructor end");
45 return funcArg.GetThisVar();
46 }
47
Start(napi_env env,napi_callback_info info)48 napi_value CloudFileDownloadNapi::Start(napi_env env, napi_callback_info info)
49 {
50 LOGI("Start begin");
51 NFuncArg funcArg(env, info);
52 if (!funcArg.InitArgs(NARG_CNT::ONE, NARG_CNT::TWO)) {
53 LOGE("Start Number of arguments unmatched");
54 NError(E_PARAMS).ThrowErr(env);
55 return nullptr;
56 }
57 auto [succUri, uri, ignore] = NVal(env, funcArg[NARG_POS::FIRST]).ToUTF8String();
58 if (!succUri) {
59 LOGE("Start get uri parameter failed!");
60 NError(E_PARAMS).ThrowErr(env);
61 return nullptr;
62 }
63
64 auto cbExec = [uri = string(uri.get()), env = env]() -> NError {
65 int32_t ret = CloudSyncManager::GetInstance().StartDownloadFile(uri);
66 if (ret != E_OK) {
67 LOGE("Start Download failed! ret = %{public}d", ret);
68 return NError(Convert2JsErrNum(ret));
69 }
70 LOGI("Start Download Success!");
71 return NError(ERRNO_NOERR);
72 };
73
74 auto cbCompl = [](napi_env env, NError err) -> NVal {
75 if (err) {
76 return {env, err.GetNapiErr(env)};
77 }
78 return NVal::CreateUndefined(env);
79 };
80
81 NVal thisVar(env, funcArg.GetThisVar());
82 string procedureName = "cloudFileDownload";
83 if (funcArg.GetArgc() == NARG_CNT::ONE) {
84 return NAsyncWorkPromise(env, thisVar).Schedule(procedureName, cbExec, cbCompl).val_;
85 } else {
86 NVal cb(env, funcArg[NARG_POS::SECOND]);
87 return NAsyncWorkCallback(env, thisVar, cb).Schedule(procedureName, cbExec, cbCompl).val_;
88 }
89 }
90
CloudDownloadCallbackImpl(napi_env env,napi_value fun)91 CloudDownloadCallbackImpl::CloudDownloadCallbackImpl(napi_env env, napi_value fun) : env_(env)
92 {
93 if (fun != nullptr) {
94 napi_create_reference(env_, fun, 1, &cbOnRef_);
95 }
96 }
97
OnComplete(UvChangeMsg * msg)98 void CloudDownloadCallbackImpl::OnComplete(UvChangeMsg *msg)
99 {
100 auto downloadcCallback = msg->CloudDownloadCallback_.lock();
101 if (downloadcCallback == nullptr || downloadcCallback->cbOnRef_ == nullptr) {
102 LOGE("downloadcCallback->cbOnRef_ is nullptr");
103 return;
104 }
105 auto env = downloadcCallback->env_;
106 auto ref = downloadcCallback->cbOnRef_;
107 napi_handle_scope scope = nullptr;
108 napi_open_handle_scope(env, &scope);
109 napi_value jsCallback = nullptr;
110 napi_status status = napi_get_reference_value(env, ref, &jsCallback);
111 if (status != napi_ok) {
112 LOGE("Create reference failed, status: %{public}d", status);
113 return;
114 }
115 NVal obj = NVal::CreateObject(env);
116 obj.AddProp("state", NVal::CreateInt32(env, (int32_t)msg->downloadProgress_.state).val_);
117 obj.AddProp("processed", NVal::CreateInt64(env, (int64_t)msg->downloadProgress_.downloadedSize).val_);
118 obj.AddProp("size", NVal::CreateInt64(env, (int64_t)msg->downloadProgress_.totalSize).val_);
119 obj.AddProp("uri", NVal::CreateUTF8String(env, msg->downloadProgress_.path).val_);
120 napi_value retVal = nullptr;
121 napi_value global = nullptr;
122 napi_get_global(env, &global);
123 status = napi_call_function(env, global, jsCallback, ARGS_ONE, &(obj.val_), &retVal);
124 if (status != napi_ok) {
125 LOGE("napi call function failed, status: %{public}d", status);
126 }
127 }
128
OnDownloadProcess(DownloadProgressObj & progress)129 void CloudDownloadCallbackImpl::OnDownloadProcess(DownloadProgressObj &progress)
130 {
131 uv_loop_s *loop = nullptr;
132 napi_get_uv_event_loop(env_, &loop);
133 if (loop == nullptr) {
134 return;
135 }
136
137 uv_work_t *work = new (nothrow) uv_work_t;
138 if (work == nullptr) {
139 LOGE("Failed to create uv work");
140 return;
141 }
142
143 UvChangeMsg *msg = new (std::nothrow) UvChangeMsg(shared_from_this(), progress);
144 if (msg == nullptr) {
145 delete work;
146 return;
147 }
148 work->data = reinterpret_cast<void *>(msg);
149 int ret = uv_queue_work(
150 loop, work, [](uv_work_t *work) {},
151 [](uv_work_t *work, int status) {
152 auto msg = reinterpret_cast<UvChangeMsg *>(work->data);
153 OnComplete(msg);
154 delete msg;
155 delete work;
156 });
157 if (ret != 0) {
158 LOGE("Failed to execute libuv work queue, ret: %{public}d", ret);
159 delete msg;
160 delete work;
161 }
162 }
163
DeleteReference()164 void CloudDownloadCallbackImpl::DeleteReference()
165 {
166 if (cbOnRef_ != nullptr) {
167 napi_delete_reference(env_, cbOnRef_);
168 cbOnRef_ = nullptr;
169 }
170 }
171
On(napi_env env,napi_callback_info info)172 napi_value CloudFileDownloadNapi::On(napi_env env, napi_callback_info info)
173 {
174 LOGI("On begin");
175 NFuncArg funcArg(env, info);
176 if (!funcArg.InitArgs(NARG_CNT::TWO)) {
177 LOGE("On Number of arguments unmatched");
178 NError(E_PARAMS).ThrowErr(env);
179 return nullptr;
180 }
181 auto [succProgress, progress, ignore] = NVal(env, funcArg[NARG_POS::FIRST]).ToUTF8String();
182 if (!(succProgress && std::string(progress.get()) == "progress")) {
183 LOGE("On get progress failed!");
184 NError(E_PARAMS).ThrowErr(env);
185 return nullptr;
186 }
187
188 if (callback_ != nullptr) {
189 LOGI("callback already exist");
190 return NVal::CreateUndefined(env).val_;
191 }
192
193 callback_ = make_shared<CloudDownloadCallbackImpl>(env, NVal(env, funcArg[(int)NARG_POS::SECOND]).val_);
194 int32_t ret = CloudSyncManager::GetInstance().RegisterDownloadFileCallback(callback_);
195 if (ret != E_OK) {
196 LOGE("RegisterDownloadFileCallback error, ret: %{public}d", ret);
197 NError(Convert2JsErrNum(ret)).ThrowErr(env);
198 return nullptr;
199 }
200
201 return NVal::CreateUndefined(env).val_;
202 }
203
Off(napi_env env,napi_callback_info info)204 napi_value CloudFileDownloadNapi::Off(napi_env env, napi_callback_info info)
205 {
206 LOGI("Off begin");
207 NFuncArg funcArg(env, info);
208 if (!funcArg.InitArgs(NARG_CNT::ONE)) {
209 LOGE("Off Number of arguments unmatched");
210 NError(E_PARAMS).ThrowErr(env);
211 return nullptr;
212 }
213 auto [succProgress, progress, ignore] = NVal(env, funcArg[NARG_POS::FIRST]).ToUTF8String();
214 if (!(succProgress && std::string(progress.get()) == "progress")) {
215 LOGE("Off get progress failed!");
216 NError(E_PARAMS).ThrowErr(env);
217 return nullptr;
218 }
219
220 int32_t ret = CloudSyncManager::GetInstance().UnregisterDownloadFileCallback();
221 if (ret != E_OK) {
222 LOGE("UnregisterDownloadFileCallback error, ret: %{public}d", ret);
223 NError(Convert2JsErrNum(ret)).ThrowErr(env);
224 return nullptr;
225 }
226 if (callback_ != nullptr) {
227 /* napi delete reference */
228 callback_->DeleteReference();
229 callback_ = nullptr;
230 }
231 return NVal::CreateUndefined(env).val_;
232 }
233
Stop(napi_env env,napi_callback_info info)234 napi_value CloudFileDownloadNapi::Stop(napi_env env, napi_callback_info info)
235 {
236 LOGI("Stop begin");
237 NFuncArg funcArg(env, info);
238 if (!funcArg.InitArgs(NARG_CNT::ONE, NARG_CNT::TWO)) {
239 LOGE("Stop Number of arguments unmatched");
240 NError(E_PARAMS).ThrowErr(env);
241 return nullptr;
242 }
243 auto [succUri, uri, ignore] = NVal(env, funcArg[NARG_POS::FIRST]).ToUTF8String();
244 if (!succUri) {
245 LOGE("Stop get uri parameter failed!");
246 NError(E_PARAMS).ThrowErr(env);
247 return nullptr;
248 }
249
250 auto cbExec = [uri = string(uri.get()), env = env]() -> NError {
251 int32_t ret = CloudSyncManager::GetInstance().StopDownloadFile(uri);
252 if (ret != E_OK) {
253 LOGE("Stop Download failed! ret = %{public}d", ret);
254 return NError(Convert2JsErrNum(ret));
255 }
256 LOGI("Stop Download Success!");
257 return NError(ERRNO_NOERR);
258 };
259
260 auto cbCompl = [](napi_env env, NError err) -> NVal {
261 if (err) {
262 return {env, err.GetNapiErr(env)};
263 }
264 return NVal::CreateUndefined(env);
265 };
266
267 NVal thisVar(env, funcArg.GetThisVar());
268 string procedureName = "cloudFileDownload";
269 if (funcArg.GetArgc() == NARG_CNT::ONE) {
270 return NAsyncWorkPromise(env, thisVar).Schedule(procedureName, cbExec, cbCompl).val_;
271 } else {
272 NVal cb(env, funcArg[NARG_POS::SECOND]);
273 return NAsyncWorkCallback(env, thisVar, cb).Schedule(procedureName, cbExec, cbCompl).val_;
274 }
275 }
276
Export()277 bool CloudFileDownloadNapi::Export()
278 {
279 LOGI("CloudFileDownloadNapi::Export begin");
280 vector<napi_property_descriptor> props = {
281 NVal::DeclareNapiFunction("start", CloudFileDownloadNapi::Start),
282 NVal::DeclareNapiFunction("on", CloudFileDownloadNapi::On),
283 NVal::DeclareNapiFunction("off", CloudFileDownloadNapi::Off),
284 NVal::DeclareNapiFunction("stop", CloudFileDownloadNapi::Stop),
285 };
286
287 auto [succ, classValue] =
288 NClass::DefineClass(exports_.env_, className, CloudFileDownloadNapi::Constructor, std::move(props));
289 if (!succ) {
290 LOGE("Failed to define Download class");
291 NError(EIO).ThrowErr(exports_.env_);
292 return false;
293 }
294 succ = NClass::SaveClass(exports_.env_, className, classValue);
295 if (!succ) {
296 LOGE("Failed to save Download class");
297 NError(EIO).ThrowErr(exports_.env_);
298 return false;
299 }
300
301 LOGI("CloudFileDownloadNapi::Export end");
302 return exports_.AddProp(className, classValue);
303 }
304
GetClassName()305 string CloudFileDownloadNapi::GetClassName()
306 {
307 return CloudFileDownloadNapi::className;
308 }
309 } // namespace OHOS::FileManagement::CloudSync