livebook/test/support/integration/enterprise_server.ex

226 lines
5 KiB
Elixir
Raw Normal View History

2022-10-11 22:27:27 +08:00
defmodule Livebook.EnterpriseServer do
@moduledoc false
use GenServer
defstruct [:token, :user, :node, :port]
2022-10-11 22:27:27 +08:00
@name __MODULE__
def start do
GenServer.start(__MODULE__, [], name: @name)
end
def url do
"http://localhost:#{app_port()}"
end
def token do
GenServer.call(@name, :fetch_token)
end
def user do
GenServer.call(@name, :fetch_user)
end
2022-10-11 22:27:27 +08:00
# GenServer Callbacks
@impl true
def init(_opts) do
state = %__MODULE__{node: enterprise_node()}
{:ok, state, {:continue, :start_enterprise}}
2022-10-11 22:27:27 +08:00
end
@impl true
def handle_continue(:start_enterprise, state) do
{:noreply, %{state | port: start_enterprise(state)}}
end
@impl true
def handle_call(:fetch_token, _from, state) do
state = if _ = state.token, do: state, else: create_enterprise_token(state)
{:reply, state.token, state}
end
@impl true
def handle_call(:fetch_user, _from, state) do
state = if _ = state.user, do: state, else: create_enterprise_user(state)
2022-10-11 22:27:27 +08:00
{:reply, state.user, state}
2022-10-11 22:27:27 +08:00
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
error("enterprise quit with status #{status}")
System.halt(status)
end
# Private
defp create_enterprise_token(state) do
if user = state.user do
token = call_erpc_function(state.node, :generate_user_session_token!, [user])
%{state | token: token}
else
user = call_erpc_function(state.node, :create_user)
token = call_erpc_function(state.node, :generate_user_session_token!, [user])
%{state | user: user, token: token}
end
end
defp create_enterprise_user(state) do
%{state | user: call_erpc_function(state.node, :create_user)}
end
defp call_erpc_function(node, function, args \\ []) do
:erpc.call(node, Enterprise.Integration, function, args)
2022-10-11 22:27:27 +08:00
end
defp start_enterprise(state) do
ensure_app_dir!()
prepare_database()
env = [
{~c"MIX_ENV", ~c"livebook"},
{~c"LIVEBOOK_ENTERPRISE_PORT", String.to_charlist(app_port())},
{~c"LIVEBOOK_ENTERPRISE_DEBUG", String.to_charlist(debug?())}
2022-10-11 22:27:27 +08:00
]
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(port)
end
defp prepare_database do
mix(["ecto.drop", "--quiet"])
mix(["ecto.create", "--quiet"])
mix(["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 enterprise repository " <>
"into it to run integration tests or set ENTERPRISE_PATH to its location"
)
System.halt(1)
end
end
defp app_dir do
System.get_env("ENTERPRISE_PATH", "../enterprise")
end
defp app_port do
System.get_env("ENTERPRISE_PORT", "4043")
end
defp debug? do
System.get_env("ENTERPRISE_DEBUG", "false")
end
2022-10-11 22:27:27 +08:00
defp wait_on_start(port) do
case :httpc.request(:get, {~c"#{url()}/public/health", []}, [], []) do
{:ok, _} ->
port
{:error, _} ->
Process.sleep(10)
wait_on_start(port)
end
end
defp mix(args, opts \\ []) do
env = [
{"MIX_ENV", "livebook"},
{"LIVEBOOK_ENTERPRISE_PORT", app_port()},
{"LIVEBOOK_ENTERPRISE_DEBUG", debug?()}
2022-10-11 22:27:27 +08:00
]
cmd_opts = [stderr_to_stdout: true, env: env, cd: app_dir()]
args = ["--erl", "-elixir ansi_enabled true", "-S", "mix" | args]
cmd_opts =
if opts[:with_return],
do: cmd_opts,
else: Keyword.put(cmd_opts, :into, IO.stream(:stdio, :line))
if opts[:with_return] do
case System.cmd(elixir_executable(), args, cmd_opts) do
{result, 0} ->
result
{message, status} ->
error("""
#{message}\
""")
System.halt(status)
end
else
{_, 0} = System.cmd(elixir_executable(), args, cmd_opts)
end
end
defp elixir_executable do
System.find_executable("elixir")
end
defp enterprise_node do
:"enterprise_#{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