Merge pull request #1647 from gilch/let-rebind

Fix let rebind bug.
This commit is contained in:
gilch 2018-07-08 21:52:32 -06:00 committed by GitHub
commit 0de8557fb5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 160 additions and 198 deletions

View File

@ -305,6 +305,7 @@ as can nested let forms.
(macro-error bindings "let bindings must be paired")) (macro-error bindings "let bindings must be paired"))
(setv g!let (gensym 'let) (setv g!let (gensym 'let)
replacements (OrderedDict) replacements (OrderedDict)
keys []
values []) values [])
(defn expander [symbol] (defn expander [symbol]
(.get replacements symbol symbol)) (.get replacements symbol symbol))
@ -313,63 +314,15 @@ as can nested let forms.
(macro-error k "bind targets must be symbols") (macro-error k "bind targets must be symbols")
(if (in '. k) (if (in '. k)
(macro-error k "binding target may not contain a dot"))) (macro-error k "binding target may not contain a dot")))
(.append values (symbolexpand (macroexpand-all v &name) expander)) (.append values (symbolexpand (macroexpand-all v &name)
(assoc replacements k `(get ~g!let ~(name k)))) expander))
(.append keys `(get ~g!let ~(name k)))
(assoc replacements k (last keys)))
`(do `(do
(setv ~g!let {} (setv ~g!let {}
~@(interleave (.values replacements) values)) ~@(interleave keys values))
~@(symbolexpand (macroexpand-all body &name) expander))) ~@(symbolexpand (macroexpand-all body &name)
expander)))
;; (defmacro macrolet []) ;; (defmacro macrolet [])
#_[special cases for let
;; Symbols containing a dot should be converted to this form.
;; attrs should not get expanded,
;; but [] lookups should.
'.',
;;; can shadow let bindings with Python locals
;; protect its bindings for the lexical scope of its body.
'fn',
'fn*',
;; protect as bindings for the lexical scope of its body
'except',
;;; changes scope of named variables
;; protect the variables they name for the lexical scope of their container
'global',
'nonlocal',
;; should we provide a protect form?
;; it's an anaphor only valid in a `let` body.
;; this would make the named variables python-scoped in its body
;; expands to a do
'protect',
;;; quoted variables must not be expanded.
;; but unprotected, unquoted variables must be.
'quasiquote',
'quote',
'unquote',
'unquote-splice',
;;;; deferred
;; should really only exist at toplevel. Ignore until someone complains?
;; raise an error? treat like fn?
;; should probably be implemented as macros in terms of fn/setv anyway.
'defmacro',
'deftag',
;;; create Python-scoped variables. It's probably hard to avoid this.
;; Best just doc this behavior for now.
;; we can't avoid clobbering enclosing python scope, unless we use a gensym,
;; but that corrupts '__name__'.
;; It could be set later, but that could mess up metaclasses!
;; Should the class name update let variables too?
'defclass',
;; should this update let variables?
;; it could be done with gensym/setv.
'import',
;; I don't understand these. Ignore until someone complains?
'eval_and_compile', 'eval_when_compile', 'require',]

View File

@ -64,16 +64,16 @@
b "b") b "b")
(let [a "x" (let [a "x"
b "y"] b "y"]
(assert (= (+ a b) (assert (= (+ a b)
"xy")) "xy"))
(let [a "z"] (let [a "z"]
(assert (= (+ a b) (assert (= (+ a b)
"zy"))) "zy")))
;; let-shadowed variable doesn't get clobbered. ;; let-shadowed variable doesn't get clobbered.
(assert (= (+ a b) (assert (= (+ a b)
"xy"))) "xy")))
(let [q "q"] (let [q "q"]
(assert (= q "q"))) (assert (= q "q")))
(assert (= a "a")) (assert (= a "a"))
(assert (= b "b")) (assert (= b "b"))
(assert (in "a" (.keys (vars)))) (assert (in "a" (.keys (vars))))
@ -85,86 +85,86 @@
(let [a "a" (let [a "a"
b "b" b "b"
ab (+ a b)] ab (+ a b)]
(assert (= ab "ab")) (assert (= ab "ab"))
(let [c "c" (let [c "c"
abc (+ ab c)] abc (+ ab c)]
(assert (= abc "abc"))))) (assert (= abc "abc")))))
(defn test-let-early [] (defn test-let-early []
(setv a "a") (setv a "a")
(let [q (+ a "x") (let [q (+ a "x")
a 2 ; should not affect q a 2 ; should not affect q
b 3] b 3]
(assert (= q "ax")) (assert (= q "ax"))
(let [q (* a b) (let [q (* a b)
a (+ a b) a (+ a b)
b (* a b)] b (* a b)]
(assert (= q 6)) (assert (= q 6))
(assert (= a 5)) (assert (= a 5))
(assert (= b 15)))) (assert (= b 15))))
(assert (= a "a"))) (assert (= a "a")))
(defn test-let-special [] (defn test-let-special []
;; special forms in function position still work as normal ;; special forms in function position still work as normal
(let [, 1] (let [, 1]
(assert (= (, , ,) (assert (= (, , ,)
(, 1 1))))) (, 1 1)))))
(defn test-let-quasiquote [] (defn test-let-quasiquote []
(setv a-symbol 'a) (setv a-symbol 'a)
(let [a "x"] (let [a "x"]
(assert (= a "x")) (assert (= a "x"))
(assert (= 'a a-symbol)) (assert (= 'a a-symbol))
(assert (= `a a-symbol)) (assert (= `a a-symbol))
(assert (= `(foo ~a) (assert (= `(foo ~a)
'(foo "x"))) '(foo "x")))
(assert (= `(foo `(bar a ~a ~~a)) (assert (= `(foo `(bar a ~a ~~a))
'(foo `(bar a ~a ~"x")))) '(foo `(bar a ~a ~"x"))))
(assert (= `(foo ~@[a]) (assert (= `(foo ~@[a])
'(foo "x"))) '(foo "x")))
(assert (= `(foo `(bar [a] ~@[a] ~@~[a 'a `a] ~~@[a])) (assert (= `(foo `(bar [a] ~@[a] ~@~[a 'a `a] ~~@[a]))
'(foo `(bar [a] ~@[a] ~@["x" a a] ~"x")))))) '(foo `(bar [a] ~@[a] ~@["x" a a] ~"x"))))))
(defn test-let-except [] (defn test-let-except []
(let [foo 42 (let [foo 42
bar 33] bar 33]
(assert (= foo 42)) (assert (= foo 42))
(try (try
(do (do
1/0 1/0
(assert False)) (assert False))
(except [foo Exception] (except [foo Exception]
;; let bindings should work in except block ;; let bindings should work in except block
(assert (= bar 33)) (assert (= bar 33))
;; but exception bindings can shadow let bindings ;; but exception bindings can shadow let bindings
(assert (instance? Exception foo)))) (assert (instance? Exception foo))))
;; let binding did not get clobbered. ;; let binding did not get clobbered.
(assert (= foo 42)))) (assert (= foo 42))))
(defn test-let-mutation [] (defn test-let-mutation []
(setv foo 42) (setv foo 42)
(setv error False) (setv error False)
(let [foo 12 (let [foo 12
bar 13] bar 13]
(assert (= foo 12)) (assert (= foo 12))
(setv foo 14) (setv foo 14)
(assert (= foo 14)) (assert (= foo 14))
(del foo) (del foo)
;; deleting a let binding should not affect others ;; deleting a let binding should not affect others
(assert (= bar 13)) (assert (= bar 13))
(try (try
;; foo=42 is still shadowed, but the let binding was deleted. ;; foo=42 is still shadowed, but the let binding was deleted.
(do (do
foo foo
(assert False)) (assert False))
(except [le LookupError] (except [le LookupError]
(setv error le))) (setv error le)))
(setv foo 16) (setv foo 16)
(assert (= foo 16)) (assert (= foo 16))
(setv [foo bar baz] [1 2 3]) (setv [foo bar baz] [1 2 3])
(assert (= foo 1)) (assert (= foo 1))
(assert (= bar 2)) (assert (= bar 2))
(assert (= baz 3))) (assert (= baz 3)))
(assert error) (assert error)
(assert (= foo 42)) (assert (= foo 42))
(assert (= baz 3))) (assert (= baz 3)))
@ -172,40 +172,40 @@
(defn test-let-break [] (defn test-let-break []
(for [x (range 3)] (for [x (range 3)]
(let [done (odd? x)] (let [done (odd? x)]
(if done (break)))) (if done (break))))
(assert (= x 1))) (assert (= x 1)))
(defn test-let-continue [] (defn test-let-continue []
(let [foo []] (let [foo []]
(for [x (range 10)] (for [x (range 10)]
(let [odd (odd? x)] (let [odd (odd? x)]
(if odd (continue)) (if odd (continue))
(.append foo x))) (.append foo x)))
(assert (= foo [0 2 4 6 8])))) (assert (= foo [0 2 4 6 8]))))
(defn test-let-yield [] (defn test-let-yield []
(defn grind [] (defn grind []
(yield 0) (yield 0)
(let [a 1 (let [a 1
b 2] b 2]
(yield a) (yield a)
(yield b))) (yield b)))
(assert (= (tuple (grind)) (assert (= (tuple (grind))
(, 0 1 2)))) (, 0 1 2))))
(defn test-let-return [] (defn test-let-return []
(defn get-answer [] (defn get-answer []
(let [answer 42] (let [answer 42]
(return answer))) (return answer)))
(assert (= (get-answer) (assert (= (get-answer)
42))) 42)))
(defn test-let-import [] (defn test-let-import []
(let [types 6] (let [types 6]
;; imports don't fail, even if using a let-bound name ;; imports don't fail, even if using a let-bound name
(import types) (import types)
;; let-bound name is not affected ;; let-bound name is not affected
(assert (= types 6))) (assert (= types 6)))
;; import happened in Python scope. ;; import happened in Python scope.
(assert (in "types" (vars))) (assert (in "types" (vars)))
(assert (instance? types.ModuleType types))) (assert (instance? types.ModuleType types)))
@ -213,13 +213,13 @@
(defn test-let-defclass [] (defn test-let-defclass []
(let [Foo 42 (let [Foo 42
quux object] quux object]
;; the name of the class is just a symbol, even if it's a let binding ;; the name of the class is just a symbol, even if it's a let binding
(defclass Foo [quux] ; let bindings apply in inheritance list (defclass Foo [quux] ; let bindings apply in inheritance list
;; let bindings apply inside class body ;; let bindings apply inside class body
(setv x Foo) (setv x Foo)
;; quux is not local ;; quux is not local
(setv quux "quux")) (setv quux "quux"))
(assert (= quux "quux"))) (assert (= quux "quux")))
;; defclass always creates a python-scoped variable, even if it's a let binding name ;; defclass always creates a python-scoped variable, even if it's a let binding name
(assert (= Foo.x 42))) (assert (= Foo.x 42)))
@ -229,82 +229,82 @@
(let [a 1 (let [a 1
b [] b []
bar (fn [])] bar (fn [])]
(setv bar.a 13) (setv bar.a 13)
(assert (= bar.a 13)) (assert (= bar.a 13))
(setv (. bar a) 14) (setv (. bar a) 14)
(assert (= bar.a 14)) (assert (= bar.a 14))
(assert (= a 1)) (assert (= a 1))
(assert (= b [])) (assert (= b []))
;; method syntax not affected ;; method syntax not affected
(.append b 2) (.append b 2)
(assert (= b [2])) (assert (= b [2]))
;; attrs access is not affected ;; attrs access is not affected
(assert (= foo.a 42)) (assert (= foo.a 42))
(assert (= (. foo a) (assert (= (. foo a)
42)) 42))
;; but indexing is ;; but indexing is
(assert (= (. [1 2 3] (assert (= (. [1 2 3]
[a]) [a])
2)))) 2))))
(defn test-let-positional [] (defn test-let-positional []
(let [a 0 (let [a 0
b 1 b 1
c 2] c 2]
(defn foo [a b] (defn foo [a b]
(, a b c)) (, a b c))
(assert (= (foo 100 200) (assert (= (foo 100 200)
(, 100 200 2))) (, 100 200 2)))
(setv c 300) (setv c 300)
(assert (= (foo 1000 2000) (assert (= (foo 1000 2000)
(, 1000 2000 300))) (, 1000 2000 300)))
(assert (= a 0)) (assert (= a 0))
(assert (= b 1)) (assert (= b 1))
(assert (= c 300)))) (assert (= c 300))))
(defn test-let-rest [] (defn test-let-rest []
(let [xs 6 (let [xs 6
a 88 a 88
c 64 c 64
&rest 12] &rest 12]
(defn foo [a b &rest xs] (defn foo [a b &rest xs]
(-= a 1) (-= a 1)
(setv xs (list xs)) (setv xs (list xs))
(.append xs 42) (.append xs 42)
(, &rest a b c xs)) (, &rest a b c xs))
(assert (= xs 6)) (assert (= xs 6))
(assert (= a 88)) (assert (= a 88))
(assert (= (foo 1 2 3 4) (assert (= (foo 1 2 3 4)
(, 12 0 2 64 [3 4 42]))) (, 12 0 2 64 [3 4 42])))
(assert (= xs 6)) (assert (= xs 6))
(assert (= c 64)) (assert (= c 64))
(assert (= a 88)))) (assert (= a 88))))
(defn test-let-kwargs [] (defn test-let-kwargs []
(let [kws 6 (let [kws 6
&kwargs 13] &kwargs 13]
(defn foo [&kwargs kws] (defn foo [&kwargs kws]
(, &kwargs kws)) (, &kwargs kws))
(assert (= kws 6)) (assert (= kws 6))
(assert (= (foo :a 1) (assert (= (foo :a 1)
(, 13 {"a" 1}))))) (, 13 {"a" 1})))))
(defn test-let-optional [] (defn test-let-optional []
(let [a 1 (let [a 1
b 6 b 6
d 2] d 2]
(defn foo [&optional [a a] b [c d]] (defn foo [&optional [a a] b [c d]]
(, a b c)) (, a b c))
(assert (= (foo) (assert (= (foo)
(, 1 None 2))) (, 1 None 2)))
(assert (= (foo 10 20 30) (assert (= (foo 10 20 30)
(, 10 20 30))))) (, 10 20 30)))))
(defn test-let-closure [] (defn test-let-closure []
(let [count 0] (let [count 0]
(defn +count [&optional [x 1]] (defn +count [&optional [x 1]]
(+= count x) (+= count x)
count)) count))
;; let bindings can still exist outside of a let body ;; let bindings can still exist outside of a let body
(assert (= 1 (+count))) (assert (= 1 (+count)))
(assert (= 2 (+count))) (assert (= 2 (+count)))
@ -323,9 +323,18 @@
(let [a 1 (let [a 1
b (triple a) b (triple a)
c (ap-triple)] c (ap-triple)]
(assert (= (triple a) (assert (= (triple a)
3)) 3))
(assert (= (ap-triple) (assert (= (ap-triple)
3)) 3))
(assert (= b 3)) (assert (= b 3))
(assert (= c 3)))) (assert (= c 3))))
(defn test-let-rebind []
(let [x "foo"
y "bar"
x (+ x y)
y (+ y x)
x (+ x x)]
(assert (= x "foobarfoobar"))
(assert (= y "barfoobar"))))