• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2022-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 #include "ipc_skeleton.h"
16 #include "napi_account_common.h"
17 
18 #include "account_log_wrapper.h"
19 #include "ipc_skeleton.h"
20 #include "js_native_api.h"
21 #include "js_native_api_types.h"
22 #include "napi_account_error.h"
23 #include "napi/native_api.h"
24 #include "napi/native_common.h"
25 #include "napi/native_node_api.h"
26 #include "securec.h"
27 #include "tokenid_kit.h"
28 
29 namespace OHOS {
30 namespace AccountJsKit {
31 namespace {
32 constexpr int32_t BUSINESS_ERROR_ARG_SIZE = 2;
33 const char BUSINESS_ERROR_CODE_NAME[] = "code";
34 const char BUSINESS_ERROR_DATA_NAME[] = "data";
35 }
36 
37 using namespace AccountSA;
38 
~CommonAsyncContext()39 CommonAsyncContext::~CommonAsyncContext()
40 {
41     if (env == nullptr) {
42         return;
43     }
44     if (callbackRef != nullptr) {
45         napi_delete_reference(env, callbackRef);
46         callbackRef = nullptr;
47     }
48     if (work != nullptr) {
49         napi_delete_async_work(env, work);
50         work = nullptr;
51     }
52 }
53 
CreateExecEnv(napi_env env,uv_loop_s ** loop,uv_work_t ** work)54 bool CreateExecEnv(napi_env env, uv_loop_s **loop, uv_work_t **work)
55 {
56     *loop = nullptr;
57     napi_get_uv_event_loop(env, loop);
58     if (*loop == nullptr) {
59         ACCOUNT_LOGE("failed to get uv event loop");
60         return false;
61     }
62     *work = new (std::nothrow) uv_work_t;
63     if (*work == nullptr) {
64         ACCOUNT_LOGE("failed to create uv_work_t");
65         return false;
66     }
67     return true;
68 }
69 
ProcessCallbackOrPromise(napi_env env,const CommonAsyncContext * asyncContext,napi_value err,napi_value data)70 void ProcessCallbackOrPromise(napi_env env, const CommonAsyncContext *asyncContext, napi_value err, napi_value data)
71 {
72     napi_value args[BUSINESS_ERROR_ARG_SIZE] = {0};
73     if (asyncContext->errCode == ERR_OK) {
74         napi_get_null(env, &args[0]);
75         args[1] = data;
76     } else {
77         napi_get_null(env, &args[1]);
78         args[0] = err;
79     }
80     if (asyncContext->deferred) {
81         if (asyncContext->errCode == ERR_OK) {
82             napi_resolve_deferred(env, asyncContext->deferred, args[1]);
83         } else {
84             napi_reject_deferred(env, asyncContext->deferred, args[0]);
85         }
86     } else {
87         NapiCallVoidFunction(env, args, BUSINESS_ERROR_ARG_SIZE, asyncContext->callbackRef);
88     }
89 }
90 
ReturnCallbackOrPromise(napi_env env,const CommonAsyncContext * asyncContext,napi_value err,napi_value data)91 void ReturnCallbackOrPromise(napi_env env, const CommonAsyncContext *asyncContext, napi_value err, napi_value data)
92 {
93     napi_value args[BUSINESS_ERROR_ARG_SIZE] = {err, data};
94     if (asyncContext->errCode == ERR_OK) {
95         NAPI_CALL_RETURN_VOID(env, napi_get_null(env, &args[0]));
96     } else {
97         NAPI_CALL_RETURN_VOID(env, napi_get_null(env, &args[1]));
98     }
99     if (asyncContext->deferred != nullptr) {
100         if (asyncContext->errCode == ERR_OK) {
101             NAPI_CALL_RETURN_VOID(env, napi_resolve_deferred(env, asyncContext->deferred, args[1]));
102         } else {
103             NAPI_CALL_RETURN_VOID(env, napi_reject_deferred(env, asyncContext->deferred, args[0]));
104         }
105         return;
106     }
107     if (asyncContext->callbackRef != nullptr) {
108         NapiCallVoidFunction(env, args, BUSINESS_ERROR_ARG_SIZE, asyncContext->callbackRef);
109     }
110 }
111 
GetIntProperty(napi_env env,napi_value obj,int32_t & property)112 bool GetIntProperty(napi_env env, napi_value obj, int32_t &property)
113 {
114     napi_valuetype valueType = napi_undefined;
115     NAPI_CALL_BASE(env, napi_typeof(env, obj, &valueType), false);
116     if (valueType != napi_number) {
117         return false;
118     }
119 
120     NAPI_CALL_BASE(env, napi_get_value_int32(env, obj, &property), false);
121     return true;
122 }
123 
GetLongIntProperty(napi_env env,napi_value obj,int64_t & property)124 bool GetLongIntProperty(napi_env env, napi_value obj, int64_t &property)
125 {
126     napi_valuetype valueType = napi_undefined;
127     NAPI_CALL_BASE(env, napi_typeof(env, obj, &valueType), false);
128     if (valueType != napi_number) {
129         return false;
130     }
131 
132     NAPI_CALL_BASE(env, napi_get_value_int64(env, obj, &property), false);
133     return true;
134 }
135 
GetBoolProperty(napi_env env,napi_value obj,bool & property)136 bool GetBoolProperty(napi_env env, napi_value obj, bool &property)
137 {
138     napi_valuetype valueType = napi_undefined;
139     NAPI_CALL_BASE(env, napi_typeof(env, obj, &valueType), false);
140     if (valueType != napi_boolean) {
141         return false;
142     }
143     NAPI_CALL_BASE(env, napi_get_value_bool(env, obj, &property), false);
144     return true;
145 }
146 
GetStringProperty(napi_env env,napi_value obj,std::string & property)147 bool GetStringProperty(napi_env env, napi_value obj, std::string &property)
148 {
149     napi_valuetype valuetype = napi_undefined;
150     NAPI_CALL_BASE(env, napi_typeof(env, obj, &valuetype), false);
151     if (valuetype != napi_string) {
152         return false;
153     }
154 
155     size_t propLen;
156     NAPI_CALL_BASE(env, napi_get_value_string_utf8(env, obj, nullptr, 0, &propLen), false);
157     property.reserve(propLen + 1);
158     property.resize(propLen);
159     NAPI_CALL_BASE(env, napi_get_value_string_utf8(env, obj, property.data(), propLen + 1, &propLen), false);
160     return true;
161 }
162 
GetStringArrayProperty(napi_env env,napi_value obj,std::vector<std::string> & property,bool allowEmpty)163 bool GetStringArrayProperty(napi_env env, napi_value obj, std::vector<std::string> &property, bool allowEmpty)
164 {
165     bool isArray = false;
166     NAPI_CALL_BASE(env, napi_is_array(env, obj, &isArray), false);
167     if (!isArray) {
168         return false;
169     }
170     uint32_t length = 0;
171     NAPI_CALL_BASE(env, napi_get_array_length(env, obj, &length), false);
172     if (!allowEmpty && (length == 0)) {
173         return false;
174     }
175 
176     for (size_t i = 0; i < length; i++) {
177         napi_value strJs = nullptr;
178         NAPI_CALL_BASE(env, napi_get_element(env, obj, i, &strJs), false);
179         std::string str;
180         if (!GetStringProperty(env, strJs, str)) {
181             return false;
182         }
183         property.emplace_back(str);
184     }
185     return true;
186 }
187 
GetCallbackProperty(napi_env env,napi_value obj,napi_ref & property,int argNum)188 bool GetCallbackProperty(napi_env env, napi_value obj, napi_ref &property, int argNum)
189 {
190     napi_valuetype valueType = napi_undefined;
191     NAPI_CALL_BASE(env, napi_typeof(env, obj, &valueType), false);
192     if ((valueType == napi_undefined) || (valueType == napi_null)) {
193         ACCOUNT_LOGI("the callback is undefined or null");
194         return true;
195     } else if (valueType == napi_function) {
196         NAPI_CALL_BASE(env, napi_create_reference(env, obj, argNum, &property), false);
197         return true;
198     }
199     ACCOUNT_LOGE("the callback is not a napi_function");
200     return false;
201 }
202 
GetStringPropertyByKey(napi_env env,napi_value obj,const std::string & propertyName,std::string & property)203 bool GetStringPropertyByKey(napi_env env, napi_value obj, const std::string &propertyName, std::string &property)
204 {
205     napi_value value = nullptr;
206     NAPI_CALL_BASE(env, napi_get_named_property(env, obj, propertyName.c_str(), &value), false);
207 
208     return GetStringProperty(env, value, property);
209 }
210 
GetOptionalStringPropertyByKey(napi_env env,napi_value obj,const std::string & propertyName,std::string & property)211 bool GetOptionalStringPropertyByKey(napi_env env, napi_value obj, const std::string &propertyName,
212     std::string &property)
213 {
214     bool hasProp = false;
215     napi_has_named_property(env, obj, propertyName.c_str(), &hasProp);
216     if (!hasProp) {
217         return true;
218     }
219     napi_value value = nullptr;
220     NAPI_CALL_BASE(env, napi_get_named_property(env, obj, propertyName.c_str(), &value), false);
221     napi_valuetype valuetype = napi_undefined;
222     NAPI_CALL_BASE(env, napi_typeof(env, value, &valuetype), false);
223     if ((valuetype == napi_undefined) || (valuetype == napi_null)) {
224         ACCOUNT_LOGI("this key's value is undefined or null");
225         return true;
226     }
227     return GetStringProperty(env, value, property);
228 }
229 
CompareOnAndOffRef(const napi_env env,napi_ref subscriberRef,napi_ref unsubscriberRef)230 bool CompareOnAndOffRef(const napi_env env, napi_ref subscriberRef, napi_ref unsubscriberRef)
231 {
232     napi_value subscriberCallback;
233     napi_get_reference_value(env, subscriberRef, &subscriberCallback);
234     napi_value unsubscriberCallback;
235     napi_get_reference_value(env, unsubscriberRef, &unsubscriberCallback);
236     bool result = false;
237     napi_strict_equals(env, subscriberCallback, unsubscriberCallback, &result);
238     return result;
239 }
240 
IsSystemApp(napi_env env)241 bool IsSystemApp(napi_env env)
242 {
243     uint64_t tokenId = IPCSkeleton::GetSelfTokenID();
244     bool isSystemApp = Security::AccessToken::TokenIdKit::IsSystemAppByFullTokenID(tokenId);
245     if (!isSystemApp) {
246         std::string errMsg = ConvertToJsErrMsg(ERR_JS_IS_NOT_SYSTEM_APP);
247         AccountNapiThrow(env, ERR_JS_IS_NOT_SYSTEM_APP, errMsg, true);
248         return false;
249     }
250     return true;
251 }
252 
CreateStringArray(napi_env env,const std::vector<std::string> & strVec)253 napi_value CreateStringArray(napi_env env, const std::vector<std::string> &strVec)
254 {
255     napi_value result = nullptr;
256     napi_create_array(env, &result);
257     for (size_t i = 0; i < strVec.size(); ++i) {
258         napi_value value = nullptr;
259         napi_create_string_utf8(env, strVec[i].c_str(), NAPI_AUTO_LENGTH, &value);
260         napi_set_element(env, result, i, value);
261     }
262     return result;
263 }
CreateUint8Array(napi_env env,const uint8_t * srcData,size_t length)264 napi_value CreateUint8Array(napi_env env, const uint8_t *srcData, size_t length)
265 {
266     napi_value result = nullptr;
267     void* dstData = nullptr;
268     napi_value napiArr = nullptr;
269     NAPI_CALL(env, napi_create_arraybuffer(env, length, &dstData, &napiArr));
270     if ((length > 0) && (memcpy_s(dstData, length, srcData, length) != EOK)) {
271         return result;
272     }
273     NAPI_CALL(env, napi_create_typedarray(env, napi_uint8_array, length, napiArr, 0, &result));
274     return result;
275 }
276 
ParseUint8TypedArray(napi_env env,napi_value value,uint8_t ** data,size_t * length)277 napi_status ParseUint8TypedArray(napi_env env, napi_value value, uint8_t **data, size_t *length)
278 {
279     *data = nullptr;
280     *length = 0;
281     bool isTypedArray = false;
282     napi_is_typedarray(env, value, &isTypedArray);
283     if (!isTypedArray) {
284         ACCOUNT_LOGE("invalid uint8 array");
285         return napi_ok;
286     }
287     napi_typedarray_type arrayType = static_cast<napi_typedarray_type>(-1);  // -1 indicates invalid type
288     napi_value buffer = nullptr;
289     size_t offset = 0;
290     napi_get_typedarray_info(env, value, &arrayType, length, reinterpret_cast<void **>(data), &buffer, &offset);
291     if (arrayType != napi_uint8_array) {
292         ACCOUNT_LOGE("invalid uint8 array");
293         *data = nullptr;
294         *length = 0;
295     }
296     return napi_ok;
297 }
298 
ParseUint8TypedArrayToVector(napi_env env,napi_value value,std::vector<uint8_t> & vec)299 napi_status ParseUint8TypedArrayToVector(napi_env env, napi_value value, std::vector<uint8_t> &vec)
300 {
301     uint8_t *data = nullptr;
302     size_t length = 0;
303     napi_status status = ParseUint8TypedArray(env, value, &data, &length);
304     if (status != napi_ok) {
305         ACCOUNT_LOGE("failed to ParseUint8TypedArray");
306         return status;
307     }
308     vec.assign(data, data + length);
309     return napi_ok;
310 }
311 
ParseUint8TypedArrayToUint64(napi_env env,napi_value value,uint64_t & result)312 napi_status ParseUint8TypedArrayToUint64(napi_env env, napi_value value, uint64_t &result)
313 {
314     uint8_t *data = nullptr;
315     size_t length = 0;
316     napi_status status = ParseUint8TypedArray(env, value, &data, &length);
317     if (status != napi_ok) {
318         ACCOUNT_LOGE("failed to ParseUint8TypedArray");
319         return status;
320     }
321     if (data == nullptr) {
322         result = 0;
323         return napi_invalid_arg;
324     }
325     if (length != sizeof(uint64_t)) {
326         ACCOUNT_LOGE("failed to convert to uint64_t value");
327         return napi_invalid_arg;
328     }
329     result = *(reinterpret_cast<uint64_t *>(data));
330     return napi_ok;
331 }
332 
ParseBusinessError(napi_env env,napi_value value,BusinessError & error)333 bool ParseBusinessError(napi_env env, napi_value value, BusinessError &error)
334 {
335     napi_valuetype valueType = napi_undefined;
336     NAPI_CALL_BASE(env, napi_typeof(env, value, &valueType), false);
337     if (valueType == napi_null || (valueType == napi_undefined)) {
338         error.code = 0;
339         return true;
340     }
341     napi_value napiCode = nullptr;
342     NAPI_CALL_BASE(env, napi_get_named_property(env, value, BUSINESS_ERROR_CODE_NAME, &napiCode), false);
343     if (napiCode == nullptr) {
344         ACCOUNT_LOGE("code is undefined");
345         return false;
346     }
347     NAPI_CALL_BASE(env, napi_get_value_int32(env, napiCode, &error.code), false);
348     bool hasData = false;
349     napi_has_named_property(env, value, BUSINESS_ERROR_DATA_NAME, &hasData);
350     if (hasData) {
351         napi_value napiData = nullptr;
352         napi_get_named_property(env, value, BUSINESS_ERROR_DATA_NAME, &napiData);
353         return GetStringProperty(env, napiData, error.data);
354     }
355     return true;
356 }
357 
GetNamedJsFunction(napi_env env,napi_value object,const std::string & name,napi_ref & callback)358 bool GetNamedJsFunction(napi_env env, napi_value object, const std::string &name, napi_ref &callback)
359 {
360     napi_valuetype valueType = napi_undefined;
361     NAPI_CALL_BASE(env, napi_typeof(env, object, &valueType), false);
362     if (valueType != napi_object) {
363         ACCOUNT_LOGE("invalid object");
364         return false;
365     }
366     napi_value result = nullptr;
367     NAPI_CALL_BASE(env, napi_get_named_property(env, object, name.c_str(), &result), false);
368     return GetCallbackProperty(env, result, callback, 1);
369 }
370 
NapiCallVoidFunction(napi_env env,napi_value * argv,size_t argc,napi_ref funcRef)371 void NapiCallVoidFunction(napi_env env, napi_value *argv, size_t argc, napi_ref funcRef)
372 {
373     napi_value undefined = nullptr;
374     NAPI_CALL_RETURN_VOID(env, napi_get_undefined(env, &undefined));
375     napi_value returnVal;
376     napi_value func = nullptr;
377     NAPI_CALL_RETURN_VOID(env, napi_get_reference_value(env, funcRef, &func));
378     napi_call_function(env, undefined, func, argc, argv, &returnVal);
379     ACCOUNT_LOGI("call js function finish");
380 }
381 
CreateAuthResult(napi_env env,const std::vector<uint8_t> & token,int32_t remainTimes,int32_t freezingTime)382 napi_value CreateAuthResult(
383     napi_env env, const std::vector<uint8_t> &token, int32_t remainTimes, int32_t freezingTime)
384 {
385     napi_value object = nullptr;
386     NAPI_CALL(env, napi_create_object(env, &object));
387     if (remainTimes >= 0) {
388         napi_value napiRemainTimes = 0;
389         NAPI_CALL(env, napi_create_uint32(env, remainTimes, &napiRemainTimes));
390         NAPI_CALL(env, napi_set_named_property(env, object, "remainTimes", napiRemainTimes));
391     }
392     if (freezingTime >= 0) {
393         napi_value napiFreezingTimes = 0;
394         NAPI_CALL(env, napi_create_uint32(env, freezingTime, &napiFreezingTimes));
395         NAPI_CALL(env, napi_set_named_property(env, object, "freezingTime", napiFreezingTimes));
396     }
397     if (token.size() > 0) {
398         napi_value napiToken = CreateUint8Array(env, token.data(), token.size());
399         NAPI_CALL(env, napi_set_named_property(env, object, "token", napiToken));
400     }
401     return object;
402 }
403 
ReleaseNapiRefAsync(napi_env env,napi_ref napiRef)404 void ReleaseNapiRefAsync(napi_env env, napi_ref napiRef)
405 {
406     ReleaseNapiRefArray(env, {napiRef});
407 }
408 
ReleaseNapiRefArray(napi_env env,const std::vector<napi_ref> & napiRefVec)409 void ReleaseNapiRefArray(napi_env env, const std::vector<napi_ref> &napiRefVec)
410 {
411     if (env == nullptr) {
412         ACCOUNT_LOGE("invalid env");
413         return;
414     }
415     std::unique_ptr<uv_work_t> work = std::make_unique<uv_work_t>();
416     std::unique_ptr<NapiRefArrayContext> context = std::make_unique<NapiRefArrayContext>();
417     uv_loop_s *loop = nullptr;
418     napi_get_uv_event_loop(env, &loop);
419     if ((loop == nullptr) || (work == nullptr) || (context == nullptr)) {
420         ACCOUNT_LOGE("fail to init execution environment");
421         return;
422     }
423     context->env = env;
424     context->napiRefVec = napiRefVec;
425     work->data = reinterpret_cast<void *>(context.get());
426     NAPI_CALL_RETURN_VOID(env, uv_queue_work_with_qos(loop, work.get(), [] (uv_work_t *work) {},
427         [] (uv_work_t *work, int status) {
428             if (work == nullptr) {
429                 ACCOUNT_LOGE("work is nullptr");
430                 return;
431             }
432             auto context = reinterpret_cast<NapiRefArrayContext *>(work->data);
433             if (context == nullptr) {
434                 ACCOUNT_LOGE("context is nullptr");
435                 delete work;
436                 return;
437             }
438             for (auto &napiRef : context->napiRefVec) {
439                 if (napiRef != nullptr) {
440                     napi_delete_reference(context->env, napiRef);
441                 }
442             }
443             delete context;
444             delete work;
445         }, uv_qos_default));
446     context.release();
447     work.release();
448 }
449 
~NapiCallbackRef()450 NapiCallbackRef::~NapiCallbackRef()
451 {
452     ReleaseNapiRefArray(env, {callbackRef});
453 }
454 
InitUvWorkCallbackEnv(uv_work_t * work,napi_handle_scope & scope)455 bool InitUvWorkCallbackEnv(uv_work_t *work, napi_handle_scope &scope)
456 {
457     if (work == nullptr) {
458         ACCOUNT_LOGE("work is nullptr");
459         return false;
460     }
461     if (work->data == nullptr) {
462         ACCOUNT_LOGE("data is nullptr");
463         return false;
464     }
465     CommonAsyncContext *data = reinterpret_cast<CommonAsyncContext *>(work->data);
466     napi_open_handle_scope(data->env, &scope);
467     if (scope == nullptr) {
468         ACCOUNT_LOGE("fail to open scope");
469         delete data;
470         work->data = nullptr;
471         return false;
472     }
473     return true;
474 }
475 } // namespace AccountJsKit
476 } // namespace OHOS