For general server architecture, see README.md
. This module implements the watchdog process.
Watchdog state #
Most LSP clients only send us file diffs, so to facilitate sending entire file contents to freshly restarted workers, the watchdog needs to maintain the current state of each file. It can also use this state to detect changes to the header and thus restart the corresponding worker, freeing its imports.
TODO(WN): We may eventually want to keep track of approximately (since this isn't knowable exactly) where in the file a worker crashed. Then on restart, we tell said worker to only parse up to that point and query the user about how to proceed (continue OR allow the user to fix the bug and then continue OR ..). Without this, if the crash is deterministic, users may be confused about why the server seemingly stopped working for a single file.
Watchdog <-> worker communication #
The watchdog process and its file worker processes communicate via LSP. If the necessity arises, we might add non-standard commands similarly based on JSON-RPC. Most requests and notifications are forwarded to the corresponding file worker process, with the exception of these notifications:
- textDocument/didOpen: Launch the file worker, create the associated watchdog state and launch a task to asynchronously receive LSP packets from the worker (e.g. request responses).
- textDocument/didChange: Update the local file state so that it can be resent to restarted workers.
Then forward the
didChange
notification. - textDocument/didClose: Signal a shutdown to the file worker and remove the associated watchdog state.
Moreover, we don't implement the full protocol at this level:
- Upon starting, the
initialize
request is forwarded to the worker, but it must not respond with its server capabilities. Consequently, the watchdog will not send aninitialized
notification to the worker. - After
initialize
, the watchdog sends the correspondingdidOpen
notification with the full current state of the file. No additionaldidOpen
notifications will be forwarded to the worker process. - File workers are always terminated with an
exit
notification, without previously receiving ashutdown
request. Similarly, they never receive adidClose
notification.
Watchdog <-> client communication #
The watchdog itself should implement the LSP standard as closely as possible. However we reserve the right to add non-standard extensions in case they're needed, for example to communicate tactic state.
Equations
- Lean.Server.Watchdog.workerCfg = { stdin := IO.Process.Stdio.piped, stdout := IO.Process.Stdio.piped }
Events that worker-specific tasks signal to the main thread.
- terminated : WorkerEvent
- importsChanged : WorkerEvent
- crashed (exitCode : UInt32) : WorkerEvent
- ioError (e : IO.Error) : WorkerEvent
- crashed : WorkerState
- cannotWrite : WorkerState
- terminating : WorkerState
- running : WorkerState
- i : Nat
- reqs : Std.TreeMap JsonRpc.RequestID (Nat × JsonRpc.Message) compare
- queue : Std.TreeMap Nat (JsonRpc.RequestID × JsonRpc.Message) compare
Equations
- Lean.Server.Watchdog.instInhabitedRequestQueueMap = { default := { i := default, reqs := default, queue := default } }
Equations
- Lean.Server.Watchdog.RequestQueueMap.empty = { i := 0, reqs := ∅, queue := ∅ }
Equations
- One or more equations did not get rendered due to their size.
- m.enqueue req = (panicWithPosWithDecl "Lean.Server.Watchdog" "Lean.Server.Watchdog.RequestQueueMap.enqueue" 118 10 "Got non-request in `RequestQueueMap.enqueue`.").run
Equations
- One or more equations did not get rendered due to their size.
- requestQueues : Std.TreeMap Lsp.DocumentUri RequestQueueMap compare
- uriByRequest : Std.TreeMap JsonRpc.RequestID Lsp.DocumentUri compare
Instances For
Equations
- Lean.Server.Watchdog.instInhabitedRequestData = { default := { requestQueues := default, uriByRequest := default } }
Equations
- Lean.Server.Watchdog.RequestData.empty = { requestQueues := ∅, uriByRequest := ∅ }
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
- d.enqueue uri req = (panicWithPosWithDecl "Lean.Server.Watchdog" "Lean.Server.Watchdog.RequestData.enqueue" 167 10 "Got non-request in `RequestData.enqueue`.").run
Equations
- d.erase uri id = { requestQueues := d.requestQueues.modify uri fun (x : Lean.Server.Watchdog.RequestQueueMap) => x.erase id, uriByRequest := d.uriByRequest.erase id }
Equations
- d.getUri? id = d.uriByRequest.get? id
Equations
- d.getRequestQueue uri = (d.requestQueues.get? uri).getD ∅
Equations
- m.clearWorkerRequestData uri = Std.Mutex.atomically m (modify fun (x : Lean.Server.Watchdog.RequestData) => x.clearWorkerRequestData uri)
Equations
- m.enqueue uri req = Std.Mutex.atomically m (modify fun (x : Lean.Server.Watchdog.RequestData) => x.enqueue uri req)
Equations
- m.erase uri id = Std.Mutex.atomically m (modify fun (x : Lean.Server.Watchdog.RequestData) => x.erase uri id)
Equations
- m.getRequestQueue uri = Std.Mutex.atomically m do let __do_lift ← get pure (__do_lift.getRequestQueue uri)
- doc : DocumentMeta
- proc : IO.Process.Child workerCfg
- commTask : ServerTask WorkerEvent
- state : Std.Mutex WorkerState
Equations
- fw.stdin = IO.FS.Stream.ofHandle fw.proc.stdin
Equations
- fw.stdout = IO.FS.Stream.ofHandle fw.proc.stdout
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Global import data for all open files managed by this watchdog.
Updates d
with the new set of imports
for the file uri
.
Equations
- One or more equations did not get rendered due to their size.
Sets the imports of uri
in d
to the empty set.
Equations
- d.eraseImportsOf uri = d.update uri ∅
- sourceUri : Lsp.DocumentUri
- localID : JsonRpc.RequestID
- pendingServerRequests : PendingServerRequestMap
- freshServerRequestID : Nat
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
- hIn : IO.FS.Stream
- hOut : IO.FS.Stream
- hLog : IO.FS.Stream
Command line arguments.
- fileWorkersRef : IO.Ref FileWorkerMap
- initParams : Lsp.InitializeParams
We store these to pass them to workers.
- workerPath : System.FilePath
- srcSearchPath : System.SearchPath
- references : IO.Ref References
- serverRequestData : IO.Ref ServerRequestData
- importData : IO.Ref ImportData
- requestData : RequestDataMutex
- srcSearchPath : System.SearchPath
- references : References
Equations
- One or more equations did not get rendered due to their size.
Equations
- Lean.Server.Watchdog.findFileWorker? uri = do let __do_lift ← read let __do_lift ← ST.Ref.get __do_lift.fileWorkersRef pure (Lean.RBMap.find? __do_lift uri)
Equations
- One or more equations did not get rendered due to their size.
Equations
- Lean.Server.Watchdog.setWorkerState fw s = fw.state.atomically (set s)
Equations
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Updates the global import data with the import closure provided by the file worker after it successfully processed its header.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Sends a notification to the file worker identified by uri
that its dependency staleDependency
is out-of-date.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Used in CallHierarchyItem.data?
to retain all the data needed to quickly re-identify the
call hierarchy item.
Extracts the CallHierarchyItemData from item.data?
and returns none
if this is not possible.
Equations
- Lean.Server.Watchdog.CallHierarchyItemData.fromItem? item = do let __do_lift ← item.data? (Lean.fromJson? __do_lift).toOption
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
When a file is saved, notifies all file workers for files that depend on this file that this specific import is now stale so that the file worker can issue a diagnostic asking users to restart the file.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
- Lean.Server.Watchdog.handleRequest id method params = Lean.Server.Watchdog.forwardRequestToWorker id method params
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
- workerEvent (fw : FileWorker) (ev : WorkerEvent) : ServerEvent
- clientMsg (msg : JsonRpc.Message) : ServerEvent
- clientError (e : IO.Error) : ServerEvent
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.
Starts loading .ileans present in the search path asynchronously in an IO task. This ensures that server startup is not blocked by loading the .ileans. In return, while the .ileans are being loaded, users will only get incomplete results in requests that need references.
Equations
- One or more equations did not get rendered due to their size.
Equations
- One or more equations did not get rendered due to their size.