• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# Node-API Development Specifications
2
3
4## Lifecycle Management
5
6**[Rule]** Properly use **napi_open_handle_scope** and **napi_close_handle_scope** to minimize the lifecycle of **napi_value** and avoid memory leakage.
7
8Each **napi_value** belongs to a specific **HandleScope**, which is opened and closed by **napi_open_handle_scope** and **napi_close_handle_scope**, respectively. After a **HandleScope** is closed, its **napi_value** is automatically released.
9
10**Example (recommended)**
11
12```
13// When the Node-API interface is frequently called to create JS objects in the for loop, use handle_scope to release resources in a timely manner when they are no longer used.
14// In the following example, the lifecycle of the local variable res ends at the end of each loop. Therefore, scope is used to release the JS object in time and prevent memory leakage.
15for (int i = 0; i < 100000; i++) {
16    napi_handle_scope scope = nullptr;
17    napi_open_handle_scope(env, &scope);
18    if (scope == nullptr) {
19        return;
20    }
21    napi_value res;
22    napi_create_object(env, &res);
23    napi_close_handle_scope(env, scope);
24}
25```
26
27## Context Sensitive
28
29**[Rule]** Do not use Node-API to access JS objects across engine instances.
30
31An engine instance is an independent running environment. Operations such as creating and accessing a JS object must be performed in the same engine instance. If an object is operated in different engine instances, the application may crash. An engine instance is represented as a value of **napi_env** in APIs.
32
33**Example (not recommended)**
34
35```
36// Create a string object with value of "bar" in env1.
37napi_create_string_utf8(env1, "bar", NAPI_AUTO_LENGTH, &string);
38// Create an object in env2 and set the string object to this object.
39napi_status status = napi_create_object(env2, &object);
40if (status != napi_ok) {
41    napi_throw_error(env, ...);
42    return;
43}
44
45status = napi_set_named_property(env2, object, "foo", string);
46if (status != napi_ok) {
47    napi_throw_error(env, ...);
48    return;
49}
50```
51
52JS objects belong to a specific **napi_env**. Therefore, you cannot set an object of env1 to an object of env2. If the object of env1 is accessed in env2, the application may crash.
53
54## Exception Handling
55
56**[Suggestion]** Any exception occurred in a Node-API call should be handled in a timely manner. Otherwise, unexpected behavior may occur.
57
58**Example (recommended)**
59
60```
61// 1. Create an object.
62napi_status status = napi_create_object(env, &object);
63if (status != napi_ok) {
64    napi_throw_error(env, ...);
65    return;
66}
67// 2. Create a property.
68status = napi_create_string_utf8(env, "bar", NAPI_AUTO_LENGTH, &string);
69if (status != napi_ok) {
70    napi_throw_error(env, ...);
71    return;
72}
73// 3. Set the result of step 2 to the value of the object property foo.
74status = napi_set_named_property(env, object, "foo", string);
75if (status != napi_ok) {
76    napi_throw_error(env, ...);
77    return;
78}
79```
80
81In this example, if an exception occurs in step 1 or step 2, step 3 will not be performed. Step 3 will be performed only when napi_ok is returned in steps 1 and 2.
82
83## Asynchronous Tasks
84
85**[Rule]** When the **uv_queue_work** method is called to throw a work to a JS thread for execution, use **napi_handle_scope** to manage the lifecycle of **napi_value** created by the JS callback.
86
87The Node-API framework will not be used when the **uv_queue_work** method is called. In this case, you must use **napi_handle_scope** to manage the lifecycle of **napi_value**.
88
89**Example (recommended)**
90
91```
92void callbackTest(CallbackContext* context)
93{
94    uv_loop_s* loop = nullptr;
95    napi_get_uv_event_loop(context->env, &loop);
96    uv_work_t* work = new uv_work_t;
97    context->retData = 1;
98    work->data = (void*)context;
99    uv_queue_work(
100        loop, work, [](uv_work_t* work) {},
101        // using callback function back to JS thread
102        [](uv_work_t* work, int status) {
103            CallbackContext* context = (CallbackContext*)work->data;
104            napi_handle_scope scope = nullptr; napi_open_handle_scope(context->env, &scope);
105            if (scope == nullptr) {
106                return;
107            }
108            napi_value callback = nullptr;
109            napi_get_reference_value(context->env, context->callbackRef, &callback);
110            napi_value retArg;
111            napi_create_int32(context->env, context->retData, &retArg);
112            napi_value ret;
113            napi_call_function(context->env, nullptr, callback, 1, &retArg, &ret);
114            napi_delete_reference(context->env, context->callbackRef);
115            napi_close_handle_scope(context->env, scope);
116            if (work != nullptr) {
117                delete work;
118            }
119            delete context;
120        }
121    );
122}
123```
124
125## Object Wrapping
126
127**[Rule]** If the value of the last parameter **result** is not **nullptr** in **napi_wrap()** , use **napi_remove_wrap()** at a proper time to delete the created **napi_ref**.
128
129The **napi_wrap** interface is defined as follows:
130
131```
132napi_wrap(napi_env env, napi_value js_object, void* native_object, napi_finalize finalize_cb, void* finalize_hint, napi_ref* result)
133```
134
135When the last parameter **result** is not null, the Node-API framework creates an **napi_ref** object pointing to **js_object**. You need to manage the lifecycle of **js_object**. Specifically, use **napi_remove_wrap** to delete **napi_ref** at a proper time so that the garbage collector (GC) can release **js_object** and trigger the destructor **finalize_cb** bound to the C++ object **native_object**.
136
137Generally, you can directly pass in **nullptr** for the last parameter **result**.
138
139**Example (recommended)**
140
141```
142// Case 1: Pass in nullptr via the last parameter in napi_wrap. In this case, the created napi_ref is a weak reference, which is managed by the system and does not need manual release.
143napi_wrap(env, jsobject, nativeObject, cb, nullptr, nullptr);
144
145// Case 2: The last parameter in napi_wrap is not nullptr. In this case, the returned napi_ref is a strong reference and needs to be manually released. Otherwise, memory leakage may occur.
146napi_ref result;
147napi_wrap(env, jsobject, nativeObject, cb, nullptr, &result);
148// When js_object and result are no longer used, call napi_remove_wrap to release result.
149napi_value result1;
150napi_remove_wrap(env, jsobject, result1);
151```
152
153
154## Data Conversion
155
156**[Suggestion]** Minimize the number of data conversions and avoid unnecessary replication.
157
158- Frequent data conversion affects performance. You are advised to use batch data processing or optimize the data structs to improve performance.
159- During data conversion, use Node-API to access the original data instead of creating a copy.
160- For the data that may be used in multiple conversions, store it in a buffer to avoid repeated data conversions. In this way, unnecessary calculations can be reduced, leading to better performance.
161
162## Others
163
164**[Rule]** Manual release is not allowed for the third parameter **data** in **napi_get_arraybuffer_info**. Its lifecycle is managed by the engine.
165
166The **napi_get_arraybuffer_info** interface is defined as follows:
167
168```
169napi_get_arraybuffer_info(napi_env env, napi_value arraybuffer, void** data, size_t* byte_length)
170```
171
172The parameter **data** specifies the buffer header pointer to ArrayBuffer. This buffer can be read and written in the given range but cannot be released. The buffer memory is managed by the ArrayBuffer Allocator in the engine and is released with the lifecycle of the JS object **ArrayBuffer**.
173
174**Example (not recommended)**
175
176```
177void* arrayBufferPtr = nullptr;
178napi_value arrayBuffer = nullptr;
179size_t createBufferSize = ARRAY_BUFFER_SIZE;
180napi_status verification = napi_create_arraybuffer(env, createBufferSize, &arrayBufferPtr, &arrayBuffer);
181size_t arrayBufferSize;
182napi_status result = napi_get_arraybuffer_info(env, arrayBuffer, &arrayBufferPtr, &arrayBufferSize);
183delete arrayBufferPtr; // This operation is not allowed and may cause a double free of the buffer. The lifecycle of the created arrayBufferPtr is managed by the engine and cannot be manually deleted.
184```
185
186**[Suggestion]** Properly use **napi_object_freeze** and **napi_object_seal**.
187
188**napi_object_freeze** is equivalent to **Object.freeze**. After an object is frozen, all its properties are immutable. **napi_object_seal** is equivalent to **Object.seal**. After an object is sealed, no properties can be added or deleted, but the existing property values are mutable.
189
190If the semantics are violated in strict mode (default), an error will be thrown.
191