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