|
|
|
@ -1,4 +1,5 @@
|
|
|
|
import Lean.Server.Watchdog
|
|
|
|
import Lean.Server.Watchdog
|
|
|
|
|
|
|
|
import GameServer.FileWorker
|
|
|
|
import GameServer.EnvExtensions
|
|
|
|
import GameServer.EnvExtensions
|
|
|
|
import GameServer.Game
|
|
|
|
import GameServer.Game
|
|
|
|
|
|
|
|
|
|
|
|
@ -11,16 +12,26 @@ open Lsp
|
|
|
|
open JsonRpc
|
|
|
|
open JsonRpc
|
|
|
|
open System.Uri
|
|
|
|
open System.Uri
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
open MyServer.FileWorker
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
structure WasmFileState :=
|
|
|
|
|
|
|
|
fileWorkerState : FileWorker.WorkerState
|
|
|
|
|
|
|
|
gameWorkerState : GameWorkerState
|
|
|
|
|
|
|
|
headerTask : Task (Except Error (Snapshots.Snapshot × SearchPath))
|
|
|
|
|
|
|
|
|
|
|
|
structure WasmServerState :=
|
|
|
|
structure WasmServerState :=
|
|
|
|
initParams? : Option InitializeParams
|
|
|
|
initParams? : Option InitializeParams
|
|
|
|
gameServerState : GameServerState
|
|
|
|
gameServerState : GameServerState
|
|
|
|
|
|
|
|
fileState : HashMap String WasmFileState := {}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def wasmSearchPath : SearchPath := ["/lib", "/gamelib"]
|
|
|
|
|
|
|
|
|
|
|
|
@[export game_make_state]
|
|
|
|
@[export game_make_state]
|
|
|
|
unsafe def makeState : IO WasmServerState := do
|
|
|
|
unsafe def makeState : IO WasmServerState := do
|
|
|
|
let e ← IO.getStderr
|
|
|
|
let e ← IO.getStderr
|
|
|
|
try
|
|
|
|
try
|
|
|
|
Lean.enableInitializersExecution
|
|
|
|
Lean.enableInitializersExecution
|
|
|
|
searchPathRef.set ["/lib", "/gamelib"]
|
|
|
|
searchPathRef.set wasmSearchPath
|
|
|
|
let env ← importModules #[
|
|
|
|
let env ← importModules #[
|
|
|
|
{ module := `GameServer : Import }
|
|
|
|
{ module := `GameServer : Import }
|
|
|
|
] {} 0
|
|
|
|
] {} 0
|
|
|
|
@ -31,7 +42,7 @@ unsafe def makeState : IO WasmServerState := do
|
|
|
|
inventory := #[]
|
|
|
|
inventory := #[]
|
|
|
|
difficulty := 0
|
|
|
|
difficulty := 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ⟨none, state⟩
|
|
|
|
return ⟨none, state, {}⟩
|
|
|
|
catch err =>
|
|
|
|
catch err =>
|
|
|
|
e.putStrLn s!"Import error: {err}"
|
|
|
|
e.putStrLn s!"Import error: {err}"
|
|
|
|
throw err
|
|
|
|
throw err
|
|
|
|
@ -71,7 +82,7 @@ def initializeServer (id : RequestID) : IO Unit := do
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ()
|
|
|
|
return ()
|
|
|
|
|
|
|
|
|
|
|
|
def mkContext (state : WasmServerState) : IO ServerContext := do
|
|
|
|
def mkServerContext (state : WasmServerState) : IO ServerContext := do
|
|
|
|
let i ← IO.getStdin
|
|
|
|
let i ← IO.getStdin
|
|
|
|
let o ← IO.getStdout
|
|
|
|
let o ← IO.getStdout
|
|
|
|
let e ← IO.getStderr
|
|
|
|
let e ← IO.getStderr
|
|
|
|
@ -96,14 +107,104 @@ def mkContext (state : WasmServerState) : IO ServerContext := do
|
|
|
|
def runGameServerM (state : WasmServerState) (x : GameServerM α) : IO (α × WasmServerState) := do
|
|
|
|
def runGameServerM (state : WasmServerState) (x : GameServerM α) : IO (α × WasmServerState) := do
|
|
|
|
let (res, gameServerState) ← ReaderT.run
|
|
|
|
let (res, gameServerState) ← ReaderT.run
|
|
|
|
(StateT.run x state.gameServerState)
|
|
|
|
(StateT.run x state.gameServerState)
|
|
|
|
(← mkContext state)
|
|
|
|
(← mkServerContext state)
|
|
|
|
return (res, {state with gameServerState})
|
|
|
|
return (res, {state with gameServerState})
|
|
|
|
|
|
|
|
|
|
|
|
def readParams (params? : Option Json.Structured) [FromJson α] : IO α :=
|
|
|
|
def mkWorkerContext (state : WasmServerState) (headerTask : Task (Except Error (Snapshots.Snapshot × SearchPath))) :
|
|
|
|
let j := toJson params?
|
|
|
|
IO FileWorker.WorkerContext := do
|
|
|
|
match fromJson? j with
|
|
|
|
let i ← IO.getStdin
|
|
|
|
| Except.ok v => pure v
|
|
|
|
let o ← IO.getStdout
|
|
|
|
| Except.error inner => throw $ userError s!"Unexpected param '{j.compress}' \n{inner}"
|
|
|
|
let e ← IO.getStderr
|
|
|
|
|
|
|
|
let some initParams := state.initParams?
|
|
|
|
|
|
|
|
| throwServerError "no yet initialized"
|
|
|
|
|
|
|
|
let clientHasWidgets := initParams.initializationOptions?.bind (·.hasWidgets?) |>.getD false
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
hIn := i
|
|
|
|
|
|
|
|
hOut := o
|
|
|
|
|
|
|
|
hLog := e
|
|
|
|
|
|
|
|
headerTask := headerTask
|
|
|
|
|
|
|
|
initParams := initParams
|
|
|
|
|
|
|
|
clientHasWidgets
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def runGameWorkerM (state : WasmServerState) (fileState : WasmFileState) (x : GameWorkerM α) :
|
|
|
|
|
|
|
|
IO (α × WasmFileState) := do
|
|
|
|
|
|
|
|
let s := fileState.fileWorkerState
|
|
|
|
|
|
|
|
let ctx ← mkWorkerContext state fileState.headerTask
|
|
|
|
|
|
|
|
let ((res, gameWorkerState), s) ← StateRefT'.run (s := s) <| ReaderT.run (r := ctx) <|
|
|
|
|
|
|
|
|
StateT.run (s := fileState.gameWorkerState) <| x
|
|
|
|
|
|
|
|
let fileState := {fileState with gameWorkerState := gameWorkerState, fileWorkerState := s}
|
|
|
|
|
|
|
|
return (res, fileState)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def parseParams {paramType : Type} [FromJson paramType] (params : Json) : IO paramType :=
|
|
|
|
|
|
|
|
match fromJson? params with
|
|
|
|
|
|
|
|
| Except.ok parsed => pure parsed
|
|
|
|
|
|
|
|
| Except.error inner => throwServerError s!"Got param with wrong structure: {params.compress}\n{inner}"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def requestWorkerUri (method : String) (params : Json) : IO (Option DocumentUri) := do
|
|
|
|
|
|
|
|
if method == "$/lean/rpc/connect" then
|
|
|
|
|
|
|
|
let ps : Lsp.RpcConnectParams ← parseParams params
|
|
|
|
|
|
|
|
pure <| fileSource ps
|
|
|
|
|
|
|
|
else match (← routeLspRequest method params) with
|
|
|
|
|
|
|
|
| Except.error e =>
|
|
|
|
|
|
|
|
throwServerError e.message
|
|
|
|
|
|
|
|
| Except.ok uri => pure uri
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
open FileWorker in
|
|
|
|
|
|
|
|
def handleDidOpen (params : DidOpenTextDocumentParams) (state : WasmServerState) : IO WasmServerState := do
|
|
|
|
|
|
|
|
let some initParams := state.initParams?
|
|
|
|
|
|
|
|
| throwServerError "no yet initialized"
|
|
|
|
|
|
|
|
let (_, state) ← runGameServerM state do
|
|
|
|
|
|
|
|
let some lvl ← GameServer.getLevelByFileName? initParams
|
|
|
|
|
|
|
|
((System.Uri.fileUriToPath? params.textDocument.uri).getD params.textDocument.uri |>.toString)
|
|
|
|
|
|
|
|
| throwServerError s!"Level not found: {params.textDocument.uri} | {initParams.rootUri?}"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let env ← importModules #[
|
|
|
|
|
|
|
|
{ module := lvl.module : Import }
|
|
|
|
|
|
|
|
] {} 0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
(← getStderr).putStr "Import for level completed"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let doc := params.textDocument
|
|
|
|
|
|
|
|
let meta : DocumentMeta := ⟨doc.uri, doc.version, doc.text.toFileMap, .always⟩
|
|
|
|
|
|
|
|
let clientHasWidgets := initParams.initializationOptions?.bind (·.hasWidgets?) |>.getD false
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let (headerStx, headerTask) ← mkHeaderTask meta (← getStdout) wasmSearchPath env {} clientHasWidgets
|
|
|
|
|
|
|
|
let cancelTk ← CancelToken.new
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let levelParams := {
|
|
|
|
|
|
|
|
uri := meta.uri
|
|
|
|
|
|
|
|
gameDir := state.gameServerState.gameDir
|
|
|
|
|
|
|
|
levelModule := lvl.module
|
|
|
|
|
|
|
|
tactics := lvl.tactics.tiles
|
|
|
|
|
|
|
|
lemmas := lvl.lemmas.tiles
|
|
|
|
|
|
|
|
definitions := lvl.definitions.tiles
|
|
|
|
|
|
|
|
inventory := state.gameServerState.inventory
|
|
|
|
|
|
|
|
difficulty := state.gameServerState.difficulty
|
|
|
|
|
|
|
|
statementName := lvl.statementName
|
|
|
|
|
|
|
|
: Game.DidOpenLevelParams
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let ctx ← mkWorkerContext state headerTask
|
|
|
|
|
|
|
|
let cmdSnaps ← EIO.mapTask (t := headerTask) (match · with
|
|
|
|
|
|
|
|
| Except.ok (s, _) => unfoldSnaps meta #[s] cancelTk levelParams ctx (startAfterMs := 0)
|
|
|
|
|
|
|
|
| Except.error e => throw (e : ElabTaskError))
|
|
|
|
|
|
|
|
let doc : EditableDocument := { meta, cmdSnaps := AsyncList.delayed cmdSnaps, cancelTk }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let s : WasmFileState := {
|
|
|
|
|
|
|
|
fileWorkerState := {
|
|
|
|
|
|
|
|
doc := doc
|
|
|
|
|
|
|
|
initHeaderStx := headerStx
|
|
|
|
|
|
|
|
pendingRequests := RBMap.empty
|
|
|
|
|
|
|
|
rpcSessions := RBMap.empty
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
gameWorkerState := { levelParams }
|
|
|
|
|
|
|
|
headerTask
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
let fileState := state.fileState.insert params.textDocument.uri s
|
|
|
|
|
|
|
|
return {state with fileState}
|
|
|
|
|
|
|
|
return state
|
|
|
|
|
|
|
|
|
|
|
|
@[export game_send_message]
|
|
|
|
@[export game_send_message]
|
|
|
|
unsafe def sendMessage (s : String) (state : WasmServerState) : IO WasmServerState := do
|
|
|
|
unsafe def sendMessage (s : String) (state : WasmServerState) : IO WasmServerState := do
|
|
|
|
@ -111,30 +212,46 @@ unsafe def sendMessage (s : String) (state : WasmServerState) : IO WasmServerSta
|
|
|
|
try
|
|
|
|
try
|
|
|
|
let m ← readMessage s
|
|
|
|
let m ← readMessage s
|
|
|
|
match m with
|
|
|
|
match m with
|
|
|
|
| Message.request id "initialize" params? =>
|
|
|
|
| Message.request id "initialize" (some params) =>
|
|
|
|
let p : InitializeParams ← readParams params?
|
|
|
|
let p : InitializeParams ← parseParams (toJson params)
|
|
|
|
initializeServer id
|
|
|
|
initializeServer id
|
|
|
|
let p := {p with rootUri? := some (toString state.gameServerState.game)}
|
|
|
|
let p := {p with rootUri? := some (toString state.gameServerState.game)}
|
|
|
|
return {state with initParams? := some p}
|
|
|
|
return {state with initParams? := some p}
|
|
|
|
| Message.notification "textDocument/didOpen" params? =>
|
|
|
|
|
|
|
|
let some initParams := state.initParams?
|
|
|
|
|
|
|
|
| throwServerError "no yet initialized"
|
|
|
|
|
|
|
|
let p : LeanDidOpenTextDocumentParams ← readParams params?
|
|
|
|
|
|
|
|
let (_, state) ← runGameServerM state do
|
|
|
|
|
|
|
|
let some lvl ← GameServer.getLevelByFileName? initParams
|
|
|
|
|
|
|
|
((System.Uri.fileUriToPath? p.textDocument.uri).getD p.textDocument.uri |>.toString)
|
|
|
|
|
|
|
|
| throwServerError s!"Level not found: {p.textDocument.uri} | {initParams.rootUri?}"
|
|
|
|
|
|
|
|
e.putStrLn s!"{lvl.module}"
|
|
|
|
|
|
|
|
return state
|
|
|
|
|
|
|
|
| _ =>
|
|
|
|
| _ =>
|
|
|
|
let (isGameEv, state) ← runGameServerM state (Game.handleServerEvent (.clientMsg m))
|
|
|
|
let (isGameEv, state) ← runGameServerM state (Game.handleServerEvent (.clientMsg m))
|
|
|
|
if isGameEv then
|
|
|
|
if isGameEv then
|
|
|
|
return state
|
|
|
|
return state
|
|
|
|
else
|
|
|
|
else
|
|
|
|
match m with
|
|
|
|
match m with
|
|
|
|
| _ =>
|
|
|
|
| Message.notification method (some params) =>
|
|
|
|
e.putStrLn s!"Expected JSON-RPC request, got: '{(toJson m).compress}'"
|
|
|
|
let handle := (fun α [FromJson α] (handler : α → WasmServerState → IO WasmServerState)
|
|
|
|
return state
|
|
|
|
=> parseParams (toJson params) >>= (handler · state))
|
|
|
|
|
|
|
|
match method with --TODO
|
|
|
|
|
|
|
|
| "textDocument/didOpen" => handle DidOpenTextDocumentParams handleDidOpen
|
|
|
|
|
|
|
|
-- | "textDocument/didChange" => handle DidChangeTextDocumentParams handleDidChange
|
|
|
|
|
|
|
|
-- | "textDocument/didClose" => handle DidCloseTextDocumentParams handleDidClose
|
|
|
|
|
|
|
|
-- | "workspace/didChangeWatchedFiles" => handle DidChangeWatchedFilesParams handleDidChangeWatchedFiles
|
|
|
|
|
|
|
|
-- | "$/cancelRequest" => handle CancelParams handleCancelRequest
|
|
|
|
|
|
|
|
-- | "$/lean/rpc/connect" => handle RpcConnectParams (forwardNotification method)
|
|
|
|
|
|
|
|
-- | "$/lean/rpc/release" => handle RpcReleaseParams (forwardNotification method)
|
|
|
|
|
|
|
|
-- | "$/lean/rpc/keepAlive" => handle RpcKeepAliveParams (forwardNotification method)
|
|
|
|
|
|
|
|
| _ => return state
|
|
|
|
|
|
|
|
| Message.request id method (some params) =>
|
|
|
|
|
|
|
|
let some uri ← requestWorkerUri method (toJson params)
|
|
|
|
|
|
|
|
| throwServerError s!"Could not find Uri for request: {method}"
|
|
|
|
|
|
|
|
let some fileState := state.fileState.find? uri
|
|
|
|
|
|
|
|
| throwServerError s!"File not open: {uri}"
|
|
|
|
|
|
|
|
let (_, fileState) ← runGameWorkerM state fileState do
|
|
|
|
|
|
|
|
MyServer.FileWorker.mainLoop1 m
|
|
|
|
|
|
|
|
let fileState := state.fileState.insert uri fileState
|
|
|
|
|
|
|
|
return {state with fileState}
|
|
|
|
|
|
|
|
| Message.responseError _ _ e .. =>
|
|
|
|
|
|
|
|
throwServerError s!"Unhandled response error: {e}"
|
|
|
|
|
|
|
|
| _ => throwServerError "Got invalid JSON-RPC message"
|
|
|
|
|
|
|
|
-- match m with
|
|
|
|
|
|
|
|
-- | _ =>
|
|
|
|
|
|
|
|
-- e.putStrLn s!"Expected JSON-RPC request, got: '{(toJson m).compress}'"
|
|
|
|
|
|
|
|
-- return state
|
|
|
|
catch err =>
|
|
|
|
catch err =>
|
|
|
|
e.putStrLn s!"Server error: {err}"
|
|
|
|
e.putStrLn s!"Server error: {err}"
|
|
|
|
return state
|
|
|
|
return state
|
|
|
|
|