quickjs 0.6.4 → 0.7.0
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 +4 -4
- data/README.md +32 -19
- data/ext/quickjsrb/extconf.rb +15 -2
- data/ext/quickjsrb/quickjsrb.c +49 -17
- data/ext/quickjsrb/quickjsrb.h +5 -0
- data/ext/quickjsrb/vendor/polyfill-intl-en.min.js +10 -0
- data/lib/quickjs/version.rb +1 -1
- metadata +4 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 44eedd9a617b8106c963b40fde4ffd9e9c3093be07a9d1528ae3cdac357abd9d
|
4
|
+
data.tar.gz: 83821e76a40a152fea3516c4720e5a5240129f866dc44db2d2ef8350672ff80b
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 5b8f7d797869c7f02a7c529f47d0340831b700e7c91fd0f351df040e8e09f4e4a0e8ae42c9da0609603cd451d50246d6f491619c512e383ecd1f4a6a53c14feb
|
7
|
+
data.tar.gz: 71bf4c8fd5d6f94db5edb6f341005dcc153b1bba13b4dbcdc1974b3eca01be2893de6dee5cf3a4e9be815afe1777febf2cd0c718df334f4e175fd25e01267ca6
|
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
|
-
####
|
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
|
-
#
|
58
|
-
Quickjs.eval_code(
|
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
|
-
#
|
61
|
-
Quickjs.eval_code(
|
61
|
+
# Expose `os.setTimouet` and `os.clearTimeout` from `os` module
|
62
|
+
vm = Quickjs.eval_code(features: [::Quickjs::FEATURES_TIMEOUT])
|
62
63
|
|
63
|
-
#
|
64
|
-
Quickjs.eval_code(
|
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
|
-
####
|
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
|
-
#
|
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
|
-
#
|
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
|
-
#
|
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
|
@@ -153,11 +156,21 @@ vm.eval_code('console.log("log me", null)')
|
|
153
156
|
vm.logs #=> Array of Quickjs::VM::Log
|
154
157
|
vm.logs.last.severity #=> :info
|
155
158
|
vm.logs.last.to_s #=> 'log me null'
|
156
|
-
vm
|
159
|
+
vm.logs.last.raw #=> ['log me', nil]
|
157
160
|
```
|
158
161
|
|
159
162
|
## License
|
160
163
|
|
161
|
-
|
162
|
-
|
163
|
-
|
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`
|
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).
|
data/ext/quickjsrb/extconf.rb
CHANGED
@@ -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
|
data/ext/quickjsrb/quickjsrb.c
CHANGED
@@ -58,17 +58,9 @@ JSValue to_js_value(JSContext *ctx, VALUE r_value)
|
|
58
58
|
{
|
59
59
|
VALUE r_json_str = rb_funcall(r_value, rb_intern("to_json"), 0, NULL);
|
60
60
|
char *str = StringValueCStr(r_json_str);
|
61
|
-
JSValue
|
62
|
-
JSValue j_jsonClass = JS_GetPropertyStr(ctx, j_global, "JSON");
|
63
|
-
JSValue j_parseFunc = JS_GetPropertyStr(ctx, j_jsonClass, "parse");
|
64
|
-
JSValue j_str = JS_NewString(ctx, str);
|
65
|
-
JSValue j_stringified = JS_Call(ctx, j_parseFunc, j_jsonClass, 1, (JSValueConst *)&j_str);
|
66
|
-
JS_FreeValue(ctx, j_global);
|
67
|
-
JS_FreeValue(ctx, j_jsonClass);
|
68
|
-
JS_FreeValue(ctx, j_parseFunc);
|
69
|
-
JS_FreeValue(ctx, j_str);
|
61
|
+
JSValue j_parsed = JS_ParseJSON(ctx, str, strlen(str), "<quickjsrb.c>");
|
70
62
|
|
71
|
-
return
|
63
|
+
return j_parsed;
|
72
64
|
}
|
73
65
|
default:
|
74
66
|
{
|
@@ -132,9 +124,7 @@ VALUE to_rb_value(JSContext *ctx, JSValue j_val)
|
|
132
124
|
{
|
133
125
|
case JS_TAG_INT:
|
134
126
|
{
|
135
|
-
|
136
|
-
JS_ToInt32(ctx, &int_res, j_val);
|
137
|
-
return INT2NUM(int_res);
|
127
|
+
return INT2NUM(JS_VALUE_GET_INT(j_val));
|
138
128
|
}
|
139
129
|
case JS_TAG_FLOAT64:
|
140
130
|
{
|
@@ -142,9 +132,7 @@ VALUE to_rb_value(JSContext *ctx, JSValue j_val)
|
|
142
132
|
{
|
143
133
|
return QUICKJSRB_SYM(nanId);
|
144
134
|
}
|
145
|
-
|
146
|
-
JS_ToFloat64(ctx, &double_res, j_val);
|
147
|
-
return DBL2NUM(double_res);
|
135
|
+
return DBL2NUM(JS_VALUE_GET_FLOAT64(j_val));
|
148
136
|
}
|
149
137
|
case JS_TAG_BOOL:
|
150
138
|
{
|
@@ -400,7 +388,40 @@ static JSValue js_quickjsrb_log(JSContext *ctx, JSValueConst _this, int argc, JS
|
|
400
388
|
for (int i = 0; i < argc; i++)
|
401
389
|
{
|
402
390
|
JSValue j_logged = JS_DupValue(ctx, argv[i]);
|
403
|
-
VALUE r_raw
|
391
|
+
VALUE r_raw;
|
392
|
+
if (JS_VALUE_GET_NORM_TAG(j_logged) == JS_TAG_OBJECT && JS_PromiseState(ctx, j_logged) != -1)
|
393
|
+
{
|
394
|
+
r_raw = rb_str_new2("Promise");
|
395
|
+
}
|
396
|
+
else if (JS_IsError(ctx, j_logged))
|
397
|
+
{
|
398
|
+
JSValue j_errorClassName = JS_GetPropertyStr(ctx, j_logged, "name");
|
399
|
+
const char *errorClassName = JS_ToCString(ctx, j_errorClassName);
|
400
|
+
JS_FreeValue(ctx, j_errorClassName);
|
401
|
+
|
402
|
+
JSValue j_errorClassMessage = JS_GetPropertyStr(ctx, j_logged, "message");
|
403
|
+
const char *errorClassMessage = JS_ToCString(ctx, j_errorClassMessage);
|
404
|
+
JS_FreeValue(ctx, j_errorClassMessage);
|
405
|
+
|
406
|
+
JSValue j_stackTrace = JS_GetPropertyStr(ctx, j_logged, "stack");
|
407
|
+
const char *stackTrace = JS_ToCString(ctx, j_stackTrace);
|
408
|
+
JS_FreeValue(ctx, j_stackTrace);
|
409
|
+
|
410
|
+
const char *headlineTemplate = "%s: %s\n%s";
|
411
|
+
int length = snprintf(NULL, 0, headlineTemplate, errorClassName, errorClassMessage, stackTrace);
|
412
|
+
char *headline = (char *)malloc(length + 1);
|
413
|
+
snprintf(headline, length + 1, headlineTemplate, errorClassName, errorClassMessage, stackTrace);
|
414
|
+
JS_FreeCString(ctx, errorClassName);
|
415
|
+
JS_FreeCString(ctx, errorClassMessage);
|
416
|
+
JS_FreeCString(ctx, stackTrace);
|
417
|
+
|
418
|
+
r_raw = rb_str_new2(headline);
|
419
|
+
free(headline);
|
420
|
+
}
|
421
|
+
else
|
422
|
+
{
|
423
|
+
r_raw = to_rb_value(ctx, j_logged);
|
424
|
+
}
|
404
425
|
const char *body = JS_ToCString(ctx, j_logged);
|
405
426
|
VALUE r_c = rb_str_new2(body);
|
406
427
|
JS_FreeCString(ctx, body);
|
@@ -504,6 +525,17 @@ static VALUE vm_m_initialize(int argc, VALUE *argv, VALUE r_self)
|
|
504
525
|
JS_FreeValue(data->context, j_timeoutEval);
|
505
526
|
}
|
506
527
|
|
528
|
+
if (RTEST(rb_funcall(r_features, rb_intern("include?"), 1, QUICKJSRB_SYM(featurePolyfillIntlId))))
|
529
|
+
{
|
530
|
+
const char *defineIntl = "Object.defineProperty(globalThis, 'Intl', { value:{} });\n";
|
531
|
+
JSValue j_defineIntl = JS_Eval(data->context, defineIntl, strlen(defineIntl), "<vm>", JS_EVAL_TYPE_GLOBAL);
|
532
|
+
JS_FreeValue(data->context, j_defineIntl);
|
533
|
+
|
534
|
+
JSValue j_polyfillIntlObject = JS_ReadObject(data->context, &qjsc_polyfill_intl_en_min, qjsc_polyfill_intl_en_min_size, JS_READ_OBJ_BYTECODE);
|
535
|
+
JSValue j_polyfillIntlResult = JS_EvalFunction(data->context, j_polyfillIntlObject); // Frees polyfillIntlObject
|
536
|
+
JS_FreeValue(data->context, j_polyfillIntlResult);
|
537
|
+
}
|
538
|
+
|
507
539
|
JSValue j_console = JS_NewObject(data->context);
|
508
540
|
JS_SetPropertyStr(
|
509
541
|
data->context, j_console, "log",
|
data/ext/quickjsrb/quickjsrb.h
CHANGED
@@ -12,9 +12,13 @@
|
|
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 *featurePolyfillIntlId = "feature_polyfill_intl";
|
18
22
|
|
19
23
|
const char *undefinedId = "undefined";
|
20
24
|
const char *nanId = "NaN";
|
@@ -133,6 +137,7 @@ static void r_define_constants(VALUE r_parent_class)
|
|
133
137
|
rb_define_const(r_parent_class, "MODULE_STD", QUICKJSRB_SYM(featureStdId));
|
134
138
|
rb_define_const(r_parent_class, "MODULE_OS", QUICKJSRB_SYM(featureOsId));
|
135
139
|
rb_define_const(r_parent_class, "FEATURES_TIMEOUT", QUICKJSRB_SYM(featureOsTimeoutId));
|
140
|
+
rb_define_const(r_parent_class, "POLYFILL_INTL", QUICKJSRB_SYM(featurePolyfillIntlId));
|
136
141
|
|
137
142
|
VALUE rb_cQuickjsValue = rb_define_class_under(r_parent_class, "Value", rb_cObject);
|
138
143
|
rb_define_const(rb_cQuickjsValue, "UNDEFINED", QUICKJSRB_SYM(undefinedId));
|