livebook/lib/livebook/hubs/personal.ex
2023-08-01 20:48:55 +02:00

210 lines
5.1 KiB
Elixir

defmodule Livebook.Hubs.Personal do
@moduledoc false
use Ecto.Schema
import Ecto.Changeset
alias Livebook.Hubs
alias Livebook.Storage
alias Livebook.Secrets.Secret
@secrets_namespace :hub_secrets
@secret_key_size 64
@type t :: %__MODULE__{
id: String.t() | nil,
hub_name: String.t() | nil,
hub_emoji: String.t() | nil,
secret_key: String.t() | nil
}
embedded_schema do
field :hub_name, :string
field :hub_emoji, :string
field :secret_key, :string, redact: true
end
@fields ~w(hub_name hub_emoji secret_key)a
@doc """
The personal hub fixed id.
"""
@spec id() :: String.t()
def id, do: "personal-hub"
@doc """
Returns an `%Ecto.Changeset{}` for tracking hub changes.
"""
@spec change_hub(t(), map()) :: Ecto.Changeset.t()
def change_hub(%__MODULE__{} = personal, attrs \\ %{}) do
changeset(personal, attrs)
end
@doc """
Returns changeset with applied validations.
"""
@spec validate_hub(t(), map()) :: Ecto.Changeset.t()
def validate_hub(%__MODULE__{} = personal, attrs \\ %{}) do
personal
|> changeset(attrs)
|> Map.put(:action, :validate)
end
@doc """
Updates a Hub.
With success, notifies interested processes about hub metadatas data change.
Otherwise, it will return an error tuple with changeset.
"""
@spec update_hub(t(), map()) :: {:ok, t()} | {:error, Ecto.Changeset.t()}
def update_hub(%__MODULE__{} = personal, attrs) do
changeset = changeset(personal, attrs)
with {:ok, struct} <- apply_action(changeset, :update) do
Hubs.save_hub(struct)
{:ok, struct}
end
end
defp changeset(personal, attrs) do
personal
|> cast(attrs, @fields)
|> validate_required(@fields)
|> validate_change(:secret_key, fn :secret_key, secret_key ->
case Base.url_decode64(secret_key, padding: false) do
{:ok, binary} when byte_size(binary) == @secret_key_size -> []
_ -> [secret_key: "must be #{@secret_key_size} bytes in Base 64 URL alphabet"]
end
end)
|> put_change(:id, id())
end
@doc """
Get the secrets list from storage.
"""
@spec get_secrets :: [Secret.t()]
def get_secrets do
Enum.map(Storage.all(@secrets_namespace), &to_secret/1)
end
@doc """
Gets a secret from storage.
Raises `RuntimeError` if the secret doesn't exist.
"""
@spec fetch_secret!(String.t()) :: Secret.t()
def fetch_secret!(id) do
Storage.fetch!(@secrets_namespace, id) |> to_secret()
end
@doc """
Stores the given secret as is, without validation.
"""
@spec set_secret(Secret.t()) :: Secret.t()
def set_secret(secret) do
attributes = Map.from_struct(secret)
:ok = Storage.insert(@secrets_namespace, secret.name, Map.to_list(attributes))
secret
end
@doc """
Unset secret from given id.
"""
@spec set_secret(String.t()) :: :ok
def unset_secret(id) do
Storage.delete(@secrets_namespace, id)
:ok
end
defp to_secret(%{name: name, value: value}) do
%Secret{
name: name,
value: value,
hub_id: Livebook.Hubs.Personal.id()
}
end
@doc """
Generates a random secret key used for stamping the notebook.
"""
@spec generate_secret_key() :: String.t()
def generate_secret_key() do
Base.url_encode64(:crypto.strong_rand_bytes(@secret_key_size), padding: false)
end
end
defimpl Livebook.Hubs.Provider, for: Livebook.Hubs.Personal do
alias Livebook.Hubs.Broadcasts
alias Livebook.Hubs.Personal
def load(personal, fields) do
%{
personal
| id: fields.id,
hub_name: fields.hub_name,
hub_emoji: fields.hub_emoji,
secret_key: fields.secret_key
}
end
def to_metadata(personal) do
%Livebook.Hubs.Metadata{
id: personal.id,
name: personal.hub_name,
provider: personal,
emoji: personal.hub_emoji,
connected?: false
}
end
def type(_personal), do: "personal"
def connection_spec(_personal), do: nil
def disconnect(_personal), do: raise("not implemented")
def capabilities(_personal), do: ~w(list_secrets create_secret)a
def get_secrets(_personal) do
Personal.get_secrets()
end
def create_secret(_personal, secret) do
Personal.set_secret(secret)
:ok = Broadcasts.secret_created(secret)
end
def update_secret(_personal, secret) do
Personal.set_secret(secret)
:ok = Broadcasts.secret_updated(secret)
end
def delete_secret(_personal, secret) do
:ok = Personal.unset_secret(secret.name)
:ok = Broadcasts.secret_deleted(secret)
end
def connection_error(_personal), do: raise("not implemented")
def notebook_stamp(_hub, _notebook_source, metadata) when metadata == %{} do
:skip
end
def notebook_stamp(personal, notebook_source, metadata) do
token = Livebook.Stamping.aead_encrypt(metadata, notebook_source, personal.secret_key)
stamp = %{"version" => 1, "token" => token}
{:ok, stamp}
end
def verify_notebook_stamp(personal, notebook_source, stamp) do
%{"version" => 1, "token" => token} = stamp
Livebook.Stamping.aead_decrypt(token, notebook_source, personal.secret_key)
end
def dump(personal) do
Map.from_struct(personal)
end
end