quickjs 0.6.5 → 0.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 31bcb05285a3c4c80c6e5c06ec043793debf795c75f82362b317d74d3eb8cf96
4
- data.tar.gz: 9032bf6cb129ea65dc52ae5df5ff394056f440c9de52947ab3029e5c0626c554
3
+ metadata.gz: 460654c8f6d8174bf3f7025c0ae6e37a0bed766f3a8bdf57da5b9f5fc4ef8e0b
4
+ data.tar.gz: 100eb2a30022baada73c5d83eb79675c19ba1df439da494278638917c3df16a2
5
5
  SHA512:
6
- metadata.gz: 2a12da5b5c71c5d350af754f5b50e47104a379aa53cde68625da76bf6b3e165a804b3ec9d764d52f7e8c87a696a870d948d17381bb044b13536b6e96992e9cea
7
- data.tar.gz: 1b0b05872129c9d3850e66b0e5f77c779bb2554801e132380a0937a2d21d393b23317364e775ab9c0779ba074da157d46fdd6779a1b5f0ba690c55e2cbedccbd
6
+ metadata.gz: 1527a557f744d35f9d79965d51c70f82cba82328351e1ec301ea66fd0293a7cd8bbe713766c470648774c6b589382d4e4357d89b01aa788a8c089e29f974f652
7
+ data.tar.gz: d06c84aac9d3a440e885d769dfceb3849276c3b93c6d43e60788c8bc476416f61fef2898f1f94c90db44a4e0bc7e847c19c23366983555ee22138feeb9889a75
data/README.md CHANGED
@@ -49,20 +49,22 @@ Quickjs.eval_code(code, { memory_limit: 1024 ** 3 })
49
49
  Quickjs.eval_code(code, { max_stack_size: 1024 ** 2 })
50
50
  ```
51
51
 
52
- #### Built-in modules
53
-
54
- To enable [std module](https://bellard.org/quickjs/quickjs.html#std-module) and [os module](https://bellard.org/quickjs/quickjs.html#os-module) selectively.
52
+ #### Toggle features
55
53
 
56
54
  ```rb
57
- # enable std module
58
- Quickjs.eval_code(code, { features: [Quickjs::MODULE_STD] })
55
+ # Enable `std` module by quickjs: https://bellard.org/quickjs/quickjs.html#std-module
56
+ vm = Quickjs.eval_code(features: [::Quickjs::MODULE_STD])
57
+
58
+ # Enable `os` module by quickjs: https://bellard.org/quickjs/quickjs.html#os-module
59
+ vm = Quickjs.eval_code(features: [::Quickjs::MODULE_OS])
59
60
 
60
- # enable os module
61
- Quickjs.eval_code(code, { features: [Quickjs::MODULE_OS] })
61
+ # Expose `os.setTimouet` and `os.clearTimeout` from `os` module
62
+ vm = Quickjs.eval_code(features: [::Quickjs::FEATURES_TIMEOUT])
62
63
 
63
- # enable timeout features `setTimeout`, `clearTimeout` from os module specifically
64
- Quickjs.eval_code(code, { features: [Quickjs::FEATURES_TIMEOUT] })
64
+ # Inject the polyfill of Intl
65
+ vm = Quickjs.eval_code(features: [::Quickjs::POLYFILL_INTL])
65
66
  ```
67
+
66
68
  </details>
67
69
 
68
70
  ### `Quickjs::VM`: Maintain a consistent VM/runtime
@@ -87,19 +89,20 @@ vm = Quickjs::VM.new(
87
89
  )
88
90
  ```
89
91
 
90
- #### Built-in modules
91
-
92
- To enable [std module](https://bellard.org/quickjs/quickjs.html#std-module) and [os module](https://bellard.org/quickjs/quickjs.html#os-module) selectively.
92
+ #### Toggle features
93
93
 
94
94
  ```rb
95
- # enable std module
95
+ # Enable `std` module by quickjs: https://bellard.org/quickjs/quickjs.html#std-module
96
96
  vm = Quickjs::VM.new(features: [::Quickjs::MODULE_STD])
97
97
 
98
- # enable os module
98
+ # Enable `os` module by quickjs: https://bellard.org/quickjs/quickjs.html#os-module
99
99
  vm = Quickjs::VM.new(features: [::Quickjs::MODULE_OS])
100
100
 
101
- # enable timeout features `setTimeout`, `clearTimeout`
101
+ # Expose `os.setTimouet` and `os.clearTimeout` from `os` module
102
102
  vm = Quickjs::VM.new(features: [::Quickjs::FEATURES_TIMEOUT])
103
+
104
+ # Inject the polyfill of Intl
105
+ vm = Quickjs::VM.new(features: [::Quickjs::POLYFILL_INTL])
103
106
  ```
104
107
 
105
108
  #### VM timeout
@@ -158,6 +161,16 @@ vm.logs.last.raw #=> ['log me', nil]
158
161
 
159
162
  ## License
160
163
 
161
- Every file in `ext/quickjsrb/quickjs` is licensed under [the MIT License Copyright 2017-2021 by Fabrice Bellard and Charlie Gordon](https://github.com/bellard/quickjs/blob/6e2e68fd0896957f92eb6c242a2e048c1ef3cae0/LICENSE).
162
-
163
- For otherwise, [the MIT License, Copyright 2024 by Kengo Hamasaki](/LICENSE).
164
+ - `ext/quickjsrb/quickjs`
165
+ - [MIT License Copyright (c) 2017-2021 by Fabrice Bellard and Charlie Gordon](https://github.com/bellard/quickjs/blob/6e2e68fd0896957f92eb6c242a2e048c1ef3cae0/LICENSE).
166
+ - `ext/quickjsrb/vendor/polyfill-intl-en.min.js` (bundled and minified)
167
+ - MIT License Copyright (c) 2023 FormatJS
168
+ - [@formatjs/intl-getcanonicallocales](https://github.com/formatjs/formatjs/blob/main/packages/intl-getcanonicallocales/LICENSE.md)
169
+ - [@formatjs/intl-locale](https://github.com/formatjs/formatjs/blob/main/packages/intl-locale/LICENSE.md)
170
+ - [@formatjs/intl-pluralrules](https://github.com/formatjs/formatjs/blob/main/packages/intl-pluralrules/LICENSE.md)
171
+ - [@formatjs/intl-numberformat](https://github.com/formatjs/formatjs/blob/main/packages/intl-numberformat/LICENSE.md)
172
+ - [@formatjs/intl-datetimeformat](https://github.com/formatjs/formatjs/blob/main/packages/intl-datetimeformat/LICENSE.md)
173
+ - MIT License Copyright (c) 2025 Michael Mclaughlin
174
+ - [decimal.js](https://www.npmjs.com/package/decimal.js)
175
+
176
+ Otherwise, [the MIT License, Copyright 2024 by Kengo Hamasaki](/LICENSE).
@@ -11,12 +11,12 @@ $srcs = [
11
11
  'cutils.c',
12
12
  'quickjs.c',
13
13
  'quickjs-libc.c',
14
+ 'polyfill-intl-en.min.c',
14
15
  'quickjsrb.c',
15
16
  ]
16
17
 
17
18
  append_cflags('-I$(srcdir)/quickjs')
18
19
 
19
-
20
20
  append_cflags('-g')
21
21
  append_cflags('-O2')
22
22
  append_cflags('-Wall')
@@ -51,4 +51,17 @@ abort('could not find quickjs-libc.h') unless find_header('quickjs-libc.h')
51
51
  append_cflags('-fvisibility=hidden')
52
52
  $warnflags = ''
53
53
 
54
- create_makefile('quickjs/quickjsrb')
54
+ create_makefile('quickjs/quickjsrb') do |conf|
55
+ conf.push <<COMPILE_POLYFILL
56
+ QJS_LIB_OBJS= quickjs.o libregexp.o libunicode.o cutils.o quickjs-libc.o libbf.o
57
+ POLYFILL_OPTS=-fno-string-normalize -fno-typedarray -fno-typedarray -fno-eval -fno-proxy -fno-module-loader -fno-bigint
58
+
59
+ qjsc: ./qjsc.o $(QJS_LIB_OBJS)
60
+ $(CC) -g -o $@ $^ -lm -ldl -lpthread
61
+ polyfill-intl-en.min.js:
62
+ $(COPY) $(srcdir)/vendor/$@ $@
63
+ polyfill-intl-en.min.c: ./qjsc polyfill-intl-en.min.js
64
+ ./qjsc $(POLYFILL_OPTS) -c -M polyfill/intl-en.so,intlen -m -o $@ polyfill-intl-en.min.js
65
+ COMPILE_POLYFILL
66
+ conf
67
+ end
@@ -4,11 +4,11 @@ JSValue j_error_from_ruby_error(JSContext *ctx, VALUE r_error)
4
4
  {
5
5
  JSValue j_error = JS_NewError(ctx); // may wanna have custom error class to determine in JS' end
6
6
 
7
- VALUE r_object_id = rb_funcall(r_error, rb_intern("object_id"), 0, NULL);
7
+ VALUE r_object_id = rb_funcall(r_error, rb_intern("object_id"), 0);
8
8
  int objectId = NUM2INT(r_object_id);
9
9
  JS_SetPropertyStr(ctx, j_error, "rb_object_id", JS_NewInt32(ctx, objectId));
10
10
 
11
- VALUE r_exception_message = rb_funcall(r_error, rb_intern("message"), 0, NULL);
11
+ VALUE r_exception_message = rb_funcall(r_error, rb_intern("message"), 0);
12
12
  const char *errorMessage = StringValueCStr(r_exception_message);
13
13
  JS_SetPropertyStr(ctx, j_error, "message", JS_NewString(ctx, errorMessage));
14
14
 
@@ -24,7 +24,7 @@ JSValue to_js_value(JSContext *ctx, VALUE r_value)
24
24
  case T_FIXNUM:
25
25
  case T_FLOAT:
26
26
  {
27
- VALUE r_str = rb_funcall(r_value, rb_intern("to_s"), 0, NULL);
27
+ VALUE r_str = rb_funcall(r_value, rb_intern("to_s"), 0);
28
28
  char *str = StringValueCStr(r_str);
29
29
  JSValue j_global = JS_GetGlobalObject(ctx);
30
30
  JSValue j_numberClass = JS_GetPropertyStr(ctx, j_global, "Number");
@@ -44,7 +44,7 @@ JSValue to_js_value(JSContext *ctx, VALUE r_value)
44
44
  }
45
45
  case T_SYMBOL:
46
46
  {
47
- VALUE r_str = rb_funcall(r_value, rb_intern("to_s"), 0, NULL);
47
+ VALUE r_str = rb_funcall(r_value, rb_intern("to_s"), 0);
48
48
  char *str = StringValueCStr(r_str);
49
49
 
50
50
  return JS_NewString(ctx, str);
@@ -56,7 +56,7 @@ JSValue to_js_value(JSContext *ctx, VALUE r_value)
56
56
  case T_HASH:
57
57
  case T_ARRAY:
58
58
  {
59
- VALUE r_json_str = rb_funcall(r_value, rb_intern("to_json"), 0, NULL);
59
+ VALUE r_json_str = rb_funcall(r_value, rb_intern("to_json"), 0);
60
60
  char *str = StringValueCStr(r_json_str);
61
61
  JSValue j_parsed = JS_ParseJSON(ctx, str, strlen(str), "<quickjsrb.c>");
62
62
 
@@ -71,7 +71,7 @@ JSValue to_js_value(JSContext *ctx, VALUE r_value)
71
71
  {
72
72
  return j_error_from_ruby_error(ctx, r_value);
73
73
  }
74
- VALUE r_inspect_str = rb_funcall(r_value, rb_intern("inspect"), 0, NULL);
74
+ VALUE r_inspect_str = rb_funcall(r_value, rb_intern("inspect"), 0);
75
75
  char *str = StringValueCStr(r_inspect_str);
76
76
 
77
77
  return JS_NewString(ctx, str);
@@ -284,7 +284,7 @@ VALUE to_rb_value(JSContext *ctx, JSValue j_val)
284
284
  JS_FreeValue(ctx, j_strigified);
285
285
  JS_FreeCString(ctx, msg);
286
286
 
287
- return rb_funcall(r_str, rb_intern("to_i"), 0, NULL);
287
+ return rb_funcall(r_str, rb_intern("to_i"), 0);
288
288
  }
289
289
  case JS_TAG_BIG_FLOAT:
290
290
  case JS_TAG_BIG_DECIMAL:
@@ -381,6 +381,37 @@ static JSValue js_quickjsrb_call_global(JSContext *ctx, JSValueConst _this, int
381
381
  }
382
382
  }
383
383
 
384
+ static JSValue js_delay_and_eval_job(JSContext *ctx, int argc, JSValueConst *argv)
385
+ {
386
+ VALUE rb_delay_msec = to_rb_value(ctx, argv[1]);
387
+ VALUE rb_delay_sec = rb_funcall(rb_delay_msec, rb_intern("/"), 1, rb_float_new(1000));
388
+ rb_thread_wait_for(rb_time_interval(rb_delay_sec));
389
+ JS_Call(ctx, argv[0], JS_UNDEFINED, 0, NULL);
390
+
391
+ return JS_UNDEFINED;
392
+ }
393
+
394
+ static JSValue js_quickjsrb_set_timeout(JSContext *ctx, JSValueConst _this, int argc, JSValueConst *argv)
395
+ {
396
+ JSValueConst func;
397
+ int64_t delay;
398
+
399
+ func = argv[0];
400
+ if (!JS_IsFunction(ctx, func))
401
+ return JS_ThrowTypeError(ctx, "not a function");
402
+ if (JS_ToInt64(ctx, &delay, argv[1])) // TODO: should be lower than global timeout
403
+ return JS_EXCEPTION;
404
+
405
+ JSValueConst args[2];
406
+ args[0] = func;
407
+ args[1] = argv[1]; // delay
408
+ // TODO: implement timer manager and poll with quickjs' queue
409
+ // Currently, queueing multiple js_delay_and_eval_job is not parallelized
410
+ JS_EnqueueJob(ctx, js_delay_and_eval_job, 2, args);
411
+
412
+ return JS_UNDEFINED;
413
+ }
414
+
384
415
  static JSValue js_quickjsrb_log(JSContext *ctx, JSValueConst _this, int argc, JSValueConst *argv, const char *severity)
385
416
  {
386
417
  VMData *data = JS_GetContextOpaque(ctx);
@@ -492,6 +523,8 @@ static VALUE vm_m_initialize(int argc, VALUE *argv, VALUE r_self)
492
523
  JS_SetModuleLoaderFunc(runtime, NULL, js_module_loader, NULL);
493
524
  js_std_init_handlers(runtime);
494
525
 
526
+ JSValue j_global = JS_GetGlobalObject(data->context);
527
+
495
528
  if (RTEST(rb_funcall(r_features, rb_intern("include?"), 1, QUICKJSRB_SYM(featureStdId))))
496
529
  {
497
530
  js_init_module_std(data->context, "std");
@@ -524,6 +557,23 @@ static VALUE vm_m_initialize(int argc, VALUE *argv, VALUE r_self)
524
557
  free(enableTimeout);
525
558
  JS_FreeValue(data->context, j_timeoutEval);
526
559
  }
560
+ else if (RTEST(rb_funcall(r_features, rb_intern("include?"), 1, QUICKJSRB_SYM(featureOsTimeoutBetaId))))
561
+ {
562
+ JS_SetPropertyStr(
563
+ data->context, j_global, "setTimeout",
564
+ JS_NewCFunction(data->context, js_quickjsrb_set_timeout, "setTimeout", 2));
565
+ }
566
+
567
+ if (RTEST(rb_funcall(r_features, rb_intern("include?"), 1, QUICKJSRB_SYM(featurePolyfillIntlId))))
568
+ {
569
+ const char *defineIntl = "Object.defineProperty(globalThis, 'Intl', { value:{} });\n";
570
+ JSValue j_defineIntl = JS_Eval(data->context, defineIntl, strlen(defineIntl), "<vm>", JS_EVAL_TYPE_GLOBAL);
571
+ JS_FreeValue(data->context, j_defineIntl);
572
+
573
+ JSValue j_polyfillIntlObject = JS_ReadObject(data->context, &qjsc_polyfill_intl_en_min, qjsc_polyfill_intl_en_min_size, JS_READ_OBJ_BYTECODE);
574
+ JSValue j_polyfillIntlResult = JS_EvalFunction(data->context, j_polyfillIntlObject); // Frees polyfillIntlObject
575
+ JS_FreeValue(data->context, j_polyfillIntlResult);
576
+ }
527
577
 
528
578
  JSValue j_console = JS_NewObject(data->context);
529
579
  JS_SetPropertyStr(
@@ -542,7 +592,6 @@ static VALUE vm_m_initialize(int argc, VALUE *argv, VALUE r_self)
542
592
  data->context, j_console, "error",
543
593
  JS_NewCFunction(data->context, js_console_error, "error", 1));
544
594
 
545
- JSValue j_global = JS_GetGlobalObject(data->context);
546
595
  JS_SetPropertyStr(data->context, j_global, "console", j_console);
547
596
  JS_FreeValue(data->context, j_global);
548
597
 
@@ -612,7 +661,7 @@ static VALUE vm_m_defineGlobalFunction(int argc, VALUE *argv, VALUE r_self)
612
661
  JS_FreeValue(data->context, ruby_data[0]);
613
662
  JS_FreeValue(data->context, ruby_data[1]);
614
663
 
615
- return rb_funcall(r_name, rb_intern("to_sym"), 0, NULL);
664
+ return rb_funcall(r_name, rb_intern("to_sym"), 0);
616
665
  }
617
666
 
618
667
  static VALUE vm_m_import(int argc, VALUE *argv, VALUE r_self)
@@ -12,9 +12,14 @@
12
12
  #include <string.h>
13
13
  #include <time.h>
14
14
 
15
+ extern const uint32_t qjsc_polyfill_intl_en_min_size;
16
+ extern const uint8_t qjsc_polyfill_intl_en_min;
17
+
15
18
  const char *featureStdId = "feature_std";
16
19
  const char *featureOsId = "feature_os";
17
20
  const char *featureOsTimeoutId = "feature_os_timeout";
21
+ const char *featureOsTimeoutBetaId = "feature_os_timeout_beta";
22
+ const char *featurePolyfillIntlId = "feature_polyfill_intl";
18
23
 
19
24
  const char *undefinedId = "undefined";
20
25
  const char *nanId = "NaN";
@@ -133,6 +138,8 @@ static void r_define_constants(VALUE r_parent_class)
133
138
  rb_define_const(r_parent_class, "MODULE_STD", QUICKJSRB_SYM(featureStdId));
134
139
  rb_define_const(r_parent_class, "MODULE_OS", QUICKJSRB_SYM(featureOsId));
135
140
  rb_define_const(r_parent_class, "FEATURES_TIMEOUT", QUICKJSRB_SYM(featureOsTimeoutId));
141
+ rb_define_const(r_parent_class, "FEATURES_TIMEOUT_BETA", QUICKJSRB_SYM(featureOsTimeoutBetaId));
142
+ rb_define_const(r_parent_class, "POLYFILL_INTL", QUICKJSRB_SYM(featurePolyfillIntlId));
136
143
 
137
144
  VALUE rb_cQuickjsValue = rb_define_class_under(r_parent_class, "Value", rb_cObject);
138
145
  rb_define_const(rb_cQuickjsValue, "UNDEFINED", QUICKJSRB_SYM(undefinedId));