defmodule LivebookWeb.Hub.New.EnterpriseComponent do use LivebookWeb, :live_component import Ecto.Changeset, only: [get_field: 2] alias Livebook.EctoTypes.HexColor alias Livebook.Hubs.{Enterprise, EnterpriseClient} @impl true def update(assigns, socket) do {:ok, socket |> assign(assigns) |> assign( base: %Enterprise{}, changeset: Enterprise.change_hub(%Enterprise{}), connected: false )} end @impl true def render(assigns) do ~H"""
<.form :let={f} id={@id} class="flex flex-col space-y-4" for={@changeset} phx-submit="save" phx-change="validate" phx-target={@myself} phx-debounce="blur" >
<.input_wrapper form={f} field={:url} class="flex flex-col space-y-1">
URL
<%= text_input(f, :url, class: "input w-full phx-form-error:border-red-300", autofocus: true, spellcheck: "false", autocomplete: "off" ) %> <.input_wrapper form={f} field={:token} class="flex flex-col space-y-1">
Token
<%= password_input(f, :token, value: get_field(@changeset, :token), class: "input w-full phx-form-error:border-red-300", spellcheck: "false", autocomplete: "off" ) %>
<%= if @connected do %>
<.input_wrapper form={f} field={:external_id} class="flex flex-col space-y-1">
ID
<%= text_input(f, :external_id, class: "input", disabled: true) %>
<.input_wrapper form={f} field={:hub_name} class="flex flex-col space-y-1">
Name
<%= text_input(f, :hub_name, class: "input", readonly: true) %> <.input_wrapper form={f} field={:hub_color} class="flex flex-col space-y-1">
Color
<.hex_color_input form={f} field={:hub_color} randomize={JS.push("randomize_color", target: @myself)} />
<%= submit("Add Hub", class: "button-base button-blue", phx_disable_with: "Add...", disabled: not @changeset.valid? ) %> <% end %>
""" end @impl true def handle_event("connect", _params, socket) do url = get_field(socket.assigns.changeset, :url) token = get_field(socket.assigns.changeset, :token) with {:ok, _info} <- EnterpriseClient.fetch_info(url, token), {:ok, %{"id" => id}} <- EnterpriseClient.fetch_me(url, token) do base = %Enterprise{ token: token, url: url, external_id: id, hub_name: "Enterprise", hub_color: HexColor.random() } changeset = Enterprise.change_hub(base) {:noreply, assign(socket, changeset: changeset, base: base, connected: true)} else {:error, _message, reason} -> {:noreply, socket |> put_flash(:error, message_from_reason(reason)) |> push_patch(to: Routes.hub_path(socket, :new))} end end def handle_event("randomize_color", _, socket) do handle_event("validate", %{"enterprise" => %{"hub_color" => HexColor.random()}}, socket) end def handle_event("save", %{"enterprise" => params}, socket) do if socket.assigns.changeset.valid? do case Enterprise.create_hub(socket.assigns.base, params) do {:ok, hub} -> {:noreply, socket |> put_flash(:success, "Hub added successfully") |> push_redirect(to: Routes.hub_path(socket, :edit, hub.id))} {:error, changeset} -> {:noreply, assign(socket, changeset: changeset)} end else {:noreply, socket} end end def handle_event("validate", %{"enterprise" => attrs}, socket) do {:noreply, assign(socket, changeset: Enterprise.change_hub(socket.assigns.base, attrs))} end defp message_from_reason(:invalid_url), do: "Failed to connect with given URL" defp message_from_reason(:unauthorized), do: "Failed to authorize with given token" defp message_from_reason(:invalid_token), do: "Failed to authenticate with given token" end