chore: Version 0.3 #136

Open
aniva wants to merge 523 commits from dev into main
5 changed files with 40 additions and 16 deletions
Showing only changes of commit a374af3a5f - Show all commits

View File

@ -104,14 +104,20 @@ structure InfoWithContext where
info: Elab.Info info: Elab.Info
context?: Option Elab.ContextInfo := .none 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 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) let .some ctx := ctx? | return (false, true)
if expr.isSorry ∧ stx.isOfKind `Lean.Parser.Term.sorry then if expr.isSorry ∧ stx.isOfKind `Lean.Parser.Term.sorry then
if expectedType?.isNone then if expectedType?.isNone then
throw $ .userError "Sorry of indeterminant type is not allowed" throw $ .userError "Sorry of indeterminant type is not allowed"
return (true, false) return (true, false)
unless options.collectTypeErrors do
return (false, true)
let .some expectedType := expectedType? | return (false, true) let .some expectedType := expectedType? | return (false, true)
let typeMatch ← ctx.runMetaM lctx do let typeMatch ← ctx.runMetaM lctx do
let type ← Meta.inferType expr let type ← Meta.inferType expr
@ -130,8 +136,9 @@ private def collectSorrysInTree (t : Elab.InfoTree) : IO (List InfoWithContext)
-- NOTE: Plural deliberately not spelled "sorries" -- NOTE: Plural deliberately not spelled "sorries"
@[export pantograph_frontend_collect_sorrys_m] @[export pantograph_frontend_collect_sorrys_m]
def collectSorrys (step: CompilationStep) : IO (List InfoWithContext) := do def collectSorrys (step: CompilationStep) (options : GoalCollectionOptions := {})
return (← step.trees.mapM collectSorrysInTree).join : IO (List InfoWithContext) := do
return (← step.trees.mapM $ λ tree => collectSorrysInTree tree options).join
structure AnnotatedGoalState where structure AnnotatedGoalState where
state : GoalState state : GoalState

View File

@ -327,11 +327,13 @@ structure FrontendProcess where
-- One of these two must be supplied: Either supply the file name or the content. -- One of these two must be supplied: Either supply the file name or the content.
fileName?: Option String := .none fileName?: Option String := .none
file?: Option String := .none file?: Option String := .none
-- If set to true, collect tactic invocations -- collect tactic invocations
invocations: Bool := false invocations: Bool := false
-- If set to true, collect `sorry`s -- collect `sorry`s
sorrys: Bool := false 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 newConstants: Bool := false
deriving Lean.FromJson deriving Lean.FromJson
structure InvokedTactic where structure InvokedTactic where

View File

@ -265,7 +265,7 @@ def execute (command: Protocol.Command): MainM Lean.Json := do
else else
pure .none pure .none
let sorrys ← if args.sorrys then let sorrys ← if args.sorrys then
Frontend.collectSorrys step Frontend.collectSorrys step (options := { collectTypeErrors := args.typeErrorsAsGoals })
else else
pure [] pure []
let messages ← step.messageStrings let messages ← step.messageStrings

View File

@ -6,11 +6,12 @@ import Test.Common
open Lean Pantograph open Lean Pantograph
namespace Pantograph.Test.Frontend 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 := "<anonymous>" let filename := "<anonymous>"
let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {} let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {}
let m := Frontend.mapCompilationSteps λ step => do 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 li ← m.run context |>.run' state
let goalStates ← li.filterMapM λ (env, sorrys) => withEnv env do let goalStates ← li.filterMapM λ (env, sorrys) => withEnv env do
if sorrys.isEmpty then if sorrys.isEmpty then
@ -181,9 +182,10 @@ def test_capture_type_mismatch : TestT MetaM Unit := do
let input := " let input := "
def mystery (k: Nat) : Nat := true 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}" 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" }, target := { pp? := "Nat" },
vars := #[{ vars := #[{
@ -193,6 +195,16 @@ 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 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)) #[
]
def collectNewConstants (source: String) : MetaM (List (List Name)) := do def collectNewConstants (source: String) : MetaM (List (List Name)) := do
let filename := "<anonymous>" let filename := "<anonymous>"
let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {} let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {}
@ -227,6 +239,7 @@ def suite (env : Environment): List (String × IO LSpec.TestSeq) :=
("sorry_in_coupled", test_sorry_in_coupled), ("sorry_in_coupled", test_sorry_in_coupled),
("environment_capture", test_environment_capture), ("environment_capture", test_environment_capture),
("capture_type_mismatch", test_capture_type_mismatch), ("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_constant", test_collect_one_constant),
("collect_one_theorem", test_collect_one_theorem), ("collect_one_theorem", test_collect_one_theorem),
] ]

View File

@ -171,10 +171,11 @@ def test_frontend_process : Test :=
let goal1 := "p q : Prop\nh : p\n⊢ p q" let goal1 := "p q : Prop\nh : p\n⊢ p q"
step "frontend.process" step "frontend.process"
[ [
("file", .str file), ("file", .str file),
("invocations", .bool true), ("invocations", .bool true),
("sorrys", .bool false), ("sorrys", .bool false),
("newConstants", .bool false), ("typeErrorsAsGoals", .bool false),
("newConstants", .bool false),
] ]
({ ({
units := [{ units := [{
@ -215,6 +216,7 @@ def test_frontend_process_sorry : Test :=
("file", .str file), ("file", .str file),
("invocations", .bool false), ("invocations", .bool false),
("sorrys", .bool true), ("sorrys", .bool true),
("typeErrorsAsGoals", .bool false),
("newConstants", .bool false), ("newConstants", .bool false),
] ]
({ ({