dop251 / goja Goto Github PK
View Code? Open in Web Editor NEWECMAScript/JavaScript engine in pure Go
License: MIT License
ECMAScript/JavaScript engine in pure Go
License: MIT License
I'm having some fun involving AST trees of JS snippets, and it feels a little silly that if I want to run one of them, I can't just chuck the already parsed tree at a runtime, but have to do the parsing twice.
Please consider sating my gluttony for microoptimizations.
Per pull request #14, (*vm).Interrupt causes a data race. This results in undefined behavior and means that interrupts are currently unsafe to use. It can be resolved in the short term by using atomics (as in #14) or other explicit synchronization to signal the interrupt from a separate goroutine, and in the long term may be fixed in another manner.
As discussed in #14, it is unacceptable for a data race to be included in a critical part of a VM, especially as part of an interrupt. According to @dop251, the data race and undefined behavior here is intentional in the pursuit of performance, and objects to the use of atomics to resolve this. The goal, then, is to find another method of causing an interrupt that both satisfies @dop251's performance requirements (currently unstated) and eliminates the data race, preferably without introducing unsafe code that cannot be checked by the race detector.
This issue is intended to discuss other approaches to removing the interrupt data race, knowing that atomics are off the table (or somehow too slow).
I am trying to register my native module, but I could not find a way to create a struct instance in js vm
type AB struct {
}
func (a *AB) A() {
fmt.Println("A")
}
o.Set("AB", reflect.TypeOf((*AB)(nil)).Elem())
.....
console.log(new(MyNativePackage.AB))
TypeError: Not a constructor at <eval>:3:13(5)
import * as CryptoJS from 'crypto-js';
function encrypt(word: string, secureKey: string, initvector: string): string {
let key = CryptoJS.enc.Utf8.parse(secureKey);
let iv = CryptoJS.enc.Utf8.parse(initvector);
let srcs = CryptoJS.enc.Utf8.parse(word);
let encrypted = CryptoJS.AES.encrypt(srcs, key, { iv: iv, mode: CryptoJS.mode.CBC });
return encrypted.toString();
}
function decrypt(cipherText: string, secureKey: string, initvector: string): string {
let key = CryptoJS.enc.Utf8.parse(secureKey);
let iv = CryptoJS.enc.Utf8.parse(initvector);
let decrypted = CryptoJS.AES.decrypt(cipherText, key, { iv: iv, mode: CryptoJS.mode.CBC });
return decrypted.toString(CryptoJS.enc.Utf8);
}
let key = '1234567887654321';
let iv = '1234567887654321';
let value1 = encrypt('bingo', key, iv);
console.log('aes encrypt:', value1);
let value2 = decrypt(value1, key, iv);
console.log('aes decrypt:', value2);
let value3 = CryptoJS.MD5("Hello, bingo");
console.log('md5:', value3.toString());
Run above code on goja, will report follow errors:
ReferenceError: unescape is not defined at crypto-js:500:42(10)
I have a rather strange use-case. I run a few goja instances in parallel, along with some other goroutines.
This can cause CPU contention. For me, the timeouts are to ensure injected code isn't running an infinite-loop, rather than having a fixed wall-clock bound.
Under high load though, a goja VM might not get around to doing much, thus running out the Wall Clock allotted time.
I'm happy to do the work, but I wanted to field the idea of "ticks". Each time Goja does some work, a tick is updated. The tick doesn't have to be precise, or meaningful, so long as it is monotonic and increments itself periodically as things get executed.
It would do two things:
I would be able to measure the inner-JS for ticks and compare two implementations which can report into performance metrics. This would help identify which scripts can be improved and when they are improved, that they really did improve.
I would be able to set an interrupt after "ticks", so under contention or maybe because the code is blocked on some input, it wouldn't expire due to the wall clock. Yet, I can guard against infinite loops.
Getting this error at runtime:
panic: TypeError: Object has no member 'error'
Is it possible to get more clues on which line this error occurred?
func TestInterrupt(t *testing.T) {
rt := goja.New()
_, err := rt.RunString(`
var fn = function() {
while (true) {}
};
`)
if !assert.NoError(t, err) {
return
}
fn, ok := goja.AssertFunction(rt.Get("fn"))
if !assert.True(t, ok) {
return
}
go func() {
<-time.After(1 * time.Second)
rt.Interrupt(errors.New("hi"))
}()
v, err := fn(goja.Undefined())
assert.Nil(t, v)
assert.EqualError(t, err, "hi")
}
Causes a panic:
panic: <nil> [recovered]
panic: <nil> [recovered]
panic: <nil>
goroutine 4 [running]:
testing.tRunner.func1(0xc4210af0e0)
/usr/lib/go/src/testing/testing.go:711 +0x2d2
panic(0xaa5820, 0xc4212d5740)
/usr/lib/go/src/runtime/panic.go:491 +0x283
github.com/loadimpact/k6/vendor/github.com/dop251/goja.(*vm).try.func1(0xc4204ceea0, 0x0, 0xc422b27e60, 0x0, 0x0, 0x0, 0xc422b27eb8)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/vm.go:356 +0x4a4
panic(0xaa5820, 0xc4212d5740)
/usr/lib/go/src/runtime/panic.go:491 +0x283
github.com/loadimpact/k6/vendor/github.com/dop251/goja.(*vm).run(0xc4204ceea0)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/vm.go:299 +0x183
github.com/loadimpact/k6/vendor/github.com/dop251/goja.(*funcObject).Call(0xc420387c80, 0xfd2b40, 0x117fd20, 0x0, 0x0, 0x0, 0x0, 0xc420024000)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/func.go:130 +0x3c8
github.com/loadimpact/k6/vendor/github.com/dop251/goja.(*funcObject).Call-fm(0xfd2b40, 0x117fd20, 0x0, 0x0, 0x0, 0x42bd99, 0xc420feca40)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/builtin_function.go:120 +0x51
github.com/loadimpact/k6/vendor/github.com/dop251/goja.AssertFunction.func1.1()
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/runtime.go:1374 +0xb6
github.com/loadimpact/k6/vendor/github.com/dop251/goja.(*vm).try(0xc4204ceea0, 0xc422b27ed8, 0x0)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/vm.go:370 +0x119
github.com/loadimpact/k6/vendor/github.com/dop251/goja.AssertFunction.func1(0xfd2b40, 0x117fd20, 0x0, 0x0, 0x0, 0x0, 0x0, 0x19829e1200000001, 0x5a222c1d)
/home/emily/go/src/github.com/loadimpact/k6/vendor/github.com/dop251/goja/runtime.go:1373 +0xdd
github.com/loadimpact/k6/js.TestInterrupt(0xc4210af0e0)
/home/emily/go/src/github.com/loadimpact/k6/js/runner_test.go:259 +0x152
testing.tRunner(0xc4210af0e0, 0xb8eb10)
/usr/lib/go/src/testing/testing.go:746 +0xd0
created by testing.(*T).Run
/usr/lib/go/src/testing/testing.go:789 +0x2de
It doesn't happen if you use a value returned from Run*
, only if you get it with Get
.
The Null()
function actually returns _undefined
instead of the canonical null
value.
Lines 1445 to 1447 in 034fb74
A quick git blame shows that this bug has been present since the initial commit, but it has probably gone unnoticed since nothing appears to call/test Null()
.
For example:
fmt.Println("IsNull(Null()): ", goja.IsNull(goja.Null()))
fmt.Println("IsUndefined(Null()): ", goja.IsUndefined(goja.Null()))
fmt.Println("IsNull(Undefined()): ", goja.IsNull(goja.Undefined()))
fmt.Println("IsUndefined(Undefined()):", goja.IsUndefined(goja.Undefined()))
Outputs:
IsNull(Null()): false
IsUndefined(Null()): true
IsNull(Undefined()): false
IsUndefined(Undefined()): true
IsNull(Null())
and IsUndefined(Undefined())
being true should probably have a tiny test each to prevent goofs like this. 😄
Hi @dop251 ! This is such a cool project.
I was just curious if there is already a REPL (interpretive prompt), or an easy way to make one?
Thank you.
Is there any chance we could have objects preserve insertion order in goja? The ES spec no longer mentions if objects should be ordered or unordered, but Chrome, Firefox and Node all preserve insertion order.
Test script:
let obj = {"a": 1, "b": 2, "c": 3}
for (key in obj) { console.log(key) }
Hi, i just run a script file ,don't care the result , but got a error:
panic: SyntaxError: test.js: Line 4:4 Illegal return statement (and 2 more errors)
package main
import (
"fmt"
"github.com/dop251/goja"
)
func main() {
filename := "test.js"
src := `
var a=true;
if(a){
return 'err';
}
return 'ok';
`
runtime := goja.New()
_, err := runtime.RunScript(filename, src)
if err != nil {
panic(err)
}
fmt.Println("done")
}
I have a piece of JS code that does the following:
var a = Object.assign({}, state, {
isAuthenticating: false,
});
where state
is an empty object, and even though it runs fine in the browser when bundled, it fails running that same bundle in goji with the following error:
TypeError: Object has no member 'assign'
I'm not sure why this is happening, but if Babel is run through goja and it processes export
statements, they will be reordered in a seemingly arbitrary (but deterministic) fashion in the generated code.
This is normally harmless, except when the exported members are classes, which need to be initialized after their superclasses.
This script reproduces the bug - the number of members are significant, as only certain positions are swapped with each other.
export const Var1 = 0;
export const Var2 = 1;
export const Var3 = 2;
export const Var4 = 3;
export class Superclass {}
export class ClassA extends Superclass {}
export class ClassB extends Superclass {}
export class ClassC extends Superclass {}
export class ClassD extends Superclass {}
Running babel-standalone through node.js correctly produces... (amended for brevity):
function _possibleConstructorReturn(self, call) { /* ... */ }
function _inherits(subClass, superClass) { /* ... */ }
function _classCallCheck(instance, Constructor) { /* ... */ }
var Var1 = exports.Var1 = 0;
var Var2 = exports.Var2 = 1;
var Var3 = exports.Var3 = 2;
var Var4 = exports.Var4 = 3;
var Superclass = exports.Superclass = function Superclass() { /* ... */ };
var ClassA = exports.ClassA = function (_Superclass) { /* ... */ }(Superclass);
var ClassB = exports.ClassB = function (_Superclass2) { /* ... */ }(Superclass);
var ClassC = exports.ClassC = function (_Superclass3) { /* ... */ }(Superclass);
var ClassD = exports.ClassD = function (_Superclass4) { /* ... */ }(Superclass);
However, running it through goja produces:
var Var2 = exports.Var2 = 1;
var Var3 = exports.Var3 = 2;
function _classCallCheck(instance, Constructor) { /* ... */ }
var Var1 = exports.Var1 = 0;
function _inherits(subClass, superClass) { /* ... */ }
function _possibleConstructorReturn(self, call) { /* ... */ }
var ClassC = exports.ClassC = function (_Superclass3) { /* ... */ }(Superclass);
var Superclass = exports.Superclass = function Superclass() { /* ... */ };
var ClassA = exports.ClassA = function (_Superclass) { /* ... */ }(Superclass);
var ClassB = exports.ClassB = function (_Superclass2) { /* ... */ }(Superclass);
var Var4 = exports.Var4 = 3;
var ClassD = exports.ClassD = function (_Superclass4) { /* ... */ }(Superclass);
In which the (individually correct) pieces are jumbled, in a way that puts ClassC above Superclass, thus making it inherit from undefined, which produces a runtime error.
Right now, ToValue
treats objects of type t map[string]interface{}
as objectGoReflect
but should probably treat as standard objectGoMapSimple
.
I get SIGSEGV when importing goja, e.g. main.go
package main
import (
_ "github.com/dop251/goja"
)
func main() {
}
$ go get
$ go run main.go
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x48 pc=0x56c37b]
goroutine 1 [running]:
golang.org/x/text/collate.newCollator(0x0, 0x0, 0x0)
/home/user/go/src/golang.org/x/text/collate/option.go:30 +0x6b
golang.org/x/text/collate.New(0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0)
/home/user/go/src/golang.org/x/text/collate/collate.go:60 +0xbc
github.com/dop251/goja.init()
/home/user/go/src/github.com/dop251/goja/builtin_string.go:15 +0x12d
main.init()
<autogenerated>:1 +0x44
exit status 2
I use ArchLinux with go package version 2:1.9-1
, go version go1.9 linux/amd64
, goja revision 0aeff75
.
Reproducible test case:
import (
"reflect"
"testing"
"github.com/dop251/goja"
"github.com/pkg/errors"
)
type FieldNameMapper struct{}
func (FieldNameMapper) FieldName(t reflect.Type, f reflect.StructField) string {
return f.Name
}
func (FieldNameMapper) MethodName(t reflect.Type, m reflect.Method) string {
return m.Name
}
func TestThrowPkgError(t *testing.T) {
rt := goja.New()
rt.SetFieldNameMapper(FieldNameMapper{})
rt.Set("fn", func() { panic(rt.NewGoError(errors.New("Error"))) })
rt.RunString("fn()")
}
Crash log:
/usr/local/Cellar/go/1.8/libexec/src/testing/testing.go:622 +0x29d
panic(0x1600520, 0xc420364000)
/usr/local/Cellar/go/1.8/libexec/src/runtime/panic.go:489 +0x2cf
github.com/dop251/goja.(*Runtime).RunProgram.func1(0xc420c6fec8)
$GOPATH/src/github.com/dop251/goja/runtime.go:756 +0x98
panic(0x1600520, 0xc420364000)
/usr/local/Cellar/go/1.8/libexec/src/runtime/panic.go:489 +0x2cf
github.com/dop251/goja.(*vm).try.func1(0xc42006ec30, 0x0, 0xc420c6fda0, 0x0, 0x0, 0x0, 0xc420c6fdf8)
$GOPATH/src/github.com/dop251/goja/vm.go:364 +0x32e
panic(0x15df480, 0xc420385ff0)
/usr/local/Cellar/go/1.8/libexec/src/runtime/panic.go:489 +0x2cf
github.com/dop251/goja.(*Object).Set.func1(0xc420c6f518)
$GOPATH/src/github.com/dop251/goja/value.go:725 +0x98
panic(0x15df480, 0xc420385ff0)
/usr/local/Cellar/go/1.8/libexec/src/runtime/panic.go:489 +0x2cf
reflect.(*rtype).NumField(0x16130a0, 0x3)
/usr/local/Cellar/go/1.8/libexec/src/reflect/type.go:1015 +0x89
github.com/dop251/goja.(*Runtime).buildFieldInfo(0xc420b34600, 0x1a02e60, 0x16130a0, 0xc420385fe0, 0x1, 0x1, 0xc420221800)
$GOPATH/src/github.com/dop251/goja/object_goreflect.go:415 +0x4f
github.com/dop251/goja.(*Runtime).buildFieldInfo(0xc420b34600, 0x1a02e60, 0x16453c0, 0x0, 0x0, 0x0, 0xc420221800)
$GOPATH/src/github.com/dop251/goja/object_goreflect.go:448 +0x373
github.com/dop251/goja.(*Runtime).buildTypeInfo(0xc420b34600, 0x1a02e60, 0x16453c0, 0x1b94b20)
$GOPATH/src/github.com/dop251/goja/object_goreflect.go:459 +0x5df
github.com/dop251/goja.(*Runtime).typeInfo(0xc420b34600, 0x1a02e60, 0x16453c0, 0x1a02e60)
$GOPATH/src/github.com/dop251/goja/object_goreflect.go:492 +0xa2
github.com/dop251/goja.(*objectGoReflect).init(0xc4200a82d0)
$GOPATH/src/github.com/dop251/goja/object_goreflect.go:69 +0x2b8
github.com/dop251/goja.(*Runtime).ToValue(0xc420b34600, 0x1620f00, 0xc420cd1fe0, 0xc420385f30, 0xc420385f20)
$GOPATH/src/github.com/dop251/goja/runtime.go:956 +0x2f8
github.com/dop251/goja.(*Object).Set(0xc420cca000, 0x16b550b, 0x5, 0x1620f00, 0xc420cd1fe0, 0x0, 0x0)
$GOPATH/src/github.com/dop251/goja/value.go:730 +0x87
github.com/dop251/goja.(*Runtime).NewGoError(0xc420b34600, 0x19f25c0, 0xc420cd1fe0, 0xc420cd1fe0)
$GOPATH/src/github.com/dop251/goja/runtime.go:295 +0xdd
[...].TestThrowPkgError.func1()
$GOPATH/src/.../crash_test.go:24 +0x61
reflect.Value.call(0x15d8e60, 0xc420385db0, 0x13, 0x16b4c19, 0x4, 0x1b93c18, 0x0, 0x0, 0x1041fdc, 0xc420385f00, ...)
/usr/local/Cellar/go/1.8/libexec/src/reflect/value.go:434 +0x91f
reflect.Value.Call(0x15d8e60, 0xc420385db0, 0x13, 0x1b93c18, 0x0, 0x0, 0x1, 0x1, 0x3)
/usr/local/Cellar/go/1.8/libexec/src/reflect/value.go:302 +0xa4
github.com/dop251/goja.(*Runtime).wrapReflectFunc.func1(0x1a00a60, 0x1b938c0, 0xc420cd1f40, 0x0, 0x0, 0xc4202217c0, 0x0)
$GOPATH/src/github.com/dop251/goja/runtime.go:1014 +0x812
github.com/dop251/goja.(*vm)._nativeCall(0xc42006ec30, 0xc4207f48f0, 0x0)
$GOPATH/src/github.com/dop251/goja/vm.go:1827 +0x2e9
github.com/dop251/goja.call.exec(0xc400000000, 0xc42006ec30)
$GOPATH/src/github.com/dop251/goja/vm.go:1808 +0x6ff
github.com/dop251/goja.(*call).exec(0xc420385dfc, 0xc42006ec30)
<autogenerated>:817 +0x55
github.com/dop251/goja.(*vm).run(0xc42006ec30)
$GOPATH/src/github.com/dop251/goja/vm.go:288 +0x7f
github.com/dop251/goja.(*vm).(github.com/dop251/goja.run)-fm()
$GOPATH/src/github.com/dop251/goja/vm.go:375 +0x2a
github.com/dop251/goja.(*vm).try(0xc42006ec30, 0xc420c6fe00, 0x0)
$GOPATH/src/github.com/dop251/goja/vm.go:370 +0x119
github.com/dop251/goja.(*vm).runTry(0xc42006ec30, 0x16d3860)
$GOPATH/src/github.com/dop251/goja/vm.go:375 +0x4f
github.com/dop251/goja.(*Runtime).RunProgram(0xc420b34600, 0xc420858900, 0x0, 0x0, 0x0, 0x0)
$GOPATH/src/github.com/dop251/goja/runtime.go:767 +0x210
github.com/dop251/goja.(*Runtime).RunScript(0xc420b34600, 0x0, 0x0, 0x16b4e8d, 0x4, 0x2, 0x1a00420, 0xc420cd1ea0, 0xc420385d00)
$GOPATH/src/github.com/dop251/goja/runtime.go:746 +0xa8
github.com/dop251/goja.(*Runtime).RunString(0xc420b34600, 0x16b4e8d, 0x4, 0x15d8e60, 0xc420385db0, 0x105ae25, 0x10759d2)
$GOPATH/src/github.com/dop251/goja/runtime.go:735 +0x51
[...].TestThrowPkgError(0xc42006e9c0)
$GOPATH/src/.../crash_test.go:25 +0x12a
testing.tRunner(0xc42006e9c0, 0x16d3a30)
/usr/local/Cellar/go/1.8/libexec/src/testing/testing.go:657 +0x96
created by testing.(*T).Run
/usr/local/Cellar/go/1.8/libexec/src/testing/testing.go:697 +0x2ca
import * as flow from 'flow';
function executeFlow() {
try {
let result = flow.invoke("download", 0, { "url": "http://www.huawei.com" });
console.log(result);
} catch (error) {
console.error(error.name, error.message);
}
}
executeFlow();
flow.invoke is an go function. the code execute very slowly on follow stack:
goroutine 1695 [runnable]:
vendor/github.com/dop251/goja.(*baseObject).init(...)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/object.go:98
vendor/github.com/dop251/goja.(*arrayObject).init(...)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:19
vendor/github.com/dop251/goja.(*sparseArrayObject).expand(0xc42f7cc3c0, 0x1d295c0)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:331 +0x14e
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f7cc3c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:200 +0x269
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f7cc280, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc4303185a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc430318500, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f7cc140, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f7cc000, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc4303183c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc430318280, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc430318140, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc430318000, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42c1786e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42c1785a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42c178460, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42c178320, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42c1781e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42c1780a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42d17a820, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42d17a6e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42d17a5a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42d17a460, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42d17a320, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42d17a1e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42d17a0a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473f40, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f473e00, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473cc0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f473b80, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473ae0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f4739a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473900, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f4737c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473680, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f473540, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473400, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f4732c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f473180, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f473040, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f472f00, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f472dc0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f472c80, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f472b40, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f472a00, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f4728c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f472640, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f472500, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f4723c0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f472280, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f472140, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f472000, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed85f40, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed85e00, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed85cc0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed85b80, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed85a40, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed859a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed85860, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed85720, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed855e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed854a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed85360, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed85220, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed850e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed84fa0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed84e60, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed84d20, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed84be0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed84aa0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed84960, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed84820, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed846e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed845a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed84460, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed84320, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42ed841e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42ed840a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f785ea0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f785d60, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f785c20, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f785ae0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f7859a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f785860, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f785720, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f7855e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f7854a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f785360, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f785220, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f7850e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f784fa0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f784e60, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f784d20, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f784be0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f784aa0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f784960, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f784820, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f7846e0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f7845a0, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
vendor/github.com/dop251/goja.(*sparseArrayObject).putIdx(0xc42f784460, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array_sparse.go:208 +0x514
vendor/github.com/dop251/goja.(*arrayObject).putIdx(0xc42f784320, 0x162d, 0x1d28f80, 0xc424cae440, 0x0, 0x0, 0x0, 0x1d295c0, 0xc4222b7790)
/data1/baas/bingo/src/vendor/github.com/dop251/goja/array.go:233 +0x2dd
created by net/http.(*Server).Serve
/data1/baas/my/go/src/net/http/server.go:2720 +0x288
Hi, I realise this is a duplicate of #27 but it was closed so I thought I'd create a new one. The line that is causing this error for me was calling a react-dom/server render to string with a react-redux Provider wrapper around a RouterContext.
An easy way to see this happen is to clone latest of https://github.com/olebedev/go-starter-kit and run 'make install' and 'make serve' to run the app. With js disabled, no server side render occurs. You just have to change line 235 to assign an error var and check if it's not nil in server/react.go. I just panicked on it to produce output similar to:
main.(*JSVM).Handle.func1(0xc420180180)
/Users/pieter/coding/go/src/teststarter/server/react.go:238 +0x232
teststarter/vendor/github.com/dop251/goja_nodejs/eventloop.(*EventLoop).RunOnLoop.func1()
/Users/pieter/coding/go/src/teststarter/vendor/github.com/dop251/goja_nodejs/eventloop/eventloop.go:109 +0x31
teststarter/vendor/github.com/dop251/goja_nodejs/eventloop.(*EventLoop).run(0xc4201b0000, 0xc4201b0201)
/Users/pieter/coding/go/src/teststarter/vendor/github.com/dop251/goja_nodejs/eventloop/eventloop.go:116 +0x77
created by teststarter/vendor/github.com/dop251/goja_nodejs/eventloop.(*EventLoop).Start
/Users/pieter/coding/go/src/teststarter/vendor/github.com/dop251/goja_nodejs/eventloop/eventloop.go:87 +0x44
I had the exact same error with a smaller test that renders out a react app, and the same error occurred when react-redux's connect function was used. I don't really know where to start looking for a cause to this. If you've got an idea of where I can start looking, I might be able to help fix the issue. Thanks!
My go env is:
GOARCH="amd64"
GOBIN=""
GOEXE=""
GOHOSTARCH="amd64"
GOHOSTOS="darwin"
GOOS="darwin"
GOPATH="/Users/pieter/coding/go"
GORACE=""
GOROOT="/usr/local/Cellar/go/1.8.1/libexec"
GOTOOLDIR="/usr/local/Cellar/go/1.8.1/libexec/pkg/tool/darwin_amd64"
GCCGO="gccgo"
CC="clang"
GOGCCFLAGS="-fPIC -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/ss/7vgx61gj12373gbg_pbst6pw0000gn/T/go-build423786578=/tmp/go-build -gno-record-gcc-switches -fno-common"
CXX="clang++"
CGO_ENABLED="1"
PKG_CONFIG="pkg-config"
CGO_CFLAGS="-g -O2"
CGO_CPPFLAGS=""
CGO_CXXFLAGS="-g -O2"
CGO_FFLAGS="-g -O2"
CGO_LDFLAGS="-g -O2"
Running this code will cause a panic in string_unicode.go (https://github.com/dop251/goja/blob/master/string_unicode.go#L270)
"abꞐcde".lastIndexOf("cd");
The indexing of the string is starting at the length of the string rather than the appropriate index. By fixing the start value before that line the function works correctly. My current version of lastIndex looks like this:
func (s unicodeString) lastIndex(substr valueString, start int64) int64 {
var ss []uint16
switch substr := substr.(type) {
case unicodeString:
ss = substr
case asciiString:
ss = make([]uint16, len(substr))
for i := 0; i < len(substr); i++ {
ss[i] = uint16(substr[i])
}
default:
panic(fmt.Errorf("Unknown string type: %T", substr))
}
if start > int64(len(s)-len(ss)) {
start = int64(len(s) - len(ss))
}
// TODO: optimise
for start >= 0 {
for i := int64(0); i < int64(len(ss)); i++ {
if s[start+i] != ss[i] {
goto nomatch
}
}
return start
nomatch:
start--
}
return -1
}
Most of the time,i just need use runtime.ToValue() convert GO value , such as
type test struct {
runtime *goja.Runtime
}
func (t *test) myFunc(call goja.FunctionCall) goja.Value {
return t.runtime.ToValue(1234)
}
but, must define a struct, if provider a runtime parameter, Things could be better.
thank you.
Hi,
It has support for tuples? Multiple return values?
Thanks.
following code panics:
package main
import (
"fmt"
"time"
"github.com/dop251/goja"
)
func main() {
vm := goja.New()
time.AfterFunc(200*time.Millisecond, func() {
vm.Interrupt("halt")
})
if v, err := vm.RunString(`i`); err == nil {
fmt.Println(v.Export())
} else {
if _, ok := err.(*goja.InterruptedError); ok {
fmt.Println("script took too long to execute")
} else {
fmt.Println(err.Error())
}
}
fmt.Println("it's all over")
}
here is the stack trace:
panic: runtime error: index out of range
goroutine 1 [running]:
github.com/dop251/goja.(*Program).sourceOffset(0x12765d10, 0x0, 0x1)
path/src/github.com/dop251/goja/compiler.go:170 +0x4e
github.com/dop251/goja.(*stackFrame).position(0x127be260, 0x65623a, 0x0)
path/src/github.com/dop251/goja/runtime.go:93 +0x2a
github.com/dop251/goja.(*stackFrame).write(0x127be260, 0x127bc9c0)
path/src/github.com/dop251/goja/runtime.go:108 +0x78
github.com/dop251/goja.(*Exception).Error(0x127c0360, 0x6557f1, 0x1)
path/src/github.com/dop251/goja/runtime.go:166 +0xc9
main.main()
path/src/testme/main.go:21 +0x196
One interesting feature Otto has is the ability to clone a VM, creating multiple, derivative VMs with their own state, but the same initial global scope. It's really handy for running scripts in parallel.
I can think of two ways this could be implemented:
When I try to run typescriptServices.js
downloaded from here through goja
's vm.RunString
method it panics.
Here is the small test program I used:
package main
import (
"fmt"
"io/ioutil"
"log"
"github.com/dop251/goja"
)
func main() {
buf, err := ioutil.ReadFile("typescriptServices.js")
if err != nil {
log.Fatalln(err)
}
vm := goja.New()
v, err := vm.RunString(string(buf))
if err != nil {
log.Fatalln(err)
}
fmt.Println(v)
str := `
interface Person {
firstName: string;
lastName: string;
}
function greeter(person: Person) {
return "Hello, " + person.firstName + " " + person.lastName;
}
var user = { firstName: "Jane", lastName: "User" };
document.body.innerHTML = greeter(user);
`
v, err = vm.RunString(`ts.compile('` + str + `')`)
if err != nil {
log.Fatalln("compile:", err)
}
fmt.Println(v)
}
Running the test program produces this output:
$ go run main.go
panic: Unknown statement type: *ast.DebuggerStatement [recovered]
panic: Unknown statement type: *ast.DebuggerStatement
goroutine 1 [running]:
panic(0x277880, 0xc42014b940)
/usr/local/go/src/runtime/panic.go:500 +0x1a1
github.com/dop251/goja.compile.func1(0xc4219c3d40, 0xc4219c3d48)
/Users/niels/go/src/github.com/dop251/goja/runtime.go:679 +0xc9
panic(0x277880, 0xc42014b940)
/usr/local/go/src/runtime/panic.go:458 +0x243
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x4023a0, 0xc420f3f658, 0xc4219c2700)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:53 +0x186
github.com/dop251/goja.(*compiler).compileStatements(0xc421db6180, 0xc420f3c2c0, 0x4, 0x4, 0xc421f75400)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:562 +0x96
github.com/dop251/goja.(*compiler).compileBlockStatement(0xc421db6180, 0xc420f378f0, 0x27fa00)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:586 +0x54
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x402220, 0xc420f378f0, 0x0)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:16 +0x397
github.com/dop251/goja.(*compiler).compileIfStatement(0xc421db6180, 0xc420f3c200, 0x0)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:517 +0x543
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x4025e0, 0xc420f3c200, 0xc41ff03b00)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:24 +0x3cc
github.com/dop251/goja.(*compiler).compileStatements(0xc421db6180, 0xc420f3fe40, 0x1, 0x1, 0x5f1000)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:567 +0x10c
github.com/dop251/goja.(*compiler).compileBlockStatement(0xc421db6180, 0xc420f37890, 0xffffffffffffff00)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:586 +0x54
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x402220, 0xc420f37890, 0x0)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:16 +0x397
github.com/dop251/goja.(*compiledFunctionLiteral).emitGetter(0xc421f89cc0, 0xa7401)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:785 +0x7bc
github.com/dop251/goja.(*compiler).emitExpr(0xc421db6180, 0x403c60, 0xc421f89cc0, 0xc421f90b01)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:990 +0x8c
github.com/dop251/goja.(*compiler).emitVarSetter.func1(0xc421f90b00)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:352 +0x41
github.com/dop251/goja.(*compiler).emitVarSetter1(0xc421db6180, 0xc42097d91a, 0x6, 0x23911, 0xc4219c2ec0)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:323 +0x408
github.com/dop251/goja.(*compiler).emitVarSetter(0xc421db6180, 0xc42097d91a, 0x6, 0x23911, 0x403c60, 0xc421f89cc0)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:353 +0x7b
github.com/dop251/goja.(*compiledIdentifierExpr).emitSetter(0xc4219c2f70, 0x403c60, 0xc421f89cc0)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:361 +0x5d
github.com/dop251/goja.(*compiler).compileFunction(0xc421db6180, 0xc4200a93d0)
/Users/niels/go/src/github.com/dop251/goja/compiler.go:416 +0xd5
github.com/dop251/goja.(*compiler).compileFunctions(0xc421db6180, 0xc420f3c340, 0x3, 0x4)
/Users/niels/go/src/github.com/dop251/goja/compiler.go:312 +0x76
github.com/dop251/goja.(*compiledFunctionLiteral).emitGetter(0xc421f89be0, 0xc4219c3301)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:784 +0x78d
github.com/dop251/goja.(*compiledCallExpr).emitGetter(0xc421f8d580, 0x402300)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:1432 +0x90d
github.com/dop251/goja.(*compiler).compileExpressionStatement(0xc421db6180, 0xc420f40670, 0xc4219c3500)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:594 +0xbe
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x4024a0, 0xc420f40670, 0x27fa00)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:18 +0x63d
github.com/dop251/goja.(*compiler).compileStatements(0xc421db6180, 0xc420f3a000, 0x11d, 0x200, 0xc4219c3500)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:562 +0x96
github.com/dop251/goja.(*compiler).compileBlockStatement(0xc421db6180, 0xc420e82cf0, 0x400500)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:586 +0x54
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x402220, 0xc420e82cf0, 0x100)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:16 +0x397
github.com/dop251/goja.(*compiledFunctionLiteral).emitGetter(0xc421ece020, 0xc4219c3901)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:785 +0x7bc
github.com/dop251/goja.(*compiledCallExpr).emitGetter(0xc421eca540, 0x402300)
/Users/niels/go/src/github.com/dop251/goja/compiler_expr.go:1432 +0x90d
github.com/dop251/goja.(*compiler).compileExpressionStatement(0xc421db6180, 0xc420f4a410, 0xc421db3600)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:594 +0xbe
github.com/dop251/goja.(*compiler).compileStatement(0xc421db6180, 0x4024a0, 0xc420f4a410, 0x0)
/Users/niels/go/src/github.com/dop251/goja/compiler_stmt.go:18 +0x63d
github.com/dop251/goja.(*compiler).compile(0xc421db6180, 0xc421db6140)
/Users/niels/go/src/github.com/dop251/goja/compiler.go:265 +0x219
github.com/dop251/goja.compile(0x0, 0x0, 0xc420546000, 0x4134b0, 0x0, 0x0, 0x0, 0x0)
/Users/niels/go/src/github.com/dop251/goja/runtime.go:684 +0x3e2
github.com/dop251/goja.Compile(0x0, 0x0, 0xc420546000, 0x4134b0, 0x0, 0x0, 0x0, 0x0)
/Users/niels/go/src/github.com/dop251/goja/runtime.go:642 +0x57
github.com/dop251/goja.(*Runtime).RunScript(0xc420096580, 0x0, 0x0, 0xc420546000, 0x4134b0, 0x0, 0x0, 0x0, 0x0)
/Users/niels/go/src/github.com/dop251/goja/runtime.go:713 +0x52
github.com/dop251/goja.(*Runtime).RunString(0xc420096580, 0xc420546000, 0x4134b0, 0x4136b0, 0xc420546000, 0x4134b0, 0x0)
/Users/niels/go/src/github.com/dop251/goja/runtime.go:708 +0x51
main.main()
/Users/niels/go/src/github.com/nwidger/goja-test/main.go:18 +0xfd
exit status 2
Here is a copy of the typescriptServices.js
file I used: typescriptServices.js.txt.
I have an index.js with
import utils from 'util'
and a blank util.js
results in
panic: SyntaxError: index.js: Line 7:1 Unexpected reserved word [recovered]
panic: Panic at 3: SyntaxError: index.js: Line 7:1 Unexpected reserved word [recovered]
panic: Panic at 3: SyntaxError: index.js: Line 7:1 Unexpected reserved word
I'm playing with trying to get hubot script running from a Go application
I've included the load() function from goja/main.go, which seems to work with a simple load('util.js')
Currently it is not possible (or I am unaware how it works) to register a prototype object within the VM. This hidden from the public API and is used in the builtin_
files.
It would be nice to also add a test/example for this.
var a = '**'
console.log(a[0])
console.log(a[1])
console.log(a.indexOf('中'))
console.log(a.indexOf('国')) // error!!! expected 1 but got -1
package main
import (
"fmt"
"io/ioutil"
"os"
"sync"
"time"
"github.com/dop251/goja"
"github.com/dop251/goja_nodejs/console"
"github.com/dop251/goja_nodejs/require"
)
func handErr(err error) {
if err != nil {
panic(err)
}
}
func main() {
if len(os.Args) == 2 {
filename := os.Args[1]
src, err := ioutil.ReadFile(filename)
handErr(err)
n := 10000
var w sync.WaitGroup
w.Add(n)
for i := 0; i < n; i++ {
go func() {
r, err := runFile(filename, string(src))
handErr(err)
fmt.Println(*r)
w.Done()
}()
}
w.Wait()
}
fmt.Println("done.")
<-time.After(10 * time.Minute)
}
func runFile(filename, src string) (*goja.Value, error) {
runtime := goja.New()
registry := new(require.Registry)
registry.Enable(runtime)
console.Enable(runtime)
prg, err := goja.Compile(filename, src, false)
if err != nil {
return nil, err
}
result, err := runtime.RunProgram(prg)
if err != nil {
return nil, err
}
return &result, nil
}
console.log("a.js init")
exports.foo = function (){
console.log("a.js foo()")
}
var a = require('./test/a.js')
a.foo()
console.log('b.js done.')
go build .\test.go
.\test.exe .\test\b.js
Do you have any plan for Ecma 402 support?
On average 6-7 times faster than otto. Also uses considerably less memory.
What benchmark code did you use for this, and what were the exact results?
Nice work so far :)
Goja panics with the following error message:
SyntaxError: (anonymous): Line 45:1 Unexpected reserved word
If it's passed a valid JS code with constant, like:
const foo = require("foo")
If you change const
to var
, it works.
The README states
This project was largely inspired by otto.
Could you clarify a little how it differs from Otto?
func TestByteToString(t *testing.T) {
rt := goja.New()
rt.Set("v", []byte("hi"))
rt.Set("fn", func(v string) {
assert.Equal(t, "hi", v)
})
_, err := rt.RunString(`fn(v)`)
assert.NoError(t, err)
}
Expected "hi", got "104,105". Looks like mangling of arrays that could use a special case for []byte
? The opposite direction appears to work just fine.
I feel this might not be possible, but something missing in otto
is being able to somehow limit how much memory a VM is allowed to use, or CPU (there's a time based interrupt thingy), and so on.
It would be pretty cool if this package allowed strict control over these.
Nice work, love to see more language interpreters!
I'm wondering how I could make a value created using vm.Set()
constant or immutible. I know that ES5 doesn't support const
but what about Object.freeze or similar methods? I am trying to make sure the JS callable Go functions and mapped vars can't be accidentally overwritten to another value.
I digged through the docs, tests and implementation but could not figure out how to "register" a function from javascript to go. This to create a callback-style application.
Something like this:
function myfunc() {
mygofunc();
}
// Register function with id 1234 for go
registerfunc(1234, myfunc);
I am aware of the Callable type but I have no idea how this should be used.
Hello!
Everything is in the question: what is the equivalent Go code for JS var obj = [10, 20, 30]
?
Thanks!
func TestInterrupt(t *testing.T) {
rt := goja.New()
go func() {
<-time.After(1 * time.Second)
rt.Interrupt(errors.New("hi"))
}()
v, err := rt.RunString(`while (true) {}`)
assert.Nil(t, v)
assert.EqualError(t, err, "hi")
}
A nil error is returned.
Hi there,
would be nice to have the possibility to destory the created vm if it makes the go-gb worker more efficent.
I would propose a Close() error
method.
Cheers,
Artan
Reproduction js:
(function(){
console.log("test");
function x(a, b) {
if (a || b) return 1;
return 0;
}
function y(b) {
return [].concat(Array.prototype.slice.call(arguments))
}
x.apply(this, y(void 0, 1));
})();
FunctionCall.Arguments that goes to x ends up having nil instead of undefined.
Hello @dop251,
first of all, need to say thank you for the project.
Just for your information.
The benchmarks were good enough to try to switch to goja from go-duktape. One critical thing I need to have is a fetch polyfill. So, now I have one written for goja, have a look. It's a window.fetch JavaScript polyfill. Would be good if you leave some feedback here.
Thanks
I've forked and started dabbling into allowing package users to create "native" constructors and instances.
I'm implementing part of the Web API (Response
, Request
, Headers
, etc.) and needed some way to create more complex instances that kept a state outside of goja.
Here's my rough implementation so far:
// constructor.go
package goja
type NativeFunction func(FunctionCall) Value
type ConstructorFunction func(args []Value, inst *Object) *Object
type Method struct {
call NativeFunction
length int
}
type Constructor struct {
Name string
r *Runtime
proto *Object
static *Object
ctor ConstructorFunction
length int
methods map[string]Method
}
func (r *Runtime) NewConstructor(name string, ctor ConstructorFunction, length int) *Constructor {
return &Constructor{
Name: name,
r: r,
ctor: ctor,
length: length,
methods: map[string]Method{},
}
}
func (c *Constructor) createProto(val *Object) objectImpl {
o := &baseObject{
class: c.Name,
val: val,
extensible: true,
prototype: c.r.global.ObjectPrototype,
}
o.init()
for name, method := range c.methods {
o._putProp(name, c.r.newNativeFunc(method.call, nil, name, nil, method.length), true, false, true)
}
return o
}
func (c *Constructor) createStatic(val *Object) objectImpl {
o := c.r.newNativeFuncConstructObj(val, c.ctor, c.Name, c.proto, c.length)
// _putProp here...
return o
}
func (c *Constructor) NewInstance(args []Value, proto *Object) *Object {
bo := c.r.newBaseObject(proto, c.Name)
val := c.ctor(args, bo.val)
return val
}
func (c *Constructor) Init() {
c.proto = c.r.newLazyObject(c.createProto)
c.static = c.r.newLazyObject(c.createStatic)
c.r.addToGlobal(c.Name, c.static)
}
func (c *Constructor) DefineFunction(name string, call NativeFunction, length int) {
c.methods[name] = Method{call, length}
}
Surprisingly, this works ok. Here's an example of how I'm using it:
func constructResponse(vm *goja.Runtime) goja.ConstructorFunction {
return func(args []goja.Value, res *goja.Object) *goja.Object {
res.Set("body", nil)
res.Set("bodyUsed", false)
res.Set("ok", true)
res.Set("redirected", false)
res.Set("status", 200)
res.Set("statusText", "OK")
res.Set("type", "default")
res.Set("url", "")
res.Set("text", bodyText(vm, strings.NewReader("")))
return res
}
}
func initResponse(vm *goja.Runtime) {
c := vm.NewConstructor("Response", constructResponse(vm), 2)
c.Init()
}
This works. Except I'm left with the same issue trying to keep state around. I have to create a bunch of closures.
For instance, for the text()
function: I'd like it to keep around a io.Reader
and read from it from the text()
function.
As far as I can tell, there's already a similar pattern in goja. Namely, the newDateObject
function seems to create a special kind of object that embed baseObject
and a few more fields. It makes sense, except that's all unexported and I'm not sure how I can make it generic enough.
I'm mostly looking for advice on how to proceed. Ideally, from a package user's perspective, the developer would only have to provide a constructor and a struct instance to make it work.
I think maybe abstracting and exporting a few more structures would help. I guess I'd mostly need to create a few structs embedding baseObject
or implementing objectImpl
.
Any thoughts?
Hi
I'm building a universal web app, where goja handles JS rendering on the server.
Now I'm in the phase of doing performance tests, and for some reason goja seems to be leaking memory.
I'm using artillery.io to perform load tests, and after the load test it seems vm.run is holding memory and not releasing it. I'm not too sure if that's goja fault (probably not), but maybe you know anything off the top of your head how to prevent memory leaks?
Here's the heap pprof BEFORE the load test:
http://jarosinski.uk/files/heap_base.svg
And here's the one after the load test:
http://jarosinski.uk/files/heap_current.svg
And this is the code that handles VM creation:
https://gist.github.com/iKonrad/a046de73c8e478b3cac07fa9e419af58
Any help or suggestions will be appreciated.
A function that takes a goja.Value
argument will, when bridged to JS and invoked with null
for that argument, receive a nil
goja.Value, which is not recognised as a proper null by goja.IsNull()
.
Test case:
func TestNullArguments(t *testing.T) {
rt := goja.New()
rt.Set("fn", func(v goja.Value) {
if v == nil {
t.Error("null becomes nil")
}
if !goja.IsNull(v) {
t.Error("null is not null")
}
})
rt.RunString(`fn(null);`)
}
Does this library support ecmascript 7?
I have what-wg-fetch loaded by one of the react modules, it seems to be expecting self to be defined.
ReferenceError: self is not defined
at bundle.js:56189:19(7)
the traced code at line 56189:
56182 /***/ function(module, exports, __webpack_require__) {
56183
56184 // the whatwg-fetch polyfill installs the fetch() function
56185 // on the global object (window or self)
56186 //
56187 // Return that as the export for use in Webpack, Browserify etc.
56188 __webpack_require__(395);
56189 module.exports = self.fetch.bind(self);
56190
56191
I tried setting 'self' as an empty object before running bundle.js, but then I get the error "Host Object is not extensible".
What can I do about this and any other future modules that expects to run on browsers? Was setting self/window an empty object a good approach?
For example:
2017/11/24 16:18:52 values: [normal throw]
2017/11/24 16:18:52 0: goja.enterFuncStashless({1 3})
2017/11/24 16:18:52 1: goja.try({15 0 false})
2017/11/24 16:18:52 2: goja._newObject({})
2017/11/24 16:18:52 3: goja.loadVal(0)
2017/11/24 16:18:52 4: goja.setProp1(type)
2017/11/24 16:18:52 5: goja.loadStack(-1)
2017/11/24 16:18:52 6: goja._dup({})
2017/11/24 16:18:52 7: goja.getPropCallee(call)
2017/11/24 16:18:52 run buildnext.then called
2017/11/24 16:18:52 8: goja.loadStack(-2)
2017/11/24 16:18:52 9: goja.loadStack(-3)
2017/11/24 16:18:52 10: goja.call(2)
2017/11/24 16:18:52 11: goja.setProp1(arg)
2017/11/24 16:18:52 12: goja._halt({})
2017/11/24 16:18:52 13: goja._retStashless({})
2017/11/24 16:18:52 14: goja._halt({})
2017/11/24 16:18:52 15: goja.jump(11)
2017/11/24 16:18:52 16: goja.storeStack(1)
2017/11/24 16:18:52 17: goja._pop({})
2017/11/24 16:18:52 18: goja._newObject({})
2017/11/24 16:18:52 19: goja.loadVal(1)
2017/11/24 16:18:52 20: goja.setProp1(type)
2017/11/24 16:18:52 21: goja.loadStack(1)
2017/11/24 16:18:52 22: goja.setProp1(arg)
2017/11/24 16:18:52 23: goja._halt({})
2017/11/24 16:18:52 24: goja._retStashless({})
2017/11/24 16:18:52 25: goja._halt({})
2017/11/24 16:18:52 26: goja._loadUndef({})
2017/11/24 16:18:52 27: goja._retStashless({})
2017/11/24 16:18:52 values: [true Promise can't be resolved itself 1 false]
2017/11/24 16:18:52 0: goja.enterFunc(1)
2017/11/24 16:18:52 1: goja.bindName(value)
2017/11/24 16:18:52 2: goja.bindName(promise)
2017/11/24 16:18:52 3: goja.bindName(then)
2017/11/24 16:18:52 4: goja.bindName( __tmp0)
2017/11/24 16:18:52 5: goja.loadStack(0)
2017/11/24 16:18:52 6: goja.setLocal(1)
2017/11/24 16:18:52 7: goja._pop({})
2017/11/24 16:18:52 8: goja.getLocal(1)
2017/11/24 16:18:52 9: goja.getProp(_d)
2017/11/24 16:18:52 10: goja.jne(3)
2017/11/24 16:18:52 11: goja._loadUndef({})
2017/11/24 16:18:52 12: goja._ret({})
2017/11/24 16:18:52 13: goja.getLocal(1)
2017/11/24 16:18:52 14: goja.loadVal(0)
2017/11/24 16:18:52 15: goja.setPropStrict(_d)
2017/11/24 16:18:52 16: goja._pop({})
2017/11/24 16:18:52 17: goja.getLocal(1)
2017/11/24 16:18:52 18: goja.getProp(_w)
2017/11/24 16:18:52 19: goja.jeq1(3)
2017/11/24 16:18:52 20: goja._pop({})
2017/11/24 16:18:52 21: goja.getLocal(1)
2017/11/24 16:18:52 22: goja.setLocal(1)
2017/11/24 16:18:52 23: goja._pop({})
2017/11/24 16:18:52 24: goja.try({38 0 false})
2017/11/24 16:18:52 25: goja.getLocal(1)
2017/11/24 16:18:52 26: goja.getLocal(0)
2017/11/24 16:18:52 27: goja._op_strict_eq({})
2017/11/24 16:18:52 28: goja.jne(6)
2017/11/24 16:18:52 29: goja._loadUndef({})
2017/11/24 16:18:52 30: goja.getLocal(16777235)
2017/11/24 16:18:52 31: goja.loadVal(1)
2017/11/24 16:18:52 32: goja.call(1)
2017/11/24 16:18:52 33: goja._throw({})
2017/11/24 16:18:52 34: goja._loadUndef({})
2017/11/24 16:18:52 35: goja.getLocal(16777247)
2017/11/24 16:18:52 36: goja.getLocal(0)
2017/11/24 16:18:52 37: goja.call(1)
2017/11/24 16:18:52 38: goja.setLocal(2)
2017/11/24 16:18:52 39: goja.jne(7)
2017/11/24 16:18:52 40: goja._loadUndef({})
2017/11/24 16:18:52 41: goja.getLocal(16777230)
2017/11/24 16:18:52 42: *goja.newFunc(&{0xc422276de0 0 true 144833 145075})
2017/11/24 16:18:52 values: [false 1]
2017/11/24 16:18:52 > 0: goja.enterFuncStashless({2 0})
2017/11/24 16:18:52 > 1: goja._newObject({})
2017/11/24 16:18:52 > 2: goja.getLocal(1)
2017/11/24 16:18:52 > 3: goja.setProp1(_w)
2017/11/24 16:18:52 > 4: goja.loadVal(0)
2017/11/24 16:18:52 > 5: goja.setProp1(_d)
2017/11/24 16:18:52 > 6: goja.storeStack(1)
2017/11/24 16:18:52 > 7: goja._pop({})
2017/11/24 16:18:52 > 8: goja.try({21 0 false})
2017/11/24 16:18:52 > 9: goja.getLocal(2)
2017/11/24 16:18:52 > 10: goja._dup({})
2017/11/24 16:18:52 > 11: goja.getPropCallee(call)
2017/11/24 16:18:52 > 12: goja.getLocal(0)
2017/11/24 16:18:52 > 13: goja._loadUndef({})
2017/11/24 16:18:52 > 14: goja.getLocal(16777221)
2017/11/24 16:18:52 > 15: goja.getLocal(16777253)
2017/11/24 16:18:52 > 16: goja.loadStack(1)
2017/11/24 16:18:52 > 17: goja.loadVal(1)
2017/11/24 16:18:52 > 18: goja.call(3)
2017/11/24 16:18:52 > 19: goja._loadUndef({})
2017/11/24 16:18:52 > 20: goja.getLocal(16777221)
2017/11/24 16:18:52 > 21: goja.getLocal(16777252)
2017/11/24 16:18:52 > 22: goja.loadStack(1)
2017/11/24 16:18:52 > 23: goja.loadVal(1)
2017/11/24 16:18:52 > 24: goja.call(3)
2017/11/24 16:18:52 > 25: goja.call(3)
2017/11/24 16:18:52 > 26: goja._pop({})
2017/11/24 16:18:52 > 27: goja._halt({})
2017/11/24 16:18:52 > 28: goja.jump(11)
2017/11/24 16:18:52 > 29: goja.storeStack(2)
2017/11/24 16:18:52 > 30: goja._pop({})
2017/11/24 16:18:52 > 31: goja.getLocal(16777252)
2017/11/24 16:18:52 > 32: goja._dup({})
2017/11/24 16:18:52 > 33: goja.getPropCallee(call)
2017/11/24 16:18:52 > 34: goja.loadStack(1)
2017/11/24 16:18:52 > 35: goja.loadStack(2)
2017/11/24 16:18:52 > 36: goja.call(2)
2017/11/24 16:18:52 > 37: goja._pop({})
2017/11/24 16:18:52 > 38: goja._halt({})
2017/11/24 16:18:52 > 39: goja._loadUndef({})
2017/11/24 16:18:52 > 40: goja._retStashless({})
2017/11/24 16:18:52 43: goja.call(1)
2017/11/24 16:18:52 44: goja._pop({})
2017/11/24 16:18:52 45: goja.jump(15)
2017/11/24 16:18:52 46: goja.getLocal(1)
2017/11/24 16:18:52 47: goja.getLocal(0)
2017/11/24 16:18:52 48: goja.setPropStrict(_v)
2017/11/24 16:18:52 49: goja._pop({})
2017/11/24 16:18:52 50: goja.getLocal(1)
2017/11/24 16:18:52 51: goja.loadVal(2)
2017/11/24 16:18:52 52: goja.setPropStrict(_s)
2017/11/24 16:18:52 53: goja._pop({})
2017/11/24 16:18:52 54: goja._loadUndef({})
2017/11/24 16:18:52 55: goja.getLocal(16777248)
2017/11/24 16:18:52 56: goja.getLocal(1)
2017/11/24 16:18:52 57: goja.loadVal(3)
2017/11/24 16:18:52 58: goja.call(2)
2017/11/24 16:18:52 59: goja._pop({})
2017/11/24 16:18:52 60: goja._halt({})
2017/11/24 16:18:52 61: goja.jump(15)
2017/11/24 16:18:52 62: goja.setLocal(3)
2017/11/24 16:18:52 63: goja._pop({})
2017/11/24 16:18:52 64: goja.getLocal(16777252)
2017/11/24 16:18:52 65: goja._dup({})
2017/11/24 16:18:52 66: goja.getPropCallee(call)
2017/11/24 16:18:52 67: goja._newObject({})
2017/11/24 16:18:52 68: goja.getLocal(1)
2017/11/24 16:18:52 69: goja.setProp1(_w)
2017/11/24 16:18:52 70: goja.loadVal(3)
2017/11/24 16:18:52 71: goja.setProp1(_d)
2017/11/24 16:18:52 72: goja.getLocal(3)
2017/11/24 16:18:52 73: goja.call(2)
2017/11/24 16:18:52 74: goja._pop({})
2017/11/24 16:18:52 75: goja._halt({})
2017/11/24 16:18:52 76: goja._loadUndef({})
2017/11/24 16:18:52 77: goja._ret({})
2017/11/24 16:18:52 values: [2 1 true Promise-chain cycle]
2017/11/24 16:18:52 0: goja.enterFuncStashless({7 1})
2017/11/24 16:18:52 1: goja.getLocal(1)
2017/11/24 16:18:52 2: goja.jne(4)
2017/11/24 16:18:52 3: goja.loadStack(-1)
2017/11/24 16:18:52 4: goja.getProp(ok)
2017/11/24 16:18:52 5: goja.jump(3)
2017/11/24 16:18:52 6: goja.loadStack(-1)
2017/11/24 16:18:52 7: goja.getProp(fail)
2017/11/24 16:18:52 8: goja.storeStack(1)
2017/11/24 16:18:52 9: goja._pop({})
2017/11/24 16:18:52 10: goja.loadStack(-1)
2017/11/24 16:18:52 11: goja.getProp(resolve)
2017/11/24 16:18:52 12: goja.storeStack(2)
2017/11/24 16:18:52 13: goja._pop({})
2017/11/24 16:18:52 14: goja.loadStack(-1)
2017/11/24 16:18:52 15: goja.getProp(reject)
2017/11/24 16:18:52 16: goja.storeStack(3)
2017/11/24 16:18:52 17: goja._pop({})
2017/11/24 16:18:52 18: goja.loadStack(-1)
2017/11/24 16:18:52 19: goja.getProp(domain)
2017/11/24 16:18:52 20: goja.storeStack(4)
2017/11/24 16:18:52 21: goja._pop({})
2017/11/24 16:18:52 22: goja.try({90 0 false})
2017/11/24 16:18:52 23: goja.loadStack(1)
2017/11/24 16:18:52 24: goja.jne(81)
2017/11/24 16:18:52 25: goja.getLocal(1)
2017/11/24 16:18:52 26: goja._not({})
2017/11/24 16:18:52 27: goja.jne(15)
2017/11/24 16:18:52 28: goja.getLocal(16777216)
2017/11/24 16:18:52 29: goja.getProp(_h)
2017/11/24 16:18:52 30: goja.loadVal(0)
2017/11/24 16:18:52 31: goja._op_eq({})
2017/11/24 16:18:52 32: goja.jne(6)
2017/11/24 16:18:52 33: goja._loadUndef({})
2017/11/24 16:18:52 34: goja.getLocal(33554467)
2017/11/24 16:18:52 35: goja.getLocal(16777216)
2017/11/24 16:18:52 36: goja.call(1)
2017/11/24 16:18:52 37: goja._pop({})
2017/11/24 16:18:52 38: goja.getLocal(16777216)
2017/11/24 16:18:52 39: goja.loadVal(1)
2017/11/24 16:18:52 40: goja.setPropStrict(_h)
2017/11/24 16:18:52 41: goja._pop({})
2017/11/24 16:18:52 42: goja.loadStack(1)
2017/11/24 16:18:52 43: goja.loadVal(2)
2017/11/24 16:18:52 44: goja._op_strict_eq({})
2017/11/24 16:18:52 45: goja.jne(5)
2017/11/24 16:18:52 46: goja.getLocal(0)
2017/11/24 16:18:52 47: goja.storeStack(5)
2017/11/24 16:18:52 48: goja._pop({})
2017/11/24 16:18:52 49: goja.jump(21)
2017/11/24 16:18:52 50: goja.loadStack(4)
2017/11/24 16:18:52 51: goja.jne(6)
2017/11/24 16:18:52 52: goja.loadStack(4)
2017/11/24 16:18:52 53: goja._dup({})
2017/11/24 16:18:52 54: goja.getPropCallee(enter)
2017/11/24 16:18:52 55: goja.call(0)
2017/11/24 16:18:52 56: goja._pop({})
2017/11/24 16:18:52 57: goja._loadUndef({})
2017/11/24 16:18:52 58: goja.loadStack(1)
2017/11/24 16:18:52 59: goja.getLocal(0)
2017/11/24 16:18:52 60: goja.call(1)
2017/11/24 16:18:52 61: goja.storeStack(5)
2017/11/24 16:18:52 62: goja._pop({})
2017/11/24 16:18:52 63: goja.loadStack(4)
2017/11/24 16:18:52 64: goja.jne(6)
2017/11/24 16:18:52 65: goja.loadStack(4)
2017/11/24 16:18:52 66: goja._dup({})
2017/11/24 16:18:52 67: goja.getPropCallee(exit)
2017/11/24 16:18:52 68: goja.call(0)
2017/11/24 16:18:52 69: goja._pop({})
2017/11/24 16:18:52 70: goja.loadStack(5)
2017/11/24 16:18:52 71: goja.loadStack(-1)
2017/11/24 16:18:52 72: goja.getProp(promise)
2017/11/24 16:18:52 73: goja._op_strict_eq({})
2017/11/24 16:18:52 74: goja.jne(10)
2017/11/24 16:18:52 75: goja._loadUndef({})
2017/11/24 16:18:52 76: goja.loadStack(3)
2017/11/24 16:18:52 77: goja._loadUndef({})
2017/11/24 16:18:52 78: goja.getLocal(33554451)
2017/11/24 16:18:52 79: goja.loadVal(3)
2017/11/24 16:18:52 80: goja.call(1)
2017/11/24 16:18:52 81: goja.call(1)
2017/11/24 16:18:52 82: goja._pop({})
2017/11/24 16:18:52 83: goja.jump(21)
2017/11/24 16:18:52 84: goja._loadUndef({})
2017/11/24 16:18:52 85: goja.getLocal(33554463)
2017/11/24 16:18:52 86: goja.loadStack(5)
2017/11/24 16:18:52 87: goja.call(1)
2017/11/24 16:18:52 88: goja.storeStack(6)
2017/11/24 16:18:52 89: goja.jne(10)
2017/11/24 16:18:52 90: goja.loadStack(6)
2017/11/24 16:18:52 91: goja._dup({})
2017/11/24 16:18:52 92: goja.getPropCallee(call)
2017/11/24 16:18:52 93: goja.loadStack(5)
2017/11/24 16:18:52 94: goja.loadStack(2)
2017/11/24 16:18:52 95: goja.loadStack(3)
2017/11/24 16:18:52 96: goja.call(3)
2017/11/24 16:18:52 97: goja._pop({})
2017/11/24 16:18:52 98: goja.jump(6)
2017/11/24 16:18:52 99: goja._loadUndef({})
2017/11/24 16:18:52 100: goja.loadStack(2)
2017/11/24 16:18:52 101: goja.loadStack(5)
2017/11/24 16:18:52 102: goja.call(1)
2017/11/24 16:18:52 103: goja._pop({})
2017/11/24 16:18:52 104: goja.jump(6)
2017/11/24 16:18:52 105: goja._loadUndef({})
2017/11/24 16:18:52 106: goja.loadStack(3)
2017/11/24 16:18:52 107: goja.getLocal(0)
2017/11/24 16:18:52 108: goja.call(1)
2017/11/24 16:18:52 109: goja._pop({})
2017/11/24 16:18:52 110: goja._halt({})
2017/11/24 16:18:52 111: goja.jump(9)
2017/11/24 16:18:52 112: goja.storeStack(7)
2017/11/24 16:18:52 113: goja._pop({})
2017/11/24 16:18:52 114: goja._loadUndef({})
2017/11/24 16:18:52 115: goja.loadStack(3)
2017/11/24 16:18:52 116: goja.loadStack(7)
2017/11/24 16:18:52 117: goja.call(1)
2017/11/24 16:18:52 118: goja._pop({})
2017/11/24 16:18:52 119: goja._halt({})
2017/11/24 16:18:52 120: goja._loadUndef({})
2017/11/24 16:18:52 121: goja._retStashless({})
2017/11/24 16:18:52 values: []
2017/11/24 16:18:52 0: goja.enterFuncStashless({3 0})
2017/11/24 16:18:52 1: goja.getLocal(16777224)
2017/11/24 16:18:52 2: goja.jneq1(5)
2017/11/24 16:18:52 3: goja._pop({})
2017/11/24 16:18:52 4: goja.getLocal(16777222)
2017/11/24 16:18:52 5: goja.getProp(domain)
2017/11/24 16:18:52 6: goja.storeStack(1)
2017/11/24 16:18:52 7: goja.jne(6)
2017/11/24 16:18:52 8: goja.loadStack(1)
2017/11/24 16:18:52 9: goja._dup({})
2017/11/24 16:18:52 10: goja.getPropCallee(exit)
2017/11/24 16:18:52 11: goja.call(0)
2017/11/24 16:18:52 12: goja._pop({})
2017/11/24 16:18:52 13: goja.getLocal(0)
2017/11/24 16:18:52 14: goja.jne(32)
2017/11/24 16:18:52 15: goja.getLocal(0)
2017/11/24 16:18:52 16: goja.getProp(fn)
2017/11/24 16:18:52 17: goja.storeStack(2)
2017/11/24 16:18:52 18: goja._pop({})
2017/11/24 16:18:52 19: goja.getLocal(0)
2017/11/24 16:18:52 20: goja.getProp(next)
2017/11/24 16:18:52 21: goja.setLocal(0)
2017/11/24 16:18:52 22: goja._pop({})
2017/11/24 16:18:52 23: goja.try({7 0 false})
2017/11/24 16:18:52 24: goja._loadUndef({})
2017/11/24 16:18:52 25: goja.loadStack(2)
2017/11/24 16:18:52 26: goja.call(0)
2017/11/24 16:18:52 27: goja._pop({})
2017/11/24 16:18:52 28: goja._halt({})
2017/11/24 16:18:52 29: goja.jump(16)
2017/11/24 16:18:52 30: goja.storeStack(3)
2017/11/24 16:18:52 31: goja._pop({})
2017/11/24 16:18:52 32: goja.getLocal(0)
2017/11/24 16:18:52 33: goja.jne(6)
2017/11/24 16:18:52 34: goja._loadUndef({})
2017/11/24 16:18:52 35: goja.getLocal(2)
2017/11/24 16:18:52 36: goja.call(0)
2017/11/24 16:18:52 37: goja._pop({})
2017/11/24 16:18:52 38: goja.jump(4)
2017/11/24 16:18:52 39: goja.getVar1(undefined)
2017/11/24 16:18:52 40: goja.setLocal(1)
2017/11/24 16:18:52 41: goja._pop({})
2017/11/24 16:18:52 42: goja.loadStack(3)
2017/11/24 16:18:52 43: goja._throw({})
2017/11/24 16:18:52 44: goja._halt({})
2017/11/24 16:18:52 45: goja.jump(-32)
2017/11/24 16:18:52 46: goja.getVar1(undefined)
2017/11/24 16:18:52 47: goja.setLocal(1)
2017/11/24 16:18:52 48: goja._pop({})
2017/11/24 16:18:52 49: goja.loadStack(1)
2017/11/24 16:18:52 50: goja.jne(6)
2017/11/24 16:18:52 51: goja.loadStack(1)
2017/11/24 16:18:52 52: goja._dup({})
2017/11/24 16:18:52 53: goja.getPropCallee(enter)
2017/11/24 16:18:52 54: goja.call(0)
2017/11/24 16:18:52 55: goja._pop({})
2017/11/24 16:18:52 56: goja._loadUndef({})
2017/11/24 16:18:52 57: goja._retStashless({})
panic: runtime error: index out of range [recovered]
panic: Panic at 10: runtime error: index out of range [recovered]
panic: Panic at 25: Panic at 10: runtime error: index out of range [recovered]
panic: Panic at 23: Panic at 25: Panic at 10: runtime error: index out of range [recovered]
panic: Panic at 24: Panic at 23: Panic at 25: Panic at 10: runtime error: index out of range
goroutine 46 [running]:
github.com/dop251/goja.(*vm).try.func1(0xc420263860, 0x0, 0xc426dc7e08, 0x0, 0x0, 0x0, 0xc426dc7e60)
github.com/dop251/goja/vm.go:364 +0x4b4
panic(0x17cff00, 0xc422820380)
/usr/local/opt/go/libexec/src/runtime/panic.go:491 +0x283
github.com/dop251/goja.(*vm).try.func1(0xc420263860, 0x6, 0xc426dc76c0, 0x19, 0x0, 0x0, 0xc426dc7718)
github.com/dop251/goja/vm.go:364 +0x4b4
panic(0x17cff00, 0xc422820360)
/usr/local/opt/go/libexec/src/runtime/panic.go:491 +0x283
github.com/dop251/goja.(*vm).try.func1(0xc420263860, 0x8, 0xc426dc74c0, 0x25, 0x0, 0x0, 0xc426dc7518)
github.com/dop251/goja/vm.go:364 +0x4b4
panic(0x17cff00, 0xc422751fe0)
/usr/local/opt/go/libexec/src/runtime/panic.go:491 +0x283
github.com/dop251/goja.(*vm).try.func1(0xc420263860, 0xb, 0xc426dc6ee8, 0x2e, 0x0, 0x0, 0xc426dc6f40)
github.com/dop251/goja/vm.go:364 +0x4b4
panic(0x17e8720, 0x1f3db30)
/usr/local/opt/go/libexec/src/runtime/panic.go:491 +0x283
github.com/dop251/goja.(*vm).popCtx(...)
github.com/dop251/goja/vm.go:427
github.com/dop251/goja.(*vm)._nativeCall(0xc420263860, 0xc420c05b80, 0x2)
github.com/dop251/goja/vm.go:1834 +0x858
github.com/dop251/goja.call.exec(0x2, 0xc420263860)
github.com/dop251/goja/vm.go:1810 +0x6b3
github.com/dop251/goja.(*call).exec(0xc42048b02c, 0xc420263860)
<autogenerated>:1 +0x44
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*funcObject).Call(0xc420b64600, 0x1d880e0, 0xc42175fb40, 0xc420f75690, 0x1, 0xe5, 0x20, 0x179c440)
github.com/dop251/goja/func.go:130 +0x3c8
github.com/dop251/goja.(*funcObject).Call-fm(0x1d880e0, 0xc42175fb40, 0xc420f75690, 0x1, 0xe5, 0x1012b70, 0x1f52360)
github.com/dop251/goja/builtin_function.go:120 +0x51
github.com/dop251/goja.(*Runtime).functionproto_call(0xc425d06f00, 0x1d880e0, 0xc4245a4120, 0xc420f75680, 0x2, 0xe6, 0xc420a11c68, 0x4)
github.com/dop251/goja/builtin_function.go:71 +0x155
github.com/dop251/goja.(*Runtime).(github.com/dop251/goja.functionproto_call)-fm(0x1d880e0, 0xc4245a4120, 0xc420f75680, 0x2, 0xe6, 0x13d3f43, 0x17e1340)
github.com/dop251/goja/builtin_function.go:160 +0x51
github.com/dop251/goja.(*vm)._nativeCall(0xc420263860, 0xc422d2aa50, 0x2)
github.com/dop251/goja/vm.go:1826 +0x2ec
github.com/dop251/goja.call.exec(0x2, 0xc420263860)
github.com/dop251/goja/vm.go:1810 +0x6b3
github.com/dop251/goja.(*call).exec(0xc4209e56fc, 0xc420263860)
<autogenerated>:1 +0x44
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*vm).(github.com/dop251/goja.run)-fm()
github.com/dop251/goja/vm.go:375 +0x2a
github.com/dop251/goja.(*vm).try(0xc420263860, 0xc427c92f48, 0x0)
github.com/dop251/goja/vm.go:370 +0x119
github.com/dop251/goja.(*vm).runTry(0xc420263860, 0xc424f19230)
github.com/dop251/goja/vm.go:375 +0x46
github.com/dop251/goja.try.exec(0x26, 0x0, 0xc420263860)
github.com/dop251/goja/vm.go:2245 +0x51
github.com/dop251/goja.(*try).exec(0xc42121f070, 0xc420263860)
<autogenerated>:1 +0x53
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*funcObject).Call(0xc420b65a40, 0x1d880e0, 0xc4235d1060, 0xc420f755b0, 0x1, 0xf3, 0x1d880e0, 0xc4235d1160)
github.com/dop251/goja/func.go:130 +0x3c8
github.com/dop251/goja.(*funcObject).Call-fm(0x1d880e0, 0xc4235d1060, 0xc420f755b0, 0x1, 0xf3, 0x1d880e0, 0xc4235d1160)
github.com/dop251/goja/builtin_function.go:120 +0x51
github.com/dop251/goja.(*Runtime).functionproto_call(0xc425d06f00, 0x1d880e0, 0xc4245a4ba0, 0xc420f755a0, 0x2, 0xf4, 0xc420a0b896, 0x4)
github.com/dop251/goja/builtin_function.go:71 +0x155
github.com/dop251/goja.(*Runtime).(github.com/dop251/goja.functionproto_call)-fm(0x1d880e0, 0xc4245a4ba0, 0xc420f755a0, 0x2, 0xf4, 0x13d3f43, 0x17e1340)
github.com/dop251/goja/builtin_function.go:160 +0x51
github.com/dop251/goja.(*vm)._nativeCall(0xc420263860, 0xc422d2aa50, 0x2)
github.com/dop251/goja/vm.go:1826 +0x2ec
github.com/dop251/goja.call.exec(0xffffffff00000002, 0xc420263860)
github.com/dop251/goja/vm.go:1810 +0x6b3
github.com/dop251/goja.(*call).exec(0xc4212380dc, 0xc420263860)
<autogenerated>:1 +0x44
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*vm).(github.com/dop251/goja.run)-fm()
github.com/dop251/goja/vm.go:375 +0x2a
github.com/dop251/goja.(*vm).try(0xc420263860, 0xc427c93520, 0x0)
github.com/dop251/goja/vm.go:370 +0x119
github.com/dop251/goja.(*vm).runTry(0xc420263860, 0xc424f19380)
github.com/dop251/goja/vm.go:375 +0x46
github.com/dop251/goja.try.exec(0x5a, 0x0, 0xc420263860)
github.com/dop251/goja/vm.go:2245 +0x51
github.com/dop251/goja.(*try).exec(0xc42121e380, 0xc420263860)
<autogenerated>:1 +0x53
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*vm).(github.com/dop251/goja.run)-fm()
github.com/dop251/goja/vm.go:375 +0x2a
github.com/dop251/goja.(*vm).try(0xc420263860, 0xc427c93720, 0x0)
github.com/dop251/goja/vm.go:370 +0x119
github.com/dop251/goja.(*vm).runTry(0xc420263860, 0xc422ab1e00)
github.com/dop251/goja/vm.go:375 +0x46
github.com/dop251/goja.try.exec(0x7, 0x0, 0xc420263860)
github.com/dop251/goja/vm.go:2245 +0x51
github.com/dop251/goja.(*try).exec(0xc4209e53e0, 0xc420263860)
<autogenerated>:1 +0x53
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*funcObject).Call(0xc420b64480, 0x1d885e0, 0xc420018d90, 0xc420f75360, 0x1, 0x118, 0x1d88720, 0x1f71e60)
github.com/dop251/goja/func.go:130 +0x3c8
github.com/dop251/goja.(*funcObject).Call-fm(0x1d885e0, 0xc420018d90, 0xc420f75360, 0x1, 0x118, 0x1d88720, 0x1f71e60)
github.com/dop251/goja/builtin_function.go:120 +0x51
github.com/dop251/goja.(*Runtime).functionproto_call(0xc425d06f00, 0x1d880e0, 0xc4245a40a0, 0xc420f75350, 0x2, 0x119, 0xc420a0b896, 0x4)
github.com/dop251/goja/builtin_function.go:71 +0x155
github.com/dop251/goja.(*Runtime).(github.com/dop251/goja.functionproto_call)-fm(0x1d880e0, 0xc4245a40a0, 0xc420f75350, 0x2, 0x119, 0x13d3f43, 0x17e1340)
github.com/dop251/goja/builtin_function.go:160 +0x51
github.com/dop251/goja.(*vm)._nativeCall(0xc420263860, 0xc422d2aa50, 0x2)
github.com/dop251/goja/vm.go:1826 +0x2ec
github.com/dop251/goja.call.exec(0xffffffff00000002, 0xc420263860)
github.com/dop251/goja/vm.go:1810 +0x6b3
github.com/dop251/goja.(*call).exec(0xc4212380dc, 0xc420263860)
<autogenerated>:1 +0x44
github.com/dop251/goja.(*vm).run(0xc420263860)
github.com/dop251/goja/vm.go:288 +0x51
github.com/dop251/goja.(*funcObject).Call(0xc425be83c0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc427c93d90, 0xc42002b900)
github.com/dop251/goja/func.go:130 +0x3c8
github.com/dop251/goja.(*funcObject).Call-fm(0x0, 0x0, 0x0, 0x0, 0x0, 0x102ad99, 0xc4201e3fb0)
github.com/dop251/goja/builtin_function.go:120 +0x51
github.com/dop251/goja.AssertFunction.func1.1()
github.com/dop251/goja/runtime.go:1374 +0xb6
github.com/dop251/goja.(*vm).try(0xc420263860, 0xc427c93e80, 0x0)
github.com/dop251/goja/vm.go:370 +0x119
github.com/dop251/goja.AssertFunction.func1(0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc427c93f30, 0x0)
github.com/dop251/goja/runtime.go:1373 +0xdd
github.com/dop251/goja_nodejs/eventloop.(*EventLoop).doTimeout(0xc42087e720, 0xc42448e030)
github.com/dop251/goja_nodejs/eventloop/eventloop.go:178 +0x6d
github.com/dop251/goja_nodejs/eventloop.(*EventLoop).addTimeout.func1.1()
github.com/dop251/goja_nodejs/eventloop/eventloop.go:156 +0xad
github.com/dop251/goja_nodejs/eventloop.(*EventLoop).run(0xc42087e720, 0x1)
github.com/dop251/goja_nodejs/eventloop/eventloop.go:137 +0x72
created by github.com/dop251/goja_nodejs/eventloop.(*EventLoop).Start
github.com/dop251/goja_nodejs/eventloop/eventloop.go:94 +0x44
Note: Line numbers may not fit the eventloop's code available on github, I've added a lot of fmt.Println to try and debug this, but I haven't changed actual code.
I think I've been running all code that interacts with a *goja.Runtime
inside RunOnLoop
. My code is transpiled (babel) and then the runtime is polyfilled heavily, but as far as I know, it should work fine. It's been working fine for the most part.
Here's an example of my xhr
polyfill: https://gist.github.com/jeromegn/cb76ba14437ab688074c6b2d607d358d (just in case you see anything that would break here)
My question is: Is there any way to make sense of this stacktrace? As far as I understand, the Panic at <n>
references the stack level in the previous trace. Except I have no idea how to correlate this with actual JS code.
Doesn't goja keep track of the line and column of the source when making its AST?
The following code produces a panic:
package main
import (
"github.com/dop251/goja"
)
func main() {
goja.New()
}
Output:
⇒ go run main.go
panic: runtime error: invalid memory address or nil pointer dereference
[signal SIGSEGV: segmentation violation code=0x1 addr=0x48 pc=0x11659fb]
goroutine 1 [running]:
github.com/deis/dhow/vendor/golang.org/x/text/collate.newCollator(0x0, 0x0, 0x0)
/Users/technosophos/Code/Go/src/github.com/deis/dhow/vendor/golang.org/x/text/collate/option.go:30 +0x6b
github.com/deis/dhow/vendor/golang.org/x/text/collate.New(0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xc42003ff78)
/Users/technosophos/Code/Go/src/github.com/deis/dhow/vendor/golang.org/x/text/collate/collate.go:60 +0xbc
github.com/deis/dhow/vendor/github.com/dop251/goja.init()
/Users/technosophos/Code/Go/src/github.com/deis/dhow/vendor/github.com/dop251/goja/builtin_string.go:15 +0x133
main.init()
/Users/technosophos/Code/Go/src/github.com/deis/dhow/cmd/main.go:10 +0x44
exit status 2
Go version:
⇒ go version
go version go1.8.3 darwin/amd64
Complete dependencies:
⇒ cat glide.lock
hash: f21c0c8c2f06ddf1910034af4875b3baa5e8a8338b33eb8da97e8e5779af078a
updated: 2017-07-01T15:23:06.031128545-06:00
imports:
- name: github.com/dlclark/regexp2
version: 902a5ce7a7812e2ba9f73b9d96c09d5136df39cd
subpackages:
- syntax
- name: github.com/dop251/goja
version: 8f3380f012a7a699e478fcff1d0e46a6f2e1abc5
subpackages:
- ast
- file
- parser
- token
- name: golang.org/x/text
version: 2910a502d2bf9e43193af9d68ca516529614eed3
subpackages:
- cases
- collate
- collate/colltab
- internal/colltab
- internal/tag
- language
- transform
- unicode/norm
testImports: []
Hi,
I am trying to debug this issue Object has no memeber 'assign'
How can I get the stacktrace when there is an error in the javascript?
I am using e.stack, but it's giving me an undefined value.
JS code:
result.error = "Render Error: " + e + "\n\t\tStackTrace: " + e.stack + "."
outcome:
Render Error: TypeError: Object has no member 'assign'
StackTrace: undefined.
And return the string representation of iface
Otherwise, for every error we will need to check if it is InterruptedError and treat it differently.
A declarative, efficient, and flexible JavaScript library for building user interfaces.
🖖 Vue.js is a progressive, incrementally-adoptable JavaScript framework for building UI on the web.
TypeScript is a superset of JavaScript that compiles to clean JavaScript output.
An Open Source Machine Learning Framework for Everyone
The Web framework for perfectionists with deadlines.
A PHP framework for web artisans
Bring data to life with SVG, Canvas and HTML. 📊📈🎉
JavaScript (JS) is a lightweight interpreted programming language with first-class functions.
Some thing interesting about web. New door for the world.
A server is a program made to process requests and deliver data to clients.
Machine learning is a way of modeling and interpreting data that allows a piece of software to respond intelligently.
Some thing interesting about visualization, use data art
Some thing interesting about game, make everyone happy.
We are working to build community through open source technology. NB: members must have two-factor auth.
Open source projects and samples from Microsoft.
Google ❤️ Open Source for everyone.
Alibaba Open Source for everyone
Data-Driven Documents codes.
China tencent open source team.