Cleaned up tactic base system

* Various style fixes
* Standardize the proof state interface/naming a little.
* Reorganized base.rkt; now easier to read
* Fixed some mismatch code/documentation
* Renamed "basic" to "standard"; basic was too close to base, and not
  really helpful.
This commit is contained in:
William J. Bowman 2015-09-15 16:02:53 -04:00
parent 13d3016c85
commit 31603bf109
No known key found for this signature in database
GPG Key ID: DDD48D26958F0D1A
3 changed files with 182 additions and 154 deletions

View File

@ -4,149 +4,183 @@
(provide (provide
proof proof
define-theorem define-theorem
define-lemma
define-tactic define-tactic
(for-syntax (for-syntax
hole
proof-state proof-state
proof-state-env proof-state-env
proof-state-goals proof-state-goals
proof-state-current-goal proof-state-proof
proof-state-current-proof proof-state-theorem
proof-state-original-goal new-proof
print-proof-state
current-theorem
empty-proof
new-proof-state new-proof-state
push-env proof-state-proof-complete?
ctxt? proof-state-goal-ref
proof-complete? proof-state-current-goal-ref
update-current-proof proof-state-current-goal-set
update-current-goal proof-state-env-ref
proof-state-env-ref-by-type
proof-state-extend-env
proof-state-fill-proof-hole
proof-state-extend-proof-context
current-hole-pretty-symbol
print-proof-state
lookup-tactic)) lookup-tactic))
;; NB: The design of Cur currently seems to prevent intermediate type-checking #| NB:
;; NB: during a tactic-based proof, unless I were to either reimplement the | The design of Cur currently seems to prevent intermediate type-checking
;; NB: type-checker or extend it with a notion of holes. | during a tactic-based proof, unless I were to either reimplement the
;; TODO: A notion of holes my be useful in general... | type-checker or extend it with a notion of holes.
| TODO: A noation of holes may be useful in general.
|#
;; NB: Per below, note that Cur already has contexts. The issue with #| NB:
;; NB: type-checking intermediate results has to do with the fact that Cur | Per below, note that Cur already has contexts. The issue with
;; NB: contexts are not exposed properly, nor is type-checking defined on | type-checking intermediate results has to do with the fact that Cur
;; NB: them. | contexts are not exposed properly, nor is type-checking defined on
| them.
|#
;;; ====================================================================== ;; ------------------------------------------------------------------------
;;; Tactic support ;;; Proof state interface
;;; A Goal is a syntax object representing a Cur proposition (type). #|
| A Goal, aka, Theorem is a syntax object representing a Cur proposition (type).
;;; A Expr is a syntax object representing Cur expression (term). |
| A Expr is a syntax object representing Cur expression (term).
;;; A Hole is a procedure of type |
| A Hole is a procedure of type
;;; A Ctxt is a procedure (Either Ctxt Expr) -> (Either Ctxt Expr) |
;;; representing a Cur expression with a hole. | A Proof is (Either Ctxt Expr)
|
;;; A Environment is a map from Symbol to Goal. | A Ctxt is a procedure Proof -> Proof representing a Cur expression
| with a hole.
;;; The compile-time proof-state interface. |
| A Environment is a map from Symbol to Theorems.
|
| A Proof-State is a struct containing:
| env: An Environment, the assumptions introduced during a tactic
| script.
| goals: A map from Natural to Goal, the goals necessary to finish
| this proof
| current-goal: A Natural, the index into goals of the current Goal
| to be proved.
| proof: A Proof, representing the proof so far. when the proof is
| an Expr, the proof is complete.
| theorem: A Theorem, the original statement of the theorem this proof is
| attempting to prove.
|#
;;; TODO: I'm partially implementing lens here. ;;; TODO: I'm partially implementing lens here.
;;; TODO: Interface is mildly inconsistent. ;;; TODO: Interface is mildly inconsistent.
(begin-for-syntax (begin-for-syntax
(define-struct proof-state (env goals current-goal proof theorem))
(define new-proof values)
(define (new-proof-state prop)
(proof-state
(make-immutable-hash)
(dict-set (make-immutable-hash) 0 prop)
0
new-proof
prop))
;; Given a Proof-State, return #t if the current-proof is complete
;; (i.e. is a Expr and not a Ctxt), #f otherwise.
(define (proof-state-proof-complete? ps) (not (procedure? (proof-state-proof ps))))
;;; Extra accessors
;; Return the goal with index i in proof-state-goals
(define (proof-state-goal-ref ps i)
(dict-ref (proof-state-goals ps) i))
;; Return the current-goal from proof-state-goals
(define (proof-state-current-goal-ref ps)
(proof-state-goal-ref ps (proof-state-current-goal ps)))
;; Return the theorem named by name in the local environment
(define (proof-state-env-ref ps name)
(dict-ref (proof-state-env ps) name))
;; Return the name of an assumption with type thm, or #f.
(define (proof-state-env-ref-by-type ps thm)
(for/first ([(k v) (in-dict (proof-state-env ps))]
#:when (cur-equal? v thm))
k))
;;; Functional updators
(define (maybe-syntax->symbol name)
(if (syntax? name)
(syntax->datum name)
name))
;; Extend the local environment with a new assumption, name, of type thm. Name can be a syntax
;; object or a symbol.
(define (proof-state-extend-env ps name thm)
(struct-copy proof-state ps
[env (dict-set (proof-state-env ps) (maybe-syntax->symbol name) thm)]))
;; Updates the current-goal to pf
(define (proof-state-current-goal-set ps pf)
(struct-copy proof-state ps
[goals (dict-set (proof-state-goals ps) (proof-state-current-goal ps) pf)]))
;; Fill the current proof hole with pf
(define (proof-state-fill-proof-hole ps pf)
;; TODO: Check for proof completion?
;; TODO: What about multiple holes?
(struct-copy proof-state ps
[proof ((proof-state-proof ps) pf)]))
;; Place the current proof in the context ctxt.
(define (proof-state-extend-proof-context ps ctxt)
(struct-copy proof-state ps
[proof (ctxt (proof-state-proof ps))]))
;; How to display a hole, for pretty printing. ;; How to display a hole, for pretty printing.
(define hole 'hole) (define current-hole-pretty-symbol (make-parameter 'hole))
;; A Proof-State is a struct containing:
;; env: An Environment, the assumptions introduced during a tactic
;; script.
;; goals: A map from Natural to Goal, the goals necessary to finish
;; this proof
;; current-goal: A Natural, the index into goals of the current Goal
;; to be proved.
;; current-proof: A (Either Ctxt Expr), representing the proof so far.
;; current-proof is an Expr, the proof is complete.
(define-struct proof-state (env goals current-goal current-proof
original-goal))
;; A pretty printer for proof-state
(define (print-proof-state ps) (define (print-proof-state ps)
(for ([(k v) (in-dict (proof-state-env ps))]) (for ([(k v) (in-dict (proof-state-env ps))])
(printf "~n~a : ~a~n" k (syntax->datum v))) (printf "~n~a : ~a~n" k (syntax->datum v)))
(printf "--------------------------------~n") (printf "--------------------------------~n")
(cond (cond
[(proof-state-current-goal ps) => [(proof-state-current-goal-ref ps)
=>
(lambda (x) (printf "~a~n~n" (syntax->datum x)))] (lambda (x) (printf "~a~n~n" (syntax->datum x)))]
[else (printf "Goal complete!~n~n")])) [else (printf "Goal complete!~n~n")])))
;; The current theorem; used internally to achieve a Coq-like notation ;; -----------------------------------------------------------------------
;; for defining theorems and tactic-based proofs. ;;; Syntax for tactic-based proofs similar to Coq.
(define current-theorem (make-parameter #f))
(define empty-proof values) #|
| A tactic is a phase 1 Racket function that manipulates the current proof state.
(define (new-proof-state prop) | Tactic : Any ... Proof-State -> Proof-State
(proof-state (make-immutable-hash) (make-immutable-hash) prop |
empty-proof prop)) | Examples:
|
;; push-env adds a mapping from name to P in (proof-state-env ps). | (define-tactic command-name function)
;; Proof-State -> Symbol -> Goal -> Proof-State | (define-tactic (command-name args ... ps) body)
(define (push-env ps name P) |#
(struct-copy proof-state ps (define-syntax define-tactic (make-rename-transformer #'define-for-syntax))
[env (hash-set (proof-state-env ps) name P)]))
;; TODO: Cur already has contexts; perhaps they should be exposed?
;; Ctxt -> (Either Ctxt Expr) -> (Either Ctxt Expr)
;; C1[C2]
(define (plug-ctxt C1 C2) (C1 C2))
(define ctxt? procedure?)
;; Given a Proof-State, return #t if the current-proof is complete
;; (i.e. is a Expr and not a Ctxt), #f otherwise.
(define (proof-complete? ps) (not (ctxt? (proof-state-current-proof ps))))
;; Given a Proof-State ps and Ctxt pf, plug pf into the current Ctxt
;; and return the modified Proof-State.
(define (update-current-proof ps pf)
;; TODO: Check for proof completion?
(struct-copy proof-state ps
[current-proof (plug-ctxt (proof-state-current-proof ps) pf)]))
;; Given a Proof-State ps and a Goal, update the current goal and
;; return the modified Proof-State.
(define (update-current-goal ps goal)
(struct-copy proof-state ps
[current-goal goal]))
(begin-for-syntax
;; Takes a symbol or syntax object naming a tactic, and returns the ;; Takes a symbol or syntax object naming a tactic, and returns the
;; procedure for that tactic. ;; procedure for that tactic.
(define (lookup-tactic syn) (define (lookup-tactic syn)
(eval (eval
(if (syntax? syn) (if (syntax? syn)
;; Ensure eval gets the right environment ;; NB: Ensure eval gets the right environment
(datum->syntax (current-theorem) (syntax->datum syn)) (datum->syntax (current-theorem) (syntax->datum syn))
syn)))) syn))))
;;; ====================================================================== (begin-for-syntax
;; The current theorem; used internally to achieve a Coq-like notation
;; A tactic is a Racket function that manipulates the current proof state. ;; for defining theorems and tactic-based proofs.
;; Tactic : Args ... Proof-State -> Proof-State (define current-theorem (make-parameter #f)))
;;; Syntax for defining tactics.
;; (define-tactic command-name function)
;; (define-tactic (command-name args ... Proof-State) body)
;; TODO: Srsly?
(define-syntax (define-tactic syn)
(syntax-case syn ()
[(_ (name args ... ps) body ...)
(quasisyntax/loc syn
(define-for-syntax (name args ... ps) body ...))]
[(_ name function)
(quasisyntax/loc syn
(define-for-syntax name function))]))
;; (define-goal-tactic command-name function)
;; (define-goal-tactic (command-name args ... ctx goal-list current-goal) body)
;; Define a theorem, with similar semantics to Coq theorems in that you ;; Define a theorem, with similar semantics to Coq theorems in that you
;; can define the theorem then define the proof next. ;; can define the theorem then define the proof next.
@ -157,29 +191,28 @@
#'(define name prop)])) #'(define name prop)]))
;; Aliases for theorems. ;; Aliases for theorems.
;; TODO: I think there's some define-syntax-rename thing that would ;; (define-syntax define-lemma (make-rename-transformer #'define-theorem))
;; TODO: prevent eta expanding these.
(define-syntax-rule (define-lemma s ...) (define-theorem s ...))
(begin-for-syntax (begin-for-syntax
;; Given a list of tactics and their arguments, run the tactic script ;; Given a list of tactics and their arguments, run the tactic script
;; on the current theorem, returning the proof if the proof is valid. ;; on the current theorem, returning the proof if the proof is valid.
(define (run-tactic-script f* args*) (define (run-tactic-script f* args*)
(unless (current-theorem) (unless (current-theorem)
(raise-syntax-error 'proof (raise-syntax-error
'proof
"You can't use proof without a first using define-theorem")) "You can't use proof without a first using define-theorem"))
(let* ([t (current-theorem)] (let* ([t (current-theorem)]
[pf (proof-state-current-proof [ps ;; Thread proof state through tactic calls, and eval
;; Thread proof state through tactic calls, and eval ;; at compile-time.
;; at compile-time. (for/fold ([ps (new-proof-state t)])
(for/fold ([ps (new-proof-state t)]) ([f f*] [args args*])
([f f*] [args args*]) (apply (lookup-tactic f) (append args (list ps))))]
(apply (lookup-tactic f) (append args (list ps)))))]) [pf (proof-state-proof ps)])
(when (ctxt? pf) (unless (proof-state-proof-complete? ps)
(raise-syntax-error 'qed "Proof contains holes" (pf hole))) (raise-syntax-error 'qed "Proof contains holes" (pf (current-hole-pretty-symbol))))
(unless (type-check/syn? pf t) (unless (type-check/syn? pf t)
(raise-syntax-error 'qed "Invalid proof" pf t)) (raise-syntax-error 'qed "Invalid proof" pf t))
pf))) pf)))
;; The proof macro starts a proof environment. Every proof should begin ;; The proof macro starts a proof environment. Every proof should begin
;; with it. ;; with it.

View File

@ -1,7 +1,7 @@
#lang s-exp "../../redex-curnel.rkt" #lang s-exp "../../redex-curnel.rkt"
(require (require
"base.rkt" "base.rkt"
(prefix-in basic: "basic.rkt") (prefix-in basic: "standard.rkt")
(for-syntax racket/syntax)) (for-syntax racket/syntax))
(provide (provide
@ -94,7 +94,7 @@
(printf "Starting interactive tactic session. Prepared to be sassed:~n") (printf "Starting interactive tactic session. Prepared to be sassed:~n")
(printf "Type (quit) to quit.~n") (printf "Type (quit) to quit.~n")
(let loop ([ps ps] [cmds '()]) (let loop ([ps ps] [cmds '()])
(if (proof-complete? ps) (if (proof-state-proof-complete? ps)
(basic:print ps) (basic:print ps)
(print ps)) (print ps))
(let ([cmd (read-syntax)]) (let ([cmd (read-syntax)])

View File

@ -13,33 +13,28 @@
by-assumption by-assumption
interactive)) interactive))
;;; TACTICS #| TODO:
;;; A tactic is a function from proof state to proof state | Tactics should probably not error on failure. Perhaps Maybe monad, or list monad, or return proof
| state unmodified, or raise exception, or ...
;; TODO: Tactics probably should not error for failure. Perhaps return |#
;; TODO: proof state unmodified, or raise an exception.
(define-tactic (intro name ps) (define-tactic (intro name ps)
;; TODO: Maybe cur-expand current-goal by default ;; TODO: Provide cur-match, which wraps syntax-parse and uses cur-expand.
(syntax-parse (cur-expand (proof-state-current-goal ps)) (syntax-parse (cur-expand (proof-state-current-goal-ref ps))
[(forall (x:id : P:expr) body:expr) [(forall (x:id : P:expr) body:expr)
(update-current-goal (let* ([ps (proof-state-extend-env ps name #'P)]
(update-current-proof [ps (proof-state-current-goal-set ps #'body)]
;; TODO: Should hide syntax-e in push-env [ps (proof-state-fill-proof-hole ps (lambda (x) #`(lambda (#,name : P) #,x)))])
(push-env ps (syntax-e name) #'P) ps)]
(lambda (x) #`(λ (#,x : P) #,x)))
#'body)]
[_ (error 'intro "Can only intro when current goal is of the form (∀ (x : P) body)")])) [_ (error 'intro "Can only intro when current goal is of the form (∀ (x : P) body)")]))
(begin-for-syntax
(define (assumption ps type)
(for/first ([(k v) (in-dict (proof-state-env ps))]
#:when (cur-equal? v type))
k)))
(define-tactic (by-assumption ps) (define-tactic (by-assumption ps)
(cond (cond
[(assumption ps (cur-expand (proof-state-current-goal ps))) [(proof-state-env-ref-by-type ps (proof-state-current-goal-ref ps))
=> (curry update-current-proof (update-current-goal ps #f))] =>
(lambda (x)
(let* ([ps (proof-state-fill-proof-hole ps x)]
[ps (proof-state-current-goal-set ps #f)])
ps))]
[else (error 'by-assumption "Cannot find an assumption that matches the goal")])) [else (error 'by-assumption "Cannot find an assumption that matches the goal")]))
;; TODO: requires more support from curnel ;; TODO: requires more support from curnel
@ -49,19 +44,17 @@
;; Do the obvious thing ;; Do the obvious thing
(define-tactic (obvious ps) (define-tactic (obvious ps)
(syntax-parse (cur-expand (proof-state-current-goal ps)) (syntax-parse (cur-expand (proof-state-current-goal-ref ps))
[(forall (x : P) t) [(forall (x : P) t)
(obvious (intro #'x ps))] (obvious (intro #'x ps))]
[t:expr [t:expr
(cond (cond
[(assumption ps #'t) (by-assumption ps)] ;; TODO: This would be cleaner if I could say "try all these things and do whichever works".
[(proof-state-env-ref-by-type ps #'t) (by-assumption ps)]
;[(inductive? ps #'t) (by-constructor ps)] ;[(inductive? ps #'t) (by-constructor ps)]
[else (error 'obvious "This is not all that obvious to me.")])])) [else (error 'obvious "This is not all that obvious to me.")])]))
(define-tactic (restart ps) (define-tactic (restart ps) (new-proof-state (proof-state-theorem ps)))
(struct-copy proof-state ps
[current-goal (proof-state-original-goal ps)]
[current-proof empty-proof]))
(define-tactic (print ps) (print-proof-state ps) ps) (define-tactic (print ps) (print-proof-state ps) ps)
@ -80,6 +73,8 @@
[(quit) [(quit)
(begin (begin
(printf "Your tactic script:~n") (printf "Your tactic script:~n")
;; TODO: Using clever trickery, could problem write a version of interactive that actually
;; modifies the file.
(pretty-print (reverse (map syntax->datum cmds))) (pretty-print (reverse (map syntax->datum cmds)))
(newline) (newline)
ps)] ps)]
@ -127,7 +122,7 @@
(proof (obvious)) (proof (obvious))
;; TODO: Fix this unit test so it doesn't require interaction ;; TODO: Fix this unit test so it doesn't require interaction
(define-theorem meow4 (forall (x : bool) bool)) (define-theorem meow4 (forall (x : bool) bool))
#;(proof (proof
(interactive)) (interactive))
;; TODO: Add check-cur-equal? for unit testing? ;; TODO: Add check-cur-equal? for unit testing?
#;(check-pred (curry cur-equal? '(lambda (x : bool) x))) #;(check-pred (curry cur-equal? '(lambda (x : bool) x)))