0
0
mirror of https://github.com/nodejs/node.git synced 2024-11-24 20:29:23 +01:00
nodejs/test/parallel/test-vm-global-get-own.js
Nicolas DUBIEN 06a211b558
vm: properly handle defining symbol props
This PR is a follow-up of #46615.

When bumping V8 for node 18, various bugs appeared around vm.
Some have been fixed along the flow, but there is at least one
remaining and described at:
https://github.com/facebook/jest/issues/13338.

The current fix does nothing on node 20: the new bump of v8 done for
node 20 seems to fix it. But it would fix the problem in both node 18
and node 19. I confirmed it would fix node 19 by cherry-picking the
commit on v19.x-staging and launching `make -j4 jstest` on it.

PR-URL: https://github.com/nodejs/node/pull/47572
Reviewed-By: James M Snell <jasnell@gmail.com>
2023-05-26 19:37:36 +00:00

106 lines
3.7 KiB
JavaScript

'use strict';
require('../common');
const assert = require('assert');
const vm = require('vm');
// These assertions check that we can set new keys to the global context,
// get them back and also list them via getOwnProperty* or in.
//
// Related to:
// - https://github.com/nodejs/node/issues/45983
const global = vm.runInContext('this', vm.createContext());
function runAssertions(data, property, viaDefine, value1, value2, value3) {
// Define the property for the first time
setPropertyAndAssert(data, property, viaDefine, value1);
// Update the property
setPropertyAndAssert(data, property, viaDefine, value2);
// Delete the property
deletePropertyAndAssert(data, property);
// Re-define the property
setPropertyAndAssert(data, property, viaDefine, value3);
// Delete the property again
deletePropertyAndAssert(data, property);
}
const fun1 = () => 1;
const fun2 = () => 2;
const fun3 = () => 3;
function runAssertionsOnSandbox(builder) {
const sandboxContext = vm.createContext({ runAssertions, fun1, fun2, fun3 });
vm.runInContext(builder('this'), sandboxContext);
vm.runInContext(builder('{}'), sandboxContext);
}
// Assertions on: define property
runAssertions(global, 'toto', true, 1, 2, 3);
runAssertions(global, Symbol.for('toto'), true, 1, 2, 3);
runAssertions(global, 'tutu', true, fun1, fun2, fun3);
runAssertions(global, Symbol.for('tutu'), true, fun1, fun2, fun3);
runAssertions(global, 'tyty', true, fun1, 2, 3);
runAssertions(global, Symbol.for('tyty'), true, fun1, 2, 3);
// Assertions on: direct assignment
runAssertions(global, 'titi', false, 1, 2, 3);
runAssertions(global, Symbol.for('titi'), false, 1, 2, 3);
runAssertions(global, 'tata', false, fun1, fun2, fun3);
runAssertions(global, Symbol.for('tata'), false, fun1, fun2, fun3);
runAssertions(global, 'tztz', false, fun1, 2, 3);
runAssertions(global, Symbol.for('tztz'), false, fun1, 2, 3);
// Assertions on: define property from sandbox
runAssertionsOnSandbox(
(variable) => `
runAssertions(${variable}, 'toto', true, 1, 2, 3);
runAssertions(${variable}, Symbol.for('toto'), true, 1, 2, 3);
runAssertions(${variable}, 'tutu', true, fun1, fun2, fun3);
runAssertions(${variable}, Symbol.for('tutu'), true, fun1, fun2, fun3);
runAssertions(${variable}, 'tyty', true, fun1, 2, 3);
runAssertions(${variable}, Symbol.for('tyty'), true, fun1, 2, 3);`
);
// Assertions on: direct assignment from sandbox
runAssertionsOnSandbox(
(variable) => `
runAssertions(${variable}, 'titi', false, 1, 2, 3);
runAssertions(${variable}, Symbol.for('titi'), false, 1, 2, 3);
runAssertions(${variable}, 'tata', false, fun1, fun2, fun3);
runAssertions(${variable}, Symbol.for('tata'), false, fun1, fun2, fun3);
runAssertions(${variable}, 'tztz', false, fun1, 2, 3);
runAssertions(${variable}, Symbol.for('tztz'), false, fun1, 2, 3);`
);
// Helpers
// Set the property on data and assert it worked
function setPropertyAndAssert(data, property, viaDefine, value) {
if (viaDefine) {
Object.defineProperty(data, property, {
enumerable: true,
writable: true,
value: value,
configurable: true,
});
} else {
data[property] = value;
}
assert.strictEqual(data[property], value);
assert.ok(property in data);
if (typeof property === 'string') {
assert.ok(Object.getOwnPropertyNames(data).includes(property));
} else {
assert.ok(Object.getOwnPropertySymbols(data).includes(property));
}
}
// Delete the property from data and assert it worked
function deletePropertyAndAssert(data, property) {
delete data[property];
assert.strictEqual(data[property], undefined);
assert.ok(!(property in data));
assert.ok(!Object.getOwnPropertyNames(data).includes(property));
assert.ok(!Object.getOwnPropertySymbols(data).includes(property));
}