• 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 "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, &copyValue);
677 
678     bool isEquals = false;
679     napi_strict_equals(env, value, copyValue, &isEquals);
680     return isEquals;
681 }
682 } // namespace OHOS::CalendarApi::NapiUtil
683