package runtime_test import ( "testing" ) func TestAssignment(t *testing.T) { expect(t, `a := 1; a = 2; out = a`, 2) expect(t, `a := 1; a = 2; out = a`, 2) expect(t, `a := 1; a = a + 4; out = a`, 5) expect(t, `a := 1; f1 := func() { a = 2; return a }; out = f1()`, 2) expect(t, `a := 1; f1 := func() { a := 3; a = 2; return a }; out = f1()`, 2) expect(t, `a := 1; out = a`, 1) expect(t, `a := 1; a = 2; out = a`, 2) expect(t, `a := 1; func() { a = 2 }(); out = a`, 2) expect(t, `a := 1; func() { a := 2 }(); out = a`, 1) // "a := 2" defines a new local variable 'a' expect(t, `a := 1; func() { b := 2; out = b }()`, 2) expect(t, ` out = func() { a := 2 func() { a = 3 // captured from outer scope }() return a }() `, 3) expect(t, ` func() { a := 5 out = func() { a := 4 return a }() }()`, 4) expectError(t, `a := 1; a := 2`, "redeclared") // redeclared in the same scope expectError(t, `func() { a := 1; a := 2 }()`, "redeclared") // redeclared in the same scope expect(t, `a := 1; a += 2; out = a`, 3) expect(t, `a := 1; a += 4 - 2;; out = a`, 3) expect(t, `a := 3; a -= 1;; out = a`, 2) expect(t, `a := 3; a -= 5 - 4;; out = a`, 2) expect(t, `a := 2; a *= 4;; out = a`, 8) expect(t, `a := 2; a *= 1 + 3;; out = a`, 8) expect(t, `a := 10; a /= 2;; out = a`, 5) expect(t, `a := 10; a /= 5 - 3;; out = a`, 5) // compound assignment operator does not define new variable expectError(t, `a += 4`, "unresolved reference") expectError(t, `a -= 4`, "unresolved reference") expectError(t, `a *= 4`, "unresolved reference") expectError(t, `a /= 4`, "unresolved reference") expect(t, ` f1 := func() { f2 := func() { a := 1 a += 2 // it's a statement, not an expression return a }; return f2(); }; out = f1();`, 3) expect(t, `f1 := func() { f2 := func() { a := 1; a += 4 - 2; return a }; return f2(); }; out = f1()`, 3) expect(t, `f1 := func() { f2 := func() { a := 3; a -= 1; return a }; return f2(); }; out = f1()`, 2) expect(t, `f1 := func() { f2 := func() { a := 3; a -= 5 - 4; return a }; return f2(); }; out = f1()`, 2) expect(t, `f1 := func() { f2 := func() { a := 2; a *= 4; return a }; return f2(); }; out = f1()`, 8) expect(t, `f1 := func() { f2 := func() { a := 2; a *= 1 + 3; return a }; return f2(); }; out = f1()`, 8) expect(t, `f1 := func() { f2 := func() { a := 10; a /= 2; return a }; return f2(); }; out = f1()`, 5) expect(t, `f1 := func() { f2 := func() { a := 10; a /= 5 - 3; return a }; return f2(); }; out = f1()`, 5) expect(t, `a := 1; f1 := func() { f2 := func() { a += 2; return a }; return f2(); }; out = f1()`, 3) expect(t, ` f1 := func(a) { return func(b) { c := a c += b * 2 return c } } out = f1(3)(4) `, 11) expect(t, ` out = func() { a := 1 func() { a = 2 func() { a = 3 func() { a := 4 // declared new }() }() }() return a }() `, 3) // write on free variables expect(t, ` f1 := func() { a := 5 return func() { a += 3 return a }() } out = f1() `, 8) 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() }()() `, 20) expect(t, ` it := func(seq, fn) { fn(seq[0]) fn(seq[1]) fn(seq[2]) } foo := func(a) { b := 0 it([1, 2, 3], func(x) { b = x + a }) return b } out = foo(2) `, 5) expect(t, ` it := func(seq, fn) { fn(seq[0]) fn(seq[1]) fn(seq[2]) } foo := func(a) { b := 0 it([1, 2, 3], func(x) { b += x + a }) return b } out = foo(2) `, 12) expect(t, ` out = func() { a := 1 func() { a = 2 }() return a }() `, 2) expect(t, ` f := func() { a := 1 return { b: func() { a += 3 }, c: func() { a += 2 }, d: func() { return a } } } m := f() m.b() m.c() out = m.d() `, 6) expect(t, ` each := func(s, x) { for i:=0; i