This PR improves metavariable pretty printing and their hovers in the InfoView. The hovers in the InfoView now include information about specific metavariables — it includes information such as the kind of the metavariable, whether it is a blocked delayed assignment and which metavariables it is blocked on, and the differences in what variables exist the metavariable's local context. Additionally, named metavariables now pretty print with tombstones if they are inaccessible. Delayed assignment pretty printing now more reliably follows chains of assignments to find the pending metavariable. **Example hovers** Hovering over a natural metavariable: > `?m.7 : Sort ?u.14` > A metavariable representing an expression that should be solved for by unification during the elaboration process. They are created during elaboration as placeholders for implicit arguments and by `_` placeholder syntax. > > Variables absent from this metavariable's local context: `a`, `b`, `y` Hovering over `?baz`, a tactic goal: > `?baz : ∀ (a : ?m.7) (a : ?m.8), True` > A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax. > > Variables absent from this metavariable's local context: `a`, `b`, `y` Hovering over `?refine_1`, which appears under a binder: > `?m.4 x : Nat` > A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax. > > This metavariable appears here via a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context. > > Additional variable in this metavariable's local context: `x` Hovering over `?m.4`: > `?m.4 : Nat → Nat` > Part of the encoding of the *delayed assignment* mechanism. Represents the metavariable `?refine_1`, which has additional local context variables. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context. > > Additional variable in this metavariable's local context: `x` The middle paragraph for `?refine_1` when it has been partially assigned: > This metavariable has been assigned, but it is a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context. Substitution is awaiting assignment of the following metavariable: `?foo`
170 lines
7.4 KiB
Text
170 lines
7.4 KiB
Text
import Lean
|
|
/-!
|
|
# Synthetic tests of the dynamic docstrings for metavariables
|
|
-/
|
|
|
|
set_option linter.unusedVariables false
|
|
|
|
open Lean Elab Tactic
|
|
|
|
elab "#delab_docstrings " t:term : command => Command.runTermElabM fun _ => do
|
|
let e ← Term.elabTerm t none
|
|
Term.synthesizeSyntheticMVars (postpone := .partial)
|
|
logInfo m!"Expression: {e}"
|
|
let { infos, .. } ← Meta.ppExprWithInfos e
|
|
let infos := Array.qsort infos.toArray (lt := fun (p, _) (p', _) => p < p')
|
|
for (p, info) in infos do
|
|
if let .ofDelabTermInfo ti := info then
|
|
if let some doc ← ti.docString? (← Meta.getPPContext) then
|
|
let header ← Meta.withLCtx ti.lctx {} do
|
|
addMessageContext m!"{ti.expr}"
|
|
logInfo m!"{p}. {header}\n{doc}"
|
|
|
|
/-!
|
|
Natural metavariable.
|
|
-/
|
|
/--
|
|
info: Expression: ?m.2
|
|
---
|
|
info: 1. ?m.2
|
|
A metavariable representing an expression that should be solved for by unification during the elaboration process. They are created during elaboration as placeholders for implicit arguments and by `_` placeholder syntax.
|
|
-/
|
|
#guard_msgs in #delab_docstrings _
|
|
|
|
/-!
|
|
Synthetic opaque metavariable.
|
|
-/
|
|
/--
|
|
info: Expression: ?m.2
|
|
---
|
|
info: 1. ?m.2
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
-/
|
|
#guard_msgs in #delab_docstrings ?_
|
|
|
|
/-!
|
|
A synthetic metavariable and a natural metavariable.
|
|
-/
|
|
/--
|
|
info: Expression: @default ?m.1 ?m.2
|
|
---
|
|
info: 5. ?m.2
|
|
A metavariable representing a typeclass instance whose synthesis is still pending. They can be solved for by unification during the elaboration process, but the inferred expression and the synthesized instance must be definitionally equal.
|
|
---
|
|
info: 17. ?m.1
|
|
A metavariable representing an expression that should be solved for by unification during the elaboration process. They are created during elaboration as placeholders for implicit arguments and by `_` placeholder syntax.
|
|
-/
|
|
#guard_msgs in set_option pp.explicit true in #delab_docstrings default
|
|
|
|
/-!
|
|
A delayed assignment pointing at a synthetic opaque metavariable.
|
|
-/
|
|
/--
|
|
info: Expression: fun n => ?foo
|
|
---
|
|
info: 5. ?foo
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
This metavariable appears here via a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context.
|
|
|
|
Additional variable in this metavariable's local context: `n`
|
|
-/
|
|
#guard_msgs in #delab_docstrings fun (n : Nat) => ?foo
|
|
|
|
/-!
|
|
An assigned synthetic opaque metavariable that can't be instantiated due to a metavariable.
|
|
-/
|
|
/--
|
|
info: Expression: let f := fun b => ?_;
|
|
f true
|
|
---
|
|
info: 21. ?_
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
This metavariable has been assigned, but it appears here via a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context. Substitution is awaiting assignment of the following metavariable: `?foo`
|
|
|
|
Additional variable in this metavariable's local context: `b`
|
|
-/
|
|
#guard_msgs in
|
|
set_option pp.mvars.anonymous false in
|
|
#delab_docstrings
|
|
let f := fun (b : _) => if b then ?foo else true
|
|
f true
|
|
|
|
elab "#delab_docstrings " t:term : tactic => withMainContext do
|
|
let e ← Tactic.elabTerm t none
|
|
logInfo m!"{e}"
|
|
let { infos, .. } ← Meta.ppExprWithInfos e
|
|
let infos := Array.qsort infos.toArray (lt := fun (p, _) (p', _) => p < p')
|
|
for (p, info) in infos do
|
|
if let .ofDelabTermInfo ti := info then
|
|
if let some doc ← ti.docString? (← Meta.getPPContext) then
|
|
let header ← Meta.withLCtx ti.lctx {} do
|
|
addMessageContext m!"{ti.expr}"
|
|
logInfo m!"{p}. {header}\n{doc}"
|
|
|
|
/-!
|
|
Metavariable shadowing. The `case'` tactic creates a new metavariablewith the same name.
|
|
This causes the original one to print as `?foo✝` and to report that the name is unreachable.
|
|
-/
|
|
/--
|
|
info: fun x => ?foo
|
|
---
|
|
info: 5. ?foo
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
This metavariable appears here via a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context.
|
|
|
|
Additional variable in this metavariable's local context: `x`
|
|
---
|
|
info: fun x => ?foo✝
|
|
---
|
|
info: 5. ?foo✝
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
This metavariable has a name but it is unreachable.
|
|
|
|
This metavariable has been assigned, but it appears here via a *delayed assignment*. Substitution is delayed until the metavariable's value contains no metavariables, since all occurrences of the variables from its local context will need to be replaced with expressions that are valid in the current context. Substitution is awaiting assignment of the following metavariable: `?foo`
|
|
|
|
Additional variable in this metavariable's local context: `x`
|
|
---
|
|
info: fun x => 0 + 1
|
|
-/
|
|
#guard_msgs in
|
|
example : True := by
|
|
let f : Nat → Nat := fun x => ?foo
|
|
#delab_docstrings value_of% f
|
|
case' foo => refine ?_ + 1
|
|
all_goals try #delab_docstrings value_of% f
|
|
exact 0
|
|
#delab_docstrings value_of% f
|
|
trivial
|
|
|
|
/-!
|
|
Checking that metavariables report which variables are absent from their local contexts.
|
|
-/
|
|
/--
|
|
info: (?foo1, ?foo2, ?foo3)
|
|
---
|
|
info: 17. ?foo1
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
Variables absent from this metavariable's local context: `x`, `y`, `z`
|
|
---
|
|
info: 21. ?foo3
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
Variable absent from this metavariable's local context: `z`
|
|
---
|
|
info: 81. ?foo2
|
|
A metavariable representing a tactic goal or an expression whose elaboration is still pending. They usually act like constants until they are completely solved for. They can be created using `?_` and `?n` synthetic placeholder syntax.
|
|
|
|
Variables absent from this metavariable's local context: `y`, `z`
|
|
-/
|
|
#guard_msgs in
|
|
example : True := by
|
|
let x : Nat := ?foo1
|
|
let y : Nat := ?foo2
|
|
let z : Nat := ?foo3
|
|
#delab_docstrings (value_of% x, value_of% y, value_of% z)
|
|
all_goals exact default
|