Skip to content

Commit dc7fb7c

Browse files
authored
feat(plugin): Throw when calling illegal process functions (#83)
* add reading umask
1 parent 47f99db commit dc7fb7c

File tree

4 files changed

+53
-4
lines changed

4 files changed

+53
-4
lines changed

.eslintignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
__tests__/integration/rplugin/node/test
1+
__tests__/integration/rplugin/node/

__tests__/integration/rplugin/node/test_2/src/index.js

+10
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,16 @@ class Test {
4545
getGlobal(name) {
4646
return global[name];
4747
}
48+
49+
@NvimFunction('Illegal', { sync: true })
50+
illegalProcessCall() {
51+
process.chdir();
52+
}
53+
54+
@NvimFunction('Umask', { sync: true })
55+
umask(arg) {
56+
return process.umask(arg);
57+
}
4858
}
4959

5060
module.exports = Test;

src/host/factory.test.ts

+21-1
Original file line numberDiff line numberDiff line change
@@ -100,8 +100,9 @@ describe('Plugin Factory (decorator api)', () => {
100100
},
101101
{ type: 'function', name: 'Func', sync: true, opts: {} },
102102
{ type: 'function', name: 'Global', sync: true, opts: {} },
103+
{ type: 'function', name: 'Illegal', sync: true, opts: {} },
103104
];
104-
expect(pluginObj.specs).toEqual(expected);
105+
expect(pluginObj.specs).toEqual(expect.arrayContaining(expected));
105106
});
106107

107108
it('should collect the handlers from a plugin', async () => {
@@ -135,4 +136,23 @@ describe('Plugin Factory (decorator api)', () => {
135136
const plugin = loadPlugin(path.join(PLUGIN_PATH, 'test_2'), nvim, {});
136137
expect(plugin.nvim).toBe(nvim);
137138
});
139+
140+
it('cannot call illegal process functions', () => {
141+
const nvim = {};
142+
const plugin = loadPlugin(path.join(PLUGIN_PATH, 'test_2'), nvim, {});
143+
expect(plugin.functions.Illegal.fn).toThrow();
144+
});
145+
146+
it('cannot write to process.umask', () => {
147+
const nvim = {};
148+
const plugin = loadPlugin(path.join(PLUGIN_PATH, 'test_2'), nvim, {});
149+
expect(() => plugin.functions.Umask.fn(123)).toThrow();
150+
});
151+
152+
it('can read process.umask()', () => {
153+
const nvim = {};
154+
const plugin = loadPlugin(path.join(PLUGIN_PATH, 'test_2'), nvim, {});
155+
expect(() => plugin.functions.Umask.fn()).not.toThrow();
156+
expect(plugin.functions.Umask.fn()).toBeDefined();
157+
});
138158
});

src/host/factory.ts

+21-2
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ export type LoadPluginOptions = {
2828

2929
const Module: IModule = require('module');
3030

31-
const BLACKLISTED_GLOBALS = [
31+
const REMOVED_GLOBALS = [
3232
'reallyExit',
3333
'abort',
3434
'chdir',
@@ -44,6 +44,12 @@ const BLACKLISTED_GLOBALS = [
4444
'kill',
4545
];
4646

47+
function removedGlobalStub(name: string) {
48+
return () => {
49+
throw new Error(`process.${name}() is not allowed in Plugin sandbox`);
50+
};
51+
}
52+
4753
// @see node/lib/internal/module.js
4854
function makeRequireFunction() {
4955
const require: any = (p: string) => this.require(p);
@@ -116,9 +122,22 @@ function createSandbox(filename: string): ISandbox {
116122

117123
// patch `require` in sandbox to run loaded module in sandbox context
118124
// if you need any of these, it might be worth discussing spawning separate processes
119-
sandbox.process = <NodeJS.Process>omit(process, BLACKLISTED_GLOBALS);
125+
sandbox.process = <NodeJS.Process>omit(process, REMOVED_GLOBALS);
126+
127+
REMOVED_GLOBALS.forEach(name => {
128+
sandbox.process[name] = removedGlobalStub(name);
129+
});
120130

121131
const devNull = new DevNull();
132+
133+
// read-only umask
134+
sandbox.process.umask = (mask: number) => {
135+
if (typeof mask !== 'undefined') {
136+
throw new Error('Cannot use process.umask() to change mask (read-only)');
137+
}
138+
return process.umask();
139+
};
140+
122141
sandbox.process.stdin = devNull;
123142
sandbox.process.stdout = devNull;
124143
sandbox.process.stderr = devNull;

0 commit comments

Comments
 (0)