2019-01-11 13:27:28 +03:00
|
|
|
package runtime_test
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestAssignment(t *testing.T) {
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
|
|
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
|
|
|
expect(t, `a := 1; a = a + 4; out = a`, nil, 5)
|
|
|
|
expect(t, `a := 1; f1 := func() { a = 2; return a }; out = f1()`, nil, 2)
|
|
|
|
expect(t, `a := 1; f1 := func() { a := 3; a = 2; return a }; out = f1()`, nil, 2)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `a := 1; out = a`, nil, 1)
|
|
|
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
|
|
|
expect(t, `a := 1; func() { a = 2 }(); out = a`, nil, 2)
|
|
|
|
expect(t, `a := 1; func() { a := 2 }(); out = a`, nil, 1) // "a := 2" defines a new local variable 'a'
|
|
|
|
expect(t, `a := 1; func() { b := 2; out = b }()`, nil, 2)
|
2019-01-09 10:17:42 +03:00
|
|
|
expect(t, `
|
|
|
|
out = func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 2
|
2019-01-09 10:17:42 +03:00
|
|
|
func() {
|
|
|
|
a = 3 // captured from outer scope
|
|
|
|
}()
|
|
|
|
return a
|
|
|
|
}()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 3)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
|
|
|
func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 5
|
2019-01-09 10:17:42 +03:00
|
|
|
out = func() {
|
|
|
|
a := 4
|
|
|
|
return a
|
|
|
|
}()
|
2019-03-18 18:15:26 +03:00
|
|
|
}()`, nil, 4)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-18 18:15:26 +03:00
|
|
|
expectError(t, `a := 1; a := 2`, nil, "redeclared") // redeclared in the same scope
|
|
|
|
expectError(t, `func() { a := 1; a := 2 }()`, nil, "redeclared") // redeclared in the same scope
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `a := 1; a += 2; out = a`, nil, 3)
|
|
|
|
expect(t, `a := 1; a += 4 - 2;; out = a`, nil, 3)
|
|
|
|
expect(t, `a := 3; a -= 1;; out = a`, nil, 2)
|
|
|
|
expect(t, `a := 3; a -= 5 - 4;; out = a`, nil, 2)
|
|
|
|
expect(t, `a := 2; a *= 4;; out = a`, nil, 8)
|
|
|
|
expect(t, `a := 2; a *= 1 + 3;; out = a`, nil, 8)
|
|
|
|
expect(t, `a := 10; a /= 2;; out = a`, nil, 5)
|
|
|
|
expect(t, `a := 10; a /= 5 - 3;; out = a`, nil, 5)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-04 02:21:28 +03:00
|
|
|
// compound assignment operator does not define new variable
|
2019-03-18 18:15:26 +03:00
|
|
|
expectError(t, `a += 4`, nil, "unresolved reference")
|
|
|
|
expectError(t, `a -= 4`, nil, "unresolved reference")
|
|
|
|
expectError(t, `a *= 4`, nil, "unresolved reference")
|
|
|
|
expectError(t, `a /= 4`, nil, "unresolved reference")
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
f1 := func() {
|
|
|
|
f2 := func() {
|
|
|
|
a := 1
|
2019-01-09 10:17:42 +03:00
|
|
|
a += 2 // it's a statement, not an expression
|
|
|
|
return a
|
|
|
|
};
|
|
|
|
|
|
|
|
return f2();
|
|
|
|
};
|
|
|
|
|
2019-03-18 18:15:26 +03:00
|
|
|
out = f1();`, nil, 3)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 1; a += 4 - 2; return a }; return f2(); }; out = f1()`, nil, 3)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 1; return a }; return f2(); }; out = f1()`, nil, 2)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 5 - 4; return a }; return f2(); }; out = f1()`, nil, 2)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 4; return a }; return f2(); }; out = f1()`, nil, 8)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 1 + 3; return a }; return f2(); }; out = f1()`, nil, 8)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 2; return a }; return f2(); }; out = f1()`, nil, 5)
|
|
|
|
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 5 - 3; return a }; return f2(); }; out = f1()`, nil, 5)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `a := 1; f1 := func() { f2 := func() { a += 2; return a }; return f2(); }; out = f1()`, nil, 3)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
f1 := func(a) {
|
2019-01-09 10:17:42 +03:00
|
|
|
return func(b) {
|
2019-01-11 09:34:28 +03:00
|
|
|
c := a
|
2019-01-09 10:17:42 +03:00
|
|
|
c += b * 2
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out = f1(3)(4)
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 11)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
|
|
|
out = func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 1
|
2019-01-09 10:17:42 +03:00
|
|
|
func() {
|
|
|
|
a = 2
|
|
|
|
func() {
|
|
|
|
a = 3
|
|
|
|
func() {
|
|
|
|
a := 4 // declared new
|
|
|
|
}()
|
|
|
|
}()
|
|
|
|
}()
|
|
|
|
return a
|
|
|
|
}()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 3)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
// write on free variables
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
f1 := func() {
|
|
|
|
a := 5
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
return func() {
|
|
|
|
a += 3
|
|
|
|
return a
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
out = f1()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 8)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
2019-03-14 11:07:19 +03:00
|
|
|
expect(t, `
|
|
|
|
out = func() {
|
|
|
|
f1 := func() {
|
|
|
|
a := 5
|
|
|
|
add1 := func() { a += 1 }
|
|
|
|
add2 := func() { a += 2 }
|
|
|
|
a += 3
|
|
|
|
return func() { a += 4; add1(); add2(); a += 5; return a }
|
|
|
|
}
|
|
|
|
return f1()
|
|
|
|
}()()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 20)
|
2019-03-14 11:07:19 +03:00
|
|
|
|
2019-01-09 10:17:42 +03:00
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
it := func(seq, fn) {
|
2019-01-09 10:17:42 +03:00
|
|
|
fn(seq[0])
|
|
|
|
fn(seq[1])
|
|
|
|
fn(seq[2])
|
|
|
|
}
|
|
|
|
|
2019-01-11 09:34:28 +03:00
|
|
|
foo := func(a) {
|
|
|
|
b := 0
|
2019-01-09 10:17:42 +03:00
|
|
|
it([1, 2, 3], func(x) {
|
|
|
|
b = x + a
|
|
|
|
})
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
out = foo(2)
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 5)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
it := func(seq, fn) {
|
2019-01-09 10:17:42 +03:00
|
|
|
fn(seq[0])
|
|
|
|
fn(seq[1])
|
|
|
|
fn(seq[2])
|
|
|
|
}
|
|
|
|
|
2019-01-11 09:34:28 +03:00
|
|
|
foo := func(a) {
|
|
|
|
b := 0
|
2019-01-09 10:17:42 +03:00
|
|
|
it([1, 2, 3], func(x) {
|
|
|
|
b += x + a
|
|
|
|
})
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
|
|
|
out = foo(2)
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 12)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
|
|
|
out = func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 1
|
2019-01-09 10:17:42 +03:00
|
|
|
func() {
|
|
|
|
a = 2
|
|
|
|
}()
|
|
|
|
return a
|
|
|
|
}()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 2)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
f := func() {
|
|
|
|
a := 1
|
2019-01-09 10:17:42 +03:00
|
|
|
return {
|
|
|
|
b: func() { a += 3 },
|
|
|
|
c: func() { a += 2 },
|
|
|
|
d: func() { return a }
|
|
|
|
}
|
|
|
|
}
|
2019-01-11 09:34:28 +03:00
|
|
|
m := f()
|
2019-01-09 10:17:42 +03:00
|
|
|
m.b()
|
|
|
|
m.c()
|
|
|
|
out = m.d()
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 6)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
each := func(s, x) { for i:=0; i<len(s); i++ { x(s[i]) } }
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
out = func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 100
|
2019-01-09 10:17:42 +03:00
|
|
|
each([1, 2, 3], func(x) {
|
|
|
|
a += x
|
|
|
|
})
|
|
|
|
a += 10
|
|
|
|
return func(b) {
|
|
|
|
return a + b
|
|
|
|
}
|
|
|
|
}()(20)
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 136)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
// assigning different type value
|
2019-03-20 11:28:40 +03:00
|
|
|
expect(t, `a := 1; a = "foo"; out = a`, nil, "foo") // global
|
|
|
|
expect(t, `func() { a := 1; a = "foo"; out = a }()`, nil, "foo") // local
|
2019-01-09 10:17:42 +03:00
|
|
|
expect(t, `
|
|
|
|
out = func() {
|
2019-01-11 09:34:28 +03:00
|
|
|
a := 5
|
2019-01-09 10:17:42 +03:00
|
|
|
return func() {
|
|
|
|
a = "foo"
|
|
|
|
return a
|
|
|
|
}()
|
2019-03-18 18:15:26 +03:00
|
|
|
}()`, nil, "foo") // free
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
// variables declared in if/for blocks
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `for a:=0; a<5; a++ {}; a := "foo"; out = a`, nil, "foo")
|
|
|
|
expect(t, `func() { for a:=0; a<5; a++ {}; a := "foo"; out = a }()`, nil, "foo")
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
// selectors
|
2019-03-18 18:15:26 +03:00
|
|
|
expect(t, `a:=[1,2,3]; a[1] = 5; out = a[1]`, nil, 5)
|
|
|
|
expect(t, `a:=[1,2,3]; a[1] += 5; out = a[1]`, nil, 7)
|
|
|
|
expect(t, `a:={b:1,c:2}; a.b = 5; out = a.b`, nil, 5)
|
|
|
|
expect(t, `a:={b:1,c:2}; a.b += 5; out = a.b`, nil, 6)
|
|
|
|
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.b`, nil, 3)
|
|
|
|
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.c`, nil, 2)
|
2019-01-09 10:17:42 +03:00
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
a := {
|
2019-01-09 10:17:42 +03:00
|
|
|
b: [1, 2, 3],
|
|
|
|
c: {
|
|
|
|
d: 8,
|
|
|
|
e: "foo",
|
|
|
|
f: [9, 8]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
a.c.f[1] += 2
|
|
|
|
out = a["c"]["f"][1]
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, 10)
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expect(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
a := {
|
2019-01-09 10:17:42 +03:00
|
|
|
b: [1, 2, 3],
|
|
|
|
c: {
|
|
|
|
d: 8,
|
|
|
|
e: "foo",
|
|
|
|
f: [9, 8]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
a.c.h = "bar"
|
|
|
|
out = a.c.h
|
2019-03-18 18:15:26 +03:00
|
|
|
`, nil, "bar")
|
2019-01-09 10:17:42 +03:00
|
|
|
|
|
|
|
expectError(t, `
|
2019-01-11 09:34:28 +03:00
|
|
|
a := {
|
2019-01-09 10:17:42 +03:00
|
|
|
b: [1, 2, 3],
|
|
|
|
c: {
|
|
|
|
d: 8,
|
|
|
|
e: "foo",
|
|
|
|
f: [9, 8]
|
|
|
|
}
|
|
|
|
}
|
2019-03-18 18:15:26 +03:00
|
|
|
a.x.e = "bar"`, nil, "not index-assignable")
|
2019-01-09 10:17:42 +03:00
|
|
|
}
|