Skip to content

Commit 8068577

Browse files
committed
src: do not add .domain to promises in VM contexts
The promises are still tracked, and their handlers will still execute in the correct domain. The creation domain is simply hidden. PR-URL: #15695 Fixes: #15673 Reviewed-By: Anna Henningsen <[email protected]> Reviewed-By: James M Snell <[email protected]> Reviewed-By: Colin Ihrig <[email protected]> Reviewed-By: Ben Noordhuis <[email protected]>
1 parent a3cd8ed commit 8068577

File tree

4 files changed

+37
-9
lines changed

4 files changed

+37
-9
lines changed

doc/api/domain.md

+6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,12 @@
11
# Domain
22
<!-- YAML
33
changes:
4+
- version: REPLACEME
5+
pr-url: https://github.com/nodejs/node/pull/REPLACEME
6+
description: Any `Promise`s created in VM contexts no longer have a
7+
`.domain` property. Their handlers are still executed in the
8+
proper domain, however, and `Promise`s created in the main
9+
context still possess a `.domain` property.
410
- version: v8.0.0
511
pr-url: https://github.com/nodejs/node/pull/12489
612
description: Handlers for `Promise`s are now invoked in the domain in which

src/env.h

+1
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,7 @@ class ModuleWrap;
9393
V(npn_buffer_private_symbol, "node:npnBuffer") \
9494
V(processed_private_symbol, "node:processed") \
9595
V(selected_npn_buffer_private_symbol, "node:selectedNpnBuffer") \
96+
V(domain_private_symbol, "node:domain") \
9697

9798
// Strings are per-isolate primitives but Environment proxies them
9899
// for the sake of convenience. Strings should be ASCII-only.

src/node.cc

+23-6
Original file line numberDiff line numberDiff line change
@@ -1156,8 +1156,19 @@ bool ShouldAbortOnUncaughtException(Isolate* isolate) {
11561156
}
11571157

11581158

1159+
Local<Value> GetDomainProperty(Environment* env, Local<Object> object) {
1160+
Local<Value> domain_v =
1161+
object->GetPrivate(env->context(), env->domain_private_symbol())
1162+
.ToLocalChecked();
1163+
if (domain_v->IsObject()) {
1164+
return domain_v;
1165+
}
1166+
return object->Get(env->context(), env->domain_string()).ToLocalChecked();
1167+
}
1168+
1169+
11591170
void DomainEnter(Environment* env, Local<Object> object) {
1160-
Local<Value> domain_v = object->Get(env->domain_string());
1171+
Local<Value> domain_v = GetDomainProperty(env, object);
11611172
if (domain_v->IsObject()) {
11621173
Local<Object> domain = domain_v.As<Object>();
11631174
Local<Value> enter_v = domain->Get(env->enter_string());
@@ -1172,7 +1183,7 @@ void DomainEnter(Environment* env, Local<Object> object) {
11721183

11731184

11741185
void DomainExit(Environment* env, v8::Local<v8::Object> object) {
1175-
Local<Value> domain_v = object->Get(env->domain_string());
1186+
Local<Value> domain_v = GetDomainProperty(env, object);
11761187
if (domain_v->IsObject()) {
11771188
Local<Object> domain = domain_v.As<Object>();
11781189
Local<Value> exit_v = domain->Get(env->exit_string());
@@ -1194,10 +1205,16 @@ void DomainPromiseHook(PromiseHookType type,
11941205
Local<Context> context = env->context();
11951206

11961207
if (type == PromiseHookType::kInit && env->in_domain()) {
1197-
promise->Set(context,
1198-
env->domain_string(),
1199-
env->domain_array()->Get(context,
1200-
0).ToLocalChecked()).FromJust();
1208+
Local<Value> domain_obj =
1209+
env->domain_array()->Get(context, 0).ToLocalChecked();
1210+
if (promise->CreationContext() == context) {
1211+
promise->Set(context, env->domain_string(), domain_obj).FromJust();
1212+
} else {
1213+
// Do not expose object from another context publicly in promises created
1214+
// in non-main contexts.
1215+
promise->SetPrivate(context, env->domain_private_symbol(), domain_obj)
1216+
.FromJust();
1217+
}
12011218
return;
12021219
}
12031220

test/parallel/test-domain-promise.js

+7-3
Original file line numberDiff line numberDiff line change
@@ -31,9 +31,13 @@ common.crashOnUnhandledRejection();
3131
const d = domain.create();
3232

3333
d.run(common.mustCall(() => {
34-
vm.runInNewContext(`Promise.resolve().then(common.mustCall(() => {
35-
assert.strictEqual(process.domain, d);
36-
}));`, { common, assert, process, d });
34+
vm.runInNewContext(`
35+
const promise = Promise.resolve();
36+
assert.strictEqual(promise.domain, undefined);
37+
promise.then(common.mustCall(() => {
38+
assert.strictEqual(process.domain, d);
39+
}));
40+
`, { common, assert, process, d });
3741
}));
3842
}
3943

0 commit comments

Comments
 (0)