livebook/lib/livebook_web/components/app_components.ex

212 lines
5.8 KiB
Elixir
Raw Normal View History

2024-01-26 12:47:56 +08:00
defmodule LivebookWeb.AppComponents do
2023-02-28 22:08:49 +08:00
use LivebookWeb, :html
alias Livebook.Hubs
@doc """
Renders page placeholder on unauthenticated dead render.
"""
def auth_placeholder(assigns) do
~H"""
<div class="flex justify-center items-center h-screen w-screen">
<img src={~p"/images/logo.png"} height="128" width="128" alt="livebook" class="animate-pulse" />
</div>
"""
end
2023-02-28 22:08:49 +08:00
@doc """
Renders app status with indicator.
"""
attr :status, :map, required: true
attr :show_label, :boolean, default: true
2023-02-28 22:08:49 +08:00
def app_status(%{status: %{lifecycle: :shutting_down}} = assigns) do
2023-02-28 22:08:49 +08:00
~H"""
<.app_status_indicator text={@show_label && "Shutting down"} variant={:inactive} />
2023-02-28 22:08:49 +08:00
"""
end
def app_status(%{status: %{lifecycle: :deactivated}} = assigns) do
2023-02-28 22:08:49 +08:00
~H"""
<.app_status_indicator text={@show_label && "Deactivated"} variant={:inactive} />
2023-02-28 22:08:49 +08:00
"""
end
def app_status(%{status: %{execution: :executing}} = assigns) do
2023-02-28 22:08:49 +08:00
~H"""
<.app_status_indicator text={@show_label && "Executing"} variant={:progressing} />
2023-02-28 22:08:49 +08:00
"""
end
def app_status(%{status: %{execution: :executed}} = assigns) do
2023-02-28 22:08:49 +08:00
~H"""
<.app_status_indicator text={@show_label && "Executed"} variant={:success} />
2023-02-28 22:08:49 +08:00
"""
end
def app_status(%{status: %{execution: :error}} = assigns) do
2023-02-28 22:08:49 +08:00
~H"""
<.app_status_indicator text={@show_label && "Error"} variant={:error} />
"""
end
def app_status(%{status: %{execution: :interrupted}} = assigns) do
~H"""
<.app_status_indicator text={@show_label && "Interrupted"} variant={:waiting} />
2023-02-28 22:08:49 +08:00
"""
end
defp app_status_indicator(assigns) do
~H"""
<span class="flex items-center space-x-2">
2023-02-28 22:08:49 +08:00
<.status_indicator variant={@variant} />
<span :if={@text}><%= @text %></span>
</span>
2023-02-28 22:08:49 +08:00
"""
end
@doc """
Shows a confirmation modal and closes the app on confirm.
"""
def confirm_app_termination(socket, app_pid, title \\ "app") do
on_confirm = fn socket ->
Livebook.App.close(app_pid)
socket
end
confirm(socket, on_confirm,
title: "Terminate #{title}",
description: "All #{title} sessions will be immediately terminated.",
confirm_text: "Terminate",
confirm_icon: "delete-bin-6-line"
)
end
@doc """
Renders form fields for Deployment Group.
"""
attr :form, Phoenix.HTML.Form, required: true
attr :hub, :map, required: true
attr :disabled, :boolean, default: false
def deployment_group_form_content(assigns) do
~H"""
<div class="grid grid-cols-1 md:grid-cols-2 gap-3">
2024-04-24 02:40:18 +08:00
<div class="flex flex-col">
<.select_field
label="Clustering"
help={
~S'''
When running multiple
instances of Livebook,
they need to be connected
into a single cluster.
You must either deploy
it as a single instance
or choose a platform to
enable clustering on.
'''
}
field={@form[:clustering]}
options={[
{"Single instance", ""},
{"Fly.io", "fly_io"},
{"DNS", "dns"}
]}
disabled={@disabled}
/>
<div :if={to_string(@form[:clustering].value) == "dns"} class="text-sm mt-1">
See the
<a
class="text-blue-800 hover:text-blue-600"
href="https://hexdocs.pm/livebook/docker.html#clustering"
>
Clustering docs
</a>
for more information.
</div>
</div>
</div>
2024-04-24 02:40:18 +08:00
<%= if Hubs.Provider.type(@hub) == "team" do %>
<div class="flex flex-col">
<div class="grid grid-cols-1 md:grid-cols-2 gap-3">
<.select_field
label="Zero Trust Authentication provider"
field={@form[:zta_provider]}
help={
~S'''
Enable this option if you want
to deploy your notebooks behind
an authentication proxy
'''
}
prompt="None"
options={zta_options()}
disabled={@disabled}
/>
2024-04-24 02:40:18 +08:00
<.text_field
:if={zta_metadata = zta_metadata(@form[:zta_provider].value)}
field={@form[:zta_key]}
2024-04-13 15:07:33 +08:00
type={Map.get(zta_metadata, :input, "text")}
label={zta_metadata.name}
placeholder={Map.get(zta_metadata, :placeholder, "")}
phx-debounce
disabled={@disabled}
/>
</div>
2024-04-24 02:40:18 +08:00
<div :if={zta_metadata = zta_metadata(@form[:zta_provider].value)} class="text-sm mt-1">
See the
<a
class="text-blue-800 hover:text-blue-600"
2024-03-07 03:36:27 +08:00
href={"https://hexdocs.pm/livebook/#{zta_metadata.type}.html"}
>
Authentication with <%= zta_metadata.name %> docs
</a>
for more information.
</div>
</div>
<% end %>
"""
end
@zta_options for provider <- Livebook.Config.identity_providers(),
do: {provider.name, provider.type}
defp zta_options(), do: @zta_options
defp zta_metadata(nil), do: nil
defp zta_metadata(provider) do
Livebook.Config.zta_metadata(provider)
end
@doc """
Lists all docker tag options.
"""
@docker_tag_options for image <- Livebook.Config.docker_images(), do: {image.tag, image.name}
def docker_tag_options(), do: @docker_tag_options
@doc """
Updates app list with the given apps event.
"""
def update_app_list(apps, event)
def update_app_list(apps, {:app_created, app}) do
if app in apps, do: apps, else: [app | apps]
end
def update_app_list(apps, {:app_updated, app}) do
Enum.map(apps, fn other ->
if other.slug == app.slug, do: app, else: other
end)
end
def update_app_list(apps, {:app_closed, app}) do
Enum.reject(apps, &(&1.slug == app.slug))
end
2023-02-28 22:08:49 +08:00
end