subtemplate/test/test-ddd.rkt
2017-02-03 12:50:05 +01:00

90 lines
2.9 KiB
Racket

#lang racket
(require subtemplate/private/ddd
stxparse-info/case
stxparse-info/parse
(only-in racket/base [... ])
rackunit
syntax/macro-testing
syntax/stx)
(check-equal? (syntax-case #'((1 2 3) (4 5)) ()
[((x ) )
(ddd (list (length (syntax->list #'(x )))
(ddd (+ (syntax-e #'x) 3))))])
'([3 (4 5 6)]
[2 (7 8)]))
(check-equal? (syntax-case #'(1 2 3) ()
[(x )
(ddd (+ (syntax-e #'x) 3))])
'(4 5 6))
(check-equal? (syntax-parse #'(1 2 3)
[(x )
(ddd (+ (syntax-e #'x) 3))])
'(4 5 6))
(check-equal? (syntax-case #'(((1 2) (3)) ((4 5 6))) ()
[(((x ) ) )
(ddd (list (length (syntax->list #'((x ) )))
(length (syntax->list #'(x )))
(ddd (ddd (- (syntax-e #'x))))))])
'([2 3 ((-1 -2) (-3))]
[1 3 ((-4 -5 -6))]))
(check-equal? (syntax-case #'([1 2 3] [a]) ()
[([x ] [y ])
(ddd (+ (syntax-e #'x) 3))])
'(4 5 6))
(check-equal? (syntax-case #'(([1 2 3] [a])) ()
[(([x ] [y ]) )
(ddd (ddd (+ (syntax-e #'x) 3)))])
'((4 5 6)))
;; The inner ddd should not make the outer one consider the variables actually
;; used. This test will break if y is considered to be used, because it does not
;; have the same shape as x anywhere, so map will complain that the lists do not
;; have the same length.
(check-equal? (syntax-case #'([#:xs (1 2 3) (4 5)]
[#:ys (a) (b) (c) (d)]) ()
[([#:xs (x ) ]
[#:ys (y ) ])
(ddd (ddd (+ (syntax-e #'x) 3)))])
'((4 5 6) (7 8)))
(check-exn
#rx"no pattern variables with depth > 0 were found in the body"
(λ ()
(convert-compile-time-error
(syntax-parse #'(1 2 3)
[(x y z)
(ddd (+ (syntax-e #'x) 3))]))))
(check-equal? (syntax-parse #'(1 2 3 4)
[(x y)
(ddd (+ (syntax-e #'x) (syntax-e #'y)))])
'(5 6 7))
(check-equal? (syntax-case #'((1 2 3) (4 5)) ()
[((x ) )
(ddd (list (length (syntax->list #'(x )))
(ddd (+ (syntax-e #'x) 3))))])
'([3 (4 5 6)]
[2 (7 8)]))
;; omitted element at the leaves = ok (should it be ok?)
(check-equal? (syntax-parse #'(1 #f 3)
[({~and {~or x:nat #f}} )
(ddd x)])
'(1 #f 3))
;; omitted element in the tree = not ok
(check-exn
#rx"attribute contains an omitted element"
(λ ()
(syntax-parse #'((1 1) #f (1 2 1 1))
[({~and {~or (x:nat ) #f}} )
(ddd (ddd x))])))