• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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