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`.
92 lines
2.1 KiB
Text
92 lines
2.1 KiB
Text
import Lean.CoreM
|
|
|
|
open Lean
|
|
|
|
structure MyState :=
|
|
(trace_state : TraceState := {})
|
|
(s : Nat := 0)
|
|
|
|
abbrev M := CoreM
|
|
|
|
def tst1 : M Unit :=
|
|
do trace[module] (m!"hello" ++ MessageData.nest 9 (m!"\n" ++ "world"));
|
|
trace[module.aux] "another message";
|
|
pure ()
|
|
|
|
def tst2 (b : Bool) : M Unit :=
|
|
withTraceNode `module (fun _ => return "message") do
|
|
tst1;
|
|
trace[bughunt] "at test2";
|
|
if b then throwError "error";
|
|
tst1;
|
|
pure ()
|
|
|
|
partial def ack : Nat → Nat → Nat
|
|
| 0, n => n+1
|
|
| m+1, 0 => ack m 1
|
|
| m+1, n+1 => ack m (ack (m+1) n)
|
|
|
|
def slow (b : Bool) : Nat :=
|
|
ack 4 (cond b 0 1)
|
|
|
|
def tst3 (b : Bool) : M Unit :=
|
|
do withTraceNode `module.slow (fun _ => return m!"slow: {slow b}") do {
|
|
tst2 b;
|
|
tst1
|
|
};
|
|
trace[bughunt] "at end of tst3";
|
|
-- Messages are computed lazily. The following message will only be computed
|
|
-- if `trace.slow is active.
|
|
trace[slow] (m!"slow message: " ++ toString (slow b))
|
|
-- This is true even if it is a monad computation:
|
|
trace[slow] (m!"slow message: " ++ (← pure (toString (slow b))))
|
|
|
|
def run (x : M Unit) : M Unit :=
|
|
withReader
|
|
(fun ctx =>
|
|
-- Try commenting/uncommeting the following `setBool`s
|
|
let opts := ctx.options;
|
|
let opts := opts.setBool `trace.module true;
|
|
-- let opts := opts.setBool `trace.module.aux false;
|
|
let opts := opts.setBool `trace.bughunt true;
|
|
-- let opts := opts.setBool `trace.slow true;
|
|
{ ctx with options := opts })
|
|
(tryCatch (tryFinally x printTraces) (fun _ => IO.println "ERROR"))
|
|
|
|
/--
|
|
info: [module] message
|
|
[module] hello
|
|
world
|
|
[bughunt] at test2
|
|
ERROR
|
|
---
|
|
info: [module] message
|
|
[module] hello
|
|
world
|
|
[bughunt] at test2
|
|
-/
|
|
#guard_msgs in
|
|
#eval run (tst3 true)
|
|
|
|
/--
|
|
info: [module] message
|
|
[module] hello
|
|
world
|
|
[bughunt] at test2
|
|
[module] hello world
|
|
[module] hello
|
|
world
|
|
[bughunt] at end of tst3
|
|
---
|
|
info: [module] message
|
|
[module] hello
|
|
world
|
|
[bughunt] at test2
|
|
[module] hello
|
|
world
|
|
[module] hello
|
|
world
|
|
[bughunt] at end of tst3
|
|
-/
|
|
#guard_msgs in
|
|
#eval run (tst3 false)
|