Function tests

This commit is contained in:
Jay McCarthy 2015-11-23 17:20:04 -05:00
parent f61e1e59be
commit 4b0a38e588
3 changed files with 55 additions and 30 deletions

View File

@ -69,7 +69,7 @@
'+ 40 '- 40 '+ 40 '- 40
'< 60 '<= 60 '< 60 '<= 60
'> 60 '>= 60 '> 60 '>= 60
'= 70 ' 70 '≙* 70)) '= 70 ' 70 '≙* 70 ' 70))
(define (shunting-yard:precendence op) (define (shunting-yard:precendence op)
(define v (syntax-local-value op (λ () #f))) (define v (syntax-local-value op (λ () #f)))
(or (and v (binary-operator? v) (binary-operator-precedence v)) (or (and v (binary-operator? v) (binary-operator-precedence v))
@ -140,6 +140,9 @@
#:declare dt (static dot-transformer? "dot transformer") #:declare dt (static dot-transformer? "dot transformer")
(dot-transform (attribute dt.value) stx)])) (dot-transform (attribute dt.value) stx)]))
(define-syntax (#%rest stx)
(raise-syntax-error '#%rest "Illegal outside of function arguments" stx))
(begin-for-syntax (begin-for-syntax
(define-syntax-class remix-λ-raw-arg (define-syntax-class remix-λ-raw-arg
#:attributes (λ-arg λ-bind) #:attributes (λ-arg λ-bind)
@ -155,36 +158,31 @@
#:attr λ-bind (list #'(def def-lhs x)))) #:attr λ-bind (list #'(def def-lhs x))))
(define-syntax-class remix-λ-maybe-def-arg (define-syntax-class remix-λ-maybe-def-arg
#:attributes (λ-arg λ-bind) #:attributes (λ-arg λ-bind)
;; xxx write a test for this
(pattern x:remix-λ-raw-arg (pattern x:remix-λ-raw-arg
#:attr λ-arg #'x.λ-arg #:attr λ-arg #'x.λ-arg
#:attr λ-bind (attribute x.λ-bind)) #:attr λ-bind (attribute x.λ-bind))
;; xxx write a test for this
(pattern (x:remix-λ-raw-arg default:expr) (pattern (x:remix-λ-raw-arg default:expr)
#:attr λ-arg #'(x.λ-arg default) #:attr λ-arg #'(x.λ-arg default)
#:attr λ-bind (attribute x.λ-bind))) #:attr λ-bind (attribute x.λ-bind)))
(define-splicing-syntax-class remix-λ-arg (define-splicing-syntax-class remix-λ-arg
#:attributes ([λ-arg 1] λ-bind) #:attributes ([λ-arg 1] λ-bind)
;; xxx write a test for this
(pattern (~seq x:remix-λ-maybe-def-arg) (pattern (~seq x:remix-λ-maybe-def-arg)
#:attr [λ-arg 1] (list #'x.λ-arg) #:attr [λ-arg 1] (list #'x.λ-arg)
#:attr λ-bind (attribute x.λ-bind)) #:attr λ-bind (attribute x.λ-bind))
;; xxx write a test for this
(pattern (~seq kw:keyword x:remix-λ-maybe-def-arg) (pattern (~seq kw:keyword x:remix-λ-maybe-def-arg)
#:attr [λ-arg 1] (list #'kw #'x.λ-arg) #:attr [λ-arg 1] (list #'kw #'x.λ-arg)
#:attr λ-bind (attribute x.λ-bind))) #:attr λ-bind (attribute x.λ-bind)))
(define-syntax-class remix-λ-args (define-syntax-class remix-λ-args
#:attributes (λ-args #:attributes (λ-args
[λ-binds 1]) [λ-binds 1])
;; xxx write a test for this #:literals (#%rest)
(pattern () (pattern ()
#:attr λ-args (syntax ()) #:attr λ-args (syntax ())
#:attr [λ-binds 1] '()) #:attr [λ-binds 1] '())
;; xxx write a test for this (pattern (~or x:remix-λ-raw-arg
(pattern x:remix-λ-raw-arg (#%rest x:remix-λ-raw-arg))
#:attr λ-args (syntax x.λ-arg) #:attr λ-args (syntax x.λ-arg)
#:attr [λ-binds 1] (list #'x.λ-bind)) #:attr [λ-binds 1] (attribute x.λ-bind))
;; xxx write a test for this
(pattern (x:remix-λ-arg . xs:remix-λ-args) (pattern (x:remix-λ-arg . xs:remix-λ-args)
#:attr λ-args #:attr λ-args
#'(x.λ-arg ... . xs.λ-args) #'(x.λ-arg ... . xs.λ-args)
@ -243,6 +241,7 @@
(rename-out [remix-λ λ] (rename-out [remix-λ λ]
[remix-cond cond] [remix-cond cond]
[remix-cut-$ $]) [remix-cut-$ $])
#%rest
#%brackets #%brackets
#%braces #%braces
(for-syntax gen:binary-operator (for-syntax gen:binary-operator

3
remix/test0.rkt Normal file
View File

@ -0,0 +1,3 @@
#lang racket/base
(require rackunit)
(provide (rename-out [check-equal? ]))

View File

@ -7,11 +7,14 @@
;; we use require to get everything else. most of it comes from stx0 ;; we use require to get everything else. most of it comes from stx0
(require remix/stx0 (require remix/stx0
remix/num/gen0) remix/num/gen0)
(module+ test
;; This introduces ≡ as a testing form
(require remix/test0))
;; define is replaced with def ;; define is replaced with def
(def z 42) (def z 42)
(module+ test (module+ test
z) {z 42})
;; when def has more forms than one, they are put inside of a block ;; when def has more forms than one, they are put inside of a block
(def x (def x
@ -19,7 +22,7 @@
(def b 2) (def b 2)
(+ a b)) (+ a b))
(module+ test (module+ test
x) {x 42})
;; but of course def supports function definitions. [] is NOT the same ;; but of course def supports function definitions. [] is NOT the same
;; as (), it parses as #%brackets and defaults to expanding to a block ;; as (), it parses as #%brackets and defaults to expanding to a block
@ -29,7 +32,7 @@
z] z]
y)) y))
(module+ test (module+ test
(f x x)) {(f x x) 126})
;; cond requires []s for the question-answer pairs. It uses this to ;; cond requires []s for the question-answer pairs. It uses this to
;; make any code in between clauses go in between the `if`s that pop ;; make any code in between clauses go in between the `if`s that pop
@ -41,9 +44,9 @@
[(< z 100) "div 100"] [(< z 100) "div 100"]
[#:else z])) [#:else z]))
(module+ test (module+ test
(g 50) {(g 50) "100"}
(g 199) {(g 199) "div 100"}
(g 200)) {(g 200) 100})
;; the @ reader is always on. One fun thing about this is that you can ;; the @ reader is always on. One fun thing about this is that you can
;; make non-() macros. I wrote a little helper function to turn the ;; make non-() macros. I wrote a little helper function to turn the
@ -65,7 +68,7 @@
(def v7 (def v7
{3 + 4}) {3 + 4})
(module+ test (module+ test
v7) {v7 7})
;; {} use C's precedence and considers the things you expect to be ;; {} use C's precedence and considers the things you expect to be
;; operators. there's a syntax-time struct property that allows you to ;; operators. there's a syntax-time struct property that allows you to
@ -73,7 +76,7 @@
(def v-26 (def v-26
{2 * 3 - 48 / 4 - 4 * 5}) {2 * 3 - 48 / 4 - 4 * 5})
(module+ test (module+ test
v-26) {v-26 -26})
;; if a symbol contains no alphabetic or numeric characters, then it ;; if a symbol contains no alphabetic or numeric characters, then it
;; is considered an operator. This means you can automatically use ;; is considered an operator. This means you can automatically use
@ -81,19 +84,19 @@
(def v85 (def v85
{z * 2 + 1}) {z * 2 + 1})
(module+ test (module+ test
v85) {v85 85})
(def v1 (def v1
(def & bitwise-and) (def & bitwise-and)
{5 & 1}) {5 & 1})
(module+ test (module+ test
v1) {v1 1})
(def v56 (def v56
(def ( x y) (+ (* x x) y)) (def ( x y) (+ (* x x) y))
{v7 v7}) {v7 v7})
(module+ test (module+ test
v56) {v56 56})
;; However, if you use , then you can force anything to be a binary ;; However, if you use , then you can force anything to be a binary
;; operator and force something that would have been a binary operator ;; operator and force something that would have been a binary operator
@ -102,18 +105,18 @@
(def (f x y) (+ x y)) (def (f x y) (+ x y))
{v7 ,f v7}) {v7 ,f v7})
(module+ test (module+ test
v14) {v14 14})
(def v14b (def v14b
{v7 ,(λ (x y) (+ x y)) v7}) {v7 ,(λ (x y) (+ x y)) v7})
(module+ test (module+ test
v14b) {v14b 14})
(def v9 (def v9
(def & 2) (def & 2)
{v7 + ,&}) {v7 + ,&})
(module+ test (module+ test
v9) {v9 9})
;; λ is a dot-transformer for cut ;; λ is a dot-transformer for cut
(def f11 (def f11
@ -121,29 +124,29 @@
(def v11 (def v11
(f11 'ignored)) (f11 'ignored))
(module+ test (module+ test
v11) {v11 11})
(def v11b (def v11b
(λ.(+ 10 1) 'ignored)) (λ.(+ 10 1) 'ignored))
(module+ test (module+ test
v11b) {v11b 11})
(def v11c (def v11c
(λ.(+ $ 1) 10)) (λ.(+ $ 1) 10))
(module+ test (module+ test
v11c) {v11c 11})
;; ≙ is a synonym for def, and because of the {} rules, is a binary ;; ≙ is a synonym for def, and because of the {} rules, is a binary
;; operator. ;; operator.
{v33 33} {v33 33}
(module+ test (module+ test
v33) {v33 33})
(def v28 (def v28
{(f x) x + x} {(f x) x + x}
(f 14)) (f 14))
(module+ test (module+ test
v28) {v28 28})
;; def* allows nested binding inside blocks ;; def* allows nested binding inside blocks
(def v64 (def v64
@ -155,6 +158,26 @@
(def* x {x + x}) (def* x {x + x})
x) x)
(module+ test (module+ test
v64) {v64 64})
;; (def [stx #%posn] (layout x y)) ;; (def [stx #%posn] (layout x y))
(def (f-no-args) 42)
(def (f-one-arg x) x)
(def (f-kw-arg #:x x) x)
(def (f-kw-args #:x x y) (+ x y))
(def (f-def-arg (x 20) (y 22)) (+ x y))
(def (f-two-arg x y) (+ x y))
(def (f-rest-args #%rest x) 42)
(module+ test
{(f-no-args) 42}
{(f-one-arg 42) 42}
{(f-kw-arg #:x 42) 42}
{(f-kw-args #:x 22 20) 42}
{(f-two-arg 20 22) 42}
{(f-def-arg) 42}
{(f-def-arg 21) 43}
{(f-def-arg 21 21) 42}
{(f-rest-args) 42}
{(f-rest-args 1) 42}
{(f-rest-args 1 2 3) 42})