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