phc-graph/dispatch-union.rkt

43 lines
1.8 KiB
Racket

#lang typed/racket/base
(require racket/require
phc-toolkit
phc-adt
(for-syntax (subtract-in racket/base "subtemplate.rkt")
phc-toolkit/untyped
racket/syntax
(subtract-in syntax/parse "subtemplate.rkt")
syntax/parse/experimental/template
type-expander/expander
"free-identifier-tree-equal.rkt"
"subtemplate.rkt")
(for-meta 2 racket/base)
(for-meta 2 phc-toolkit/untyped)
(for-meta 2 syntax/parse))
(provide dispatch-union)
(define-syntax/parse (dispatch-union v
([type-to-replaceᵢ Aᵢ predicateᵢ] )
[Xⱼ result] )
((λ (x) (local-require racket/pretty) #;(pretty-write (syntax->datum x)) x)
(quasisyntax/top-loc stx
(cond
;; TODO: put first the type-to-replaceᵢ, then afterwards the other Xⱼ, otherwise it can fail to typecheck.
. #,(stx-map
(λ (Xⱼ result)
(syntax-parse Xⱼ
#:literals (tagged)
[t
#:with (_ predicate)
(findf (λ (r) (free-identifier-tree=? #'t (stx-car r)))
(syntax->list
(subtemplate ([type-to-replaceᵢ predicateᵢ] ))))
#`[(predicate v) #,result]]
[(tagged name [fieldₖ (~optional :colon) typeₖ] )
#`[((tagged? name fieldₖ ) v) #,result]]
[other (raise-syntax-error 'graph
"Unhandled union type"
#'other)]))
#'(Xⱼ )
#'(result ))))))