24
24
const {
25
25
ContextifyScript,
26
26
kParsingContext,
27
-
28
27
makeContext,
29
28
isContext : _isContext ,
30
29
} = process . binding ( 'contextify' ) ;
31
30
32
31
const {
33
32
ERR_INVALID_ARG_TYPE ,
34
- ERR_MISSING_ARGS
33
+ ERR_OUT_OF_RANGE
35
34
} = require ( 'internal/errors' ) . codes ;
36
-
37
- // The binding provides a few useful primitives:
38
- // - Script(code, { filename = "evalmachine.anonymous",
39
- // displayErrors = true } = {})
40
- // with methods:
41
- // - runInThisContext({ displayErrors = true } = {})
42
- // - runInContext(sandbox, { displayErrors = true, timeout = undefined } = {})
43
- // - makeContext(sandbox)
44
- // - isContext(sandbox)
45
- // From this we build the entire documented API.
35
+ const { isUint8Array } = require ( 'internal/util/types' ) ;
46
36
47
37
class Script extends ContextifyScript {
48
- constructor ( code , options ) {
38
+ constructor ( code , options = { } ) {
39
+ code = `${ code } ` ;
40
+ if ( typeof options === 'string' ) {
41
+ options = { filename : options } ;
42
+ }
43
+ if ( typeof options !== 'object' || options === null ) {
44
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , 'Object' , options ) ;
45
+ }
46
+
47
+ const {
48
+ filename = 'evalmachine.<anonymous>' ,
49
+ lineOffset = 0 ,
50
+ columnOffset = 0 ,
51
+ cachedData,
52
+ produceCachedData = false ,
53
+ [ kParsingContext ] : parsingContext
54
+ } = options ;
55
+
56
+ if ( typeof filename !== 'string' ) {
57
+ throw new ERR_INVALID_ARG_TYPE ( 'options.filename' , 'string' , filename ) ;
58
+ }
59
+ validateInteger ( lineOffset , 'options.lineOffset' ) ;
60
+ validateInteger ( columnOffset , 'options.columnOffset' ) ;
61
+ if ( cachedData !== undefined && ! isUint8Array ( cachedData ) ) {
62
+ throw new ERR_INVALID_ARG_TYPE ( 'options.cachedData' ,
63
+ [ 'Buffer' , 'Uint8Array' ] , cachedData ) ;
64
+ }
65
+ if ( typeof produceCachedData !== 'boolean' ) {
66
+ throw new ERR_INVALID_ARG_TYPE ( 'options.produceCachedData' , 'boolean' ,
67
+ produceCachedData ) ;
68
+ }
69
+
49
70
// Calling `ReThrow()` on a native TryCatch does not generate a new
50
71
// abort-on-uncaught-exception check. A dummy try/catch in JS land
51
72
// protects against that.
52
73
try {
53
- super ( code , options ) ;
74
+ super ( code ,
75
+ filename ,
76
+ lineOffset ,
77
+ columnOffset ,
78
+ cachedData ,
79
+ produceCachedData ,
80
+ parsingContext ) ;
54
81
} catch ( e ) {
55
82
throw e ; /* node-do-not-add-exception-line */
56
83
}
57
84
}
58
- }
59
85
60
- const realRunInThisContext = Script . prototype . runInThisContext ;
61
- const realRunInContext = Script . prototype . runInContext ;
86
+ runInThisContext ( options ) {
87
+ const { breakOnSigint, args } = getRunInContextArgs ( options ) ;
88
+ if ( breakOnSigint && process . listenerCount ( 'SIGINT' ) > 0 ) {
89
+ return sigintHandlersWrap ( super . runInThisContext , this , args ) ;
90
+ } else {
91
+ return super . runInThisContext ( ...args ) ;
92
+ }
93
+ }
62
94
63
- Script . prototype . runInThisContext = function ( options ) {
64
- if ( options && options . breakOnSigint && process . listenerCount ( 'SIGINT' ) > 0 ) {
65
- return sigintHandlersWrap ( realRunInThisContext , this , [ options ] ) ;
66
- } else {
67
- return realRunInThisContext . call ( this , options ) ;
95
+ runInContext ( contextifiedSandbox , options ) {
96
+ validateContext ( contextifiedSandbox ) ;
97
+ const { breakOnSigint, args } = getRunInContextArgs ( options ) ;
98
+ if ( breakOnSigint && process . listenerCount ( 'SIGINT' ) > 0 ) {
99
+ return sigintHandlersWrap ( super . runInContext , this ,
100
+ [ contextifiedSandbox , ...args ] ) ;
101
+ } else {
102
+ return super . runInContext ( contextifiedSandbox , ...args ) ;
103
+ }
68
104
}
69
- } ;
70
105
71
- Script . prototype . runInContext = function ( contextifiedSandbox , options ) {
72
- if ( options && options . breakOnSigint && process . listenerCount ( 'SIGINT' ) > 0 ) {
73
- return sigintHandlersWrap ( realRunInContext , this ,
74
- [ contextifiedSandbox , options ] ) ;
75
- } else {
76
- return realRunInContext . call ( this , contextifiedSandbox , options ) ;
106
+ runInNewContext ( sandbox , options ) {
107
+ const context = createContext ( sandbox , getContextOptions ( options ) ) ;
108
+ return this . runInContext ( context , options ) ;
77
109
}
78
- } ;
110
+ }
79
111
80
- Script . prototype . runInNewContext = function ( sandbox , options ) {
81
- const context = createContext ( sandbox , getContextOptions ( options ) ) ;
82
- return this . runInContext ( context , options ) ;
83
- } ;
112
+ function validateContext ( sandbox ) {
113
+ if ( typeof sandbox !== 'object' || sandbox === null ) {
114
+ throw new ERR_INVALID_ARG_TYPE ( 'contextifiedSandbox' , 'Object' , sandbox ) ;
115
+ }
116
+ if ( ! _isContext ( sandbox ) ) {
117
+ throw new ERR_INVALID_ARG_TYPE ( 'contextifiedSandbox' , 'vm.Context' ,
118
+ sandbox ) ;
119
+ }
120
+ }
121
+
122
+ function validateInteger ( prop , propName ) {
123
+ if ( ! Number . isInteger ( prop ) ) {
124
+ throw new ERR_INVALID_ARG_TYPE ( propName , 'integer' , prop ) ;
125
+ }
126
+ if ( ( prop >> 0 ) !== prop ) {
127
+ throw new ERR_OUT_OF_RANGE ( propName , '32-bit integer' , prop ) ;
128
+ }
129
+ }
84
130
85
131
function validateString ( prop , propName ) {
86
132
if ( prop !== undefined && typeof prop !== 'string' )
@@ -97,6 +143,39 @@ function validateObject(prop, propName) {
97
143
throw new ERR_INVALID_ARG_TYPE ( propName , 'Object' , prop ) ;
98
144
}
99
145
146
+ function getRunInContextArgs ( options = { } ) {
147
+ if ( typeof options !== 'object' || options === null ) {
148
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , 'Object' , options ) ;
149
+ }
150
+
151
+ let timeout = options . timeout ;
152
+ if ( timeout === undefined ) {
153
+ timeout = - 1 ;
154
+ } else if ( ! Number . isInteger ( timeout ) || timeout <= 0 ) {
155
+ throw new ERR_INVALID_ARG_TYPE ( 'options.timeout' , 'a positive integer' ,
156
+ timeout ) ;
157
+ }
158
+
159
+ const {
160
+ displayErrors = true ,
161
+ breakOnSigint = false
162
+ } = options ;
163
+
164
+ if ( typeof displayErrors !== 'boolean' ) {
165
+ throw new ERR_INVALID_ARG_TYPE ( 'options.displayErrors' , 'boolean' ,
166
+ displayErrors ) ;
167
+ }
168
+ if ( typeof breakOnSigint !== 'boolean' ) {
169
+ throw new ERR_INVALID_ARG_TYPE ( 'options.breakOnSigint' , 'boolean' ,
170
+ breakOnSigint ) ;
171
+ }
172
+
173
+ return {
174
+ breakOnSigint,
175
+ args : [ timeout , displayErrors , breakOnSigint ]
176
+ } ;
177
+ }
178
+
100
179
function getContextOptions ( options ) {
101
180
if ( options ) {
102
181
validateObject ( options . contextCodeGeneration ,
@@ -123,57 +202,43 @@ function getContextOptions(options) {
123
202
}
124
203
125
204
function isContext ( sandbox ) {
126
- if ( arguments . length < 1 ) {
127
- throw new ERR_MISSING_ARGS ( 'sandbox' ) ;
205
+ if ( typeof sandbox !== 'object' || sandbox === null ) {
206
+ throw new ERR_INVALID_ARG_TYPE ( 'sandbox' , 'Object' , sandbox ) ;
128
207
}
129
-
130
- if ( typeof sandbox !== 'object' && typeof sandbox !== 'function' ||
131
- sandbox === null ) {
132
- throw new ERR_INVALID_ARG_TYPE ( 'sandbox' , 'object' , sandbox ) ;
133
- }
134
-
135
208
return _isContext ( sandbox ) ;
136
209
}
137
210
138
211
let defaultContextNameIndex = 1 ;
139
- function createContext ( sandbox , options ) {
140
- if ( sandbox === undefined ) {
141
- sandbox = { } ;
142
- } else if ( isContext ( sandbox ) ) {
212
+ function createContext ( sandbox = { } , options = { } ) {
213
+ if ( isContext ( sandbox ) ) {
143
214
return sandbox ;
144
215
}
145
216
146
- if ( options !== undefined ) {
147
- if ( typeof options !== 'object' || options === null ) {
148
- throw new ERR_INVALID_ARG_TYPE ( 'options' , 'object' , options ) ;
149
- }
150
- validateObject ( options . codeGeneration , 'options.codeGeneration' ) ;
151
- options = {
152
- name : options . name ,
153
- origin : options . origin ,
154
- codeGeneration : typeof options . codeGeneration === 'object' ? {
155
- strings : options . codeGeneration . strings ,
156
- wasm : options . codeGeneration . wasm ,
157
- } : undefined ,
158
- } ;
159
- if ( options . codeGeneration !== undefined ) {
160
- validateBool ( options . codeGeneration . strings ,
161
- 'options.codeGeneration.strings' ) ;
162
- validateBool ( options . codeGeneration . wasm ,
163
- 'options.codeGeneration.wasm' ) ;
164
- }
165
- if ( options . name === undefined ) {
166
- options . name = `VM Context ${ defaultContextNameIndex ++ } ` ;
167
- } else if ( typeof options . name !== 'string' ) {
168
- throw new ERR_INVALID_ARG_TYPE ( 'options.name' , 'string' , options . name ) ;
169
- }
170
- validateString ( options . origin , 'options.origin' ) ;
171
- } else {
172
- options = {
173
- name : `VM Context ${ defaultContextNameIndex ++ } `
174
- } ;
217
+ if ( typeof options !== 'object' || options === null ) {
218
+ throw new ERR_INVALID_ARG_TYPE ( 'options' , 'Object' , options ) ;
175
219
}
176
- makeContext ( sandbox , options ) ;
220
+
221
+ const {
222
+ name = `VM Context ${ defaultContextNameIndex ++ } ` ,
223
+ origin,
224
+ codeGeneration
225
+ } = options ;
226
+
227
+ if ( typeof name !== 'string' ) {
228
+ throw new ERR_INVALID_ARG_TYPE ( 'options.name' , 'string' , options . name ) ;
229
+ }
230
+ validateString ( origin , 'options.origin' ) ;
231
+ validateObject ( codeGeneration , 'options.codeGeneration' ) ;
232
+
233
+ let strings = true ;
234
+ let wasm = true ;
235
+ if ( codeGeneration !== undefined ) {
236
+ ( { strings = true , wasm = true } = codeGeneration ) ;
237
+ validateBool ( strings , 'options.codeGeneration.strings' ) ;
238
+ validateBool ( wasm , 'options.codeGeneration.wasm' ) ;
239
+ }
240
+
241
+ makeContext ( sandbox , name , origin , strings , wasm ) ;
177
242
return sandbox ;
178
243
}
179
244
@@ -200,6 +265,7 @@ function sigintHandlersWrap(fn, thisArg, argsArray) {
200
265
}
201
266
202
267
function runInContext ( code , contextifiedSandbox , options ) {
268
+ validateContext ( contextifiedSandbox ) ;
203
269
if ( typeof options === 'string' ) {
204
270
options = {
205
271
filename : options ,
@@ -226,6 +292,9 @@ function runInNewContext(code, sandbox, options) {
226
292
}
227
293
228
294
function runInThisContext ( code , options ) {
295
+ if ( typeof options === 'string' ) {
296
+ options = { filename : options } ;
297
+ }
229
298
return createScript ( code , options ) . runInThisContext ( options ) ;
230
299
}
231
300
0 commit comments