Reflection in static interfaces & keyword members
This commit is contained in:
parent
f7f1a57334
commit
9476f32ce1
|
@ -149,6 +149,12 @@ functions!
|
||||||
|
|
||||||
TODO how to make sure cases of cond/etc are complete?
|
TODO how to make sure cases of cond/etc are complete?
|
||||||
|
|
||||||
|
A bare minimum dynamic solution is to do a module-lift to a test
|
||||||
|
property that use enumeration to enumerate inputs and that sort of
|
||||||
|
thing. A bare minimum static solution is to have the data and the
|
||||||
|
variants and a mapping between cases and the variants. It may require
|
||||||
|
too much knowledge on the data structure than is Rackety.
|
||||||
|
|
||||||
TODO operator overloading definitions of + and ones that check the
|
TODO operator overloading definitions of + and ones that check the
|
||||||
types for float/int/fixnum/etc
|
types for float/int/fixnum/etc
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,8 @@
|
||||||
racket/base
|
racket/base
|
||||||
syntax/parse
|
syntax/parse
|
||||||
racket/syntax
|
racket/syntax
|
||||||
|
racket/generic
|
||||||
|
racket/format
|
||||||
(prefix-in remix: remix/stx0)
|
(prefix-in remix: remix/stx0)
|
||||||
remix/stx/singleton-struct0
|
remix/stx/singleton-struct0
|
||||||
(for-syntax racket/base
|
(for-syntax racket/base
|
||||||
|
@ -11,11 +13,24 @@
|
||||||
(prefix-in remix: remix/stx0))
|
(prefix-in remix: remix/stx0))
|
||||||
|
|
||||||
(begin-for-syntax
|
(begin-for-syntax
|
||||||
(define-syntax (static-interface stx)
|
(define-generics static-interface
|
||||||
|
(static-interface-members static-interface))
|
||||||
|
|
||||||
|
(module interface-member racket/base
|
||||||
|
(require syntax/parse)
|
||||||
|
(define-syntax-class interface-member
|
||||||
|
(pattern x:id)
|
||||||
|
(pattern x:keyword))
|
||||||
|
(provide interface-member))
|
||||||
|
(require (submod "." interface-member)
|
||||||
|
(for-syntax
|
||||||
|
(submod "." interface-member)))
|
||||||
|
|
||||||
|
(define-syntax (:static-interface stx)
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
#:literals (remix:#%brackets)
|
#:literals (remix:#%brackets)
|
||||||
[(_si (remix:#%brackets
|
[(_si (remix:#%brackets
|
||||||
lhs:id rhs:id
|
lhs:interface-member rhs:id
|
||||||
(~optional
|
(~optional
|
||||||
(~seq #:is rhs-dt:id)
|
(~seq #:is rhs-dt:id)
|
||||||
#:defaults ([rhs-dt #'#f])))
|
#:defaults ([rhs-dt #'#f])))
|
||||||
|
@ -31,7 +46,7 @@
|
||||||
(define available-ids
|
(define available-ids
|
||||||
(sort (hash-keys int-id->orig)
|
(sort (hash-keys int-id->orig)
|
||||||
string<=?
|
string<=?
|
||||||
#:key symbol->string))
|
#:key ~a))
|
||||||
(define (get-rhs stx x)
|
(define (get-rhs stx x)
|
||||||
(define xv (syntax->datum x))
|
(define xv (syntax->datum x))
|
||||||
(hash-ref int-id->orig
|
(hash-ref int-id->orig
|
||||||
|
@ -64,12 +79,15 @@
|
||||||
#:property prop:procedure
|
#:property prop:procedure
|
||||||
(λ (_ stx)
|
(λ (_ stx)
|
||||||
(raise-syntax-error 'int-name "Illegal in expression context" stx))
|
(raise-syntax-error 'int-name "Illegal in expression context" stx))
|
||||||
|
#:methods gen:static-interface
|
||||||
|
[(define (static-interface-members _)
|
||||||
|
available-ids)]
|
||||||
#:methods remix:gen:dot-transformer
|
#:methods remix:gen:dot-transformer
|
||||||
[(define (dot-transform _ stx)
|
[(define (dot-transform _ stx)
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
[(_dot me:id x:id)
|
[(_dot me:id x:interface-member)
|
||||||
(get-rhs-id stx #'x)]
|
(get-rhs-id stx #'x)]
|
||||||
[(_dot me:id x:id . more:expr)
|
[(_dot me:id x:interface-member . more:expr)
|
||||||
(quasisyntax/loc stx
|
(quasisyntax/loc stx
|
||||||
(remix:block
|
(remix:block
|
||||||
#,(get-rhs-def stx #'x)
|
#,(get-rhs-def stx #'x)
|
||||||
|
@ -77,10 +95,10 @@
|
||||||
#:methods remix:gen:app-dot-transformer
|
#:methods remix:gen:app-dot-transformer
|
||||||
[(define (app-dot-transform _ stx)
|
[(define (app-dot-transform _ stx)
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
[(_app (_dot me:id x:id) . body:expr)
|
[(_app (_dot me:id x:interface-member) . body:expr)
|
||||||
(quasisyntax/loc stx
|
(quasisyntax/loc stx
|
||||||
(#,(get-rhs-id stx #'x) . body))]
|
(#,(get-rhs-id stx #'x) . body))]
|
||||||
[(_app (_dot me:id x:id . more:expr) . body:expr)
|
[(_app (_dot me:id x:interface-member . more:expr) . body:expr)
|
||||||
(quasisyntax/loc stx
|
(quasisyntax/loc stx
|
||||||
(remix:block
|
(remix:block
|
||||||
#,(get-rhs-def stx #'x)
|
#,(get-rhs-def stx #'x)
|
||||||
|
@ -98,8 +116,10 @@
|
||||||
(remix:#%app rhs real-i . blah))
|
(remix:#%app rhs real-i . blah))
|
||||||
...
|
...
|
||||||
(remix:def (remix:#%brackets remix:stx i)
|
(remix:def (remix:#%brackets remix:stx i)
|
||||||
(static-interface
|
(:static-interface
|
||||||
(remix:#%brackets lhs def-rhs)
|
(remix:#%brackets lhs def-rhs)
|
||||||
...)))))]))]))))])))
|
...)))))]))]))))])))
|
||||||
|
|
||||||
(provide (for-syntax static-interface))
|
(provide (for-syntax (rename-out [:static-interface static-interface])
|
||||||
|
gen:static-interface
|
||||||
|
static-interface?))
|
||||||
|
|
|
@ -297,3 +297,15 @@
|
||||||
{(example3^.fg.f 2) ≡ 1}
|
{(example3^.fg.f 2) ≡ 1}
|
||||||
{(example3^.fg.g 2) ≡ 2}
|
{(example3^.fg.g 2) ≡ 2}
|
||||||
{example3^.h ≡ 19})
|
{example3^.h ≡ 19})
|
||||||
|
|
||||||
|
;; The syntax of interface members is not limited to identifiers. In
|
||||||
|
;; particular, #:keywords are useful.
|
||||||
|
(def example4-kw-key '#:key)
|
||||||
|
(def example4-key 'key)
|
||||||
|
(def [stx example4^]
|
||||||
|
(static-interface
|
||||||
|
[#:key example4-kw-key]
|
||||||
|
[key example4-key]))
|
||||||
|
(module+ test
|
||||||
|
{example4^.#:key ≡ '#:key}
|
||||||
|
{example4^.key ≡ 'key})
|
||||||
|
|
Loading…
Reference in New Issue
Block a user