2023-02-18 08:16:42 +08:00
|
|
|
defmodule LivebookWeb.AppAuthHook do
|
|
|
|
import Phoenix.Component
|
|
|
|
import Phoenix.LiveView
|
|
|
|
|
2023-02-23 02:34:54 +08:00
|
|
|
use LivebookWeb, :verified_routes
|
2023-02-18 08:16:42 +08:00
|
|
|
|
|
|
|
# For apps with password, we want to store the hashed password
|
|
|
|
# (let's call it token) in the session, as we do for the main auth.
|
|
|
|
# However, the session uses cookies, which have a ~4kb size limit.
|
|
|
|
# We could use multiple cookies, however there are also limits on
|
|
|
|
# the number of cookies, and we don't want the browser to clear
|
|
|
|
# them all at some point. Additionally, accumulating cookies for
|
|
|
|
# all apps would imply larger payloads on every regular request.
|
|
|
|
#
|
|
|
|
# Since we don't have any persistence on the server side, the other
|
|
|
|
# option is to use the browser local storage and manage it through
|
|
|
|
# JavaScript. Therefore, the whole auth is built using LiveView.
|
|
|
|
# The flows are:
|
|
|
|
#
|
|
|
|
# * unauthenticated - the auth LiveView shows a regular form and
|
|
|
|
# validates the user-provided password. Once the password is
|
|
|
|
# correct, it pushes an event to the client to store the token.
|
|
|
|
# Once the token is stored it redirects to the app page.
|
|
|
|
#
|
|
|
|
# * authenticated - on dead render the app LiveView renders just
|
|
|
|
# a loading screen. On the client side, provided it's the app
|
|
|
|
# page, we read the token from local storage (if stored) and
|
|
|
|
# send it in mount connect params via the socket. Then on the
|
|
|
|
# server we use that token to authenticate.
|
|
|
|
#
|
2023-05-30 21:44:13 +08:00
|
|
|
# This module defines a hook that sets the following assigns:
|
|
|
|
#
|
|
|
|
# * `:app_authenticated?` - reflects the current authentication.
|
|
|
|
# For public apps (or in case the user has full access) it is
|
|
|
|
# set to `true` on both dead and live render
|
|
|
|
#
|
|
|
|
# * `:livebook_authenticated?` - if the user has full Livebook
|
|
|
|
# access
|
|
|
|
#
|
|
|
|
# * `:app_settings` - the current app settings
|
|
|
|
#
|
2023-02-18 08:16:42 +08:00
|
|
|
|
|
|
|
def on_mount(:default, %{"slug" => slug}, session, socket) do
|
2023-05-30 21:44:13 +08:00
|
|
|
livebook_authenticated? = livebook_authenticated?(session, socket)
|
|
|
|
|
|
|
|
socket = assign(socket, livebook_authenticated?: livebook_authenticated?)
|
|
|
|
|
2023-05-20 01:40:56 +08:00
|
|
|
case Livebook.Apps.fetch_settings(slug) do
|
2023-02-18 08:16:42 +08:00
|
|
|
{:ok, %{access_type: :public} = app_settings} ->
|
|
|
|
{:cont, assign(socket, app_authenticated?: true, app_settings: app_settings)}
|
|
|
|
|
|
|
|
{:ok, %{access_type: :protected} = app_settings} ->
|
2023-05-30 21:44:13 +08:00
|
|
|
app_authenticated? = livebook_authenticated? or has_valid_token?(socket, app_settings)
|
2023-02-18 08:16:42 +08:00
|
|
|
|
|
|
|
{:cont,
|
|
|
|
assign(socket, app_authenticated?: app_authenticated?, app_settings: app_settings)}
|
|
|
|
|
|
|
|
:error ->
|
2023-02-23 02:34:54 +08:00
|
|
|
{:halt, redirect(socket, to: ~p"/")}
|
2023-02-18 08:16:42 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-11-06 16:08:28 +08:00
|
|
|
# Skip auth for non-app-specific routes
|
|
|
|
def on_mount(:default, %{}, _session, socket) do
|
|
|
|
{:cont, socket}
|
|
|
|
end
|
|
|
|
|
2023-02-18 08:16:42 +08:00
|
|
|
defp livebook_authenticated?(session, socket) do
|
|
|
|
uri = get_connect_info(socket, :uri)
|
2024-06-15 00:59:54 +08:00
|
|
|
LivebookWeb.AuthPlug.authenticated?(session, uri.port)
|
2023-02-18 08:16:42 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
defp has_valid_token?(socket, app_settings) do
|
|
|
|
connect_params = get_connect_params(socket) || %{}
|
|
|
|
|
|
|
|
if token = connect_params["app_auth_token"] do
|
|
|
|
valid_auth_token?(token, app_settings)
|
|
|
|
else
|
|
|
|
false
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
@doc """
|
|
|
|
Generates auth token that can be sent to the client.
|
|
|
|
"""
|
|
|
|
@spec get_auth_token(Livebook.Notebook.AppSettings.t()) :: String.t()
|
|
|
|
def get_auth_token(app_settings) do
|
|
|
|
:crypto.hash(:sha256, app_settings.password) |> Base.encode64()
|
|
|
|
end
|
|
|
|
|
|
|
|
@doc """
|
|
|
|
Checks the given token is valid.
|
|
|
|
"""
|
2023-02-23 02:34:54 +08:00
|
|
|
@spec valid_auth_token?(String.t(), Livebook.Notebook.AppSettings.t()) :: boolean()
|
2023-02-18 08:16:42 +08:00
|
|
|
def valid_auth_token?(token, app_settings) do
|
|
|
|
Plug.Crypto.secure_compare(token, get_auth_token(app_settings))
|
|
|
|
end
|
|
|
|
|
|
|
|
@doc """
|
|
|
|
Checks if the given password is valid.
|
|
|
|
"""
|
2023-02-23 02:34:54 +08:00
|
|
|
@spec valid_password?(String.t(), Livebook.Notebook.AppSettings.t()) :: boolean()
|
2023-02-18 08:16:42 +08:00
|
|
|
def valid_password?(password, app_settings) do
|
|
|
|
Plug.Crypto.secure_compare(password, app_settings.password)
|
|
|
|
end
|
|
|
|
end
|