2023-05-17 00:21:49 +08:00
|
|
|
defmodule Livebook.TeamsServer do
|
|
|
|
@moduledoc false
|
|
|
|
use GenServer
|
|
|
|
|
|
|
|
defstruct [:node, :token, :user, :org, :teams_key, :port, :app_port, :url, :env]
|
|
|
|
|
|
|
|
@name __MODULE__
|
|
|
|
@timeout 10_000
|
|
|
|
@default_teams_dir "../hub"
|
|
|
|
|
|
|
|
def available?() do
|
|
|
|
System.get_env("TEAMS_PATH") != nil or File.exists?(@default_teams_dir)
|
|
|
|
end
|
|
|
|
|
|
|
|
def start(opts \\ []) do
|
|
|
|
GenServer.start(__MODULE__, opts, name: @name)
|
|
|
|
end
|
|
|
|
|
|
|
|
def url() do
|
|
|
|
GenServer.call(@name, :fetch_url, @timeout)
|
|
|
|
end
|
|
|
|
|
|
|
|
def token() do
|
|
|
|
GenServer.call(@name, :fetch_token, @timeout)
|
|
|
|
end
|
|
|
|
|
|
|
|
def user() do
|
|
|
|
GenServer.call(@name, :fetch_user, @timeout)
|
|
|
|
end
|
|
|
|
|
|
|
|
def get_node() do
|
|
|
|
GenServer.call(@name, :fetch_node, @timeout)
|
|
|
|
end
|
|
|
|
|
|
|
|
def drop_database() do
|
|
|
|
app_port = GenServer.call(@name, :fetch_port)
|
|
|
|
state_env = GenServer.call(@name, :fetch_env)
|
|
|
|
|
|
|
|
app_port |> env(state_env) |> mix(["ecto.drop", "--quiet"])
|
|
|
|
end
|
|
|
|
|
|
|
|
# GenServer Callbacks
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def init(opts) do
|
|
|
|
state = struct!(__MODULE__, opts)
|
|
|
|
{:ok, %{state | node: app_node()}, {:continue, :start_app}}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_continue(:start_app, state) do
|
|
|
|
ensure_app_dir!()
|
|
|
|
prepare_database(state)
|
|
|
|
|
|
|
|
{:noreply, %{state | port: start_app(state)}}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_call(:fetch_token, _from, state) do
|
|
|
|
state = if state.token, do: state, else: ensure_session_token(state)
|
|
|
|
|
|
|
|
{:reply, state.token, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_call(:fetch_user, _from, state) do
|
|
|
|
state = if state.user, do: state, else: ensure_user(state)
|
|
|
|
|
|
|
|
{:reply, state.user, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_call(:fetch_url, _from, state) do
|
|
|
|
state = if state.app_port, do: state, else: %{state | app_port: app_port()}
|
|
|
|
url = state.url || fetch_url(state)
|
|
|
|
|
|
|
|
{:reply, url, %{state | url: url}}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_call(:fetch_node, _from, state) do
|
|
|
|
{:reply, state.node, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_call(:fetch_port, _from, state) do
|
|
|
|
app_port = state.app_port || app_port()
|
|
|
|
|
|
|
|
{:reply, app_port, %{state | app_port: app_port}}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_call(:fetch_env, _from, state) do
|
|
|
|
{:reply, state.env, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
# Port Callbacks
|
|
|
|
|
|
|
|
@impl true
|
|
|
|
def handle_info({_port, {:data, message}}, state) do
|
|
|
|
info(message)
|
|
|
|
{:noreply, state}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({_port, {:exit_status, status}}, _state) do
|
2023-06-01 23:01:43 +08:00
|
|
|
error("team quit with status #{status}")
|
2023-05-17 00:21:49 +08:00
|
|
|
System.halt(status)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Private
|
|
|
|
|
|
|
|
defp call_erpc_function(node, function, args \\ []) do
|
|
|
|
:erpc.call(node, Hub.Integration, function, args)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp ensure_session_token(state) do
|
|
|
|
state =
|
|
|
|
state
|
|
|
|
|> ensure_user()
|
|
|
|
|> ensure_org()
|
|
|
|
|> ensure_teams_key()
|
|
|
|
|
|
|
|
token = call_erpc_function(state.node, :associate_user_with_org, [state.user, state.org])
|
|
|
|
|
|
|
|
%{state | token: token}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp ensure_user(state) do
|
|
|
|
if state.user,
|
|
|
|
do: state,
|
|
|
|
else: %{state | user: call_erpc_function(state.node, :create_user)}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp ensure_org(state) do
|
|
|
|
if state.org,
|
|
|
|
do: state,
|
|
|
|
else: %{state | org: call_erpc_function(state.node, :create_org)}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp ensure_teams_key(state) do
|
|
|
|
if state.teams_key,
|
|
|
|
do: state,
|
|
|
|
else: %{
|
|
|
|
state
|
|
|
|
| teams_key: call_erpc_function(state.node, :create_org_key, [[org: state.org]]).key_hash
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp start_app(state) do
|
|
|
|
env =
|
|
|
|
for {key, value} <- env(state), into: [] do
|
|
|
|
{String.to_charlist(key), String.to_charlist(value)}
|
|
|
|
end
|
|
|
|
|
|
|
|
args = [
|
|
|
|
"-e",
|
|
|
|
"spawn(fn -> IO.gets([]) && System.halt(0) end)",
|
|
|
|
"--sname",
|
|
|
|
to_string(state.node),
|
|
|
|
"--cookie",
|
|
|
|
to_string(Node.get_cookie()),
|
|
|
|
"-S",
|
|
|
|
"mix",
|
|
|
|
"phx.server"
|
|
|
|
]
|
|
|
|
|
|
|
|
port =
|
|
|
|
Port.open({:spawn_executable, elixir_executable()}, [
|
|
|
|
:exit_status,
|
|
|
|
:use_stdio,
|
|
|
|
:stderr_to_stdout,
|
|
|
|
:binary,
|
|
|
|
:hide,
|
|
|
|
env: env,
|
|
|
|
cd: app_dir(),
|
|
|
|
args: args
|
|
|
|
])
|
|
|
|
|
|
|
|
wait_on_start(state, port)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp fetch_url(state) do
|
|
|
|
port = state.app_port || app_port()
|
|
|
|
"http://localhost:#{port}"
|
|
|
|
end
|
|
|
|
|
|
|
|
defp prepare_database(state) do
|
|
|
|
:ok = mix(state, ["ecto.drop", "--quiet"])
|
|
|
|
:ok = mix(state, ["ecto.create", "--quiet"])
|
|
|
|
:ok = mix(state, ["ecto.migrate", "--quiet"])
|
|
|
|
end
|
|
|
|
|
|
|
|
defp ensure_app_dir! do
|
|
|
|
dir = app_dir()
|
|
|
|
|
|
|
|
unless File.exists?(dir) do
|
|
|
|
IO.puts(
|
|
|
|
"Unable to find #{dir}, make sure to clone the hub repository " <>
|
|
|
|
"into it to run integration tests or set TEAMS_PATH to its location"
|
|
|
|
)
|
|
|
|
|
|
|
|
System.halt(1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp app_dir do
|
|
|
|
System.get_env("TEAMS_PATH", @default_teams_dir)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp app_port do
|
2023-05-30 18:19:51 +08:00
|
|
|
System.get_env("TEAMS_PORT", "4123")
|
2023-05-17 00:21:49 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
defp debug do
|
|
|
|
System.get_env("TEAMS_DEBUG", "false")
|
|
|
|
end
|
|
|
|
|
2023-06-01 23:01:43 +08:00
|
|
|
defp proto do
|
|
|
|
System.get_env("TEAMS_LIVEBOOK_PROTO_PATH")
|
|
|
|
end
|
|
|
|
|
2023-05-17 00:21:49 +08:00
|
|
|
defp wait_on_start(state, port) do
|
|
|
|
url = state.url || fetch_url(state)
|
|
|
|
|
|
|
|
case :httpc.request(:get, {~c"#{url}/public/health", []}, [], []) do
|
|
|
|
{:ok, _} ->
|
|
|
|
port
|
|
|
|
|
|
|
|
{:error, _} ->
|
|
|
|
Process.sleep(10)
|
|
|
|
wait_on_start(state, port)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp mix(state, args) when is_struct(state) do
|
|
|
|
state |> env() |> mix(args)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp mix(env, args) do
|
|
|
|
cmd_opts = [
|
|
|
|
stderr_to_stdout: true,
|
|
|
|
env: env,
|
|
|
|
cd: app_dir(),
|
|
|
|
into: IO.stream(:stdio, :line)
|
|
|
|
]
|
|
|
|
|
|
|
|
args = ["--erl", "-elixir ansi_enabled true", "-S", "mix" | args]
|
|
|
|
|
|
|
|
case System.cmd(elixir_executable(), args, cmd_opts) do
|
|
|
|
{_, 0} -> :ok
|
|
|
|
_ -> :error
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp env(state) do
|
|
|
|
app_port = state.app_port || app_port()
|
|
|
|
env(app_port, state.env)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp env(app_port, state_env) do
|
|
|
|
env = %{
|
|
|
|
"MIX_ENV" => "livebook",
|
|
|
|
"PORT" => to_string(app_port),
|
|
|
|
"DEBUG" => debug()
|
|
|
|
}
|
|
|
|
|
2023-06-01 23:01:43 +08:00
|
|
|
env = if proto(), do: Map.merge(env, %{"LIVEBOOK_PROTO_PATH" => proto()}), else: env
|
|
|
|
|
2023-05-17 00:21:49 +08:00
|
|
|
if state_env do
|
|
|
|
Map.merge(env, state_env)
|
|
|
|
else
|
|
|
|
env
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp elixir_executable do
|
|
|
|
System.find_executable("elixir")
|
|
|
|
end
|
|
|
|
|
|
|
|
defp app_node do
|
|
|
|
:"teams_#{Livebook.Utils.random_short_id()}@#{hostname()}"
|
|
|
|
end
|
|
|
|
|
|
|
|
defp hostname do
|
|
|
|
[nodename, hostname] =
|
|
|
|
node()
|
|
|
|
|> Atom.to_charlist()
|
|
|
|
|> :string.split(~c"@")
|
|
|
|
|
|
|
|
with {:ok, nodenames} <- :erl_epmd.names(hostname),
|
|
|
|
true <- List.keymember?(nodenames, nodename, 0) do
|
|
|
|
hostname
|
|
|
|
else
|
|
|
|
_ ->
|
|
|
|
raise "Error"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp info(message), do: log([:blue, message <> "\n"])
|
|
|
|
defp error(message), do: log([:red, message <> "\n"])
|
|
|
|
defp log(data), do: data |> IO.ANSI.format() |> IO.write()
|
|
|
|
end
|