Skip to content

Commit 49cd4fa

Browse files
mafintoshMylesBorins
authored andcommittedApr 16, 2018
n-api: add napi_fatal_exception
Add function to trigger and uncaught exception. Useful if an async callback throws an exception with no way to recover. Backport-PR-URL: #19447 PR-URL: #19337 Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Colin Ihrig <cjihrig@gmail.com> Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Michael Dawson <michael_dawson@ca.ibm.com>
1 parent eb29266 commit 49cd4fa

File tree

8 files changed

+104
-4
lines changed

8 files changed

+104
-4
lines changed
 

‎doc/api/n-api.md

+14
Original file line numberDiff line numberDiff line change
@@ -539,6 +539,20 @@ This API returns true if an exception is pending.
539539

540540
This API can be called even if there is a pending JavaScript exception.
541541

542+
#### napi_fatal_exception
543+
<!-- YAML
544+
added: REPLACEME
545+
-->
546+
```C
547+
napi_status napi_fatal_exception(napi_env env, napi_value err);
548+
```
549+
550+
- `[in] env`: The environment that the API is invoked under.
551+
- `[in] err`: The error you want to pass to `uncaughtException`.
552+
553+
Trigger an `uncaughtException` in JavaScript. Useful if an async
554+
callback throws an exception with no way to recover.
555+
542556
### Fatal Errors
543557

544558
In the event of an unrecoverable error in a native module, a fatal error can be

‎src/node_api.cc

+20-4
Original file line numberDiff line numberDiff line change
@@ -293,6 +293,13 @@ v8::Local<v8::Value> V8LocalValueFromJsValue(napi_value v) {
293293
return local;
294294
}
295295

296+
static inline void trigger_fatal_exception(
297+
napi_env env, v8::Local<v8::Value> local_err) {
298+
v8::Local<v8::Message> local_msg =
299+
v8::Exception::CreateMessage(env->isolate, local_err);
300+
node::FatalException(env->isolate, local_err, local_msg);
301+
}
302+
296303
static inline napi_status V8NameFromPropertyDescriptor(napi_env env,
297304
const napi_property_descriptor* p,
298305
v8::Local<v8::Name>* result) {
@@ -967,6 +974,16 @@ napi_status napi_get_last_error_info(napi_env env,
967974
return napi_ok;
968975
}
969976

977+
napi_status napi_fatal_exception(napi_env env, napi_value err) {
978+
NAPI_PREAMBLE(env);
979+
CHECK_ARG(env, err);
980+
981+
v8::Local<v8::Value> local_err = v8impl::V8LocalValueFromJsValue(err);
982+
v8impl::trigger_fatal_exception(env, local_err);
983+
984+
return napi_clear_last_error(env);
985+
}
986+
970987
NAPI_NO_RETURN void napi_fatal_error(const char* location,
971988
size_t location_len,
972989
const char* message,
@@ -3457,10 +3474,9 @@ class Work : public node::AsyncResource {
34573474
// report it as a fatal exception. (There is no JavaScript on the
34583475
// callstack that can possibly handle it.)
34593476
if (!env->last_exception.IsEmpty()) {
3460-
v8::TryCatch try_catch(env->isolate);
3461-
env->isolate->ThrowException(
3462-
v8::Local<v8::Value>::New(env->isolate, env->last_exception));
3463-
node::FatalException(env->isolate, try_catch);
3477+
v8::Local<v8::Value> local_err = v8::Local<v8::Value>::New(
3478+
env->isolate, env->last_exception);
3479+
v8impl::trigger_fatal_exception(env, local_err);
34643480
}
34653481
}
34663482
}

‎src/node_api.h

+2
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,8 @@ NAPI_EXTERN napi_status
103103
napi_get_last_error_info(napi_env env,
104104
const napi_extended_error_info** result);
105105

106+
NAPI_EXTERN napi_status napi_fatal_exception(napi_env env, napi_value err);
107+
106108
NAPI_EXTERN NAPI_NO_RETURN void napi_fatal_error(const char* location,
107109
size_t location_len,
108110
const char* message,

‎src/node_internals.h

+5
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,11 @@ void GetSockOrPeerName(const v8::FunctionCallbackInfo<v8::Value>& args) {
118118
args.GetReturnValue().Set(err);
119119
}
120120

121+
void FatalException(v8::Isolate* isolate,
122+
v8::Local<v8::Value> error,
123+
v8::Local<v8::Message> message);
124+
125+
121126
void SignalExit(int signo);
122127
#ifdef __POSIX__
123128
void RegisterSignalHandler(int signal,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
'use strict';
2+
const common = require('../../common');
3+
const assert = require('assert');
4+
const test_async = require(`./build/${common.buildType}/test_async`);
5+
6+
process.on('uncaughtException', common.mustCall(function(err) {
7+
try {
8+
throw new Error('should not fail');
9+
} catch (err) {
10+
assert.strictEqual(err.message, 'should not fail');
11+
}
12+
assert.strictEqual(err.message, 'uncaught');
13+
}));
14+
15+
// Successful async execution and completion callback.
16+
test_async.Test(5, {}, common.mustCall(function() {
17+
throw new Error('uncaught');
18+
}));
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"targets": [
3+
{
4+
"target_name": "test_fatal_exception",
5+
"sources": [ "test_fatal_exception.c" ]
6+
}
7+
]
8+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
'use strict';
2+
const common = require('../../common');
3+
const assert = require('assert');
4+
const test_fatal = require(`./build/${common.buildType}/test_fatal_exception`);
5+
6+
process.on('uncaughtException', common.mustCall(function(err) {
7+
assert.strictEqual(err.message, 'fatal error');
8+
}));
9+
10+
const err = new Error('fatal error');
11+
test_fatal.Test(err);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
#include <node_api.h>
2+
#include "../common.h"
3+
4+
napi_value Test(napi_env env, napi_callback_info info) {
5+
napi_value err;
6+
size_t argc = 1;
7+
8+
NAPI_CALL(env, napi_get_cb_info(env, info, &argc, &err, NULL, NULL));
9+
10+
NAPI_CALL(env, napi_fatal_exception(env, err));
11+
12+
return NULL;
13+
}
14+
15+
napi_value Init(napi_env env, napi_value exports) {
16+
napi_property_descriptor properties[] = {
17+
DECLARE_NAPI_PROPERTY("Test", Test),
18+
};
19+
20+
NAPI_CALL(env, napi_define_properties(
21+
env, exports, sizeof(properties) / sizeof(*properties), properties));
22+
23+
return exports;
24+
}
25+
26+
NAPI_MODULE(NODE_GYP_MODULE_NAME, Init)

0 commit comments

Comments
 (0)
Please sign in to comment.