This refactors and improves the `#eval` command, introducing some new
features.
* Now evaluated results can be represented using `ToExpr` and pretty
printing. This means **hoverable output**. If `ToExpr` fails, it then
tries `Repr` and then `ToString`. The `eval.pp` option controls whether
or not to try `ToExpr`.
* There is now **auto-derivation** of `Repr` instances, enabled with the
`pp.derive.repr` option (default to **true**). For example:
```lean
inductive Baz
| a | b
#eval Baz.a
-- Baz.a
```
It simply does `deriving instance Repr for Baz` when there's no way to
represent `Baz`. If core Lean gets `ToExpr` derive handlers, they could
be used here as well.
* The option `eval.type` controls whether or not to include the type in
the output. For now the default is false.
* Now things like `#eval do return 2` work. It tries using
`CommandElabM`, `TermElabM`, or `IO` when the monad is unknown.
* Now there is no longer `Lean.Eval` or `Lean.MetaEval`. These each used
to be responsible for both adapting monads and printing results. The
concerns have been split into two. (1) The `MonadEval` class is
responsible for adapting monads for evaluation (it is similar to
`MonadLift`, but instances are allowed to use default data when
initializing state) and (2) finding a way to represent results is
handled separately.
* Error messages about failed instance synthesis are now more precise.
Once it detects that a `MonadEval` class applies, then the error message
will be specific about missing `ToExpr`/`Repr`/`ToString` instances.
* Fixes a bug where `Repr`/`ToString` instances can't be found by
unfolding types "under the monad". For example, this works now:
```lean
def Foo := List Nat
def Foo.mk (l : List Nat) : Foo := l
#eval show Lean.CoreM Foo from do return Foo.mk [1,2,3]
```
* Elaboration errors now abort evaluation. This eliminates some
not-so-relevant error messages.
* Now evaluating a value of type `m Unit` never prints a blank message.
* Fixes bugs where evaluating `MetaM` and `CoreM` wouldn't collect log
messages.
The `run_cmd`, `run_elab`, and `run_meta` commands are now frontends for
`#eval`.
99 lines
1.8 KiB
Text
99 lines
1.8 KiB
Text
def f (v : Nat) : StateRefT Nat IO Nat := do
|
||
IO.println "hello"
|
||
modify fun s => s - v
|
||
get
|
||
|
||
def g : IO Nat :=
|
||
f 5 |>.run' 20
|
||
|
||
/--
|
||
info: hello
|
||
---
|
||
info: 15
|
||
-/
|
||
#guard_msgs in
|
||
#eval (f 5).run' 20
|
||
|
||
/--
|
||
info: hello
|
||
---
|
||
info: 95
|
||
-/
|
||
#guard_msgs in
|
||
#eval (do set 100; f 5 : StateRefT Nat IO Nat).run' 0
|
||
def f2 : ReaderT Nat (StateRefT Nat IO) Nat := do
|
||
let v ← read
|
||
IO.println $ "context " ++ toString v
|
||
modify fun s => s + v
|
||
get
|
||
|
||
/--
|
||
info: context 10
|
||
---
|
||
info: 30
|
||
-/
|
||
#guard_msgs in
|
||
#eval (f2.run 10).run' 20
|
||
|
||
def f3 : StateT String (StateRefT Nat IO) Nat := do
|
||
let s ← get
|
||
let n ← getThe Nat
|
||
set $ s ++ ", " ++ toString n
|
||
let s ← get
|
||
IO.println s
|
||
set (n+1)
|
||
getThe Nat
|
||
|
||
/--
|
||
info: test, 10
|
||
---
|
||
info: 11
|
||
-/
|
||
#guard_msgs in
|
||
#eval (f3.run' "test").run' 10
|
||
|
||
structure Label {β : Type} (v : β) (α : Type) :=
|
||
(val : α)
|
||
|
||
class HasGetAt {β : Type} (v : β) (α : outParam Type) (m : Type → Type) :=
|
||
(getAt : m α)
|
||
|
||
instance monadState.hasGetAt (β : Type) (v : β) (α : Type) (m : Type → Type) [Monad m] [MonadStateOf (Label v α) m] : HasGetAt v α m :=
|
||
{ getAt := do let a ← getThe (Label v α); pure a.val }
|
||
|
||
export HasGetAt (getAt)
|
||
|
||
abbrev M := StateRefT (Label 0 Nat) $ StateRefT (Label 1 Nat) $ StateRefT (Label 2 Nat) IO
|
||
|
||
def f4 : M Nat := do
|
||
let a0 : Nat ← getAt 0
|
||
let a1 ← getAt 1
|
||
let a2 ← getAt 2
|
||
IO.println $ "state0 " ++ toString a0
|
||
IO.println $ "state1 " ++ toString a1
|
||
IO.println $ "state1 " ++ toString a2
|
||
pure (a0 + a1 + a2)
|
||
|
||
/--
|
||
info: state0 10
|
||
state1 20
|
||
state1 30
|
||
---
|
||
info: 60
|
||
-/
|
||
#guard_msgs in
|
||
#eval f4.run' ⟨10⟩ |>.run' ⟨20⟩ |>.run' ⟨30⟩
|
||
|
||
abbrev S (ω : Type) := StateRefT Nat $ StateRefT String $ ST ω
|
||
|
||
def f5 {ω} : S ω Unit := do
|
||
let s ← getThe String
|
||
modify fun n => n + s.length
|
||
pure ()
|
||
|
||
def f5Pure (n : Nat) (s : String) :=
|
||
runST fun _ => f5.run n |>.run s
|
||
|
||
/-- info: (((), 21), "hello world") -/
|
||
#guard_msgs in
|
||
#eval f5Pure 10 "hello world"
|