Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 12 additions & 3 deletions lib/repl.js
Original file line number Diff line number Diff line change
Expand Up @@ -1836,7 +1836,7 @@ function includesProxiesOrGetters(expr, exprStr, evalFn, ctx, callback) {
if (astProp.type === 'Literal') {
// We have something like `obj['foo'].x` where `x` is the literal

if (isProxy(obj[astProp.value])) {
if (safeIsProxyAccess(obj, astProp.value)) {
return cb(true);
}

Expand All @@ -1854,7 +1854,7 @@ function includesProxiesOrGetters(expr, exprStr, evalFn, ctx, callback) {
) {
// We have something like `obj.foo.x` where `foo` is the identifier

if (isProxy(obj[astProp.name])) {
if (safeIsProxyAccess(obj, astProp.name)) {
return cb(true);
}

Expand All @@ -1881,7 +1881,7 @@ function includesProxiesOrGetters(expr, exprStr, evalFn, ctx, callback) {
}

if (typeof evaledProp === 'string') {
if (isProxy(obj[evaledProp])) {
if (safeIsProxyAccess(obj, evaledProp)) {
return cb(true);
}

Expand All @@ -1898,6 +1898,15 @@ function includesProxiesOrGetters(expr, exprStr, evalFn, ctx, callback) {
);
}

function safeIsProxyAccess(obj, prop) {
// Accessing `prop` may trigger a getter that throws, so we use try-catch to guard against it
try {
return isProxy(obj[prop]);
} catch {
return false;
}
}

return callback(false);
}

Expand Down
51 changes: 51 additions & 0 deletions test/parallel/test-repl-tab-complete-getter-error.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
'use strict';

const common = require('../common');
const repl = require('repl');
const ArrayStream = require('../common/arraystream');
const assert = require('assert');

(async function() {
await runTest();
})().then(common.mustCall());

async function runTest() {
const input = new ArrayStream();
const output = new ArrayStream();

const replServer = repl.start({
prompt: '',
input,
output: output,
allowBlockingCompletions: true,
terminal: true
});

replServer._domain.on('error', (e) => {
assert.fail(`Error in REPL domain: ${e}`);
});

await new Promise((resolve, reject) => {
replServer.eval(`
const getNameText = () => "name";
const foo = { get name() { throw new Error(); } };
`, replServer.context, '', (err) => {
if (err) {
reject(err);
} else {
resolve();
}
});
});

['foo.name.', 'foo["name"].', 'foo[getNameText()].'].forEach((test) => {
replServer.complete(
test,
common.mustCall((error, data) => {
assert.strictEqual(error, null);
assert.strictEqual(data.length, 2);
assert.strictEqual(data[1], test);
})
);
});
}
Loading