fix: Conditional handling of `.proj`
This commit is contained in:
parent
c1f63af019
commit
5994f0ddf0
|
@ -17,15 +17,18 @@ structure ProjectionApplication where
|
|||
numParams: Nat
|
||||
inner: Expr
|
||||
|
||||
/-- Converts a `.proj` expression to a function application if possible. Not all
|
||||
such expressions are convertible. -/
|
||||
@[export pantograph_expr_proj_to_app]
|
||||
def exprProjToApp (env: Environment) (e: Expr): ProjectionApplication :=
|
||||
def exprProjToApp (env: Environment) (e: Expr): Option ProjectionApplication := do
|
||||
let (typeName, idx, inner) := match e with
|
||||
| .proj typeName idx inner => (typeName, idx, inner)
|
||||
| _ => panic! "Argument must be proj"
|
||||
let _ ← getStructureInfo? env typeName
|
||||
let ctor := getStructureCtor env typeName
|
||||
let fieldName := getStructureFields env typeName |>.get! idx
|
||||
let projector := getProjFnForField? env typeName fieldName |>.get!
|
||||
{
|
||||
return {
|
||||
projector,
|
||||
numParams := ctor.numParams,
|
||||
inner,
|
||||
|
@ -375,12 +378,10 @@ partial def serializeExpressionSexp (expr: Expr) : MetaM String := do
|
|||
-- NOTE: Equivalent to expr itself, but mdata influences the prettyprinter
|
||||
-- It may become necessary to incorporate the metadata.
|
||||
self inner
|
||||
| .proj _ _ _ => do
|
||||
let env ← getEnv
|
||||
let projApp := exprProjToApp env e
|
||||
let autos := String.intercalate " " (List.replicate projApp.numParams "_")
|
||||
let inner ← self projApp.inner
|
||||
pure s!"((:c {projApp.projector}) {autos} {inner})"
|
||||
| .proj typeName idx e => do
|
||||
let typeName' := serializeName typeName (sanitize := false)
|
||||
let e' ← self e
|
||||
pure s!"(:proj {typeName'} {idx} {e'})"
|
||||
-- Elides all unhygenic names
|
||||
binderInfoSexp : Lean.BinderInfo → String
|
||||
| .default => ""
|
||||
|
|
|
@ -77,7 +77,7 @@ def test_sexp_of_expr (env: Environment): IO LSpec.TestSeq := do
|
|||
.default)
|
||||
.implicit)
|
||||
.implicit,
|
||||
"(:lambda p (:sort 0) (:lambda q (:sort 0) (:lambda k ((:c And) 1 0) ((:c And.right) _ _ 0)) :i) :i)"
|
||||
"(:lambda p (:sort 0) (:lambda q (:sort 0) (:lambda k ((:c And) 1 0) (:proj And 1 0)) :i) :i)"
|
||||
),
|
||||
]
|
||||
let termElabM: Elab.TermElabM LSpec.TestSeq := entries.foldlM (λ suites (expr, target) => do
|
||||
|
@ -96,6 +96,18 @@ def test_instance (env: Environment): IO LSpec.TestSeq :=
|
|||
let _expr := (← runTermElabMInMeta <| elabTerm s) |>.toOption |>.get!
|
||||
return LSpec.TestSeq.done
|
||||
|
||||
def test_projection (env: Environment) : IO LSpec.TestSeq:= runTest do
|
||||
let prod := .app (.bvar 1) (.bvar 0)
|
||||
let expr := .proj `Prod 1 prod
|
||||
let .some { projector, numParams, inner }:= exprProjToApp env expr |
|
||||
fail "`Prod should have projection function"
|
||||
checkEq "projector" projector `Prod.snd
|
||||
checkEq "numParams" numParams 2
|
||||
checkTrue "inner" $ inner.equal prod
|
||||
|
||||
let expr := .proj `Exists 1 prod
|
||||
checkTrue "Exists" (exprProjToApp env expr).isNone
|
||||
|
||||
def suite (env: Environment): List (String × IO LSpec.TestSeq) :=
|
||||
[
|
||||
("serializeName", do pure test_serializeName),
|
||||
|
@ -104,6 +116,7 @@ def suite (env: Environment): List (String × IO LSpec.TestSeq) :=
|
|||
("Sexp from elaborated expr", test_sexp_of_elab env),
|
||||
("Sexp from expr", test_sexp_of_expr env),
|
||||
("Instance", test_instance env),
|
||||
("Projection", test_projection env),
|
||||
]
|
||||
|
||||
end Pantograph.Test.Delate
|
||||
|
|
Loading…
Reference in New Issue