2021-10-05 06:26:41 +08:00
|
|
|
defmodule Livebook do
|
|
|
|
@moduledoc """
|
|
|
|
Livebook is an interactive notebook system for Elixir.
|
|
|
|
|
|
|
|
This module includes the public API.
|
2022-04-02 02:13:37 +08:00
|
|
|
|
|
|
|
## Configuration
|
|
|
|
|
|
|
|
See `config_runtime/0` for bootstrapping the default runtime
|
|
|
|
configuration. There are several public configuration entries that
|
|
|
|
you can customize.
|
|
|
|
|
|
|
|
### Custom plugs
|
|
|
|
|
|
|
|
You can list a number of plugs to call directly before the Livebook
|
|
|
|
router
|
|
|
|
|
|
|
|
config :livebook, :plugs, [{CustomPlug, []}]
|
|
|
|
|
|
|
|
### Embedded runtime dependencies
|
|
|
|
|
2022-04-30 23:14:10 +08:00
|
|
|
In case you use the Embedded runtime and support installing
|
|
|
|
dependencies with `Mix.install/2`, you can make those discoverable
|
|
|
|
in the package search, by configuring a loader function:
|
2022-04-02 02:13:37 +08:00
|
|
|
|
|
|
|
config :livebook, Livebook.Runtime.Embedded,
|
2022-04-30 23:14:10 +08:00
|
|
|
load_packages: {Loader, :packages, []}
|
2022-04-02 02:13:37 +08:00
|
|
|
|
|
|
|
The function should return a list of entries like this:
|
|
|
|
|
|
|
|
[
|
|
|
|
%{
|
2022-12-03 23:23:43 +08:00
|
|
|
dependency: %{dep: {:kino, "~> 0.6.1"}, config: []},
|
2022-04-02 02:13:37 +08:00
|
|
|
description: "Interactive widgets for Livebook",
|
|
|
|
name: "kino",
|
|
|
|
url: "https://hex.pm/packages/kino",
|
2022-05-03 21:09:37 +08:00
|
|
|
version: "0.6.1"
|
2022-04-02 02:13:37 +08:00
|
|
|
}
|
|
|
|
]
|
|
|
|
|
2022-09-21 18:06:22 +08:00
|
|
|
### Custom learn notebooks
|
2022-04-02 02:13:37 +08:00
|
|
|
|
|
|
|
**Note that this is compile time configuration.**
|
|
|
|
|
2022-09-21 18:06:22 +08:00
|
|
|
A list of additional notebooks to include in the Learn section.
|
2022-04-02 02:13:37 +08:00
|
|
|
|
|
|
|
Note that the notebooks are loaded and embedded in a compiled module,
|
|
|
|
so the paths are accessed at compile time only.
|
|
|
|
|
2022-09-21 18:06:22 +08:00
|
|
|
config :livebook, :learn_notebooks, [
|
2022-04-02 02:13:37 +08:00
|
|
|
%{
|
|
|
|
# Required notebook path
|
|
|
|
path: "/path/to/notebook.livemd",
|
2022-09-21 18:06:22 +08:00
|
|
|
# Optional notebook identifier for URLs, as in /learn/notebooks/{slug}
|
2022-04-02 02:13:37 +08:00
|
|
|
# By default the slug is inferred from file name, so there is no need to set it
|
|
|
|
slug: "my-notebook"
|
|
|
|
# Optional list of images
|
|
|
|
image_paths: [
|
|
|
|
# This image can be sourced as images/myimage.jpg in the notebook
|
|
|
|
"/path/to/myimage.jpg"
|
|
|
|
],
|
|
|
|
# Optional details for the notebook card. If omitted, the notebook
|
2022-09-21 18:06:22 +08:00
|
|
|
# is hidden in the UI, but still accessible under /learn/notebooks/{slug}
|
2022-04-02 02:13:37 +08:00
|
|
|
details: %{
|
|
|
|
cover_path: "/path/to/logo.png",
|
|
|
|
description: "My custom notebook that showcases some amazing stuff."
|
|
|
|
}
|
|
|
|
},
|
|
|
|
%{
|
|
|
|
path: "/path/to/other_notebook.livemd"
|
|
|
|
}
|
|
|
|
]
|
2021-10-05 06:26:41 +08:00
|
|
|
"""
|
|
|
|
|
2021-10-05 06:49:01 +08:00
|
|
|
@doc """
|
2022-04-02 02:13:37 +08:00
|
|
|
Executes Livebook's `config/runtime.exs`.
|
2021-10-05 06:49:01 +08:00
|
|
|
|
|
|
|
If you use Livebook as a dependency, you can add the following
|
2022-04-02 02:13:37 +08:00
|
|
|
to your `config/runtime.exs` to trigger Livebook's `config/runtime.exs`
|
2021-10-05 06:49:01 +08:00
|
|
|
configuration:
|
|
|
|
|
|
|
|
Livebook.config_runtime()
|
|
|
|
|
|
|
|
"""
|
|
|
|
def config_runtime do
|
|
|
|
import Config
|
|
|
|
|
|
|
|
config :livebook, LivebookWeb.Endpoint,
|
|
|
|
secret_key_base:
|
|
|
|
Livebook.Config.secret!("LIVEBOOK_SECRET_KEY_BASE") ||
|
|
|
|
Base.encode64(:crypto.strong_rand_bytes(48))
|
|
|
|
|
|
|
|
if port = Livebook.Config.port!("LIVEBOOK_PORT") do
|
|
|
|
config :livebook, LivebookWeb.Endpoint, http: [port: port]
|
|
|
|
end
|
|
|
|
|
|
|
|
if ip = Livebook.Config.ip!("LIVEBOOK_IP") do
|
2022-05-13 21:22:41 +08:00
|
|
|
host = Livebook.Utils.ip_to_host(ip)
|
|
|
|
config :livebook, LivebookWeb.Endpoint, http: [ip: ip], url: [host: host]
|
2021-10-05 06:49:01 +08:00
|
|
|
end
|
|
|
|
|
2022-11-30 06:34:38 +08:00
|
|
|
if base_url_path = Livebook.Config.base_url_path!("LIVEBOOK_BASE_URL_PATH") do
|
|
|
|
config :livebook, LivebookWeb.Endpoint, url: [path: base_url_path]
|
|
|
|
end
|
|
|
|
|
2021-11-10 20:54:50 +08:00
|
|
|
cond do
|
|
|
|
password = Livebook.Config.password!("LIVEBOOK_PASSWORD") ->
|
|
|
|
config :livebook, authentication_mode: :password, password: password
|
|
|
|
|
2022-04-14 21:52:28 +08:00
|
|
|
Livebook.Config.boolean!("LIVEBOOK_TOKEN_ENABLED", true) ->
|
2021-11-10 20:54:50 +08:00
|
|
|
config :livebook, token: Livebook.Utils.random_id()
|
|
|
|
|
|
|
|
true ->
|
|
|
|
config :livebook, authentication_mode: :disabled
|
2021-10-05 06:49:01 +08:00
|
|
|
end
|
|
|
|
|
2022-02-08 21:45:58 +08:00
|
|
|
if port = Livebook.Config.port!("LIVEBOOK_IFRAME_PORT") do
|
|
|
|
config :livebook, :iframe_port, port
|
|
|
|
end
|
|
|
|
|
2022-05-26 08:39:55 +08:00
|
|
|
if url = Livebook.Config.iframe_url!("LIVEBOOK_IFRAME_URL") do
|
|
|
|
config :livebook, :iframe_url, url
|
|
|
|
end
|
|
|
|
|
2023-05-17 00:21:49 +08:00
|
|
|
if url = Livebook.Config.teams_url!("LIVEBOOK_TEAMS_URL") do
|
|
|
|
config :livebook, :teams_url, url
|
|
|
|
end
|
|
|
|
|
2022-04-14 21:52:28 +08:00
|
|
|
if Livebook.Config.boolean!("LIVEBOOK_SHUTDOWN_ENABLED", false) do
|
2022-11-11 01:09:01 +08:00
|
|
|
config :livebook, :shutdown_callback, {System, :stop, []}
|
2022-04-14 21:52:28 +08:00
|
|
|
end
|
|
|
|
|
2022-11-10 23:02:59 +08:00
|
|
|
if Livebook.Config.boolean!("LIVEBOOK_WITHIN_IFRAME", false) do
|
|
|
|
config :livebook, :within_iframe, true
|
|
|
|
end
|
|
|
|
|
2022-04-02 02:13:37 +08:00
|
|
|
config :livebook,
|
|
|
|
:default_runtime,
|
|
|
|
Livebook.Config.default_runtime!("LIVEBOOK_DEFAULT_RUNTIME") ||
|
|
|
|
Livebook.Runtime.ElixirStandalone.new()
|
2021-10-05 06:49:01 +08:00
|
|
|
|
2023-02-16 20:47:46 +08:00
|
|
|
config :livebook, :default_app_runtime, Livebook.Runtime.ElixirStandalone.new()
|
|
|
|
|
2022-04-04 18:20:36 +08:00
|
|
|
config :livebook,
|
|
|
|
:runtime_modules,
|
|
|
|
[
|
|
|
|
Livebook.Runtime.ElixirStandalone,
|
|
|
|
Livebook.Runtime.Attached
|
|
|
|
]
|
|
|
|
|
2022-01-31 18:51:57 +08:00
|
|
|
if home = Livebook.Config.writable_dir!("LIVEBOOK_HOME") do
|
|
|
|
config :livebook, :home, home
|
|
|
|
end
|
|
|
|
|
|
|
|
if data_path = Livebook.Config.writable_dir!("LIVEBOOK_DATA_PATH") do
|
|
|
|
config :livebook, :data_path, data_path
|
|
|
|
end
|
|
|
|
|
2023-03-21 19:33:53 +08:00
|
|
|
if apps_path = System.get_env("LIVEBOOK_APPS_PATH") do
|
2023-03-01 04:37:43 +08:00
|
|
|
config :livebook, :apps_path, apps_path
|
|
|
|
end
|
|
|
|
|
|
|
|
if apps_path_password = Livebook.Config.password!("LIVEBOOK_APPS_PATH_PASSWORD") do
|
|
|
|
config :livebook, :apps_path_password, apps_path_password
|
|
|
|
end
|
|
|
|
|
2022-03-24 01:24:58 +08:00
|
|
|
if force_ssl_host = Livebook.Config.force_ssl_host!("LIVEBOOK_FORCE_SSL_HOST") do
|
|
|
|
config :livebook, :force_ssl_host, force_ssl_host
|
|
|
|
end
|
|
|
|
|
2021-10-05 06:49:01 +08:00
|
|
|
config :livebook,
|
|
|
|
:cookie,
|
|
|
|
Livebook.Config.cookie!("LIVEBOOK_COOKIE") ||
|
|
|
|
Livebook.Config.cookie!("RELEASE_COOKIE") ||
|
|
|
|
Livebook.Utils.random_cookie()
|
2022-04-15 19:46:26 +08:00
|
|
|
|
2023-02-02 04:22:49 +08:00
|
|
|
if node = Livebook.Config.node!("LIVEBOOK_NODE", "LIVEBOOK_DISTRIBUTION") do
|
|
|
|
config :livebook, :node, node
|
|
|
|
end
|
|
|
|
|
2022-04-15 19:46:26 +08:00
|
|
|
if app_service_name = Livebook.Config.app_service_name!("LIVEBOOK_APP_SERVICE_NAME") do
|
|
|
|
config :livebook, :app_service_name, app_service_name
|
|
|
|
|
|
|
|
config :livebook,
|
|
|
|
:app_service_url,
|
|
|
|
Livebook.Config.app_service_url!("LIVEBOOK_APP_SERVICE_URL")
|
|
|
|
end
|
2022-04-18 06:55:08 +08:00
|
|
|
|
|
|
|
if update_instructions_url =
|
|
|
|
Livebook.Config.update_instructions_url!("LIVEBOOK_UPDATE_INSTRUCTIONS_URL") do
|
|
|
|
config :livebook, :update_instructions_url, update_instructions_url
|
|
|
|
end
|
2023-02-16 05:16:38 +08:00
|
|
|
|
|
|
|
if allowed_uri_schemes = Livebook.Config.allowed_uri_schemes!("LIVEBOOK_ALLOW_URI_SCHEMES") do
|
|
|
|
config :livebook, :allowed_uri_schemes, allowed_uri_schemes
|
|
|
|
end
|
2021-10-05 06:49:01 +08:00
|
|
|
end
|
|
|
|
|
2021-10-05 06:26:41 +08:00
|
|
|
@doc """
|
|
|
|
Parses the given Live Markdown document and converts it to Elixir
|
|
|
|
source code.
|
|
|
|
|
|
|
|
## Limitations
|
|
|
|
|
|
|
|
Note that the resulting script may not compile in some cases, for
|
|
|
|
example if you define a macro in one cell and import it in another
|
|
|
|
cell, it works fine in Livebook, because each cell is compiled
|
|
|
|
separately. However, when running the script it gets compiled as a
|
|
|
|
whole and consequently doing so doesn't work.
|
|
|
|
|
|
|
|
Additionally, branching sections are commented out.
|
|
|
|
"""
|
|
|
|
@spec live_markdown_to_elixir(String.t()) :: String.t()
|
|
|
|
def live_markdown_to_elixir(markdown) do
|
2022-02-10 19:15:41 +08:00
|
|
|
{notebook, _messages} = Livebook.LiveMarkdown.notebook_from_livemd(markdown)
|
2021-10-05 06:26:41 +08:00
|
|
|
Livebook.Notebook.Export.Elixir.notebook_to_elixir(notebook)
|
|
|
|
end
|
|
|
|
end
|