]>
jfr.im git - yt-dlp.git/blob - test/test_jsinterp.py
3 # Allow direct execution
8 sys
.path
.insert(0, os
.path
.dirname(os
.path
.dirname(os
.path
.abspath(__file__
))))
13 from yt_dlp
.jsinterp
import JS_Undefined
, JSInterpreter
16 class TestJSInterpreter(unittest
.TestCase
):
18 jsi
= JSInterpreter('function x(){;}')
19 self
.assertEqual(jsi
.call_function('x'), None)
21 jsi
= JSInterpreter('function x3(){return 42;}')
22 self
.assertEqual(jsi
.call_function('x3'), 42)
24 jsi
= JSInterpreter('function x3(){42}')
25 self
.assertEqual(jsi
.call_function('x3'), None)
27 jsi
= JSInterpreter('var x5 = function(){return 42;}')
28 self
.assertEqual(jsi
.call_function('x5'), 42)
31 jsi
= JSInterpreter('function x4(a){return 2*a+1;}')
32 self
.assertEqual(jsi
.call_function('x4', 3), 7)
34 def test_empty_return(self
):
35 jsi
= JSInterpreter('function f(){return; y()}')
36 self
.assertEqual(jsi
.call_function('f'), None)
38 def test_morespace(self
):
39 jsi
= JSInterpreter('function x (a) { return 2 * a + 1 ; }')
40 self
.assertEqual(jsi
.call_function('x', 3), 7)
42 jsi
= JSInterpreter('function f () { x = 2 ; return x; }')
43 self
.assertEqual(jsi
.call_function('f'), 2)
45 def test_strange_chars(self
):
46 jsi
= JSInterpreter('function $_xY1 ($_axY1) { var $_axY2 = $_axY1 + 1; return $_axY2; }')
47 self
.assertEqual(jsi
.call_function('$_xY1', 20), 21)
49 def test_operators(self
):
50 jsi
= JSInterpreter('function f(){return 1 << 5;}')
51 self
.assertEqual(jsi
.call_function('f'), 32)
53 jsi
= JSInterpreter('function f(){return 2 ** 5}')
54 self
.assertEqual(jsi
.call_function('f'), 32)
56 jsi
= JSInterpreter('function f(){return 19 & 21;}')
57 self
.assertEqual(jsi
.call_function('f'), 17)
59 jsi
= JSInterpreter('function f(){return 11 >> 2;}')
60 self
.assertEqual(jsi
.call_function('f'), 2)
62 jsi
= JSInterpreter('function f(){return []? 2+3: 4;}')
63 self
.assertEqual(jsi
.call_function('f'), 5)
65 jsi
= JSInterpreter('function f(){return 1 == 2}')
66 self
.assertEqual(jsi
.call_function('f'), False)
68 jsi
= JSInterpreter('function f(){return 0 && 1 || 2;}')
69 self
.assertEqual(jsi
.call_function('f'), 2)
71 jsi
= JSInterpreter('function f(){return 0 ?? 42;}')
72 self
.assertEqual(jsi
.call_function('f'), 0)
74 jsi
= JSInterpreter('function f(){return "life, the universe and everything" < 42;}')
75 self
.assertFalse(jsi
.call_function('f'))
77 def test_array_access(self
):
78 jsi
= JSInterpreter('function f(){var x = [1,2,3]; x[0] = 4; x[0] = 5; x[2.0] = 7; return x;}')
79 self
.assertEqual(jsi
.call_function('f'), [5, 2, 7])
81 def test_parens(self
):
82 jsi
= JSInterpreter('function f(){return (1) + (2) * ((( (( (((((3)))))) )) ));}')
83 self
.assertEqual(jsi
.call_function('f'), 7)
85 jsi
= JSInterpreter('function f(){return (1 + 2) * 3;}')
86 self
.assertEqual(jsi
.call_function('f'), 9)
88 def test_quotes(self
):
89 jsi
= JSInterpreter(R
'function f(){return "a\"\\("}')
90 self
.assertEqual(jsi
.call_function('f'), R
'a"\(')
92 def test_assignments(self
):
93 jsi
= JSInterpreter('function f(){var x = 20; x = 30 + 1; return x;}')
94 self
.assertEqual(jsi
.call_function('f'), 31)
96 jsi
= JSInterpreter('function f(){var x = 20; x += 30 + 1; return x;}')
97 self
.assertEqual(jsi
.call_function('f'), 51)
99 jsi
= JSInterpreter('function f(){var x = 20; x -= 30 + 1; return x;}')
100 self
.assertEqual(jsi
.call_function('f'), -11)
102 def test_comments(self
):
103 'Skipping: Not yet fully implemented'
105 jsi
= JSInterpreter('''
113 self
.assertEqual(jsi
.call_function('x'), 52)
115 jsi
= JSInterpreter('''
118 var y = 1 /* comment */ + 2;
122 self
.assertEqual(jsi
.call_function('f'), 3)
124 def test_precedence(self
):
125 jsi
= JSInterpreter('''
127 var a = [10, 20, 30, 40, 50];
132 self
.assertEqual(jsi
.call_function('x'), [20, 20, 30, 40, 50])
134 def test_builtins(self
):
135 jsi
= JSInterpreter('''
136 function x() { return NaN }
138 self
.assertTrue(math
.isnan(jsi
.call_function('x')))
140 jsi
= JSInterpreter('''
141 function x() { return new Date('Wednesday 31 December 1969 18:01:26 MDT') - 0; }
143 self
.assertEqual(jsi
.call_function('x'), 86000)
144 jsi
= JSInterpreter('''
145 function x(dt) { return new Date(dt) - 0; }
147 self
.assertEqual(jsi
.call_function('x', 'Wednesday 31 December 1969 18:01:26 MDT'), 86000)
150 jsi
= JSInterpreter('''
151 function x() { return 2; }
152 function y(a) { return x() + (a?a:0); }
153 function z() { return y(3); }
155 self
.assertEqual(jsi
.call_function('z'), 5)
156 self
.assertEqual(jsi
.call_function('y'), 2)
158 def test_for_loop(self
):
159 jsi
= JSInterpreter('''
160 function x() { a=0; for (i=0; i-10; i++) {a++} return a }
162 self
.assertEqual(jsi
.call_function('x'), 10)
164 def test_switch(self
):
165 jsi
= JSInterpreter('''
166 function x(f) { switch(f){
174 self
.assertEqual(jsi
.call_function('x', 1), 7)
175 self
.assertEqual(jsi
.call_function('x', 3), 6)
176 self
.assertEqual(jsi
.call_function('x', 5), 0)
178 def test_switch_default(self
):
179 jsi
= JSInterpreter('''
180 function x(f) { switch(f){
189 self
.assertEqual(jsi
.call_function('x', 1), 2)
190 self
.assertEqual(jsi
.call_function('x', 5), 11)
191 self
.assertEqual(jsi
.call_function('x', 9), 14)
194 jsi
= JSInterpreter('''
195 function x() { try{return 10} catch(e){return 5} }
197 self
.assertEqual(jsi
.call_function('x'), 10)
199 def test_catch(self
):
200 jsi
= JSInterpreter('''
201 function x() { try{throw 10} catch(e){return 5} }
203 self
.assertEqual(jsi
.call_function('x'), 5)
205 def test_finally(self
):
206 jsi
= JSInterpreter('''
207 function x() { try{throw 10} finally {return 42} }
209 self
.assertEqual(jsi
.call_function('x'), 42)
210 jsi
= JSInterpreter('''
211 function x() { try{throw 10} catch(e){return 5} finally {return 42} }
213 self
.assertEqual(jsi
.call_function('x'), 42)
215 def test_nested_try(self
):
216 jsi
= JSInterpreter('''
218 try{throw 10} finally {throw 42}
219 } catch(e){return 5} }
221 self
.assertEqual(jsi
.call_function('x'), 5)
223 def test_for_loop_continue(self
):
224 jsi
= JSInterpreter('''
225 function x() { a=0; for (i=0; i-10; i++) { continue; a++ } return a }
227 self
.assertEqual(jsi
.call_function('x'), 0)
229 def test_for_loop_break(self
):
230 jsi
= JSInterpreter('''
231 function x() { a=0; for (i=0; i-10; i++) { break; a++ } return a }
233 self
.assertEqual(jsi
.call_function('x'), 0)
235 def test_for_loop_try(self
):
236 jsi
= JSInterpreter('''
238 for (i=0; i-10; i++) { try { if (i == 5) throw i} catch {return 10} finally {break} };
241 self
.assertEqual(jsi
.call_function('x'), 42)
243 def test_literal_list(self
):
244 jsi
= JSInterpreter('''
245 function x() { return [1, 2, "asdf", [5, 6, 7]][3] }
247 self
.assertEqual(jsi
.call_function('x'), [5, 6, 7])
249 def test_comma(self
):
250 jsi
= JSInterpreter('''
251 function x() { a=5; a -= 1, a+=3; return a }
253 self
.assertEqual(jsi
.call_function('x'), 7)
255 jsi
= JSInterpreter('''
256 function x() { a=5; return (a -= 1, a+=3, a); }
258 self
.assertEqual(jsi
.call_function('x'), 7)
260 jsi
= JSInterpreter('''
261 function x() { return (l=[0,1,2,3], function(a, b){return a+b})((l[1], l[2]), l[3]) }
263 self
.assertEqual(jsi
.call_function('x'), 5)
266 jsi
= JSInterpreter('''
267 function x() { return void 42; }
269 self
.assertEqual(jsi
.call_function('x'), None)
271 def test_return_function(self
):
272 jsi
= JSInterpreter('''
273 function x() { return [1, function(){return 1}][1] }
275 self
.assertEqual(jsi
.call_function('x')([]), 1)
278 jsi
= JSInterpreter('''
279 function x() { return null; }
281 self
.assertEqual(jsi
.call_function('x'), None)
283 jsi
= JSInterpreter('''
284 function x() { return [null > 0, null < 0, null == 0, null === 0]; }
286 self
.assertEqual(jsi
.call_function('x'), [False, False, False, False])
288 jsi
= JSInterpreter('''
289 function x() { return [null >= 0, null <= 0]; }
291 self
.assertEqual(jsi
.call_function('x'), [True, True])
293 def test_undefined(self
):
294 jsi
= JSInterpreter('''
295 function x() { return undefined === undefined; }
297 self
.assertEqual(jsi
.call_function('x'), True)
299 jsi
= JSInterpreter('''
300 function x() { return undefined; }
302 self
.assertEqual(jsi
.call_function('x'), JS_Undefined
)
304 jsi
= JSInterpreter('''
305 function x() { let v; return v; }
307 self
.assertEqual(jsi
.call_function('x'), JS_Undefined
)
309 jsi
= JSInterpreter('''
310 function x() { return [undefined === undefined, undefined == undefined, undefined < undefined, undefined > undefined]; }
312 self
.assertEqual(jsi
.call_function('x'), [True, True, False, False])
314 jsi
= JSInterpreter('''
315 function x() { return [undefined === 0, undefined == 0, undefined < 0, undefined > 0]; }
317 self
.assertEqual(jsi
.call_function('x'), [False, False, False, False])
319 jsi
= JSInterpreter('''
320 function x() { return [undefined >= 0, undefined <= 0]; }
322 self
.assertEqual(jsi
.call_function('x'), [False, False])
324 jsi
= JSInterpreter('''
325 function x() { return [undefined > null, undefined < null, undefined == null, undefined === null]; }
327 self
.assertEqual(jsi
.call_function('x'), [False, False, True, False])
329 jsi
= JSInterpreter('''
330 function x() { return [undefined === null, undefined == null, undefined < null, undefined > null]; }
332 self
.assertEqual(jsi
.call_function('x'), [False, True, False, False])
334 jsi
= JSInterpreter('''
335 function x() { let v; return [42+v, v+42, v**42, 42**v, 0**v]; }
337 for y
in jsi
.call_function('x'):
338 self
.assertTrue(math
.isnan(y
))
340 jsi
= JSInterpreter('''
341 function x() { let v; return v**0; }
343 self
.assertEqual(jsi
.call_function('x'), 1)
345 jsi
= JSInterpreter('''
346 function x() { let v; return [v>42, v<=42, v&&42, 42&&v]; }
348 self
.assertEqual(jsi
.call_function('x'), [False, False, JS_Undefined
, JS_Undefined
])
350 jsi
= JSInterpreter('function x(){return undefined ?? 42; }')
351 self
.assertEqual(jsi
.call_function('x'), 42)
353 def test_object(self
):
354 jsi
= JSInterpreter('''
355 function x() { return {}; }
357 self
.assertEqual(jsi
.call_function('x'), {})
359 jsi
= JSInterpreter('''
360 function x() { let a = {m1: 42, m2: 0 }; return [a["m1"], a.m2]; }
362 self
.assertEqual(jsi
.call_function('x'), [42, 0])
364 jsi
= JSInterpreter('''
365 function x() { let a; return a?.qq; }
367 self
.assertEqual(jsi
.call_function('x'), JS_Undefined
)
369 jsi
= JSInterpreter('''
370 function x() { let a = {m1: 42, m2: 0 }; return a?.qq; }
372 self
.assertEqual(jsi
.call_function('x'), JS_Undefined
)
374 def test_regex(self
):
375 jsi
= JSInterpreter('''
376 function x() { let a=/,,[/,913,/](,)}/; }
378 self
.assertEqual(jsi
.call_function('x'), None)
380 jsi
= JSInterpreter('''
381 function x() { let a=/,,[/,913,/](,)}/; return a; }
383 self
.assertIsInstance(jsi
.call_function('x'), re
.Pattern
)
385 jsi
= JSInterpreter('''
386 function x() { let a=/,,[/,913,/](,)}/i; return a; }
388 self
.assertEqual(jsi
.call_function('x').flags
& re
.I
, re
.I
)
390 jsi
= JSInterpreter(R
'''
391 function x() { let a=/,][}",],()}(\[)/; return a; }
393 self
.assertEqual(jsi
.call_function('x').pattern
, r
',][}",],()}(\[)')
395 jsi
= JSInterpreter(R
'''
396 function x() { let a=[/[)\\]/]; return a[0]; }
398 self
.assertEqual(jsi
.call_function('x').pattern
, r
'[)\\]')
400 def test_char_code_at(self
):
401 jsi
= JSInterpreter('function x(i){return "test".charCodeAt(i)}')
402 self
.assertEqual(jsi
.call_function('x', 0), 116)
403 self
.assertEqual(jsi
.call_function('x', 1), 101)
404 self
.assertEqual(jsi
.call_function('x', 2), 115)
405 self
.assertEqual(jsi
.call_function('x', 3), 116)
406 self
.assertEqual(jsi
.call_function('x', 4), None)
407 self
.assertEqual(jsi
.call_function('x', 'not_a_number'), 116)
409 def test_bitwise_operators_overflow(self
):
410 jsi
= JSInterpreter('function x(){return -524999584 << 5}')
411 self
.assertEqual(jsi
.call_function('x'), 379882496)
413 jsi
= JSInterpreter('function x(){return 1236566549 << 5}')
414 self
.assertEqual(jsi
.call_function('x'), 915423904)
417 if __name__
== '__main__':