Closes FB case 176 Fix the type of nodes for (non-)polymorphic graph types
This commit is contained in:
parent
7991ed7d7e
commit
693ab9e84e
|
@ -265,7 +265,7 @@ data.
|
||||||
(syntax->datum (datum->syntax #f v)))
|
(syntax->datum (datum->syntax #f v)))
|
||||||
(case mode
|
(case mode
|
||||||
[(#t)
|
[(#t)
|
||||||
(display "#(~#t~" port)
|
(display "#(" port)
|
||||||
(display name port)
|
(display name port)
|
||||||
(for-each (λ (f)
|
(for-each (λ (f)
|
||||||
(display " " port)
|
(display " " port)
|
||||||
|
@ -273,7 +273,7 @@ data.
|
||||||
fields)
|
fields)
|
||||||
(display ")" port)]
|
(display ")" port)]
|
||||||
[(#f)
|
[(#f)
|
||||||
(display "#(~#f~" port)
|
(display "#(" port)
|
||||||
(display name port)
|
(display name port)
|
||||||
(for-each (λ (f)
|
(for-each (λ (f)
|
||||||
(display " " port)
|
(display " " port)
|
||||||
|
|
|
@ -13,33 +13,67 @@
|
||||||
'("(lib phc-graph/scribblings/phc-graph-implementation.scrbl)"
|
'("(lib phc-graph/scribblings/phc-graph-implementation.scrbl)"
|
||||||
"phc-graph/graph-type"))
|
"phc-graph/graph-type"))
|
||||||
|
|
||||||
@CHUNK[<define-graph-type>
|
The @racket[define-graph-type] form binds @racket[name] to a
|
||||||
|
@racket[graph-info] struct. The @racket[name] therefore contains metadata
|
||||||
|
describing among other things the types of nodes, the invariants that
|
||||||
|
instances of this graph type will satisfy.
|
||||||
|
|
||||||
|
@chunk[<signature>
|
||||||
(begin-for-syntax
|
(begin-for-syntax
|
||||||
(define-syntax-class signature
|
(define-syntax-class signature
|
||||||
#:datum-literals (∈ ∋ ≡ ≢ ∉)
|
#:datum-literals (∈ ∋ ≡ ≢ ∉)
|
||||||
#:literals (:)
|
#:literals (:)
|
||||||
(pattern (name
|
(pattern
|
||||||
{~maybe #:∀ (tvar …)}
|
(~no-order {~once name}
|
||||||
(~and {~seq [nodeᵢ:id [fieldᵢⱼ:id : τ] …] …}
|
{~maybe #:∀ (tvar …)}
|
||||||
{~seq [root-node . _] _ …})
|
{~once (~and {~seq (nodeᵢ:id [fieldᵢⱼ:id : τᵢⱼ:type]
|
||||||
{~seq #:invariant a {~and op {~or ∈ ∋ ≡ ≢ ∉}} b} …
|
…) …}
|
||||||
{~seq #:invariant p} …))))
|
{~seq [root-node . _] _ …})}
|
||||||
|
{~seq #:invariant a {~and op {~or ∈ ∋ ≡ ≢ ∉}} b}
|
||||||
|
{~seq #:invariant p}))))]
|
||||||
|
|
||||||
(define-for-syntax (compute-graph-info stx)
|
@section{Implementation}
|
||||||
|
|
||||||
|
The @racket[define-graph-type] macro expands to code which defines names for
|
||||||
|
the node types. It then binds the given @racket[name] to the
|
||||||
|
@racket[graph-info] instance built by @racket[build-graph-info].
|
||||||
|
|
||||||
|
@CHUNK[<define-graph-type>
|
||||||
|
(begin-for-syntax
|
||||||
|
(define-template-metafunction (!check-remembered-node! stx)
|
||||||
|
(syntax-case stx ()
|
||||||
|
[(_ nodeᵢ fieldᵢⱼ …)
|
||||||
|
(syntax-local-template-metafunction-introduce
|
||||||
|
(check-remembered-node! #'(nodeᵢ fieldᵢⱼ …)))])))
|
||||||
|
|
||||||
|
(define-syntax/parse (define-graph-type . {~and whole :signature})
|
||||||
|
;; fire off the eventual delayed errors added by build-graph-info
|
||||||
|
(lift-maybe-delayed-errors)
|
||||||
|
#`(begin
|
||||||
|
<declare-node-types>
|
||||||
|
(define-syntax name
|
||||||
|
(build-graph-info (quote-syntax whole)))))]
|
||||||
|
|
||||||
|
@section{Declaring the node types}
|
||||||
|
|
||||||
|
@chunk[<declare-node-types>
|
||||||
|
(define-type nodeᵢ
|
||||||
|
(Promise
|
||||||
|
((!check-remembered-node! nodeᵢ fieldᵢⱼ …) τᵢⱼ …
|
||||||
|
'Database
|
||||||
|
'Index)))
|
||||||
|
…]
|
||||||
|
|
||||||
|
@section{Creating the @racket[graph-info] instance}
|
||||||
|
|
||||||
|
@CHUNK[<build-graph-info>
|
||||||
|
(define-for-syntax (build-graph-info stx)
|
||||||
(parameterize ([disable-remember-immediate-error #t])
|
(parameterize ([disable-remember-immediate-error #t])
|
||||||
(syntax-parse stx
|
(syntax-parse stx
|
||||||
[:signature
|
[:signature
|
||||||
<graph-info>])))
|
<graph-info>])))]
|
||||||
(define-syntax/parse (define-graph-type . whole:signature)
|
|
||||||
;; fire off the eventual delayed errors added by compute-graph-info
|
|
||||||
(lift-maybe-delayed-errors)
|
|
||||||
#`(begin
|
|
||||||
(define-syntax whole.name
|
|
||||||
(compute-graph-info (quote-syntax whole)))))]
|
|
||||||
|
|
||||||
@chunk[<graph-info>
|
@chunk[<graph-info>
|
||||||
#:with (node-incompleteᵢ …) (stx-map #λ(format-id % " ~a-incomplete" %)
|
|
||||||
#'(nodeᵢ …))
|
|
||||||
(graph-info #'name
|
(graph-info #'name
|
||||||
(syntax->list (if (attribute tvar) #'(tvar …) #'()))
|
(syntax->list (if (attribute tvar) #'(tvar …) #'()))
|
||||||
#'root-node
|
#'root-node
|
||||||
|
@ -69,7 +103,7 @@
|
||||||
(stx-map (λ/syntax-case (fieldᵢⱼ τᵢⱼ) ()
|
(stx-map (λ/syntax-case (fieldᵢⱼ τᵢⱼ) ()
|
||||||
<field-info>)
|
<field-info>)
|
||||||
#'([fieldᵢⱼ τᵢⱼ] …))))
|
#'([fieldᵢⱼ τᵢⱼ] …))))
|
||||||
(check-remembered-node! #'(nodeᵢ fieldᵢⱼ …))
|
#'nodeᵢ ; promise type
|
||||||
#;(meta-struct-constructor
|
#;(meta-struct-constructor
|
||||||
(check-remembered-tagged! #'(node-incompleteᵢ fieldᵢⱼ …)))
|
(check-remembered-tagged! #'(node-incompleteᵢ fieldᵢⱼ …)))
|
||||||
#;(check-remembered-tagged! #'(node-incompleteᵢ fieldᵢⱼ …)))]
|
#;(check-remembered-tagged! #'(node-incompleteᵢ fieldᵢⱼ …)))]
|
||||||
|
@ -85,19 +119,26 @@
|
||||||
(invariant-info #'predicateTODO
|
(invariant-info #'predicateTODO
|
||||||
#'witnessTODO)]
|
#'witnessTODO)]
|
||||||
|
|
||||||
|
@section{Putting it all together}
|
||||||
|
|
||||||
@chunk[<*>
|
@chunk[<*>
|
||||||
(require racket/require
|
(require racket/require
|
||||||
phc-toolkit
|
phc-toolkit
|
||||||
remember
|
remember
|
||||||
(lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
(lib "phc-adt/tagged-structure-low-level.hl.rkt")
|
||||||
(for-syntax "graph-info.hl.rkt"
|
(for-syntax "graph-info.hl.rkt"
|
||||||
|
type-expander/expander
|
||||||
phc-toolkit/untyped
|
phc-toolkit/untyped
|
||||||
(subtract-in syntax/parse phc-graph/subtemplate)
|
(subtract-in syntax/parse phc-graph/subtemplate)
|
||||||
racket/set
|
racket/set
|
||||||
phc-graph/subtemplate-override
|
phc-graph/subtemplate-override
|
||||||
racket/syntax)
|
racket/syntax
|
||||||
|
extensible-parser-specifications
|
||||||
|
backport-template-pr1514/experimental/template)
|
||||||
(for-meta 2 racket/base))
|
(for-meta 2 racket/base))
|
||||||
|
|
||||||
(provide define-graph-type)
|
(provide define-graph-type)
|
||||||
|
|
||||||
|
<signature>
|
||||||
|
<build-graph-info>
|
||||||
<define-graph-type>]
|
<define-graph-type>]
|
3
info.rkt
3
info.rkt
|
@ -14,7 +14,8 @@
|
||||||
"typed-map"
|
"typed-map"
|
||||||
"scribble-lib"
|
"scribble-lib"
|
||||||
"pconvert-lib"
|
"pconvert-lib"
|
||||||
"remember"))
|
"remember"
|
||||||
|
"extensible-parser-specifications"))
|
||||||
(define build-deps '("scribble-lib"
|
(define build-deps '("scribble-lib"
|
||||||
"racket-doc"
|
"racket-doc"
|
||||||
"remember"
|
"remember"
|
||||||
|
|
49
test/assumption-equivalent-types-same-type.rkt
Normal file
49
test/assumption-equivalent-types-same-type.rkt
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
#lang typed/racket
|
||||||
|
|
||||||
|
;; Check that equivalent type specifications are correctly interpreted as
|
||||||
|
;; being the same type by Typed/Racket.
|
||||||
|
;;
|
||||||
|
;; This was not the case in some situations in older versions of Typed/Racket,
|
||||||
|
;; but I am not sure whether this reproduces the same issue, or whether this
|
||||||
|
;; file would typecheck in older versions too.
|
||||||
|
|
||||||
|
(let ()
|
||||||
|
(define-type (Foo X)
|
||||||
|
(U X (List 'foo (Bar X) (Foo X))))
|
||||||
|
|
||||||
|
(define-type (Bar Y)
|
||||||
|
(List 'bar (Foo Y)))
|
||||||
|
|
||||||
|
(define-type (Foo2 X)
|
||||||
|
(U X (List 'foo (Bar2 X) (Foo2 X))))
|
||||||
|
|
||||||
|
(define-type (Bar2 Y)
|
||||||
|
(List 'bar (Foo2 Y)))
|
||||||
|
|
||||||
|
(λ #:∀ (A) ([x : (Foo A)])
|
||||||
|
;; Check here:
|
||||||
|
(ann (ann x (Foo2 A)) (Foo A)))
|
||||||
|
|
||||||
|
(void))
|
||||||
|
|
||||||
|
(struct (a b) st-foo ([a : a] [b : b]))
|
||||||
|
(struct (a) st-bar ([a : a]))
|
||||||
|
|
||||||
|
(let ()
|
||||||
|
(define-type (Foo X)
|
||||||
|
(U X (st-foo (Bar X) (Foo X))))
|
||||||
|
|
||||||
|
(define-type (Bar Y)
|
||||||
|
(st-bar (Foo Y)))
|
||||||
|
|
||||||
|
(define-type (Foo2 X)
|
||||||
|
(U X (st-foo (Bar2 X) (Foo2 X))))
|
||||||
|
|
||||||
|
(define-type (Bar2 Y)
|
||||||
|
(st-bar (Foo2 Y)))
|
||||||
|
|
||||||
|
(λ #:∀ (A) ([x : (Foo A)])
|
||||||
|
;; Check here:
|
||||||
|
(ann (ann x (Foo2 A)) (Foo A)))
|
||||||
|
|
||||||
|
(void))
|
|
@ -17,13 +17,23 @@
|
||||||
#:invariant City.citizens._ ∈ City.streets._.houses._.owner
|
#:invariant City.citizens._ ∈ City.streets._.houses._.owner
|
||||||
#:invariant City.citizens._ ∋ City.streets._.houses._.owner)
|
#:invariant City.citizens._ ∋ City.streets._.houses._.owner)
|
||||||
|
|
||||||
(module* test racket/base
|
(require (for-syntax racket/pretty
|
||||||
(require (for-syntax racket/pretty
|
racket/base))
|
||||||
racket/base)
|
(eval #'(begin
|
||||||
(submod ".."))
|
(define-syntax (dbg _stx)
|
||||||
(eval #'(begin
|
(parameterize ([pretty-print-columns 188])
|
||||||
(define-syntax (dbg _stx)
|
(pretty-print (syntax-local-value #'g1)))
|
||||||
(parameterize ([pretty-print-columns 188])
|
#'(void))
|
||||||
(pretty-print (syntax-local-value #'g1)))
|
(dbg)))
|
||||||
#'(void))
|
|
||||||
(dbg))))
|
(require (for-syntax syntax/parse
|
||||||
|
"../graph-info.hl.rkt"))
|
||||||
|
|
||||||
|
(define-syntax dbg
|
||||||
|
(syntax-parser
|
||||||
|
[(_ t)
|
||||||
|
#`(define-type t
|
||||||
|
#,(node-info-promise-type
|
||||||
|
(hash-ref (graph-info-nodes (syntax-local-value #'g1)) 'City)))]))
|
||||||
|
(dbg t-city)
|
||||||
|
;(define-type expected (t-city Number String Symbol 'Database 'Index))
|
Loading…
Reference in New Issue
Block a user