chore: Version 0.3 #136
|
@ -52,12 +52,12 @@ def describe (_: Protocol.EnvDescribe): CoreM Protocol.EnvDescribeResult := do
|
||||||
imports := env.header.imports.map toString,
|
imports := env.header.imports.map toString,
|
||||||
modules := env.header.moduleNames.map (·.toString),
|
modules := env.header.moduleNames.map (·.toString),
|
||||||
}
|
}
|
||||||
def moduleRead (args: Protocol.EnvModuleRead): CoreM (Protocol.CR Protocol.EnvModuleReadResult) := do
|
def moduleRead (args: Protocol.EnvModuleRead): CoreM Protocol.EnvModuleReadResult := do
|
||||||
let env ← Lean.MonadEnv.getEnv
|
let env ← Lean.MonadEnv.getEnv
|
||||||
let .some i := env.header.moduleNames.findIdx? (· == args.module.toName) |
|
let .some i := env.header.moduleNames.findIdx? (· == args.module.toName) |
|
||||||
return .error $ Protocol.errorIndex s!"Module not found {args.module}"
|
throwError s!"Module not found {args.module}"
|
||||||
let data := env.header.moduleData[i]!
|
let data := env.header.moduleData[i]!
|
||||||
return .ok {
|
return {
|
||||||
imports := data.imports.map toString,
|
imports := data.imports.map toString,
|
||||||
constNames := data.constNames.map (·.toString),
|
constNames := data.constNames.map (·.toString),
|
||||||
extraConstNames := data.extraConstNames.map (·.toString),
|
extraConstNames := data.extraConstNames.map (·.toString),
|
||||||
|
@ -69,13 +69,11 @@ def catalog (_: Protocol.EnvCatalog): CoreM Protocol.EnvCatalogResult := do
|
||||||
| .some x => acc.push x
|
| .some x => acc.push x
|
||||||
| .none => acc)
|
| .none => acc)
|
||||||
return { symbols := names }
|
return { symbols := names }
|
||||||
def inspect (args: Protocol.EnvInspect) (options: @&Protocol.Options): CoreM (Protocol.CR Protocol.EnvInspectResult) := do
|
def inspect (args: Protocol.EnvInspect) (options: @&Protocol.Options): Protocol.FallibleT CoreM Protocol.EnvInspectResult := do
|
||||||
let env ← Lean.MonadEnv.getEnv
|
let env ← Lean.MonadEnv.getEnv
|
||||||
let name := args.name.toName
|
let name := args.name.toName
|
||||||
let info? := env.find? name
|
let info? := env.find? name
|
||||||
let info ← match info? with
|
let .some info := info? | Protocol.throw $ Protocol.errorIndex s!"Symbol not found {args.name}"
|
||||||
| none => return .error $ Protocol.errorIndex s!"Symbol not found {args.name}"
|
|
||||||
| some info => pure info
|
|
||||||
let module? := env.getModuleIdxFor? name >>=
|
let module? := env.getModuleIdxFor? name >>=
|
||||||
(λ idx => env.allImportedModuleNames.get? idx.toNat)
|
(λ idx => env.allImportedModuleNames.get? idx.toNat)
|
||||||
let value? := match args.value?, info with
|
let value? := match args.value?, info with
|
||||||
|
@ -145,17 +143,19 @@ def inspect (args: Protocol.EnvInspect) (options: @&Protocol.Options): CoreM (Pr
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
.pure result
|
.pure result
|
||||||
return .ok result
|
return result
|
||||||
def addDecl (args: Protocol.EnvAdd): CoreM (Protocol.CR Protocol.EnvAddResult) := do
|
@[export pantograph_env_add_m]
|
||||||
|
def addDecl (name: String) (levels: Array String := #[]) (type: String) (value: String) (isTheorem: Bool)
|
||||||
|
: Protocol.FallibleT CoreM Protocol.EnvAddResult := do
|
||||||
let env ← Lean.MonadEnv.getEnv
|
let env ← Lean.MonadEnv.getEnv
|
||||||
let tvM: Elab.TermElabM (Except String (Expr × Expr)) := do
|
let tvM: Elab.TermElabM (Except String (Expr × Expr)) := do
|
||||||
let type ← match parseTerm env args.type with
|
let type ← match parseTerm env type with
|
||||||
| .ok syn => do
|
| .ok syn => do
|
||||||
match ← elabTerm syn with
|
match ← elabTerm syn with
|
||||||
| .error e => return .error e
|
| .error e => return .error e
|
||||||
| .ok expr => pure expr
|
| .ok expr => pure expr
|
||||||
| .error e => return .error e
|
| .error e => return .error e
|
||||||
let value ← match parseTerm env args.value with
|
let value ← match parseTerm env value with
|
||||||
| .ok syn => do
|
| .ok syn => do
|
||||||
try
|
try
|
||||||
let expr ← Elab.Term.elabTerm (stx := syn) (expectedType? := .some type)
|
let expr ← Elab.Term.elabTerm (stx := syn) (expectedType? := .some type)
|
||||||
|
@ -167,16 +167,25 @@ def addDecl (args: Protocol.EnvAdd): CoreM (Protocol.CR Protocol.EnvAddResult) :
|
||||||
pure $ .ok (type, value)
|
pure $ .ok (type, value)
|
||||||
let (type, value) ← match ← tvM.run' (ctx := {}) |>.run' with
|
let (type, value) ← match ← tvM.run' (ctx := {}) |>.run' with
|
||||||
| .ok t => pure t
|
| .ok t => pure t
|
||||||
| .error e => return .error $ Protocol.errorExpr e
|
| .error e => Protocol.throw $ Protocol.errorExpr e
|
||||||
let decl := Lean.Declaration.defnDecl <| Lean.mkDefinitionValEx
|
let levelParams := levels.toList.map (·.toName)
|
||||||
(name := args.name.toName)
|
let decl := if isTheorem then
|
||||||
(levelParams := [])
|
Lean.Declaration.thmDecl <| Lean.mkTheoremValEx
|
||||||
(type := type)
|
(name := name.toName)
|
||||||
(value := value)
|
(levelParams := levelParams)
|
||||||
(hints := Lean.mkReducibilityHintsRegularEx 1)
|
(type := type )
|
||||||
(safety := Lean.DefinitionSafety.safe)
|
(value := value)
|
||||||
(all := [])
|
(all := [])
|
||||||
|
else
|
||||||
|
Lean.Declaration.defnDecl <| Lean.mkDefinitionValEx
|
||||||
|
(name := name.toName)
|
||||||
|
(levelParams := levelParams)
|
||||||
|
(type := type)
|
||||||
|
(value := value)
|
||||||
|
(hints := Lean.mkReducibilityHintsRegularEx 1)
|
||||||
|
(safety := Lean.DefinitionSafety.safe)
|
||||||
|
(all := [])
|
||||||
Lean.addDecl decl
|
Lean.addDecl decl
|
||||||
return .ok {}
|
return {}
|
||||||
|
|
||||||
end Pantograph.Environment
|
end Pantograph.Environment
|
||||||
|
|
|
@ -5,8 +5,6 @@ import Pantograph.Delate
|
||||||
import Pantograph.Version
|
import Pantograph.Version
|
||||||
|
|
||||||
import Lean
|
import Lean
|
||||||
import Std.Time
|
|
||||||
import Std.Internal.Async
|
|
||||||
|
|
||||||
namespace Lean
|
namespace Lean
|
||||||
|
|
||||||
|
@ -78,59 +76,53 @@ def createCoreState (imports: Array String): IO Core.State := do
|
||||||
(trustLevel := 1)
|
(trustLevel := 1)
|
||||||
return { env := env }
|
return { env := env }
|
||||||
|
|
||||||
@[export pantograph_env_add_m]
|
|
||||||
def envAdd (name: String) (type: String) (value: String) (isTheorem: Bool):
|
|
||||||
CoreM (Protocol.CR Protocol.EnvAddResult) :=
|
|
||||||
Environment.addDecl { name, type, value, isTheorem }
|
|
||||||
|
|
||||||
@[export pantograph_parse_elab_type_m]
|
@[export pantograph_parse_elab_type_m]
|
||||||
def parseElabType (type: String): Elab.TermElabM (Protocol.CR Expr) := do
|
def parseElabType (type: String): Protocol.FallibleT Elab.TermElabM Expr := do
|
||||||
let env ← MonadEnv.getEnv
|
let env ← MonadEnv.getEnv
|
||||||
let syn ← match parseTerm env type with
|
let syn ← match parseTerm env type with
|
||||||
| .error str => return .error $ errorI "parsing" str
|
| .error str => Protocol.throw $ errorI "parsing" str
|
||||||
| .ok syn => pure syn
|
| .ok syn => pure syn
|
||||||
match ← elabType syn with
|
match ← elabType syn with
|
||||||
| .error str => return .error $ errorI "elab" str
|
| .error str => Protocol.throw $ errorI "elab" str
|
||||||
| .ok expr => return .ok (← instantiateMVars expr)
|
| .ok expr => return (← instantiateMVars expr)
|
||||||
|
|
||||||
/-- This must be a TermElabM since the parsed expr contains extra information -/
|
/-- This must be a TermElabM since the parsed expr contains extra information -/
|
||||||
@[export pantograph_parse_elab_expr_m]
|
@[export pantograph_parse_elab_expr_m]
|
||||||
def parseElabExpr (expr: String) (expectedType?: Option String := .none): Elab.TermElabM (Protocol.CR Expr) := do
|
def parseElabExpr (expr: String) (expectedType?: Option String := .none): Protocol.FallibleT Elab.TermElabM Expr := do
|
||||||
let env ← MonadEnv.getEnv
|
let env ← MonadEnv.getEnv
|
||||||
let expectedType? ← match ← expectedType?.mapM parseElabType with
|
let expectedType? ← expectedType?.mapM parseElabType
|
||||||
| .none => pure $ .none
|
|
||||||
| .some (.ok t) => pure $ .some t
|
|
||||||
| .some (.error e) => return .error e
|
|
||||||
let syn ← match parseTerm env expr with
|
let syn ← match parseTerm env expr with
|
||||||
| .error str => return .error $ errorI "parsing" str
|
| .error str => Protocol.throw $ errorI "parsing" str
|
||||||
| .ok syn => pure syn
|
| .ok syn => pure syn
|
||||||
match ← elabTerm syn expectedType? with
|
match ← elabTerm syn expectedType? with
|
||||||
| .error str => return .error $ errorI "elab" str
|
| .error str => Protocol.throw $ errorI "elab" str
|
||||||
| .ok expr => return .ok (← instantiateMVars expr)
|
| .ok expr => return (← instantiateMVars expr)
|
||||||
|
|
||||||
@[export pantograph_expr_echo_m]
|
@[export pantograph_expr_echo_m]
|
||||||
def exprEcho (expr: String) (expectedType?: Option String := .none) (levels: Array String := #[]) (options: @&Protocol.Options := {}):
|
def exprEcho (expr: String) (expectedType?: Option String := .none) (levels: Array String := #[]) (options: @&Protocol.Options := {}):
|
||||||
CoreM (Protocol.CR Protocol.ExprEchoResult) :=
|
Protocol.FallibleT CoreM Protocol.ExprEchoResult := do
|
||||||
runTermElabM $ Elab.Term.withLevelNames (levels.toList.map (·.toName)) do
|
let e : Except Protocol.InteractionError _ ← runTermElabM $ Elab.Term.withLevelNames (levels.toList.map (·.toName)) do
|
||||||
let expr ← match ← parseElabExpr expr expectedType? with
|
let expr ← match ← parseElabExpr expr expectedType? |>.run with
|
||||||
| .error e => return .error e
|
| .error e => return Except.error e
|
||||||
| .ok expr => pure expr
|
| .ok expr => pure expr
|
||||||
try
|
try
|
||||||
let type ← unfoldAuxLemmas (← Meta.inferType expr)
|
let type ← unfoldAuxLemmas (← Meta.inferType expr)
|
||||||
return .ok {
|
return .ok $ .ok ({
|
||||||
type := (← serializeExpression options type),
|
type := (← serializeExpression options type),
|
||||||
expr := (← serializeExpression options expr)
|
expr := (← serializeExpression options expr),
|
||||||
}
|
}: Protocol.ExprEchoResult)
|
||||||
catch exception =>
|
catch exception =>
|
||||||
return .error $ errorI "typing" (← exception.toMessageData.toString)
|
return Except.error $ errorI "typing" (← exception.toMessageData.toString)
|
||||||
|
liftExcept e
|
||||||
|
|
||||||
@[export pantograph_goal_start_expr_m]
|
@[export pantograph_goal_start_expr_m]
|
||||||
def goalStartExpr (expr: String) (levels: Array String): CoreM (Protocol.CR GoalState) :=
|
def goalStartExpr (expr: String) (levels: Array String): Protocol.FallibleT CoreM GoalState := do
|
||||||
runTermElabM $ Elab.Term.withLevelNames (levels.toList.map (·.toName)) do
|
let e : Except Protocol.InteractionError _ ← runTermElabM $ Elab.Term.withLevelNames (levels.toList.map (·.toName)) do
|
||||||
let expr ← match ← parseElabType expr with
|
let expr ← match ← parseElabType expr |>.run with
|
||||||
| .error e => return .error e
|
| .error e => return .error e
|
||||||
| .ok expr => pure $ expr
|
| .ok expr => pure $ expr
|
||||||
return .ok $ ← GoalState.create expr
|
return .ok $ .ok $ ← GoalState.create expr
|
||||||
|
liftExcept e
|
||||||
|
|
||||||
@[export pantograph_goal_resume]
|
@[export pantograph_goal_resume]
|
||||||
def goalResume (target: GoalState) (goals: Array String): Except String GoalState :=
|
def goalResume (target: GoalState) (goals: Array String): Except String GoalState :=
|
||||||
|
@ -217,9 +209,9 @@ def goalCalc (state: GoalState) (goal: MVarId) (pred: String): CoreM TacticResul
|
||||||
|
|
||||||
-- Cancel the token after a timeout.
|
-- Cancel the token after a timeout.
|
||||||
@[export pantograph_run_cancel_token_with_timeout_m]
|
@[export pantograph_run_cancel_token_with_timeout_m]
|
||||||
def runCancelTokenWithTimeout (cancelToken : IO.CancelToken) (timeout : Std.Time.Millisecond.Offset) : IO Unit := do
|
def runCancelTokenWithTimeout (cancelToken : IO.CancelToken) (timeout : UInt32) : IO Unit := do
|
||||||
let _ ← EIO.asTask do
|
let _ ← EIO.asTask do
|
||||||
let () ← (← Std.Internal.IO.Async.sleep timeout).block
|
IO.sleep timeout
|
||||||
cancelToken.set
|
cancelToken.set
|
||||||
return ()
|
return ()
|
||||||
|
|
||||||
|
|
|
@ -30,6 +30,8 @@ structure Options where
|
||||||
printImplementationDetailHyps: Bool := false
|
printImplementationDetailHyps: Bool := false
|
||||||
-- If this is set to `true`, goals will never go dormant, so you don't have to manage resumption
|
-- If this is set to `true`, goals will never go dormant, so you don't have to manage resumption
|
||||||
automaticMode: Bool := true
|
automaticMode: Bool := true
|
||||||
|
-- Timeout for tactics and operations that could potentially execute a tactic
|
||||||
|
timeout: Nat := 0
|
||||||
deriving Lean.ToJson
|
deriving Lean.ToJson
|
||||||
|
|
||||||
abbrev OptionsT := ReaderT Options
|
abbrev OptionsT := ReaderT Options
|
||||||
|
@ -202,9 +204,10 @@ structure EnvInspectResult where
|
||||||
|
|
||||||
structure EnvAdd where
|
structure EnvAdd where
|
||||||
name: String
|
name: String
|
||||||
|
levels: Array String := #[]
|
||||||
type: String
|
type: String
|
||||||
value: String
|
value: String
|
||||||
isTheorem: Bool
|
isTheorem: Bool := false
|
||||||
deriving Lean.FromJson
|
deriving Lean.FromJson
|
||||||
structure EnvAddResult where
|
structure EnvAddResult where
|
||||||
deriving Lean.ToJson
|
deriving Lean.ToJson
|
||||||
|
@ -225,6 +228,7 @@ structure OptionsSet where
|
||||||
printAuxDecls?: Option Bool := .none
|
printAuxDecls?: Option Bool := .none
|
||||||
printImplementationDetailHyps?: Option Bool := .none
|
printImplementationDetailHyps?: Option Bool := .none
|
||||||
automaticMode?: Option Bool := .none
|
automaticMode?: Option Bool := .none
|
||||||
|
timeout?: Option Nat := .none
|
||||||
deriving Lean.FromJson
|
deriving Lean.FromJson
|
||||||
structure OptionsSetResult where
|
structure OptionsSetResult where
|
||||||
deriving Lean.ToJson
|
deriving Lean.ToJson
|
||||||
|
@ -386,6 +390,9 @@ structure FrontendProcessResult where
|
||||||
units: List CompilationUnit
|
units: List CompilationUnit
|
||||||
deriving Lean.ToJson
|
deriving Lean.ToJson
|
||||||
|
|
||||||
abbrev CR α := Except InteractionError α
|
abbrev FallibleT := ExceptT InteractionError
|
||||||
|
|
||||||
|
abbrev throw {m : Type v → Type w} [MonadExceptOf InteractionError m] {α : Type v} (e : InteractionError) : m α :=
|
||||||
|
throwThe InteractionError e
|
||||||
|
|
||||||
end Pantograph.Protocol
|
end Pantograph.Protocol
|
||||||
|
|
177
Repl.lean
177
Repl.lean
|
@ -20,9 +20,8 @@ structure State where
|
||||||
|
|
||||||
/-- Main state monad for executing commands -/
|
/-- Main state monad for executing commands -/
|
||||||
abbrev MainM := ReaderT Context $ StateRefT State IO
|
abbrev MainM := ReaderT Context $ StateRefT State IO
|
||||||
|
abbrev EMainM := Protocol.FallibleT $ ReaderT Context $ StateRefT State IO
|
||||||
def getMainState : MainM State := get
|
def getMainState : MainM State := get
|
||||||
/-- Fallible subroutine return type -/
|
|
||||||
abbrev CR α := Except Protocol.InteractionError α
|
|
||||||
|
|
||||||
instance : MonadEnv MainM where
|
instance : MonadEnv MainM where
|
||||||
getEnv := return (← get).env
|
getEnv := return (← get).env
|
||||||
|
@ -37,24 +36,42 @@ def newGoalState (goalState: GoalState) : MainM Nat := do
|
||||||
}
|
}
|
||||||
return stateId
|
return stateId
|
||||||
|
|
||||||
def runCoreM { α } (coreM : CoreM α) : MainM α := do
|
def runCoreM { α } (coreM : CoreM α) : EMainM α := do
|
||||||
let scope := (← get).scope
|
let scope := (← get).scope
|
||||||
|
let options :=(← get).options
|
||||||
|
let cancelTk? ← match options.timeout with
|
||||||
|
| 0 => pure .none
|
||||||
|
| _ => .some <$> IO.CancelToken.new
|
||||||
let coreCtx : Core.Context := {
|
let coreCtx : Core.Context := {
|
||||||
(← read).coreContext with
|
(← read).coreContext with
|
||||||
currNamespace := scope.currNamespace
|
currNamespace := scope.currNamespace
|
||||||
openDecls := scope.openDecls
|
openDecls := scope.openDecls
|
||||||
options := scope.opts
|
options := scope.opts
|
||||||
|
cancelTk?,
|
||||||
}
|
}
|
||||||
let coreState : Core.State := {
|
let coreState : Core.State := {
|
||||||
env := (← get).env
|
env := (← get).env
|
||||||
}
|
}
|
||||||
let (result, state') ← coreM.toIO coreCtx coreState
|
-- Remap the coreM to capture every exception
|
||||||
modifyEnv λ _ => state'.env
|
let coreM' : CoreM _ :=
|
||||||
return result
|
Core.tryCatchRuntimeEx (Except.ok <$> coreM) λ ex => do
|
||||||
|
let desc ← ex.toMessageData.toString
|
||||||
|
return Except.error $ ({ error := "exception", desc } : Protocol.InteractionError)
|
||||||
|
let io := coreM'.toIO coreCtx coreState
|
||||||
|
if let .some token := cancelTk? then
|
||||||
|
runCancelTokenWithTimeout token (timeout := .mk options.timeout)
|
||||||
|
let (result, state') ← io
|
||||||
|
if result matches .ok _ then
|
||||||
|
modifyEnv λ _ => state'.env
|
||||||
|
liftExcept result
|
||||||
|
|
||||||
def liftMetaM { α } (metaM : MetaM α): MainM α :=
|
def runCoreM' { α } (coreM : Protocol.FallibleT CoreM α) : EMainM α := do
|
||||||
|
liftExcept $ ← runCoreM coreM.run
|
||||||
|
|
||||||
|
|
||||||
|
def liftMetaM { α } (metaM : MetaM α): EMainM α :=
|
||||||
runCoreM metaM.run'
|
runCoreM metaM.run'
|
||||||
def liftTermElabM { α } (termElabM: Elab.TermElabM α) : MainM α :=
|
def liftTermElabM { α } (termElabM: Elab.TermElabM α) : EMainM α :=
|
||||||
runCoreM $ termElabM.run' (ctx := defaultElabContext) |>.run'
|
runCoreM $ termElabM.run' (ctx := defaultElabContext) |>.run'
|
||||||
|
|
||||||
section Frontend
|
section Frontend
|
||||||
|
@ -68,7 +85,7 @@ structure CompilationUnit where
|
||||||
messages : Array String
|
messages : Array String
|
||||||
newConstants : List Name
|
newConstants : List Name
|
||||||
|
|
||||||
def frontend_process_inner (args: Protocol.FrontendProcess): MainM (CR Protocol.FrontendProcessResult) := do
|
def frontend_process_inner (args: Protocol.FrontendProcess): EMainM Protocol.FrontendProcessResult := do
|
||||||
let options := (← getMainState).options
|
let options := (← getMainState).options
|
||||||
let (fileName, file) ← match args.fileName?, args.file? with
|
let (fileName, file) ← match args.fileName?, args.file? with
|
||||||
| .some fileName, .none => do
|
| .some fileName, .none => do
|
||||||
|
@ -76,7 +93,7 @@ def frontend_process_inner (args: Protocol.FrontendProcess): MainM (CR Protocol.
|
||||||
pure (fileName, file)
|
pure (fileName, file)
|
||||||
| .none, .some file =>
|
| .none, .some file =>
|
||||||
pure ("<anonymous>", file)
|
pure ("<anonymous>", file)
|
||||||
| _, _ => return .error <| errorI "arguments" "Exactly one of {fileName, file} must be supplied"
|
| _, _ => Protocol.throw $ errorI "arguments" "Exactly one of {fileName, file} must be supplied"
|
||||||
let env?: Option Environment ← if args.readHeader then
|
let env?: Option Environment ← if args.readHeader then
|
||||||
pure .none
|
pure .none
|
||||||
else do
|
else do
|
||||||
|
@ -137,19 +154,23 @@ def frontend_process_inner (args: Protocol.FrontendProcess): MainM (CR Protocol.
|
||||||
goalSrcBoundaries?,
|
goalSrcBoundaries?,
|
||||||
newConstants?,
|
newConstants?,
|
||||||
}
|
}
|
||||||
return .ok { units }
|
return { units }
|
||||||
|
|
||||||
end Frontend
|
end Frontend
|
||||||
|
|
||||||
/-- Main loop command of the REPL -/
|
/-- Main loop command of the REPL -/
|
||||||
def execute (command: Protocol.Command): MainM Json := do
|
def execute (command: Protocol.Command): MainM Json := do
|
||||||
let run { α β: Type } [FromJson α] [ToJson β] (comm: α → MainM (CR β)): MainM Json :=
|
let run { α β: Type } [FromJson α] [ToJson β] (comm: α → EMainM β): MainM Json :=
|
||||||
match fromJson? command.payload with
|
try
|
||||||
| .ok args => do
|
match fromJson? command.payload with
|
||||||
match (← comm args) with
|
| .ok args => do
|
||||||
| .ok result => return toJson result
|
match (← comm args |>.run) with
|
||||||
| .error ierror => return toJson ierror
|
| .ok result => return toJson result
|
||||||
| .error error => return toJson $ errorCommand s!"Unable to parse json: {error}"
|
| .error ierror => return toJson ierror
|
||||||
|
| .error error => return toJson $ errorCommand s!"Unable to parse json: {error}"
|
||||||
|
catch ex : IO.Error =>
|
||||||
|
let error : Protocol.InteractionError := { error := "io", desc := ex.toString }
|
||||||
|
return toJson error
|
||||||
match command.cmd with
|
match command.cmd with
|
||||||
| "reset" => run reset
|
| "reset" => run reset
|
||||||
| "stat" => run stat
|
| "stat" => run stat
|
||||||
|
@ -180,40 +201,40 @@ def execute (command: Protocol.Command): MainM Json := do
|
||||||
errorIndex := errorI "index"
|
errorIndex := errorI "index"
|
||||||
errorIO := errorI "io"
|
errorIO := errorI "io"
|
||||||
-- Command Functions
|
-- Command Functions
|
||||||
reset (_: Protocol.Reset): MainM (CR Protocol.StatResult) := do
|
reset (_: Protocol.Reset): EMainM Protocol.StatResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let nGoals := state.goalStates.size
|
let nGoals := state.goalStates.size
|
||||||
set { state with nextId := 0, goalStates := .empty }
|
set { state with nextId := 0, goalStates := .empty }
|
||||||
return .ok { nGoals }
|
return { nGoals }
|
||||||
stat (_: Protocol.Stat): MainM (CR Protocol.StatResult) := do
|
stat (_: Protocol.Stat): EMainM Protocol.StatResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let nGoals := state.goalStates.size
|
let nGoals := state.goalStates.size
|
||||||
return .ok { nGoals }
|
return { nGoals }
|
||||||
env_describe (args: Protocol.EnvDescribe): MainM (CR Protocol.EnvDescribeResult) := do
|
env_describe (args: Protocol.EnvDescribe): EMainM Protocol.EnvDescribeResult := do
|
||||||
let result ← runCoreM $ Environment.describe args
|
let result ← runCoreM $ Environment.describe args
|
||||||
return .ok result
|
return result
|
||||||
env_module_read (args: Protocol.EnvModuleRead): MainM (CR Protocol.EnvModuleReadResult) := do
|
env_module_read (args: Protocol.EnvModuleRead): EMainM Protocol.EnvModuleReadResult := do
|
||||||
runCoreM $ Environment.moduleRead args
|
runCoreM $ Environment.moduleRead args
|
||||||
env_catalog (args: Protocol.EnvCatalog): MainM (CR Protocol.EnvCatalogResult) := do
|
env_catalog (args: Protocol.EnvCatalog): EMainM Protocol.EnvCatalogResult := do
|
||||||
let result ← runCoreM $ Environment.catalog args
|
let result ← runCoreM $ Environment.catalog args
|
||||||
return .ok result
|
return result
|
||||||
env_inspect (args: Protocol.EnvInspect): MainM (CR Protocol.EnvInspectResult) := do
|
env_inspect (args: Protocol.EnvInspect): EMainM Protocol.EnvInspectResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
runCoreM $ Environment.inspect args state.options
|
runCoreM' $ Environment.inspect args state.options
|
||||||
env_add (args: Protocol.EnvAdd): MainM (CR Protocol.EnvAddResult) := do
|
env_add (args: Protocol.EnvAdd): EMainM Protocol.EnvAddResult := do
|
||||||
runCoreM $ Environment.addDecl args
|
runCoreM' $ Environment.addDecl args.name args.levels args.type args.value args.isTheorem
|
||||||
env_save (args: Protocol.EnvSaveLoad): MainM (CR Protocol.EnvSaveLoadResult) := do
|
env_save (args: Protocol.EnvSaveLoad): EMainM Protocol.EnvSaveLoadResult := do
|
||||||
let env ← MonadEnv.getEnv
|
let env ← MonadEnv.getEnv
|
||||||
environmentPickle env args.path
|
environmentPickle env args.path
|
||||||
return .ok {}
|
return {}
|
||||||
env_load (args: Protocol.EnvSaveLoad): MainM (CR Protocol.EnvSaveLoadResult) := do
|
env_load (args: Protocol.EnvSaveLoad): EMainM Protocol.EnvSaveLoadResult := do
|
||||||
let (env, _) ← environmentUnpickle args.path
|
let (env, _) ← environmentUnpickle args.path
|
||||||
setEnv env
|
setEnv env
|
||||||
return .ok {}
|
return {}
|
||||||
expr_echo (args: Protocol.ExprEcho): MainM (CR Protocol.ExprEchoResult) := do
|
expr_echo (args: Protocol.ExprEcho): EMainM Protocol.ExprEchoResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
runCoreM $ exprEcho args.expr (expectedType? := args.type?) (levels := args.levels.getD #[]) (options := state.options)
|
runCoreM' $ exprEcho args.expr (expectedType? := args.type?) (levels := args.levels.getD #[]) (options := state.options)
|
||||||
options_set (args: Protocol.OptionsSet): MainM (CR Protocol.OptionsSetResult) := do
|
options_set (args: Protocol.OptionsSet): EMainM Protocol.OptionsSetResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let options := state.options
|
let options := state.options
|
||||||
set { state with
|
set { state with
|
||||||
|
@ -227,14 +248,15 @@ def execute (command: Protocol.Command): MainM Json := do
|
||||||
printAuxDecls := args.printAuxDecls?.getD options.printAuxDecls,
|
printAuxDecls := args.printAuxDecls?.getD options.printAuxDecls,
|
||||||
printImplementationDetailHyps := args.printImplementationDetailHyps?.getD options.printImplementationDetailHyps
|
printImplementationDetailHyps := args.printImplementationDetailHyps?.getD options.printImplementationDetailHyps
|
||||||
automaticMode := args.automaticMode?.getD options.automaticMode,
|
automaticMode := args.automaticMode?.getD options.automaticMode,
|
||||||
|
timeout := args.timeout?.getD options.timeout,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return .ok { }
|
return { }
|
||||||
options_print (_: Protocol.OptionsPrint): MainM (CR Protocol.Options) := do
|
options_print (_: Protocol.OptionsPrint): EMainM Protocol.Options := do
|
||||||
return .ok (← getMainState).options
|
return (← getMainState).options
|
||||||
goal_start (args: Protocol.GoalStart): MainM (CR Protocol.GoalStartResult) := do
|
goal_start (args: Protocol.GoalStart): EMainM Protocol.GoalStartResult := do
|
||||||
let expr?: Except _ GoalState ← liftTermElabM (match args.expr, args.copyFrom with
|
let expr?: Except _ GoalState ← liftTermElabM (match args.expr, args.copyFrom with
|
||||||
| .some expr, .none => goalStartExpr expr (args.levels.getD #[])
|
| .some expr, .none => goalStartExpr expr (args.levels.getD #[]) |>.run
|
||||||
| .none, .some copyFrom => do
|
| .none, .some copyFrom => do
|
||||||
(match (← getEnv).find? <| copyFrom.toName with
|
(match (← getEnv).find? <| copyFrom.toName with
|
||||||
| .none => return .error <| errorIndex s!"Symbol not found: {copyFrom}"
|
| .none => return .error <| errorIndex s!"Symbol not found: {copyFrom}"
|
||||||
|
@ -242,16 +264,16 @@ def execute (command: Protocol.Command): MainM Json := do
|
||||||
| _, _ =>
|
| _, _ =>
|
||||||
return .error <| errorI "arguments" "Exactly one of {expr, copyFrom} must be supplied")
|
return .error <| errorI "arguments" "Exactly one of {expr, copyFrom} must be supplied")
|
||||||
match expr? with
|
match expr? with
|
||||||
| .error error => return .error error
|
| .error error => Protocol.throw error
|
||||||
| .ok goalState =>
|
| .ok goalState =>
|
||||||
let stateId ← newGoalState goalState
|
let stateId ← newGoalState goalState
|
||||||
return .ok { stateId, root := goalState.root.name.toString }
|
return { stateId, root := goalState.root.name.toString }
|
||||||
goal_tactic (args: Protocol.GoalTactic): MainM (CR Protocol.GoalTacticResult) := do
|
goal_tactic (args: Protocol.GoalTactic): EMainM Protocol.GoalTacticResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let .some goalState := state.goalStates[args.stateId]? |
|
let .some goalState := state.goalStates[args.stateId]? |
|
||||||
return .error $ errorIndex s!"Invalid state index {args.stateId}"
|
Protocol.throw $ errorIndex s!"Invalid state index {args.stateId}"
|
||||||
let .some goal := goalState.goals.get? args.goalId |
|
let .some goal := goalState.goals.get? args.goalId |
|
||||||
return .error $ errorIndex s!"Invalid goal index {args.goalId}"
|
Protocol.throw $ errorIndex s!"Invalid goal index {args.goalId}"
|
||||||
let nextGoalState?: Except _ TacticResult ← liftTermElabM do
|
let nextGoalState?: Except _ TacticResult ← liftTermElabM do
|
||||||
-- NOTE: Should probably use a macro to handle this...
|
-- NOTE: Should probably use a macro to handle this...
|
||||||
match args.tactic?, args.expr?, args.have?, args.let?, args.calc?, args.conv?, args.draft? with
|
match args.tactic?, args.expr?, args.have?, args.let?, args.calc?, args.conv?, args.draft? with
|
||||||
|
@ -277,63 +299,63 @@ def execute (command: Protocol.Command): MainM Json := do
|
||||||
let error := errorI "arguments" "Exactly one of {tactic, expr, have, let, calc, conv, draft} must be supplied"
|
let error := errorI "arguments" "Exactly one of {tactic, expr, have, let, calc, conv, draft} must be supplied"
|
||||||
pure $ .error error
|
pure $ .error error
|
||||||
match nextGoalState? with
|
match nextGoalState? with
|
||||||
| .error error => return .error error
|
| .error error => Protocol.throw error
|
||||||
| .ok (.success nextGoalState) => do
|
| .ok (.success nextGoalState) => do
|
||||||
let nextGoalState ← match state.options.automaticMode, args.conv? with
|
let nextGoalState ← match state.options.automaticMode, args.conv? with
|
||||||
| true, .none => do
|
| true, .none => do
|
||||||
let .ok result := nextGoalState.resume (nextGoalState.goals ++ goalState.goals) |
|
let .ok result := nextGoalState.resume (nextGoalState.goals ++ goalState.goals) |
|
||||||
return .error $ errorIO "Resuming known goals"
|
Protocol.throw $ errorIO "Resuming known goals"
|
||||||
pure result
|
pure result
|
||||||
| true, .some true => pure nextGoalState
|
| true, .some true => pure nextGoalState
|
||||||
| true, .some false => do
|
| true, .some false => do
|
||||||
let .some (_, _, dormantGoals) := goalState.convMVar? |
|
let .some (_, _, dormantGoals) := goalState.convMVar? |
|
||||||
return .error $ errorIO "If conv exit succeeded this should not fail"
|
Protocol.throw $ errorIO "If conv exit succeeded this should not fail"
|
||||||
let .ok result := nextGoalState.resume (nextGoalState.goals ++ dormantGoals) |
|
let .ok result := nextGoalState.resume (nextGoalState.goals ++ dormantGoals) |
|
||||||
return .error $ errorIO "Resuming known goals"
|
Protocol.throw $ errorIO "Resuming known goals"
|
||||||
pure result
|
pure result
|
||||||
| false, _ => pure nextGoalState
|
| false, _ => pure nextGoalState
|
||||||
let nextStateId ← newGoalState nextGoalState
|
let nextStateId ← newGoalState nextGoalState
|
||||||
let goals ← runCoreM $ nextGoalState.serializeGoals (parent := .some goalState) (options := state.options) |>.run'
|
let goals ← runCoreM $ nextGoalState.serializeGoals (parent := .some goalState) (options := state.options) |>.run'
|
||||||
return .ok {
|
return {
|
||||||
nextStateId? := .some nextStateId,
|
nextStateId? := .some nextStateId,
|
||||||
goals? := .some goals,
|
goals? := .some goals,
|
||||||
}
|
}
|
||||||
| .ok (.parseError message) =>
|
| .ok (.parseError message) =>
|
||||||
return .ok { parseError? := .some message }
|
return { parseError? := .some message }
|
||||||
| .ok (.invalidAction message) =>
|
| .ok (.invalidAction message) =>
|
||||||
return .error $ errorI "invalid" message
|
Protocol.throw $ errorI "invalid" message
|
||||||
| .ok (.failure messages) =>
|
| .ok (.failure messages) =>
|
||||||
return .ok { tacticErrors? := .some messages }
|
return { tacticErrors? := .some messages }
|
||||||
goal_continue (args: Protocol.GoalContinue): MainM (CR Protocol.GoalContinueResult) := do
|
goal_continue (args: Protocol.GoalContinue): EMainM Protocol.GoalContinueResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let .some target := state.goalStates[args.target]? |
|
let .some target := state.goalStates[args.target]? |
|
||||||
return .error $ errorIndex s!"Invalid state index {args.target}"
|
Protocol.throw $ errorIndex s!"Invalid state index {args.target}"
|
||||||
let nextState? ← match args.branch?, args.goals? with
|
let nextGoalState? : GoalState ← match args.branch?, args.goals? with
|
||||||
| .some branchId, .none => do
|
| .some branchId, .none => do
|
||||||
match state.goalStates[branchId]? with
|
match state.goalStates[branchId]? with
|
||||||
| .none => return .error $ errorIndex s!"Invalid state index {branchId}"
|
| .none => Protocol.throw $ errorIndex s!"Invalid state index {branchId}"
|
||||||
| .some branch => pure $ target.continue branch
|
| .some branch => pure $ target.continue branch
|
||||||
| .none, .some goals =>
|
| .none, .some goals =>
|
||||||
pure $ goalResume target goals
|
pure $ goalResume target goals
|
||||||
| _, _ => return .error <| errorI "arguments" "Exactly one of {branch, goals} must be supplied"
|
| _, _ => Protocol.throw $ errorI "arguments" "Exactly one of {branch, goals} must be supplied"
|
||||||
match nextState? with
|
match nextGoalState? with
|
||||||
| .error error => return .error <| errorI "structure" error
|
| .error error => Protocol.throw $ errorI "structure" error
|
||||||
| .ok nextGoalState =>
|
| .ok nextGoalState =>
|
||||||
let nextStateId ← newGoalState nextGoalState
|
let nextStateId ← newGoalState nextGoalState
|
||||||
let goals ← liftMetaM $ goalSerialize nextGoalState (options := state.options)
|
let goals ← liftMetaM $ goalSerialize nextGoalState (options := state.options)
|
||||||
return .ok {
|
return {
|
||||||
nextStateId,
|
nextStateId,
|
||||||
goals,
|
goals,
|
||||||
}
|
}
|
||||||
goal_delete (args: Protocol.GoalDelete): MainM (CR Protocol.GoalDeleteResult) := do
|
goal_delete (args: Protocol.GoalDelete): EMainM Protocol.GoalDeleteResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let goalStates := args.stateIds.foldl (λ map id => map.erase id) state.goalStates
|
let goalStates := args.stateIds.foldl (λ map id => map.erase id) state.goalStates
|
||||||
set { state with goalStates }
|
set { state with goalStates }
|
||||||
return .ok {}
|
return {}
|
||||||
goal_print (args: Protocol.GoalPrint): MainM (CR Protocol.GoalPrintResult) := do
|
goal_print (args: Protocol.GoalPrint): EMainM Protocol.GoalPrintResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let .some goalState := state.goalStates[args.stateId]? |
|
let .some goalState := state.goalStates[args.stateId]? |
|
||||||
return .error $ errorIndex s!"Invalid state index {args.stateId}"
|
Protocol.throw $ errorIndex s!"Invalid state index {args.stateId}"
|
||||||
let result ← liftMetaM <| goalPrint
|
let result ← liftMetaM <| goalPrint
|
||||||
goalState
|
goalState
|
||||||
(rootExpr := args.rootExpr?.getD False)
|
(rootExpr := args.rootExpr?.getD False)
|
||||||
|
@ -341,21 +363,18 @@ def execute (command: Protocol.Command): MainM Json := do
|
||||||
(goals := args.goals?.getD False)
|
(goals := args.goals?.getD False)
|
||||||
(extraMVars := args.extraMVars?.getD #[])
|
(extraMVars := args.extraMVars?.getD #[])
|
||||||
(options := state.options)
|
(options := state.options)
|
||||||
return .ok result
|
return result
|
||||||
goal_save (args: Protocol.GoalSave): MainM (CR Protocol.GoalSaveResult) := do
|
goal_save (args: Protocol.GoalSave): EMainM Protocol.GoalSaveResult := do
|
||||||
let state ← getMainState
|
let state ← getMainState
|
||||||
let .some goalState := state.goalStates[args.id]? |
|
let .some goalState := state.goalStates[args.id]? |
|
||||||
return .error $ errorIndex s!"Invalid state index {args.id}"
|
Protocol.throw $ errorIndex s!"Invalid state index {args.id}"
|
||||||
goalStatePickle goalState args.path
|
goalStatePickle goalState args.path
|
||||||
return .ok {}
|
return {}
|
||||||
goal_load (args: Protocol.GoalLoad): MainM (CR Protocol.GoalLoadResult) := do
|
goal_load (args: Protocol.GoalLoad): EMainM Protocol.GoalLoadResult := do
|
||||||
let (goalState, _) ← goalStateUnpickle args.path (← MonadEnv.getEnv)
|
let (goalState, _) ← goalStateUnpickle args.path (← MonadEnv.getEnv)
|
||||||
let id ← newGoalState goalState
|
let id ← newGoalState goalState
|
||||||
return .ok { id }
|
return { id }
|
||||||
frontend_process (args: Protocol.FrontendProcess): MainM (CR Protocol.FrontendProcessResult) := do
|
frontend_process (args: Protocol.FrontendProcess): EMainM Protocol.FrontendProcessResult := do
|
||||||
try
|
frontend_process_inner args
|
||||||
frontend_process_inner args
|
|
||||||
catch e =>
|
|
||||||
return .error $ errorI "frontend" e.toString
|
|
||||||
|
|
||||||
end Pantograph.Repl
|
end Pantograph.Repl
|
||||||
|
|
|
@ -103,14 +103,14 @@ def test_symbol_location : TestT IO Unit := do
|
||||||
(opts := {})
|
(opts := {})
|
||||||
(trustLevel := 1)
|
(trustLevel := 1)
|
||||||
addTest $ ← runTestCoreM env do
|
addTest $ ← runTestCoreM env do
|
||||||
let .ok result ← Environment.inspect { name := "Nat.le_of_succ_le", source? := .some true } (options := {}) | fail "Inspect failed"
|
let .ok result ← (Environment.inspect { name := "Nat.le_of_succ_le", source? := .some true } (options := {})).run | fail "Inspect failed"
|
||||||
checkEq "module" result.module? <| .some "Init.Data.Nat.Basic"
|
checkEq "module" result.module? <| .some "Init.Data.Nat.Basic"
|
||||||
|
|
||||||
-- Extraction of source doesn't work for symbols in `Init` for some reason
|
-- Extraction of source doesn't work for symbols in `Init` for some reason
|
||||||
checkTrue "file" result.sourceUri?.isNone
|
checkTrue "file" result.sourceUri?.isNone
|
||||||
checkEq "pos" (result.sourceStart?.map (·.column)) <| .some 0
|
checkEq "pos" (result.sourceStart?.map (·.column)) <| .some 0
|
||||||
checkEq "pos" (result.sourceEnd?.map (·.column)) <| .some 88
|
checkEq "pos" (result.sourceEnd?.map (·.column)) <| .some 88
|
||||||
let .ok { imports, constNames, .. } ← Environment.moduleRead ⟨"Init.Data.Nat.Basic"⟩ | fail "Module read failed"
|
let { imports, constNames, .. } ← Environment.moduleRead ⟨"Init.Data.Nat.Basic"⟩
|
||||||
checkEq "imports" imports #["Init.SimpLemmas", "Init.Data.NeZero"]
|
checkEq "imports" imports #["Init.SimpLemmas", "Init.Data.NeZero"]
|
||||||
checkTrue "constNames" $ constNames.contains "Nat.succ_add"
|
checkTrue "constNames" $ constNames.contains "Nat.succ_add"
|
||||||
|
|
||||||
|
|
|
@ -91,6 +91,24 @@ def test_tactic : Test :=
|
||||||
({ tacticErrors? := .some #["tactic 'apply' failed, failed to unify\n ∀ {m : Nat}, Nat.succ ?n ≤ m → ?n ≤ m\nwith\n ∀ (q : Prop), x ∨ q → q ∨ x\nx : Prop\n⊢ ∀ (q : Prop), x ∨ q → q ∨ x"] }:
|
({ tacticErrors? := .some #["tactic 'apply' failed, failed to unify\n ∀ {m : Nat}, Nat.succ ?n ≤ m → ?n ≤ m\nwith\n ∀ (q : Prop), x ∨ q → q ∨ x\nx : Prop\n⊢ ∀ (q : Prop), x ∨ q → q ∨ x"] }:
|
||||||
Protocol.GoalTacticResult)
|
Protocol.GoalTacticResult)
|
||||||
]
|
]
|
||||||
|
example : (1 : Nat) + (2 * 3) = 1 + (4 - 3) + (6 - 4) + 3 := by
|
||||||
|
simp
|
||||||
|
def test_tactic_timeout : Test :=
|
||||||
|
[
|
||||||
|
step "goal.start" ({ expr := "(1 : Nat) + (2 * 3) = 1 + (4 - 3) + (6 - 4) + 3" }: Protocol.GoalStart)
|
||||||
|
({ stateId := 0, root := "_uniq.319" }: Protocol.GoalStartResult),
|
||||||
|
-- timeout of 10 milliseconds
|
||||||
|
step "options.set" ({ timeout? := .some 10 } : Protocol.OptionsSet)
|
||||||
|
({ }: Protocol.OptionsSetResult),
|
||||||
|
step "goal.tactic" ({ stateId := 0, expr? := .some "by\nsleep 1000; simp" }: Protocol.GoalTactic)
|
||||||
|
({ error := "io", desc := "internal exception #6" }: Protocol.InteractionError),
|
||||||
|
-- ensure graceful recovery
|
||||||
|
step "options.set" ({ timeout? := .some 0 } : Protocol.OptionsSet)
|
||||||
|
({ }: Protocol.OptionsSetResult),
|
||||||
|
step "goal.tactic" ({ stateId := 0, tactic? := .some "simp" }: Protocol.GoalTactic)
|
||||||
|
({ nextStateId? := .some 1, goals? := .some #[], }: Protocol.GoalTacticResult),
|
||||||
|
]
|
||||||
|
|
||||||
def test_automatic_mode (automatic: Bool): Test :=
|
def test_automatic_mode (automatic: Bool): Test :=
|
||||||
let varsPQ := #[
|
let varsPQ := #[
|
||||||
{ name := "_uniq.10", userName := "p", type? := .some { pp? := .some "Prop" }},
|
{ name := "_uniq.10", userName := "p", type? := .some { pp? := .some "Prop" }},
|
||||||
|
@ -280,6 +298,7 @@ def suite (env : Lean.Environment): List (String × IO LSpec.TestSeq) :=
|
||||||
("options.set options.print", test_option_modify),
|
("options.set options.print", test_option_modify),
|
||||||
("Malformed command", test_malformed_command),
|
("Malformed command", test_malformed_command),
|
||||||
("Tactic", test_tactic),
|
("Tactic", test_tactic),
|
||||||
|
("Tactic Timeout", test_tactic_timeout),
|
||||||
("Manual Mode", test_automatic_mode false),
|
("Manual Mode", test_automatic_mode false),
|
||||||
("Automatic Mode", test_automatic_mode true),
|
("Automatic Mode", test_automatic_mode true),
|
||||||
("env.add env.inspect", test_env_add_inspect),
|
("env.add env.inspect", test_env_add_inspect),
|
||||||
|
|
|
@ -22,6 +22,9 @@ See `Pantograph/Protocol.lean` for a description of the parameters and return va
|
||||||
automatic mode (flag: `"automaticMode"`). By default it is turned on, with
|
automatic mode (flag: `"automaticMode"`). By default it is turned on, with
|
||||||
all goals automatically resuming. This makes Pantograph act like a gym,
|
all goals automatically resuming. This makes Pantograph act like a gym,
|
||||||
with no resumption necessary to manage your goals.
|
with no resumption necessary to manage your goals.
|
||||||
|
|
||||||
|
Set `timeout` to a non-zero number to specify timeout (milliseconds) for all `CoreM`
|
||||||
|
operations.
|
||||||
* `options.print`: Display the current set of options
|
* `options.print`: Display the current set of options
|
||||||
* `goal.start {["name": <name>], ["expr": <expr>], ["levels": [<levels>]], ["copyFrom": <symbol>]}`:
|
* `goal.start {["name": <name>], ["expr": <expr>], ["levels": [<levels>]], ["copyFrom": <symbol>]}`:
|
||||||
Start a new proof from a given expression or symbol
|
Start a new proof from a given expression or symbol
|
||||||
|
|
Loading…
Reference in New Issue