2024-10-02 22:22:20 -07:00
|
|
|
|
import LSpec
|
|
|
|
|
import Pantograph
|
|
|
|
|
import Repl
|
|
|
|
|
import Test.Common
|
|
|
|
|
|
|
|
|
|
open Lean Pantograph
|
|
|
|
|
namespace Pantograph.Test.Frontend
|
|
|
|
|
|
2025-01-07 02:40:03 -08:00
|
|
|
|
def collectSorrysFromSource (source: String) (options : Frontend.GoalCollectionOptions := {})
|
|
|
|
|
: MetaM (List GoalState) := do
|
2024-10-02 22:22:20 -07:00
|
|
|
|
let filename := "<anonymous>"
|
|
|
|
|
let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {}
|
|
|
|
|
let m := Frontend.mapCompilationSteps λ step => do
|
2025-01-07 02:40:03 -08:00
|
|
|
|
return (step.before, ← Frontend.collectSorrys step options)
|
2024-10-02 22:22:20 -07:00
|
|
|
|
let li ← m.run context |>.run' state
|
2024-10-12 16:52:36 -07:00
|
|
|
|
let goalStates ← li.filterMapM λ (env, sorrys) => withEnv env do
|
2024-10-02 22:22:20 -07:00
|
|
|
|
if sorrys.isEmpty then
|
|
|
|
|
return .none
|
2024-12-09 20:38:27 -08:00
|
|
|
|
let { state, .. } ← Frontend.sorrysToGoalState sorrys
|
|
|
|
|
return .some state
|
2024-10-02 22:22:20 -07:00
|
|
|
|
return goalStates
|
|
|
|
|
|
2024-10-03 01:29:46 -07:00
|
|
|
|
def test_multiple_sorrys_in_proof : TestT MetaM Unit := do
|
2024-10-02 22:22:20 -07:00
|
|
|
|
let sketch := "
|
|
|
|
|
theorem plus_n_Sm_proved_formal_sketch : ∀ n m : Nat, n + (m + 1) = (n + m) + 1 := by
|
|
|
|
|
have h_nat_add_succ: ∀ n m : Nat, n = m := sorry
|
|
|
|
|
sorry
|
|
|
|
|
"
|
|
|
|
|
let goalStates ← (collectSorrysFromSource sketch).run' {}
|
2024-10-03 11:35:54 -07:00
|
|
|
|
let [goalState] := goalStates | panic! "Incorrect number of states"
|
|
|
|
|
addTest $ LSpec.check "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize) = #[
|
2024-10-02 22:22:20 -07:00
|
|
|
|
{
|
|
|
|
|
target := { pp? := "∀ (n m : Nat), n = m" },
|
|
|
|
|
vars := #[
|
|
|
|
|
]
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
target := { pp? := "∀ (n m : Nat), n + (m + 1) = n + m + 1" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "h_nat_add_succ",
|
|
|
|
|
type? := .some { pp? := "∀ (n m : Nat), n = m" },
|
|
|
|
|
}],
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
2024-10-03 01:29:46 -07:00
|
|
|
|
def test_sorry_in_middle: TestT MetaM Unit := do
|
|
|
|
|
let sketch := "
|
|
|
|
|
example : ∀ (n m: Nat), n + m = m + n := by
|
|
|
|
|
intros n m
|
|
|
|
|
sorry
|
|
|
|
|
"
|
|
|
|
|
let goalStates ← (collectSorrysFromSource sketch).run' {}
|
2024-10-03 11:35:54 -07:00
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
|
|
|
|
addTest $ LSpec.check "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize) = #[
|
2024-10-03 01:29:46 -07:00
|
|
|
|
{
|
|
|
|
|
target := { pp? := "n + m = m + n" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "n",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "m",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
def test_sorry_in_induction : TestT MetaM Unit := do
|
|
|
|
|
let sketch := "
|
|
|
|
|
example : ∀ (n m: Nat), n + m = m + n := by
|
|
|
|
|
intros n m
|
|
|
|
|
induction n with
|
|
|
|
|
| zero =>
|
|
|
|
|
have h1 : 0 + m = m := sorry
|
|
|
|
|
sorry
|
|
|
|
|
| succ n ih =>
|
|
|
|
|
have h2 : n + m = m := sorry
|
|
|
|
|
sorry
|
|
|
|
|
"
|
|
|
|
|
let goalStates ← (collectSorrysFromSource sketch).run' {}
|
2024-10-03 11:35:54 -07:00
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
|
|
|
|
addTest $ LSpec.check "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize) = #[
|
2024-10-03 01:29:46 -07:00
|
|
|
|
{
|
|
|
|
|
target := { pp? := "0 + m = m" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "m",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}]
|
|
|
|
|
},
|
|
|
|
|
{
|
2024-10-03 11:35:54 -07:00
|
|
|
|
userName? := .some "zero",
|
2024-10-03 01:29:46 -07:00
|
|
|
|
target := { pp? := "0 + m = m + 0" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "m",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "h1",
|
|
|
|
|
type? := .some { pp? := "0 + m = m" },
|
|
|
|
|
}]
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
target := { pp? := "n + m = m" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "m",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "n",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "ih",
|
|
|
|
|
type? := .some { pp? := "n + m = m + n" },
|
|
|
|
|
}]
|
|
|
|
|
},
|
|
|
|
|
{
|
2024-10-03 11:35:54 -07:00
|
|
|
|
userName? := .some "succ",
|
2024-10-03 01:29:46 -07:00
|
|
|
|
target := { pp? := "n + 1 + m = m + (n + 1)" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "m",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "n",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "ih",
|
|
|
|
|
type? := .some { pp? := "n + m = m + n" },
|
|
|
|
|
}, {
|
|
|
|
|
userName := "h2",
|
|
|
|
|
type? := .some { pp? := "n + m = m" },
|
|
|
|
|
}]
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
2024-10-03 11:35:54 -07:00
|
|
|
|
def test_sorry_in_coupled: TestT MetaM Unit := do
|
|
|
|
|
let sketch := "
|
|
|
|
|
example : ∀ (y: Nat), ∃ (x: Nat), y + 1 = x := by
|
|
|
|
|
intro y
|
|
|
|
|
apply Exists.intro
|
|
|
|
|
case h => sorry
|
|
|
|
|
case w => sorry
|
|
|
|
|
"
|
|
|
|
|
let goalStates ← (collectSorrysFromSource sketch).run' {}
|
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
|
|
|
|
addTest $ LSpec.check "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize) = #[
|
|
|
|
|
{
|
|
|
|
|
target := { pp? := "y + 1 = ?w" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "y",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
userName? := .some "w",
|
|
|
|
|
target := { pp? := "Nat" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "y",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}
|
|
|
|
|
],
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
2024-10-12 16:52:36 -07:00
|
|
|
|
def test_environment_capture: TestT MetaM Unit := do
|
|
|
|
|
let sketch := "
|
|
|
|
|
def mystery (n: Nat) := n + 1
|
|
|
|
|
|
|
|
|
|
example (n: Nat) : mystery n + 1 = n + 2 := sorry
|
|
|
|
|
"
|
|
|
|
|
let goalStates ← (collectSorrysFromSource sketch).run' {}
|
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
|
|
|
|
addTest $ LSpec.check "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize) = #[
|
|
|
|
|
{
|
|
|
|
|
target := { pp? := "mystery n + 1 = n + 2" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "n",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}],
|
|
|
|
|
}
|
|
|
|
|
])
|
|
|
|
|
|
2024-12-08 23:21:36 -08:00
|
|
|
|
def test_capture_type_mismatch : TestT MetaM Unit := do
|
|
|
|
|
let input := "
|
2024-12-09 20:15:53 -08:00
|
|
|
|
def mystery (k: Nat) : Nat := true
|
2024-12-08 23:21:36 -08:00
|
|
|
|
"
|
2025-01-07 02:40:03 -08:00
|
|
|
|
let options := { collectTypeErrors := true }
|
|
|
|
|
let goalStates ← (collectSorrysFromSource input options).run' {}
|
2024-12-08 23:21:36 -08:00
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
2025-01-07 02:40:03 -08:00
|
|
|
|
checkEq "goals" ((← goalState.serializeGoals).map (·.devolatilize)) #[
|
2024-12-09 20:15:53 -08:00
|
|
|
|
{
|
|
|
|
|
target := { pp? := "Nat" },
|
|
|
|
|
vars := #[{
|
|
|
|
|
userName := "k",
|
|
|
|
|
type? := .some { pp? := "Nat" },
|
|
|
|
|
}],
|
|
|
|
|
}
|
|
|
|
|
]
|
2024-12-08 23:21:36 -08:00
|
|
|
|
|
2024-12-21 15:47:38 -08:00
|
|
|
|
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)
|
|
|
|
|
"
|
2025-01-07 02:40:03 -08:00
|
|
|
|
let options := { collectTypeErrors := true }
|
|
|
|
|
let goalStates ← (collectSorrysFromSource input options).run' {}
|
2024-12-21 15:47:38 -08:00
|
|
|
|
let [goalState] := goalStates | panic! s!"Incorrect number of states: {goalStates.length}"
|
|
|
|
|
checkEq "goals" ((← goalState.serializeGoals (options := {})).map (·.devolatilize)) #[
|
|
|
|
|
]
|
|
|
|
|
|
2024-12-09 19:40:00 -08:00
|
|
|
|
def collectNewConstants (source: String) : MetaM (List (List Name)) := do
|
|
|
|
|
let filename := "<anonymous>"
|
|
|
|
|
let (context, state) ← do Frontend.createContextStateFromFile source filename (← getEnv) {}
|
|
|
|
|
let m := Frontend.mapCompilationSteps λ step => do
|
|
|
|
|
Frontend.collectNewDefinedConstants step
|
|
|
|
|
m.run context |>.run' state
|
|
|
|
|
|
|
|
|
|
def test_collect_one_constant : TestT MetaM Unit := do
|
|
|
|
|
let input := "
|
|
|
|
|
def mystery : Nat := 123
|
|
|
|
|
"
|
|
|
|
|
let names ← collectNewConstants input
|
|
|
|
|
checkEq "constants" names [[`mystery]]
|
|
|
|
|
def test_collect_one_theorem : TestT MetaM Unit := do
|
|
|
|
|
let input := "
|
|
|
|
|
theorem mystery [SizeOf α] (as : List α) (i : Fin as.length) : sizeOf (as.get i) < sizeOf as := by
|
|
|
|
|
match as, i with
|
|
|
|
|
| a::as, ⟨0, _⟩ => simp_arith [get]
|
|
|
|
|
| a::as, ⟨i+1, h⟩ =>
|
|
|
|
|
have ih := sizeOf_get as ⟨i, Nat.le_of_succ_le_succ h⟩
|
|
|
|
|
apply Nat.lt_trans ih
|
|
|
|
|
simp_arith
|
|
|
|
|
"
|
|
|
|
|
let names ← collectNewConstants input
|
|
|
|
|
checkEq "constants" names [[`mystery]]
|
2024-10-02 22:22:20 -07:00
|
|
|
|
|
|
|
|
|
def suite (env : Environment): List (String × IO LSpec.TestSeq) :=
|
|
|
|
|
let tests := [
|
2024-10-03 01:29:46 -07:00
|
|
|
|
("multiple_sorrys_in_proof", test_multiple_sorrys_in_proof),
|
|
|
|
|
("sorry_in_middle", test_sorry_in_middle),
|
|
|
|
|
("sorry_in_induction", test_sorry_in_induction),
|
2024-10-03 11:35:54 -07:00
|
|
|
|
("sorry_in_coupled", test_sorry_in_coupled),
|
2024-10-12 16:52:36 -07:00
|
|
|
|
("environment_capture", test_environment_capture),
|
2024-12-08 23:21:36 -08:00
|
|
|
|
("capture_type_mismatch", test_capture_type_mismatch),
|
2025-01-07 02:40:03 -08:00
|
|
|
|
--("capture_type_mismatch_in_binder", test_capture_type_mismatch_in_binder),
|
2024-12-09 19:40:00 -08:00
|
|
|
|
("collect_one_constant", test_collect_one_constant),
|
|
|
|
|
("collect_one_theorem", test_collect_one_theorem),
|
2024-10-02 22:22:20 -07:00
|
|
|
|
]
|
|
|
|
|
tests.map (fun (name, test) => (name, runMetaMSeq env $ runTest test))
|
|
|
|
|
|
|
|
|
|
end Pantograph.Test.Frontend
|