Wrote parser for paths
This commit is contained in:
parent
120746442e
commit
32379533ed
|
@ -134,8 +134,8 @@ the union never becomes empty.
|
||||||
This solution also has the advantage that the size of the run-time witness is
|
This solution also has the advantage that the size of the run-time witness is
|
||||||
constant, and does not depend on the number of checked contracts (unlike the
|
constant, and does not depend on the number of checked contracts (unlike the
|
||||||
representation using a list). In practice the function should never be called.
|
representation using a list). In practice the function should never be called.
|
||||||
It can however simply be implemented in a way which pleases the type checked
|
It can however simply be implemented, in a way which will match all witness
|
||||||
as a function accepting anything and returning void.
|
types, as a function accepting anything and returning void.
|
||||||
|
|
||||||
In addition to testifying that a graph node was checked against multiple,
|
In addition to testifying that a graph node was checked against multiple,
|
||||||
separate contracts, there might be some contracts which check stronger
|
separate contracts, there might be some contracts which check stronger
|
||||||
|
@ -328,8 +328,8 @@ As mentioned earlier, we group invariants together using an untagged union
|
||||||
@racket[U], which must appear in a contravariant position. We wish to express
|
@racket[U], which must appear in a contravariant position. We wish to express
|
||||||
witnesses for stronger invariants as subtypes of witnesses for weaker
|
witnesses for stronger invariants as subtypes of witnesses for weaker
|
||||||
invariants, and therefore use a second nested function type to flip again the
|
invariants, and therefore use a second nested function type to flip again the
|
||||||
variance direction. We always include the Or element in the union, to avoid
|
variance direction. We always include the @racket[Or] element in the union, to
|
||||||
having an empty union.
|
avoid ever having an empty union.
|
||||||
|
|
||||||
@chunk[<grouping-invariants>
|
@chunk[<grouping-invariants>
|
||||||
(struct Or ())
|
(struct Or ())
|
||||||
|
@ -371,6 +371,95 @@ start on an element of type @racket[A], and follow its fields @racket[b] then
|
||||||
@racket[c]. The syntax @racket[.x:T.y.z] indicates paths @racket[.x.y.z], where
|
@racket[c]. The syntax @racket[.x:T.y.z] indicates paths @racket[.x.y.z], where
|
||||||
the element accessed by @racket[.x] has the type @racket[T].
|
the element accessed by @racket[.x] has the type @racket[T].
|
||||||
|
|
||||||
|
The @racket[?] path element indicates that any field can be used at the given
|
||||||
|
point. The syntax @racket[.x.?.y] therefore indicates the paths
|
||||||
|
@racket[.x.f.y], @racket[.x.g.y], @racket[.x.h.y] and so on, where @racket[f],
|
||||||
|
@racket[g] and @racket[h] are the fields of the value obtained after
|
||||||
|
@racket[.x]. The @racket[?] path element can be used to describe all fields,
|
||||||
|
including those hidden away by row polymorphism.
|
||||||
|
|
||||||
|
It would be possible to represent sets of path concisely by reversing all
|
||||||
|
these paths so that they start with their target element, and building a
|
||||||
|
prefix tree. Unfortunately, Typed Racket does not currently does not
|
||||||
|
automatically detect the equivalence between the types
|
||||||
|
@racket[(U (Pairof A B) (Pairof A C))] and @racket[(Pairof A (U B C))]. @;{
|
||||||
|
TODO: insert reference to the relevant bug.} In other words, at the type
|
||||||
|
level, unions do not implicitly distribute onto pairs. When a set of
|
||||||
|
properties @racket[S] is extended with a new property @racket[p], the
|
||||||
|
resulting set of properties will be encoded as @racket[(U S p)]. If @racket[S]
|
||||||
|
is encoded as a prefix tree, @racket[p] will not implicitly be merged into the
|
||||||
|
prefix tree. This means that if prefix trees were to be used, extending a set
|
||||||
|
of properties with a new property would give one representation, and directly
|
||||||
|
encoding that set would give another representation, and the two
|
||||||
|
representations would be incomparable.
|
||||||
|
|
||||||
|
We therefore represent sets of paths using a more costly representation, by
|
||||||
|
making a union of all paths, without factoring out common parts.
|
||||||
|
|
||||||
|
@subsection{Parsing paths}
|
||||||
|
|
||||||
|
@; TODO: make sure the literal "dot" is provided.
|
||||||
|
|
||||||
|
@; .a.b(.c.d)*.e
|
||||||
|
@; ⇒ (λdot a b) ((λdot c d)) * (λdot e)
|
||||||
|
|
||||||
|
@chunk[<parse>
|
||||||
|
(begin-for-syntax
|
||||||
|
(define-syntax-class dot-ish
|
||||||
|
#:literals (dot λdot)
|
||||||
|
(pattern ({~and d dot} e …)
|
||||||
|
#:with full
|
||||||
|
(add-between (syntax->list #'(e …))
|
||||||
|
(datum->syntax #'d '|.| #'d #'d)))
|
||||||
|
(pattern ({~and d λdot} e …)
|
||||||
|
#:with full
|
||||||
|
(cons #'d (add-between (syntax->list #'(e …))
|
||||||
|
(datum->syntax #'d '|.| #'d #'d)))))
|
||||||
|
(define-syntax ~dot-ish
|
||||||
|
(pattern-expander
|
||||||
|
(λ (stx)
|
||||||
|
(syntax-case stx ()
|
||||||
|
[(_ pat …)
|
||||||
|
#'(~and x:dot-ish
|
||||||
|
(~parse (pat …) #'x))]))))
|
||||||
|
|
||||||
|
(define (match-id rx id)
|
||||||
|
(let ([m (regexp-match rx (identifier→string id))])
|
||||||
|
(and m (map (λ (%) (datum->syntax id (string->symbol %) id id))
|
||||||
|
(cdr m)))))
|
||||||
|
(define-syntax ~rx-id
|
||||||
|
(pattern-expander
|
||||||
|
(λ (stx)
|
||||||
|
(syntax-case stx ()
|
||||||
|
[(_ rx g …)
|
||||||
|
#'(~and x:id
|
||||||
|
{~parse (g …) (match-id rx #'x)})]))))
|
||||||
|
|
||||||
|
(define-syntax-class f+τ
|
||||||
|
#:attributes (f τ)
|
||||||
|
(pattern {~rx-id #px"^([^:]+):([^:]+)$" f τ})
|
||||||
|
(pattern {~rx-id #px"^([^:]+)$" f} #:with ({~optional τ}) #'())
|
||||||
|
(pattern {~rx-id #px"^:([^:]+)$" τ} #:with ({~optional f}) #'()))
|
||||||
|
(define-syntax-class just-τ
|
||||||
|
#:attributes (τ)
|
||||||
|
(pattern {~rx-id #px"^:([^:]+)$" τ}))
|
||||||
|
(define-syntax-class π-elements
|
||||||
|
#:literals (|.|)
|
||||||
|
(pattern
|
||||||
|
(~dot-ish {~or {~seq |.| :f+τ} elt:π-elements} …)))
|
||||||
|
(define-syntax-class whole-π
|
||||||
|
#:literals (|.|)
|
||||||
|
(pattern
|
||||||
|
(~dot-ish {~optional τ₀:just-τ}
|
||||||
|
{~or {~seq |.| :f+τ} elt:π-elements} …)))
|
||||||
|
)]
|
||||||
|
|
||||||
|
@chunk[<parse>
|
||||||
|
(define-for-syntax parse-path
|
||||||
|
(syntax-parser
|
||||||
|
[:whole-π
|
||||||
|
'ok]))]
|
||||||
|
|
||||||
@subsubsection{Comparison operator tokens}
|
@subsubsection{Comparison operator tokens}
|
||||||
|
|
||||||
We define some tokens which will be used to identify the operator which
|
We define some tokens which will be used to identify the operator which
|
||||||
|
@ -447,6 +536,7 @@ relates two nodes in the graph.
|
||||||
@; becomes (Pairof (∩ 'a 'x) (∩ 'b 'c)), which is equivalent to have all four
|
@; becomes (Pairof (∩ 'a 'x) (∩ 'b 'c)), which is equivalent to have all four
|
||||||
@; elements of {a,x} × {b,c}, but we only want three out of these four.
|
@; elements of {a,x} × {b,c}, but we only want three out of these four.
|
||||||
|
|
||||||
|
@;{
|
||||||
Two sorts of paths inside (in)equality constraints:
|
Two sorts of paths inside (in)equality constraints:
|
||||||
|
|
||||||
@itemlist[
|
@itemlist[
|
||||||
|
@ -461,12 +551,18 @@ Two sorts of paths inside (in)equality constraints:
|
||||||
∀\ \textit{node}₂ : \textit{NodeType}₂,\\
|
∀\ \textit{node}₂ : \textit{NodeType}₂,\\
|
||||||
\textit{node}₁.\textit{path}₁ ≡ \textit{node}₂.\textit{path}₂
|
\textit{node}₁.\textit{path}₁ ≡ \textit{node}₂.\textit{path}₂
|
||||||
\end{array}}}]
|
\end{array}}}]
|
||||||
|
}
|
||||||
|
|
||||||
@subsection{Putting it all together}
|
@subsection{Putting it all together}
|
||||||
|
|
||||||
@chunk[<*>
|
@chunk[<*>
|
||||||
(require (for-syntax phc-toolkit/untyped
|
(require (for-syntax racket/base
|
||||||
syntax/parse))
|
racket/list
|
||||||
|
phc-toolkit/untyped
|
||||||
|
syntax/parse)
|
||||||
|
(for-meta 2 racket/base)
|
||||||
|
(for-meta 2 phc-toolkit/untyped/aliases)
|
||||||
|
(for-meta 3 racket/base))
|
||||||
|
|
||||||
;; For testing:
|
;; For testing:
|
||||||
(provide Invariants
|
(provide Invariants
|
||||||
|
@ -479,6 +575,8 @@ Two sorts of paths inside (in)equality constraints:
|
||||||
ε
|
ε
|
||||||
witness-value)
|
witness-value)
|
||||||
|
|
||||||
|
<parse>
|
||||||
|
|
||||||
<witness-value>
|
<witness-value>
|
||||||
<grouping-invariants>
|
<grouping-invariants>
|
||||||
<cycles>
|
<cycles>
|
||||||
|
|
|
@ -4,6 +4,42 @@
|
||||||
"util.rkt"
|
"util.rkt"
|
||||||
phc-toolkit)
|
phc-toolkit)
|
||||||
|
|
||||||
|
#|
|
||||||
|
|
||||||
|
(check-ann witness-value (Invariants)) ;; No invariants
|
||||||
|
(check-ann witness-value (Invariants (≡ (_ a) (_ a b c))))
|
||||||
|
|
||||||
|
(check-a-stronger-than-b (Invariants (≡ (_ a) (_ a b c)))
|
||||||
|
(Invariants))
|
||||||
|
|
||||||
|
(check-a-same-as-b (Invariants (≡ (_ a) (_ a b c)))
|
||||||
|
(Invariants (≡ (_ a b c) (_ a))))
|
||||||
|
|
||||||
|
(check-a-stronger-than-b (Invariants (≡ (_) (_ b c))
|
||||||
|
(≡ (_) (_ b d)))
|
||||||
|
(Invariants (≡ (_) (_ b c))))
|
||||||
|
(check-a-stronger-than-b (Invariants (≡ (_) (_ b d))
|
||||||
|
(≡ (_) (_ b c)))
|
||||||
|
(Invariants (≡ (_) (_ b c))))
|
||||||
|
|
||||||
|
;; ∀ .b.d(.a.b.>d)* of length ≥ 5
|
||||||
|
;; is stronger than
|
||||||
|
;; ∀ .b.d(.a.b.>d)* of length ≥ 8
|
||||||
|
;; as the elements of the latter are included in the former, but
|
||||||
|
;; the first element (length = 5) is missing in the latter, so the
|
||||||
|
;; former constrains more paths.
|
||||||
|
(check-a-stronger-than-b (Invariants (≡ (_)
|
||||||
|
(_ b d ↙ a b (d))))
|
||||||
|
(Invariants (≡ (_)
|
||||||
|
(_ b d a b d ↙ a b (d)))))
|
||||||
|
|
||||||
|
(check-a-stronger-than-b (Invariants (≡ (_)
|
||||||
|
(_ a b c ↙ d (e))))
|
||||||
|
(Invariants (≡ (_)
|
||||||
|
(_ a b c d e))))
|
||||||
|
|#
|
||||||
|
|
||||||
|
|
||||||
(check-ann witness-value (Invariants)) ;; No invariants
|
(check-ann witness-value (Invariants)) ;; No invariants
|
||||||
(check-ann witness-value (Invariants (≡ (_ a) (_ a b c))))
|
(check-ann witness-value (Invariants (≡ (_ a) (_ a b c))))
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user