Skip to content

Commit

Permalink
fix: start runtime under a supervisor (elixir-tools#124)
Browse files Browse the repository at this point in the history
  • Loading branch information
mhanberg authored Jul 24, 2023
1 parent 639493c commit df331dc
Show file tree
Hide file tree
Showing 3 changed files with 70 additions and 49 deletions.
75 changes: 26 additions & 49 deletions lib/next_ls.ex
Original file line number Diff line number Diff line change
Expand Up @@ -28,10 +28,9 @@ defmodule NextLS do
alias GenLSP.Structures.TextDocumentItem
alias GenLSP.Structures.TextDocumentSyncOptions
alias GenLSP.Structures.TextEdit
alias GenLSP.Structures.WorkDoneProgressBegin
alias GenLSP.Structures.WorkDoneProgressEnd
alias NextLS.Definition
alias NextLS.DiagnosticCache
alias NextLS.Progress
alias NextLS.Runtime
alias NextLS.SymbolTable

Expand Down Expand Up @@ -280,30 +279,32 @@ defmodule NextLS do

for %{uri: uri, name: name} <- lsp.assigns.workspace_folders do
token = token()
progress_start(lsp, token, "Initializing NextLS runtime for folder #{name}...")
Progress.start(lsp, token, "Initializing NextLS runtime for folder #{name}...")
parent = self()

{:ok, runtime} =
DynamicSupervisor.start_child(
lsp.assigns.dynamic_supervisor,
{NextLS.Runtime,
name: name,
task_supervisor: lsp.assigns.runtime_task_supervisor,
registry: lsp.assigns.registry,
working_dir: URI.parse(uri).path,
uri: uri,
parent: self(),
on_initialized: fn status ->
if status == :ready do
progress_end(lsp, token, "NextLS runtime for folder #{name} has initialized!")
GenLSP.log(lsp, "[NextLS] Runtime for folder #{name} is ready...")
send(parent, {:runtime_ready, name, self()})
else
progress_end(lsp, token)
GenLSP.error(lsp, "[NextLS] Runtime for folder #{name} failed to initialize")
end
end,
logger: lsp.assigns.logger}
{NextLS.RuntimeSupervisor,
runtime: [
name: name,
task_supervisor: lsp.assigns.runtime_task_supervisor,
registry: lsp.assigns.registry,
working_dir: URI.parse(uri).path,
uri: uri,
parent: parent,
on_initialized: fn status ->
if status == :ready do
Progress.stop(lsp, token, "NextLS runtime for folder #{name} has initialized!")
GenLSP.log(lsp, "[NextLS] Runtime for folder #{name} is ready...")
send(parent, {:runtime_ready, name, self()})
else
Progress.stop(lsp, token)
GenLSP.error(lsp, "[NextLS] Runtime for folder #{name} failed to initialize")
end
end,
logger: lsp.assigns.logger
]}
)

ref = Process.monitor(runtime)
Expand Down Expand Up @@ -335,7 +336,7 @@ defmodule NextLS do
dispatch(lsp.assigns.registry, :runtimes, fn entries ->
for {pid, %{name: name, uri: wuri}} <- entries, String.starts_with?(uri, wuri), into: %{} do
token = token()
progress_start(lsp, token, "Compiling...")
Progress.start(lsp, token, "Compiling...")

task =
Task.Supervisor.async_nolink(lsp.assigns.task_supervisor, fn ->
Expand Down Expand Up @@ -418,7 +419,7 @@ defmodule NextLS do

def handle_info({:runtime_ready, name, runtime_pid}, lsp) do
token = token()
progress_start(lsp, token, "Compiling...")
Progress.start(lsp, token, "Compiling...")

task =
Task.Supervisor.async_nolink(lsp.assigns.task_supervisor, fn ->
Expand All @@ -434,7 +435,7 @@ defmodule NextLS do
Process.demonitor(ref, [:flush])
{{token, msg}, refs} = Map.pop(refs, ref)

progress_end(lsp, token, msg)
Progress.stop(lsp, token, msg)

{:noreply, assign(lsp, refresh_refs: refs)}
end
Expand All @@ -443,7 +444,7 @@ defmodule NextLS do
when is_map_key(refs, ref) do
{{token, _}, refs} = Map.pop(refs, ref)

progress_end(lsp, token)
Progress.stop(lsp, token)

{:noreply, assign(lsp, refresh_refs: refs)}
end
Expand All @@ -462,30 +463,6 @@ defmodule NextLS do
{:noreply, lsp}
end

defp progress_start(lsp, token, msg) do
GenLSP.notify(lsp, %GenLSP.Notifications.DollarProgress{
params: %GenLSP.Structures.ProgressParams{
token: token,
value: %WorkDoneProgressBegin{
kind: "begin",
title: msg
}
}
})
end

defp progress_end(lsp, token, msg \\ nil) do
GenLSP.notify(lsp, %GenLSP.Notifications.DollarProgress{
params: %GenLSP.Structures.ProgressParams{
token: token,
value: %WorkDoneProgressEnd{
kind: "end",
message: msg
}
}
})
end

defp token do
8
|> :crypto.strong_rand_bytes()
Expand Down
26 changes: 26 additions & 0 deletions lib/next_ls/progress.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
defmodule NextLS.Progress do
@moduledoc false
def start(lsp, token, msg) do
GenLSP.notify(lsp, %GenLSP.Notifications.DollarProgress{
params: %GenLSP.Structures.ProgressParams{
token: token,
value: %GenLSP.Structures.WorkDoneProgressBegin{
kind: "begin",
title: msg
}
}
})
end

def stop(lsp, token, msg \\ nil) do
GenLSP.notify(lsp, %GenLSP.Notifications.DollarProgress{
params: %GenLSP.Structures.ProgressParams{
token: token,
value: %GenLSP.Structures.WorkDoneProgressEnd{
kind: "end",
message: msg
}
}
})
end
end
18 changes: 18 additions & 0 deletions lib/next_ls/runtime_supervisor.ex
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
defmodule NextLS.RuntimeSupervisor do
@moduledoc false

use Supervisor

def start_link(init_arg) do
Supervisor.start_link(__MODULE__, init_arg)
end

@impl true
def init(init_arg) do
children = [
{NextLS.Runtime, init_arg[:runtime]}
]

Supervisor.init(children, strategy: :one_for_one)
end
end

0 comments on commit df331dc

Please sign in to comment.