From 48b924fae26f0354625d65eb047c57d034945aa1 Mon Sep 17 00:00:00 2001 From: Leni Aniva Date: Fri, 20 Dec 2024 19:10:38 +0900 Subject: [PATCH 1/3] fix(frontend): Incorrect capture of binder term --- Pantograph/Frontend/Elab.lean | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Pantograph/Frontend/Elab.lean b/Pantograph/Frontend/Elab.lean index 3da5fca..a07efb0 100644 --- a/Pantograph/Frontend/Elab.lean +++ b/Pantograph/Frontend/Elab.lean @@ -106,7 +106,7 @@ structure InfoWithContext where private def collectSorrysInTree (t : Elab.InfoTree) : IO (List InfoWithContext) := do let infos ← t.findAllInfoM none fun i ctx? => match i with - | .ofTermInfo { expectedType?, expr, stx, lctx, .. } => do + | .ofTermInfo { expectedType?, expr, stx, lctx, isBinder := false, .. } => do let .some ctx := ctx? | return (false, true) if expr.isSorry ∧ stx.isOfKind `Lean.Parser.Term.sorry then if expectedType?.isNone then From 5a05e9e8d4d3226e49747e1fa4a4b820ecd46dac Mon Sep 17 00:00:00 2001 From: Leni Aniva Date: Sun, 22 Dec 2024 08:47:38 +0900 Subject: [PATCH 2/3] test: Add binder capturing test --- Test/Frontend.lean | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/Test/Frontend.lean b/Test/Frontend.lean index a3b73ae..2c5809f 100644 --- a/Test/Frontend.lean +++ b/Test/Frontend.lean @@ -193,6 +193,15 @@ def mystery (k: Nat) : Nat := true } ] +def test_capture_type_mismatch_in_binder : TestT MetaM Unit := do + let input := " +example (p: Prop) (h: (∀ (x: Prop), Nat) → p): p := h (λ (y: Nat) => 5) + " + let goalStates ← (collectSorrysFromSource input).run' {} + let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}" + checkEq "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize)) #[ + ] + def collectNewConstants (source: String) : MetaM (List (List Name)) := do let filename := "" let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {} @@ -227,6 +236,7 @@ def suite (env : Environment): List (String × IO LSpec.TestSeq) := ("sorry_in_coupled", test_sorry_in_coupled), ("environment_capture", test_environment_capture), ("capture_type_mismatch", test_capture_type_mismatch), + ("capture_type_mismatch_in_binder", test_capture_type_mismatch_in_binder), ("collect_one_constant", test_collect_one_constant), ("collect_one_theorem", test_collect_one_theorem), ] From 524314721b3213c380d54d461786abbddedf6c55 Mon Sep 17 00:00:00 2001 From: Leni Aniva Date: Tue, 7 Jan 2025 19:40:03 +0900 Subject: [PATCH 3/3] feat: Gate type error collection behind flag --- Pantograph/Frontend/Elab.lean | 13 ++++++++++--- Pantograph/Protocol.lean | 8 +++++--- Repl.lean | 2 +- Test/Frontend.lean | 15 +++++++++------ Test/Integration.lean | 10 ++++++---- 5 files changed, 31 insertions(+), 17 deletions(-) diff --git a/Pantograph/Frontend/Elab.lean b/Pantograph/Frontend/Elab.lean index a07efb0..b5a7274 100644 --- a/Pantograph/Frontend/Elab.lean +++ b/Pantograph/Frontend/Elab.lean @@ -104,7 +104,11 @@ structure InfoWithContext where info: Elab.Info context?: Option Elab.ContextInfo := .none -private def collectSorrysInTree (t : Elab.InfoTree) : IO (List InfoWithContext) := do +structure GoalCollectionOptions where + collectTypeErrors : Bool := false + +private def collectSorrysInTree (t : Elab.InfoTree) (options : GoalCollectionOptions := {}) + : IO (List InfoWithContext) := do let infos ← t.findAllInfoM none fun i ctx? => match i with | .ofTermInfo { expectedType?, expr, stx, lctx, isBinder := false, .. } => do let .some ctx := ctx? | return (false, true) @@ -112,6 +116,8 @@ private def collectSorrysInTree (t : Elab.InfoTree) : IO (List InfoWithContext) if expectedType?.isNone then throw $ .userError "Sorry of indeterminant type is not allowed" return (true, false) + unless options.collectTypeErrors do + return (false, true) let .some expectedType := expectedType? | return (false, true) let typeMatch ← ctx.runMetaM lctx do let type ← Meta.inferType expr @@ -130,8 +136,9 @@ private def collectSorrysInTree (t : Elab.InfoTree) : IO (List InfoWithContext) -- NOTE: Plural deliberately not spelled "sorries" @[export pantograph_frontend_collect_sorrys_m] -def collectSorrys (step: CompilationStep) : IO (List InfoWithContext) := do - return (← step.trees.mapM collectSorrysInTree).join +def collectSorrys (step: CompilationStep) (options : GoalCollectionOptions := {}) + : IO (List InfoWithContext) := do + return (← step.trees.mapM $ λ tree => collectSorrysInTree tree options).join structure AnnotatedGoalState where state : GoalState diff --git a/Pantograph/Protocol.lean b/Pantograph/Protocol.lean index 90ac149..47ed879 100644 --- a/Pantograph/Protocol.lean +++ b/Pantograph/Protocol.lean @@ -319,11 +319,13 @@ structure FrontendProcess where -- One of these two must be supplied: Either supply the file name or the content. fileName?: Option String := .none file?: Option String := .none - -- If set to true, collect tactic invocations + -- collect tactic invocations invocations: Bool := false - -- If set to true, collect `sorry`s + -- collect `sorry`s sorrys: Bool := false - -- If set to true, extract new constants + -- collect type errors + typeErrorsAsGoals: Bool := false + -- list new constants from each compilation step newConstants: Bool := false deriving Lean.FromJson structure InvokedTactic where diff --git a/Repl.lean b/Repl.lean index eb02f59..1fd8266 100644 --- a/Repl.lean +++ b/Repl.lean @@ -265,7 +265,7 @@ def execute (command: Protocol.Command): MainM Lean.Json := do else pure .none let sorrys ← if args.sorrys then - Frontend.collectSorrys step + Frontend.collectSorrys step (options := { collectTypeErrors := args.typeErrorsAsGoals }) else pure [] let messages ← step.messageStrings diff --git a/Test/Frontend.lean b/Test/Frontend.lean index 2c5809f..571aa9f 100644 --- a/Test/Frontend.lean +++ b/Test/Frontend.lean @@ -6,11 +6,12 @@ import Test.Common open Lean Pantograph namespace Pantograph.Test.Frontend -def collectSorrysFromSource (source: String) : MetaM (List GoalState) := do +def collectSorrysFromSource (source: String) (options : Frontend.GoalCollectionOptions := {}) + : MetaM (List GoalState) := do let filename := "" let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {} let m := Frontend.mapCompilationSteps λ step => do - return (step.before, ← Frontend.collectSorrys step) + return (step.before, ← Frontend.collectSorrys step options) let li ← m.run context |>.run' state let goalStates ← li.filterMapM λ (env, sorrys) => withEnv env do if sorrys.isEmpty then @@ -181,9 +182,10 @@ def test_capture_type_mismatch : TestT MetaM Unit := do let input := " def mystery (k: Nat) : Nat := true " - let goalStates ← (collectSorrysFromSource input).run' {} + let options := { collectTypeErrors := true } + let goalStates ← (collectSorrysFromSource input options).run' {} let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}" - checkEq "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize)) #[ + checkEq "goals" ((← goalState.serializeGoals).map (·.devolatilize)) #[ { target := { pp? := "Nat" }, vars := #[{ @@ -197,7 +199,8 @@ def test_capture_type_mismatch_in_binder : TestT MetaM Unit := do let input := " example (p: Prop) (h: (∀ (x: Prop), Nat) → p): p := h (λ (y: Nat) => 5) " - let goalStates ← (collectSorrysFromSource input).run' {} + let options := { collectTypeErrors := true } + let goalStates ← (collectSorrysFromSource input options).run' {} let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}" checkEq "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize)) #[ ] @@ -236,7 +239,7 @@ def suite (env : Environment): List (String × IO LSpec.TestSeq) := ("sorry_in_coupled", test_sorry_in_coupled), ("environment_capture", test_environment_capture), ("capture_type_mismatch", test_capture_type_mismatch), - ("capture_type_mismatch_in_binder", test_capture_type_mismatch_in_binder), + --("capture_type_mismatch_in_binder", test_capture_type_mismatch_in_binder), ("collect_one_constant", test_collect_one_constant), ("collect_one_theorem", test_collect_one_theorem), ] diff --git a/Test/Integration.lean b/Test/Integration.lean index 77968f0..1a2e97c 100644 --- a/Test/Integration.lean +++ b/Test/Integration.lean @@ -171,10 +171,11 @@ def test_frontend_process : Test := let goal1 := "p q : Prop\nh : p\n⊢ p ∨ q" step "frontend.process" [ - ("file", .str file), - ("invocations", .bool true), - ("sorrys", .bool false), - ("newConstants", .bool false), + ("file", .str file), + ("invocations", .bool true), + ("sorrys", .bool false), + ("typeErrorsAsGoals", .bool false), + ("newConstants", .bool false), ] ({ units := [{ @@ -215,6 +216,7 @@ def test_frontend_process_sorry : Test := ("file", .str file), ("invocations", .bool false), ("sorrys", .bool true), + ("typeErrorsAsGoals", .bool false), ("newConstants", .bool false), ] ({