2023-02-16 20:47:46 +08:00
|
|
|
defmodule LivebookWeb.AppLive do
|
|
|
|
use LivebookWeb, :live_view
|
|
|
|
|
|
|
|
alias Livebook.Session
|
|
|
|
alias Livebook.Notebook
|
|
|
|
alias Livebook.Notebook.Cell
|
|
|
|
|
|
|
|
@impl true
|
2023-02-18 08:16:42 +08:00
|
|
|
def mount(%{"slug" => slug}, _session, socket) when socket.assigns.app_authenticated? do
|
|
|
|
{:ok, %{pid: session_pid, id: session_id}} = Livebook.Apps.fetch_session_by_slug(slug)
|
|
|
|
|
|
|
|
{data, client_id} =
|
|
|
|
if connected?(socket) do
|
2023-02-16 20:47:46 +08:00
|
|
|
{data, client_id} =
|
2023-02-18 08:16:42 +08:00
|
|
|
Session.register_client(session_pid, self(), socket.assigns.current_user)
|
2023-02-16 20:47:46 +08:00
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
Session.subscribe(session_id)
|
2023-02-16 20:47:46 +08:00
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
{data, client_id}
|
|
|
|
else
|
|
|
|
data = Session.get_data(session_pid)
|
|
|
|
{data, nil}
|
|
|
|
end
|
2023-02-16 20:47:46 +08:00
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
session = Session.get_by_pid(session_pid)
|
2023-02-16 20:47:46 +08:00
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
{:ok,
|
|
|
|
socket
|
|
|
|
|> assign(
|
|
|
|
slug: slug,
|
|
|
|
session: session,
|
|
|
|
page_title: get_page_title(data.notebook.name),
|
|
|
|
client_id: client_id,
|
|
|
|
data_view: data_to_view(data)
|
|
|
|
)
|
|
|
|
|> assign_private(data: data)}
|
|
|
|
end
|
2023-02-16 20:47:46 +08:00
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
def mount(%{"slug" => slug}, _session, socket) do
|
|
|
|
if connected?(socket) do
|
2023-02-23 02:34:54 +08:00
|
|
|
{:ok, push_navigate(socket, to: ~p"/apps/#{slug}/authenticate")}
|
2023-02-18 08:16:42 +08:00
|
|
|
else
|
|
|
|
{:ok, socket}
|
2023-02-16 20:47:46 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Puts the given assigns in `socket.private`,
|
|
|
|
# to ensure they are not used for rendering.
|
|
|
|
defp assign_private(socket, assigns) do
|
|
|
|
Enum.reduce(assigns, socket, fn {key, value}, socket ->
|
|
|
|
put_in(socket.private[key], value)
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
@impl true
|
2023-02-18 08:16:42 +08:00
|
|
|
def render(assigns) when assigns.app_authenticated? do
|
2023-02-16 20:47:46 +08:00
|
|
|
~H"""
|
2023-03-24 22:22:50 +08:00
|
|
|
<div class="h-full relative overflow-y-auto px-4 md:px-20" data-el-notebook>
|
|
|
|
<div class="w-full max-w-screen-lg py-4 mx-auto" data-el-notebook-content>
|
|
|
|
<div class="absolute md:fixed right-8 md:left-4 top-3 w-10 h-10">
|
|
|
|
<.menu id="app-menu" position={:bottom_left}>
|
|
|
|
<:toggle>
|
|
|
|
<button class="flex items-center text-gray-900">
|
|
|
|
<img src={~p"/images/logo.png"} height="40" width="40" alt="logo livebook" />
|
|
|
|
<.remix_icon icon="arrow-down-s-line" />
|
|
|
|
</button>
|
|
|
|
</:toggle>
|
|
|
|
<.menu_item>
|
|
|
|
<.link navigate={~p"/"} role="menuitem">
|
|
|
|
<.remix_icon icon="home-6-line" />
|
|
|
|
<span>Home</span>
|
|
|
|
</.link>
|
|
|
|
</.menu_item>
|
|
|
|
<.menu_item :if={@data_view.show_source}>
|
|
|
|
<.link patch={~p"/apps/#{@data_view.slug}/source"} role="menuitem">
|
|
|
|
<.remix_icon icon="code-line" />
|
|
|
|
<span>View source</span>
|
|
|
|
</.link>
|
|
|
|
</.menu_item>
|
|
|
|
</.menu>
|
|
|
|
</div>
|
|
|
|
<div data-el-js-view-iframes phx-update="ignore" id="js-view-iframes"></div>
|
|
|
|
<div class="flex items-center pb-4 mb-2 space-x-4 border-b border-gray-200 pr-20 md:pr-0">
|
|
|
|
<h1 class="text-3xl font-semibold text-gray-800">
|
|
|
|
<%= @data_view.notebook_name %>
|
|
|
|
</h1>
|
|
|
|
</div>
|
|
|
|
<div :if={@data_view.app_status == :booting} class="flex items-center space-x-2">
|
|
|
|
<span class="relative flex h-3 w-3">
|
|
|
|
<span class="animate-ping absolute inline-flex h-full w-full rounded-full bg-blue-400 opacity-75">
|
2023-02-16 20:47:46 +08:00
|
|
|
</span>
|
2023-03-24 22:22:50 +08:00
|
|
|
<span class="relative inline-flex rounded-full h-3 w-3 bg-blue-500"></span>
|
|
|
|
</span>
|
|
|
|
<div class="text-gray-700 font-medium">
|
|
|
|
Booting
|
2023-02-16 20:47:46 +08:00
|
|
|
</div>
|
2023-03-24 22:22:50 +08:00
|
|
|
</div>
|
|
|
|
<div :if={@data_view.app_status == :error} class="flex items-center space-x-2">
|
|
|
|
<span class="relative flex h-3 w-3">
|
|
|
|
<span class="relative inline-flex rounded-full h-3 w-3 bg-red-500"></span>
|
|
|
|
</span>
|
|
|
|
<div class="text-gray-700 font-medium">
|
|
|
|
Error
|
2023-02-16 20:47:46 +08:00
|
|
|
</div>
|
2023-03-24 22:22:50 +08:00
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
:if={@data_view.app_status in [:running, :shutting_down]}
|
|
|
|
class="pt-4 flex flex-col space-y-6"
|
|
|
|
data-el-outputs-container
|
|
|
|
id="outputs"
|
|
|
|
>
|
|
|
|
<div :for={output_view <- Enum.reverse(@data_view.output_views)}>
|
|
|
|
<LivebookWeb.Output.outputs
|
|
|
|
outputs={[output_view.output]}
|
|
|
|
dom_id_map={%{}}
|
|
|
|
session_id={@session.id}
|
|
|
|
session_pid={@session.pid}
|
|
|
|
client_id={@client_id}
|
|
|
|
input_values={output_view.input_values}
|
|
|
|
/>
|
2023-02-16 20:47:46 +08:00
|
|
|
</div>
|
2023-05-08 17:26:21 +08:00
|
|
|
<div :if={@data_view.output_views == []} class="info-box">
|
|
|
|
This deployed notebook is empty. Deployed apps only render Kino outputs.
|
|
|
|
Ensure you use Kino for interactive visualizations and dynamic content.
|
|
|
|
</div>
|
2023-02-23 02:34:54 +08:00
|
|
|
</div>
|
2023-03-24 22:22:50 +08:00
|
|
|
<div style="height: 80vh"></div>
|
2023-02-16 20:47:46 +08:00
|
|
|
</div>
|
|
|
|
</div>
|
2023-02-28 00:05:36 +08:00
|
|
|
|
|
|
|
<.modal
|
|
|
|
:if={@live_action == :source and @data_view.show_source}
|
|
|
|
id="source-modal"
|
|
|
|
show
|
2023-03-08 05:19:16 +08:00
|
|
|
width={:big}
|
2023-02-28 00:05:36 +08:00
|
|
|
patch={~p"/apps/#{@data_view.slug}"}
|
|
|
|
>
|
|
|
|
<.live_component module={LivebookWeb.AppLive.SourceComponent} id="source" session={@session} />
|
|
|
|
</.modal>
|
2023-02-16 20:47:46 +08:00
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
def render(assigns) do
|
|
|
|
~H"""
|
|
|
|
<div class="flex justify-center items-center h-screen w-screen">
|
2023-02-23 02:34:54 +08:00
|
|
|
<img src={~p"/images/logo.png"} height="128" width="128" alt="livebook" class="animate-pulse" />
|
2023-02-18 08:16:42 +08:00
|
|
|
</div>
|
|
|
|
"""
|
|
|
|
end
|
|
|
|
|
2023-02-16 20:47:46 +08:00
|
|
|
defp get_page_title(notebook_name) do
|
|
|
|
"Livebook - #{notebook_name}"
|
|
|
|
end
|
|
|
|
|
2023-02-28 00:05:36 +08:00
|
|
|
@impl true
|
|
|
|
def handle_params(_params, _url, socket), do: {:noreply, socket}
|
|
|
|
|
2023-02-16 20:47:46 +08:00
|
|
|
@impl true
|
|
|
|
def handle_info({:operation, operation}, socket) do
|
|
|
|
{:noreply, handle_operation(socket, operation)}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info({:set_input_values, values, _local = true}, socket) do
|
|
|
|
socket =
|
|
|
|
Enum.reduce(values, socket, fn {input_id, value}, socket ->
|
|
|
|
operation = {:set_input_value, socket.assigns.client_id, input_id, value}
|
|
|
|
handle_operation(socket, operation)
|
|
|
|
end)
|
|
|
|
|
|
|
|
{:noreply, socket}
|
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info(:session_closed, socket) do
|
|
|
|
{:noreply,
|
|
|
|
socket
|
|
|
|
|> put_flash(:info, "Session has been closed")
|
2023-02-23 02:34:54 +08:00
|
|
|
|> push_navigate(to: ~p"/")}
|
2023-02-16 20:47:46 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def handle_info(_message, socket), do: {:noreply, socket}
|
|
|
|
|
|
|
|
defp handle_operation(socket, operation) do
|
|
|
|
case Session.Data.apply_operation(socket.private.data, operation) do
|
|
|
|
{:ok, data, _actions} ->
|
|
|
|
socket
|
|
|
|
|> assign_private(data: data)
|
|
|
|
|> assign(
|
|
|
|
data_view:
|
|
|
|
update_data_view(socket.assigns.data_view, socket.private.data, data, operation)
|
|
|
|
)
|
|
|
|
|
|
|
|
:error ->
|
|
|
|
socket
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp update_data_view(data_view, _prev_data, data, operation) do
|
|
|
|
case operation do
|
|
|
|
# See LivebookWeb.SessionLive for more details
|
|
|
|
{:add_cell_evaluation_output, _client_id, _cell_id,
|
|
|
|
{:frame, _outputs, %{type: type, ref: ref}}}
|
|
|
|
when type != :default ->
|
|
|
|
for {idx, {:frame, frame_outputs, _}} <- Notebook.find_frame_outputs(data.notebook, ref) do
|
|
|
|
send_update(LivebookWeb.Output.FrameComponent,
|
|
|
|
id: "output-#{idx}",
|
|
|
|
outputs: frame_outputs,
|
|
|
|
update_type: type
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
data_view
|
|
|
|
|
|
|
|
_ ->
|
|
|
|
data_to_view(data)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp data_to_view(data) do
|
|
|
|
%{
|
|
|
|
notebook_name: data.notebook.name,
|
|
|
|
output_views:
|
|
|
|
for(
|
|
|
|
output <- visible_outputs(data.notebook),
|
|
|
|
do: %{
|
|
|
|
output: output,
|
|
|
|
input_values: input_values_for_output(output, data)
|
|
|
|
}
|
|
|
|
),
|
2023-02-28 00:05:36 +08:00
|
|
|
app_status: data.app_data.status,
|
|
|
|
show_source: data.notebook.app_settings.show_source,
|
|
|
|
slug: data.notebook.app_settings.slug
|
2023-02-16 20:47:46 +08:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp input_values_for_output(output, data) do
|
|
|
|
input_ids = for attrs <- Cell.find_inputs_in_output(output), do: attrs.id
|
|
|
|
Map.take(data.input_values, input_ids)
|
|
|
|
end
|
|
|
|
|
|
|
|
defp visible_outputs(notebook) do
|
|
|
|
for section <- Enum.reverse(notebook.sections),
|
|
|
|
cell <- Enum.reverse(section.cells),
|
|
|
|
Cell.evaluable?(cell),
|
|
|
|
output <- filter_outputs(cell.outputs),
|
|
|
|
do: output
|
|
|
|
end
|
|
|
|
|
|
|
|
defp filter_outputs(outputs) do
|
|
|
|
for output <- outputs, output = filter_output(output), do: output
|
|
|
|
end
|
|
|
|
|
|
|
|
defp filter_output({idx, output})
|
2023-03-16 00:48:38 +08:00
|
|
|
when elem(output, 0) in [:plain_text, :markdown, :image, :js, :control],
|
2023-02-16 20:47:46 +08:00
|
|
|
do: {idx, output}
|
|
|
|
|
|
|
|
defp filter_output({idx, {:tabs, outputs, metadata}}) do
|
|
|
|
outputs_with_labels =
|
|
|
|
for {output, label} <- Enum.zip(outputs, metadata.labels),
|
|
|
|
output = filter_output(output),
|
|
|
|
do: {output, label}
|
|
|
|
|
|
|
|
{outputs, labels} = Enum.unzip(outputs_with_labels)
|
|
|
|
|
|
|
|
{idx, {:tabs, outputs, %{metadata | labels: labels}}}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp filter_output({idx, {:grid, outputs, metadata}}) do
|
|
|
|
outputs = filter_outputs(outputs)
|
|
|
|
|
|
|
|
if outputs != [] do
|
|
|
|
{idx, {:grid, outputs, metadata}}
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
defp filter_output({idx, {:frame, outputs, metadata}}) do
|
|
|
|
outputs = filter_outputs(outputs)
|
|
|
|
{idx, {:frame, outputs, metadata}}
|
|
|
|
end
|
|
|
|
|
|
|
|
defp filter_output(_output), do: nil
|
|
|
|
end
|