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 "napi_util.h"
17 #include <endian.h>
18 #include <securec.h>
19 #include <sstream>
20 #include "calendar_log.h"
21 #include "napi_queue.h"
22 #include "native_util.h"
23 #include "event_filter_napi.h"
24
25 namespace OHOS::CalendarApi::NapiUtil {
26 constexpr int32_t STR_MAX_LENGTH = 4096;
27 constexpr size_t STR_TAIL_LENGTH = 1;
28
GetValue(napi_env env,napi_value in,napi_value & out)29 napi_status GetValue(napi_env env, napi_value in, napi_value& out)
30 {
31 out = in;
32 return napi_ok;
33 }
34
SetValue(napi_env env,napi_value in,napi_value & out)35 napi_status SetValue(napi_env env, napi_value in, napi_value& out)
36 {
37 out = in;
38 return napi_ok;
39 }
40
41 /* napi_value <-> bool */
GetValue(napi_env env,napi_value in,bool & out)42 napi_status GetValue(napi_env env, napi_value in, bool& out)
43 {
44 return napi_get_value_bool(env, in, &out);
45 }
46
SetValue(napi_env env,const bool & in,napi_value & out)47 napi_status SetValue(napi_env env, const bool& in, napi_value& out)
48 {
49 return napi_get_boolean(env, in, &out);
50 }
51
52 /* napi_value <-> int32_t */
GetValue(napi_env env,napi_value in,int32_t & out)53 napi_status GetValue(napi_env env, napi_value in, int32_t& out)
54 {
55 return napi_get_value_int32(env, in, &out);
56 }
57
SetValue(napi_env env,const int32_t & in,napi_value & out)58 napi_status SetValue(napi_env env, const int32_t& in, napi_value& out)
59 {
60 return napi_create_int32(env, in, &out);
61 }
62
63 /* napi_value <-> uint32_t */
GetValue(napi_env env,napi_value in,uint32_t & out)64 napi_status GetValue(napi_env env, napi_value in, uint32_t& out)
65 {
66 return napi_get_value_uint32(env, in, &out);
67 }
68
SetValue(napi_env env,const uint32_t & in,napi_value & out)69 napi_status SetValue(napi_env env, const uint32_t& in, napi_value& out)
70 {
71 return napi_create_uint32(env, in, &out);
72 }
73
74 /* napi_value <-> int64_t */
GetValue(napi_env env,napi_value in,int64_t & out)75 napi_status GetValue(napi_env env, napi_value in, int64_t& out)
76 {
77 return napi_get_value_int64(env, in, &out);
78 }
79
SetValue(napi_env env,const int64_t & in,napi_value & out)80 napi_status SetValue(napi_env env, const int64_t& in, napi_value& out)
81 {
82 return napi_create_int64(env, in, &out);
83 }
84
GetValue(napi_env env,napi_value in,uint64_t & out)85 napi_status GetValue(napi_env env, napi_value in, uint64_t& out)
86 {
87 bool lossless = true;
88 return napi_get_value_bigint_uint64(env, in, &out, &lossless);
89 }
90
SetValue(napi_env env,const uint64_t & in,napi_value & out)91 napi_status SetValue(napi_env env, const uint64_t& in, napi_value& out)
92 {
93 return napi_create_bigint_uint64(env, in, &out);
94 }
95
96 /* napi_value <-> double */
GetValue(napi_env env,napi_value in,double & out)97 napi_status GetValue(napi_env env, napi_value in, double& out)
98 {
99 return napi_get_value_double(env, in, &out);
100 }
101
SetValue(napi_env env,const double & in,napi_value & out)102 napi_status SetValue(napi_env env, const double& in, napi_value& out)
103 {
104 return napi_create_double(env, in, &out);
105 }
106
107 /* napi_value <-> std::string */
GetValue(napi_env env,napi_value in,std::string & out)108 napi_status GetValue(napi_env env, napi_value in, std::string& out)
109 {
110 napi_valuetype type = napi_undefined;
111 napi_status status = napi_typeof(env, in, &type);
112 CHECK_RETURN((status == napi_ok) && (type == napi_string), "invalid type", napi_invalid_arg);
113
114 size_t maxLen = STR_MAX_LENGTH;
115 status = napi_get_value_string_utf8(env, in, NULL, 0, &maxLen);
116 if (maxLen <= 0) {
117 return status;
118 }
119 LOG_DEBUG("napi_value -> std::string get length %{public}d", (int)maxLen);
120 char* buf = new (std::nothrow) char[STR_TAIL_LENGTH + maxLen];
121 if (buf != nullptr) {
122 size_t len = 0;
123 status = napi_get_value_string_utf8(env, in, buf, STR_TAIL_LENGTH + maxLen, &len);
124 if (status == napi_ok) {
125 buf[len] = 0;
126 out = std::string(buf);
127 }
128 delete[] buf;
129 } else {
130 status = napi_generic_failure;
131 }
132 return status;
133 }
134
SetValue(napi_env env,const std::string & in,napi_value & out)135 napi_status SetValue(napi_env env, const std::string& in, napi_value& out)
136 {
137 return napi_create_string_utf8(env, in.c_str(), in.size(), &out);
138 }
139
140 /* napi_value <-> std::vector<std::string> */
GetValue(napi_env env,napi_value in,std::vector<std::string> & out)141 napi_status GetValue(napi_env env, napi_value in, std::vector<std::string>& out)
142 {
143 LOG_DEBUG("napi_value -> std::vector<std::string>");
144 return GetValueArray(env, in, out);
145 }
146
SetValue(napi_env env,const std::vector<std::string> & in,napi_value & out)147 napi_status SetValue(napi_env env, const std::vector<std::string>& in, napi_value& out)
148 {
149 LOG_DEBUG("std::vector<std::string> -> napi_value");
150 napi_status status = napi_create_array_with_length(env, in.size(), &out);
151 CHECK_RETURN(status == napi_ok, "create array failed!", status);
152 int index = 0;
153 for (auto& item : in) {
154 napi_value element = nullptr;
155 SetValue(env, item, element);
156 status = napi_set_element(env, out, index++, element);
157 CHECK_RETURN((status == napi_ok), "napi_set_element failed!", status);
158 }
159 return status;
160 }
161
162 /* napi_value <-> std::vector<uint8_t> */
GetValue(napi_env env,napi_value in,std::vector<uint8_t> & out)163 napi_status GetValue(napi_env env, napi_value in, std::vector<uint8_t>& out)
164 {
165 out.clear();
166 LOG_DEBUG("napi_value -> std::vector<uint8_t> ");
167 napi_typedarray_type type = napi_biguint64_array;
168 size_t length = 0;
169 napi_value buffer = nullptr;
170 size_t offset = 0;
171 void* data = nullptr;
172 napi_status status = napi_get_typedarray_info(env, in, &type, &length, &data, &buffer, &offset);
173 LOG_DEBUG("array type=%{public}d length=%{public}d offset=%{public}d", (int)type, (int)length, (int)offset);
174 CHECK_RETURN(status == napi_ok, "napi_get_typedarray_info failed!", napi_invalid_arg);
175 CHECK_RETURN(type == napi_uint8_array, "is not Uint8Array!", napi_invalid_arg);
176 CHECK_RETURN((length > 0) && (data != nullptr), "invalid data!", napi_invalid_arg);
177 out.assign((uint8_t*)data, ((uint8_t*)data) + length);
178 return status;
179 }
180
SetValue(napi_env env,const std::vector<uint8_t> & in,napi_value & out)181 napi_status SetValue(napi_env env, const std::vector<uint8_t>& in, napi_value& out)
182 {
183 LOG_DEBUG("napi_value <- std::vector<uint8_t> ");
184 CHECK_RETURN(in.size() > 0, "invalid std::vector<uint8_t>", napi_invalid_arg);
185 void* data = nullptr;
186 napi_value buffer = nullptr;
187 napi_status status = napi_create_arraybuffer(env, in.size(), &data, &buffer);
188 CHECK_RETURN((status == napi_ok), "create array buffer failed!", status);
189
190 if (memcpy_s(data, in.size(), in.data(), in.size()) != EOK) {
191 LOG_ERROR("memcpy_s not EOK");
192 return napi_invalid_arg;
193 }
194 status = napi_create_typedarray(env, napi_uint8_array, in.size(), buffer, 0, &out);
195 CHECK_RETURN((status == napi_ok), "napi_value <- std::vector<uint8_t> invalid value", status);
196 return status;
197 }
198
199 /* napi_value <-> std::vector<int32_t> */
GetValue(napi_env env,napi_value in,std::vector<int32_t> & out)200 napi_status GetValue(napi_env env, napi_value in, std::vector<int32_t>& out)
201 {
202 LOG_DEBUG("napi_value -> std::vector<int32_t> ");
203 return GetValueArray(env, in, out);
204 }
205
SetValue(napi_env env,const std::vector<int32_t> & in,napi_value & out)206 napi_status SetValue(napi_env env, const std::vector<int32_t>& in, napi_value& out)
207 {
208 LOG_DEBUG("napi_value <- std::vector<int32_t> ");
209 return SetValueArray(env, in, out);
210 }
211
212 /* napi_value <-> std::vector<uint32_t> */
GetValue(napi_env env,napi_value in,std::vector<uint32_t> & out)213 napi_status GetValue(napi_env env, napi_value in, std::vector<uint32_t>& out)
214 {
215 LOG_DEBUG("napi_value -> std::vector<uint32_t> ");
216 return GetValueArray(env, in, out);
217 }
218
SetValue(napi_env env,const std::vector<uint32_t> & in,napi_value & out)219 napi_status SetValue(napi_env env, const std::vector<uint32_t>& in, napi_value& out)
220 {
221 LOG_DEBUG("napi_value <- std::vector<uint32_t> ");
222 size_t bytes = in.size() * sizeof(uint32_t);
223 CHECK_RETURN(bytes > 0, "invalid std::vector<uint32_t>", napi_invalid_arg);
224 void* data = nullptr;
225 napi_value buffer = nullptr;
226 napi_status status = napi_create_arraybuffer(env, bytes, &data, &buffer);
227 CHECK_RETURN((status == napi_ok), "invalid buffer", status);
228
229 if (memcpy_s(data, bytes, in.data(), bytes) != EOK) {
230 LOG_ERROR("memcpy_s not EOK");
231 return napi_invalid_arg;
232 }
233 status = napi_create_typedarray(env, napi_uint32_array, in.size(), buffer, 0, &out);
234 CHECK_RETURN((status == napi_ok), "invalid buffer", status);
235 return status;
236 }
237
238 /* napi_value <-> std::vector<int64_t> */
GetValue(napi_env env,napi_value in,std::vector<int64_t> & out)239 napi_status GetValue(napi_env env, napi_value in, std::vector<int64_t>& out)
240 {
241 LOG_DEBUG("napi_value -> std::vector<int64_t> ");
242 return GetValueArray(env, in, out);
243 }
244
SetValue(napi_env env,const std::vector<int64_t> & in,napi_value & out)245 napi_status SetValue(napi_env env, const std::vector<int64_t>& in, napi_value& out)
246 {
247 LOG_DEBUG("napi_value <- std::vector<int64_t> ");
248 size_t bytes = in.size() * sizeof(int64_t);
249 CHECK_RETURN(bytes > 0, "invalid std::vector<uint32_t>", napi_invalid_arg);
250 void* data = nullptr;
251 napi_value buffer = nullptr;
252 napi_status status = napi_create_arraybuffer(env, bytes, &data, &buffer);
253 CHECK_RETURN((status == napi_ok), "invalid buffer", status);
254
255 if (memcpy_s(data, bytes, in.data(), bytes) != EOK) {
256 LOG_ERROR("memcpy_s not EOK");
257 return napi_invalid_arg;
258 }
259 status = napi_create_typedarray(env, napi_bigint64_array, in.size(), buffer, 0, &out);
260 CHECK_RETURN((status == napi_ok), "invalid buffer", status);
261 return status;
262 }
263
GetValue(napi_env env,napi_value in,CalendarAccount & out)264 napi_status GetValue(napi_env env, napi_value in, CalendarAccount& out)
265 {
266 LOG_DEBUG("napi_value -> CalendarAccount ");
267 napi_status status = NapiUtil::GetNamedProperty(env, in, "name", out.name);
268 CHECK_RETURN((status == napi_ok), "invalid name", status);
269 status = NapiUtil::GetNamedProperty(env, in, "type", out.type);
270 CHECK_RETURN((status == napi_ok), "invalid type", status);
271 NapiUtil::GetNamedPropertyOptional(env, in, "displayName", out.displayName);
272 return napi_ok;
273 }
274
SetValue(napi_env env,const CalendarAccount & in,napi_value & out)275 napi_status SetValue(napi_env env, const CalendarAccount& in, napi_value& out)
276 {
277 LOG_DEBUG("CalendarAccount -> napi_value ");
278 napi_status status = napi_create_object(env, &out);
279 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
280
281 napi_value nameValue = nullptr;
282 status = SetValue(env, in.name, nameValue);
283 CHECK_RETURN((status == napi_ok), "invalid entry name", status);
284 napi_set_named_property(env, out, "name", nameValue);
285 napi_value typeValue = nullptr;
286 status = SetValue(env, in.type, typeValue);
287 CHECK_RETURN((status == napi_ok), "invalid entry type", status);
288 napi_set_named_property(env, out, "type", typeValue);
289
290 if (in.displayName) {
291 napi_value displayNameValue = nullptr;
292 status = SetValue(env, in.displayName.value(), displayNameValue);
293 CHECK_RETURN((status == napi_ok), "invalid entry displayName", status);
294 napi_set_named_property(env, out, "displayName", displayNameValue);
295 }
296 return status;
297 }
298
299 /* napi_value <-> CalendarConfig */
GetValue(napi_env env,napi_value in,CalendarConfig & out)300 napi_status GetValue(napi_env env, napi_value in, CalendarConfig& out)
301 {
302 LOG_DEBUG("napi_value -> CalendarConfig ");
303 NapiUtil::GetNamedPropertyOptional(env, in, "enableReminder", out.enableReminder);
304 bool result = true;
305 napi_status status = napi_has_named_property(env, in, "color", &result);
306 if (status == napi_ok && !result) {
307 const int64_t defaultColor = 0xFF0A59F7;
308 LOG_DEBUG("napi_value color is null, use default color: 0xFF0A59F7");
309 out.color.emplace<1>(defaultColor);
310 return napi_ok;
311 }
312 napi_value value = NULL;
313 napi_valuetype valueType = napi_undefined;
314 napi_get_named_property(env, in, "color", &value);
315 napi_typeof(env, value, &valueType);
316 if (valueType == napi_string) {
317 LOG_DEBUG("napi_value color is string");
318 string str = "";
319 NapiUtil::GetValue(env, value, str);
320 LOG_DEBUG("napi_value color: %{public}s", str.c_str());
321 bool ok = Native::ColorParse(str, out.color);
322 if (!ok) {
323 return napi_string_expected;
324 }
325 return napi_ok;
326 }
327 LOG_DEBUG("napi_value color is number");
328 int64_t colorValue;
329 napi_status statusToGetInt64 = napi_get_value_int64(env, value, &colorValue);
330 if (statusToGetInt64 == napi_ok) {
331 out.color.emplace<1>(colorValue);
332 LOG_DEBUG("color: %{public}s", std::to_string(std::get<1>(out.color)).c_str());
333 } else {
334 LOG_DEBUG("color number -> int_64 err");
335 }
336 return statusToGetInt64;
337 }
338
SetValue(napi_env env,const CalendarConfig & in,napi_value & out)339 napi_status SetValue(napi_env env, const CalendarConfig& in, napi_value& out)
340 {
341 LOG_DEBUG("CalendarConfig -> napi_value ");
342 napi_status status = napi_create_object(env, &out);
343 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
344
345 if (in.enableReminder.has_value()) {
346 LOG_DEBUG("config.enableReminder: %{public}d", in.enableReminder.value_or(-1));
347 napi_value enableRemindValue = nullptr;
348 status = SetValue(env, in.enableReminder.value(), enableRemindValue);
349 CHECK_RETURN((status == napi_ok), "invalid entry enableReminder", status);
350 napi_set_named_property(env, out, "enableReminder", enableRemindValue);
351 }
352 if (std::get_if<1>(&in.color)) {
353 napi_value colorValue = nullptr;
354 string color;
355 std::stringstream ss;
356 ss << std::hex << std::uppercase << std::get<1>(in.color);
357 ss >> color;
358 const int rgbLen = 5;
359 const int argbLen = 7;
360 if (color.size() == rgbLen || color.size() == argbLen) {
361 color = '0' + color;
362 }
363 color = '#' + color;
364 LOG_DEBUG("config.color: %{public}s", color.c_str());
365 status = SetValue(env, color, colorValue);
366 CHECK_RETURN((status == napi_ok), "invalid entry color", status);
367 napi_set_named_property(env, out, "color", colorValue);
368 }
369 return status;
370 }
371
GetValue(napi_env env,napi_value in,Location & out)372 napi_status GetValue(napi_env env, napi_value in, Location& out)
373 {
374 LOG_DEBUG("napi_value -> Location ");
375 NapiUtil::GetNamedPropertyOptional(env, in, "location", out.location);
376 NapiUtil::GetNamedPropertyOptional(env, in, "longitude", out.longitude);
377 NapiUtil::GetNamedPropertyOptional(env, in, "latitude", out.latitude);
378 return napi_ok;
379 }
380
SetValue(napi_env env,const Location & in,napi_value & out)381 napi_status SetValue(napi_env env, const Location& in, napi_value& out)
382 {
383 LOG_DEBUG("Location -> napi_value ");
384 napi_status status = napi_create_object(env, &out);
385 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
386
387 if (in.location) {
388 napi_value locationValue = nullptr;
389 status = SetValue(env, in.location.value(), locationValue);
390 CHECK_RETURN((status == napi_ok), "invalid location", status);
391 napi_set_named_property(env, out, "location", locationValue);
392 }
393 if (in.longitude) {
394 napi_value value = nullptr;
395 status = SetValue(env, in.longitude.value(), value);
396 CHECK_RETURN((status == napi_ok), "invalid longitude", status);
397 napi_set_named_property(env, out, "longitude", value);
398 }
399 if (in.latitude) {
400 napi_value value = nullptr;
401 status = SetValue(env, in.latitude.value(), value);
402 CHECK_RETURN((status == napi_ok), "invalid latitude", status);
403 napi_set_named_property(env, out, "latitude", value);
404 }
405 return napi_ok;
406 }
407
408 /* napi_value <-> RecurrenceRule */
GetValue(napi_env env,napi_value in,RecurrenceRule & out)409 napi_status GetValue(napi_env env, napi_value in, RecurrenceRule& out)
410 {
411 LOG_DEBUG("napi_value -> RecurrenceRule ");
412 return napi_ok;
413 }
414
SetValue(napi_env env,const RecurrenceRule & in,napi_value & out)415 napi_status SetValue(napi_env env, const RecurrenceRule& in, napi_value& out)
416 {
417 LOG_DEBUG("RecurrenceRule -> napi_value ");
418 return napi_ok;
419 }
420
421 /* napi_value <-> Attendee */
GetValue(napi_env env,napi_value in,Attendee & out)422 napi_status GetValue(napi_env env, napi_value in, Attendee& out)
423 {
424 LOG_DEBUG("Attendee -> napi_value ");
425 NapiUtil::GetNamedProperty(env, in, "name", out.name);
426 NapiUtil::GetNamedProperty(env, in, "email", out.email);
427 return napi_ok;
428 }
429
SetValue(napi_env env,const Attendee & in,napi_value & out)430 napi_status SetValue(napi_env env, const Attendee& in, napi_value& out)
431 {
432 LOG_DEBUG("napi_value -> Attendee ");
433 napi_status status = napi_create_object(env, &out);
434 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
435
436 napi_value nameValue = nullptr;
437 status = SetValue(env, in.name, nameValue);
438 CHECK_RETURN((status == napi_ok), "invalid entry name", status);
439 napi_set_named_property(env, out, "name", nameValue);
440 napi_value emailValue = nullptr;
441 status = SetValue(env, in.email, emailValue);
442 CHECK_RETURN((status == napi_ok), "invalid entry type", status);
443 napi_set_named_property(env, out, "email", emailValue);
444 return napi_ok;
445 }
446
447 /* napi_value <-> std::vector<Attendee> */
GetValue(napi_env env,napi_value in,std::vector<Attendee> & out)448 napi_status GetValue(napi_env env, napi_value in, std::vector<Attendee>& out)
449 {
450 LOG_DEBUG("napi_value -> std::vector<Attendee> ");
451 return GetValueArray(env, in, out);
452 }
453
SetValue(napi_env env,const std::vector<Attendee> & in,napi_value & out)454 napi_status SetValue(napi_env env, const std::vector<Attendee>& in, napi_value& out)
455 {
456 LOG_DEBUG("std::vector<Attendee> -> napi_value ");
457 return SetValueArray(env, in, out);
458 }
459
GetValue(napi_env env,napi_value in,EventService & out)460 napi_status GetValue(napi_env env, napi_value in, EventService& out)
461 {
462 LOG_DEBUG("napi_value -> EventService");
463 NapiUtil::GetNamedPropertyOptional(env, in, "description", out.description);
464 NapiUtil::GetNamedProperty(env, in, "type", out.type);
465 return NapiUtil::GetNamedProperty(env, in, "uri", out.uri);
466 }
467
SetValue(napi_env env,const EventService & in,napi_value & out)468 napi_status SetValue(napi_env env, const EventService& in, napi_value& out)
469 {
470 LOG_DEBUG("EventService -> napi_value");
471 napi_status status = napi_create_object(env, &out);
472 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
473
474 napi_value typeValue = nullptr;
475 status = SetValue(env, in.type, typeValue);
476 CHECK_RETURN((status == napi_ok), "invalid typeValue", status);
477 status = napi_set_named_property(env, out, "type", typeValue);
478 CHECK_RETURN((status == napi_ok), "set type failed", status);
479
480 napi_value uriValue = nullptr;
481 status = SetValue(env, in.uri, uriValue);
482 CHECK_RETURN((status == napi_ok), "invalid entry type", status);
483 status = napi_set_named_property(env, out, "uri", uriValue);
484 CHECK_RETURN((status == napi_ok), "set uri failed", status);
485
486 if (in.description) {
487 napi_value descriptionValue = nullptr;
488 status = SetValue(env, in.description.value(), descriptionValue);
489 CHECK_RETURN((status == napi_ok), "invalid description", status);
490 status = napi_set_named_property(env, out, "description", descriptionValue);
491 CHECK_RETURN((status == napi_ok), "set description failed", status);
492 }
493 return status;
494 }
495
496 /* napi_value <-> EventFilter */
GetValue(napi_env env,napi_value in,EventFilterNapi * & out)497 napi_status GetValue(napi_env env, napi_value in, EventFilterNapi*& out)
498 {
499 LOG_DEBUG("napi_value -> EventFilter ");
500 return EventFilterNapi::ToJson(env, in, out);
501 }
502
SetValue(napi_env env,const EventFilterNapi & in,napi_value & out)503 napi_status SetValue(napi_env env, const EventFilterNapi& in, napi_value& out)
504 {
505 LOG_DEBUG("EventFilterNapi -> napi_value ");
506 return napi_ok;
507 }
508
509 /* napi_value <-> EventType */
GetValue(napi_env env,napi_value in,EventType & out)510 napi_status GetValue(napi_env env, napi_value in, EventType& out)
511 {
512 LOG_DEBUG("napi_value -> EventType ");
513 return napi_ok;
514 }
515
SetValue(napi_env env,const EventType & in,napi_value & out)516 napi_status SetValue(napi_env env, const EventType& in, napi_value& out)
517 {
518 LOG_DEBUG("EventType -> napi_value ");
519 return napi_ok;
520 }
521
522 /* napi_value <-> Event */
GetValue(napi_env env,napi_value in,Event & out)523 napi_status GetValue(napi_env env, napi_value in, Event& out)
524 {
525 LOG_DEBUG("napi_value -> Event ");
526 GetNamedPropertyOptional(env, in, "id", out.id);
527 int type = -1;
528 napi_status status = GetNamedProperty(env, in, "type", type);
529 out.type = static_cast<EventType>(type);
530 CHECK_RETURN((status == napi_ok), "invalid entry type", status);
531 GetNamedPropertyOptional(env, in, "title", out.title);
532 GetNamedPropertyOptional(env, in, "location", out.location);
533 status = GetNamedProperty(env, in, "startTime", out.startTime);
534 CHECK_RETURN((status == napi_ok), "invalid entry startTime", status);
535 status = GetNamedProperty(env, in, "endTime", out.endTime);
536 CHECK_RETURN((status == napi_ok), "invalid entry endTime", status);
537 GetNamedPropertyOptional(env, in, "isAllDay", out.isAllDay);
538 GetNamedProperty(env, in, "attendee", out.attendees); // colud be empty not check result
539 GetNamedPropertyOptional(env, in, "timeZone", out.timeZone);
540 GetNamedPropertyOptional(env, in, "reminderTime", out.reminderTime);
541 GetNamedPropertyOptional(env, in, "recurrenceRule", out.recurrenceRule);
542 GetNamedPropertyOptional(env, in, "description", out.description);
543 GetNamedPropertyOptional(env, in, "service", out.service);
544 return status;
545 }
546
SetValue(napi_env env,const Event & in,napi_value & out)547 napi_status SetValue(napi_env env, const Event& in, napi_value& out)
548 {
549 LOG_DEBUG("Event -> napi_value");
550 napi_status status = napi_create_object(env, &out);
551 CHECK_RETURN((status == napi_ok), "invalid entry object", status);
552
553 status = SetNamedProperty(env, "id", in.id.value(), out);
554 CHECK_RETURN((status == napi_ok), "invalid entry id", status);
555
556 status = SetNamedProperty(env, "type", static_cast<int>(in.type), out);
557 CHECK_RETURN((status == napi_ok), "invalid entry type", status);
558
559 SetNamedPropertyOptional(env, "location", in.location, out);
560 SetNamedPropertyOptional(env, "title", in.title, out);
561
562 status = SetNamedProperty(env, "startTime", in.startTime, out);
563 CHECK_RETURN((status == napi_ok), "invalid entry startTime", status);
564
565 status = SetNamedProperty(env, "endTime", in.endTime, out);
566 CHECK_RETURN((status == napi_ok), "invalid entry endTime", status);
567
568 SetNamedPropertyOptional(env, "isAllDay", in.isAllDay, out);
569 if (!in.attendees.empty()) {
570 status = SetNamedProperty(env, "attendee", in.attendees, out);
571 CHECK_RETURN((status == napi_ok), "invalid entry attendee", status);
572 }
573 SetNamedPropertyOptional(env, "timeZone", in.timeZone, out);
574 SetNamedPropertyOptional(env, "reminderTime", in.reminderTime, out);
575 SetNamedPropertyOptional(env, "description", in.description, out);
576 SetNamedPropertyOptional(env, "service", in.service, out);
577 return status;
578 }
579
580 /* napi_value <-> std::vector<Event> */
GetValue(napi_env env,napi_value in,std::vector<Event> & out)581 napi_status GetValue(napi_env env, napi_value in, std::vector<Event>& out)
582 {
583 LOG_DEBUG("napi_value -> std::vector<Event>");
584 return GetValueArray(env, in, out);
585 }
586
SetValue(napi_env env,const std::vector<Event> & in,napi_value & out)587 napi_status SetValue(napi_env env, const std::vector<Event>& in, napi_value& out)
588 {
589 LOG_DEBUG("std::vector<Event> -> napi_value");
590 return SetValueArray(env, in, out);
591 }
592
593
DefineClass(napi_env env,const std::string & name,const napi_property_descriptor * properties,size_t count,napi_callback newcb)594 napi_value DefineClass(napi_env env, const std::string& name,
595 const napi_property_descriptor* properties, size_t count, napi_callback newcb)
596 {
597 // base64("calendar.calendarmanager") as rootPropName, i.e. global.<root>
598 const std::string rootPropName = "Y2FsZW5kYXIuY2FsZW5kYXJtYW5hZ2Vy";
599 napi_value root = nullptr;
600 bool hasRoot = false;
601 napi_value global = nullptr;
602 napi_get_global(env, &global);
603 napi_has_named_property(env, global, rootPropName.c_str(), &hasRoot);
604 if (hasRoot) {
605 napi_get_named_property(env, global, rootPropName.c_str(), &root);
606 } else {
607 napi_create_object(env, &root);
608 napi_set_named_property(env, global, rootPropName.c_str(), root);
609 }
610
611 std::string propName = "constructor_of_" + name;
612 napi_value constructor = nullptr;
613 bool hasProp = false;
614 napi_has_named_property(env, root, propName.c_str(), &hasProp);
615 if (hasProp) {
616 napi_get_named_property(env, root, propName.c_str(), &constructor);
617 if (constructor != nullptr) {
618 LOG_DEBUG("got calendar.calendarmanager.%{public}s as constructor", propName.c_str());
619 return constructor;
620 }
621 hasProp = false; // no constructor.
622 }
623
624 NAPI_CALL(env, napi_define_class(env, name.c_str(), name.size(), newcb, nullptr, count, properties, &constructor));
625 NAPI_ASSERT(env, constructor != nullptr, "napi_define_class failed!");
626
627 if (!hasProp) {
628 napi_set_named_property(env, root, propName.c_str(), constructor);
629 LOG_DEBUG("save constructor to calendar.calendarmanager.%{public}s", propName.c_str());
630 }
631 return constructor;
632 }
633
NewWithRef(napi_env env,size_t argc,napi_value * argv,void ** out,napi_value constructor)634 napi_ref NewWithRef(napi_env env, size_t argc, napi_value* argv, void** out, napi_value constructor)
635 {
636 napi_value object = nullptr;
637 napi_status status = napi_new_instance(env, constructor, argc, argv, &object);
638 CHECK_RETURN(status == napi_ok, "napi_new_instance failed", nullptr);
639 CHECK_RETURN(object != nullptr, "napi_new_instance failed", nullptr);
640
641 status = napi_unwrap(env, object, out);
642 CHECK_RETURN(status == napi_ok, "napi_unwrap failed", nullptr);
643 CHECK_RETURN(out != nullptr, "napi_unwrap failed", nullptr);
644
645 napi_ref ref = nullptr;
646 status = napi_create_reference(env, object, 1, &ref);
647 CHECK_RETURN(status == napi_ok, "napi_create_reference failed!", nullptr);
648 CHECK_RETURN(ref != nullptr, "napi_create_reference failed!", nullptr);
649 return ref;
650 }
651
Unwrap(napi_env env,napi_value in,void ** out,napi_value constructor)652 napi_status Unwrap(napi_env env, napi_value in, void** out, napi_value constructor)
653 {
654 if (constructor != nullptr) {
655 bool isInstance = false;
656 auto status = napi_instanceof(env, in, constructor, &isInstance);
657 if (status != napi_ok) {
658 LOG_ERROR("napi_instanceof failed");
659 return status;
660 }
661 if (!isInstance) {
662 LOG_ERROR("not a instance of *");
663 return napi_invalid_arg;
664 }
665 }
666 return napi_unwrap(env, in, out);
667 }
668
Equals(napi_env env,napi_value value,napi_ref copy)669 bool Equals(napi_env env, napi_value value, napi_ref copy)
670 {
671 if (copy == nullptr) {
672 return value == nullptr? true : false;
673 }
674
675 napi_value copyValue = nullptr;
676 napi_get_reference_value(env, copy, ©Value);
677
678 bool isEquals = false;
679 napi_strict_equals(env, value, copyValue, &isEquals);
680 return isEquals;
681 }
682 } // namespace OHOS::CalendarApi::NapiUtil
683