mirror of
https://github.com/livebook-dev/livebook.git
synced 2025-01-08 16:07:37 +08:00
2249 lines
74 KiB
Elixir
2249 lines
74 KiB
Elixir
defmodule LivebookWeb.SessionLiveTest do
|
|
use LivebookWeb.ConnCase, async: true
|
|
|
|
import Livebook.SessionHelpers
|
|
import Livebook.TestHelpers
|
|
import Phoenix.LiveViewTest
|
|
|
|
alias Livebook.{Sessions, Session, Settings, Runtime, Users, FileSystem}
|
|
alias Livebook.Notebook.Cell
|
|
|
|
setup do
|
|
{:ok, session} = Sessions.create_session(notebook: Livebook.Notebook.new())
|
|
|
|
on_exit(fn ->
|
|
Session.close(session.pid)
|
|
end)
|
|
|
|
%{session: session}
|
|
end
|
|
|
|
test "disconnected and connected render", %{conn: conn, session: session} do
|
|
{:ok, view, disconnected_html} = live(conn, ~p"/sessions/#{session.id}")
|
|
assert disconnected_html =~ "Untitled notebook"
|
|
assert render(view) =~ "Untitled notebook"
|
|
end
|
|
|
|
describe "asynchronous updates" do
|
|
test "renders an updated notebook name", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
Session.set_notebook_name(session.pid, "My notebook")
|
|
wait_for_session_update(session.pid)
|
|
|
|
assert render(view) =~ "My notebook"
|
|
end
|
|
|
|
test "renders an updated notebook name in title", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert page_title(view) =~ "Untitled notebook"
|
|
|
|
Session.set_notebook_name(session.pid, "My notebook")
|
|
wait_for_session_update(session.pid)
|
|
|
|
# Wait for LV to update
|
|
_ = render(view)
|
|
|
|
assert page_title(view) =~ "My notebook"
|
|
end
|
|
|
|
test "renders a newly inserted section", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
section_id = insert_section(session.pid)
|
|
|
|
assert render(view) =~ section_id
|
|
end
|
|
|
|
test "renders an updated section name", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
Session.set_section_name(session.pid, section_id, "My section")
|
|
wait_for_session_update(session.pid)
|
|
|
|
assert render(view) =~ "My section"
|
|
end
|
|
|
|
test "renders a newly inserted cell", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
cell_id = insert_text_cell(session.pid, section_id, :markdown)
|
|
|
|
assert render(view) =~ "cell-" <> cell_id
|
|
end
|
|
|
|
test "un-renders a deleted cell", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :markdown)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
Session.delete_cell(session.pid, cell_id)
|
|
wait_for_session_update(session.pid)
|
|
|
|
refute render(view) =~ "cell-" <> cell_id
|
|
end
|
|
end
|
|
|
|
describe "UI events update the shared state" do
|
|
test "adding a new section", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element("button", "New section")
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{sections: [_section]}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "adding a new cell", %{conn: conn, session: session} do
|
|
Session.insert_section(session.pid, 0)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element("button", "Markdown")
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{sections: [%{cells: [%Cell.Markdown{}]}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "queueing cell evaluation", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
{source, continue_fun} = source_for_blocking()
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, source)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert %{cell_infos: %{^cell_id => %{eval: %{status: :evaluating}}}} =
|
|
Session.get_data(session.pid)
|
|
|
|
continue_fun.()
|
|
end
|
|
|
|
test "reevaluting the setup cell", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => "setup"})
|
|
|
|
assert_receive {:operation, {:set_runtime, _pid, %{} = _runtime}}
|
|
end
|
|
|
|
test "reevaluting the setup cell with dependencies cache disabled",
|
|
%{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
|
|
# Start the standalone runtime, to encapsulate env var changes
|
|
{:ok, runtime} = Runtime.ElixirStandalone.new() |> Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
evaluate_setup(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{
|
|
"cell_id" => "setup",
|
|
"disable_dependencies_cache" => true
|
|
})
|
|
|
|
section_id = insert_section(session.pid)
|
|
|
|
cell_id =
|
|
insert_text_cell(session.pid, section_id, :code, ~s/System.get_env("MIX_INSTALL_FORCE")/)
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id,
|
|
terminal_text("\e[32m\"true\"\e[0m"), _}}
|
|
end
|
|
|
|
test "cancelling cell evaluation", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, "Process.sleep(2000)")
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("cancel_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert %{cell_infos: %{^cell_id => %{eval: %{status: :ready}}}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "setting cell to always reevaluating", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element("#cell-#{cell_id}-evaluation-menu button", "Reevaluate automatically")
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{sections: [%{cells: [%Cell.Code{reevaluate_automatically: true}]}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "inserting a cell below the given cell", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("insert_cell_below", %{"cell_id" => cell_id, "type" => "markdown"})
|
|
|
|
assert %{notebook: %{sections: [%{cells: [_first_cell, %Cell.Markdown{}]}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "inserting a cell at section start", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
_cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("insert_cell_below", %{"section_id" => section_id, "type" => "markdown"})
|
|
|
|
assert %{notebook: %{sections: [%{cells: [%Cell.Markdown{}, _first_cell]}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "inserting an image", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(
|
|
~s/[phx-click="insert_image"][phx-value-section_id="#{section_id}"][phx-value-cell_id="#{cell_id}"]/
|
|
)
|
|
|> render_click()
|
|
|
|
view
|
|
|> element(~s/#insert-image-modal form/)
|
|
|> render_change(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
view
|
|
|> file_input(~s/#insert-image-modal form/, :image, [
|
|
%{
|
|
last_modified: 1_594_171_879_000,
|
|
name: "image.jpg",
|
|
content: "content",
|
|
size: 7,
|
|
type: "text/plain"
|
|
}
|
|
])
|
|
|> render_upload("image.jpg")
|
|
|
|
view
|
|
|> element(~s/#insert-image-modal form/)
|
|
|> render_submit(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
assert %{
|
|
notebook: %{
|
|
sections: [
|
|
%{cells: [_first_cell, %Cell.Markdown{source: "![](files/image.jpg)"}]}
|
|
]
|
|
}
|
|
} =
|
|
Session.get_data(session.pid)
|
|
|
|
assert FileSystem.File.resolve(session.files_dir, "image.jpg") |> FileSystem.File.read() ==
|
|
{:ok, "content"}
|
|
end
|
|
|
|
test "inserting a file", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
%{files_dir: files_dir} = session
|
|
image_file = FileSystem.File.resolve(files_dir, "file.bin")
|
|
:ok = FileSystem.File.write(image_file, "content")
|
|
Session.add_file_entries(session.pid, [%{type: :attachment, name: "file.bin"}])
|
|
|
|
{:ok, runtime} = Livebook.Runtime.NoopRuntime.new() |> Livebook.Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("insert_file", %{
|
|
"file_entry_name" => "file.bin",
|
|
"section_id" => section_id,
|
|
"cell_id" => cell_id
|
|
})
|
|
|
|
view
|
|
|> element(~s/#insert-file-modal [phx-click]/, "Read file content")
|
|
|> render_click()
|
|
|
|
assert %{
|
|
notebook: %{
|
|
sections: [
|
|
%{
|
|
cells: [
|
|
_first_cell,
|
|
%Cell.Code{
|
|
source: """
|
|
content =
|
|
Kino.FS.file_path("file.bin")
|
|
|> File.read!()\
|
|
"""
|
|
}
|
|
]
|
|
}
|
|
]
|
|
}
|
|
} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "inserting a file as markdown image", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
%{files_dir: files_dir} = session
|
|
image_file = FileSystem.File.resolve(files_dir, "image.jpg")
|
|
:ok = FileSystem.File.write(image_file, "content")
|
|
Session.add_file_entries(session.pid, [%{type: :attachment, name: "image.jpg"}])
|
|
|
|
{:ok, runtime} = Livebook.Runtime.NoopRuntime.new() |> Livebook.Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("insert_file", %{
|
|
"file_entry_name" => "image.jpg",
|
|
"section_id" => section_id,
|
|
"cell_id" => cell_id
|
|
})
|
|
|
|
view
|
|
|> element(~s/#insert-file-modal [phx-click]/, "Insert as Markdown image")
|
|
|> render_click()
|
|
|
|
assert %{
|
|
notebook: %{
|
|
sections: [
|
|
%{cells: [_first_cell, %Cell.Markdown{source: "![](files/image.jpg)"}]}
|
|
]
|
|
}
|
|
} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "inserting file after file drop upload", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, runtime} = Livebook.Runtime.NoopRuntime.new() |> Livebook.Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> render_hook("handle_file_drop", %{"section_id" => section_id, "cell_id" => cell_id})
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_change(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
view
|
|
|> file_input(~s{#add-file-entry-form}, :file, [
|
|
%{
|
|
last_modified: 1_594_171_879_000,
|
|
name: "image.jpg",
|
|
content: "content",
|
|
size: 7,
|
|
type: "text/plain"
|
|
}
|
|
])
|
|
|> render_upload("image.jpg")
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
view
|
|
|> element(~s/#insert-file-modal [phx-click]/, "Insert as Markdown image")
|
|
|> render_click()
|
|
|
|
assert %{
|
|
notebook: %{
|
|
sections: [
|
|
%{cells: [_first_cell, %Cell.Markdown{source: "![](files/image.jpg)"}]}
|
|
]
|
|
}
|
|
} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "deleting section with no cells requires no confirmation",
|
|
%{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(
|
|
~s{[data-section-id="#{section_id}"] [data-el-section-headline] [aria-label="delete section"]}
|
|
)
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{sections: []}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "deleting section with cells requires confirmation", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(
|
|
~s{[data-section-id="#{section_id}"] [data-el-section-headline] [aria-label="delete section"]}
|
|
)
|
|
|> render_click()
|
|
|
|
render_confirm(view)
|
|
|
|
assert %{notebook: %{sections: []}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "deleting the given cell", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("delete_cell", %{"cell_id" => cell_id})
|
|
|
|
render_confirm(view)
|
|
|
|
assert %{notebook: %{sections: [%{cells: []}]}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "restoring a deleted cell", %{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.delete_cell(session.pid, cell_id)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
refute render(view) =~ "cell-" <> cell_id
|
|
|
|
view
|
|
|> element(~s{a[aria-label="Bin (sb)"]})
|
|
|> render_click()
|
|
|
|
view
|
|
|> element("button", "Restore")
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{sections: [%{cells: [%{id: ^cell_id}]}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "editing input field in cell output", %{conn: conn, session: session, test: test} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
Process.register(self(), test)
|
|
|
|
input = %{
|
|
type: :input,
|
|
ref: "ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{type: :number, default: 1, label: "Name", debounce: :blur}
|
|
}
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
insert_cell_with_output(session.pid, section_id, input)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s/[data-el-outputs-container] form/)
|
|
|> render_change(%{"html_value" => "10"})
|
|
|
|
assert %{input_infos: %{"input1" => %{value: 10}}} = Session.get_data(session.pid)
|
|
|
|
assert_receive {:event, "ref1", %{value: 10, type: :change}}
|
|
end
|
|
|
|
test "newlines in text input are normalized", %{conn: conn, session: session, test: test} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
Process.register(self(), test)
|
|
|
|
input = %{
|
|
type: :input,
|
|
ref: "ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{
|
|
type: :textarea,
|
|
default: "hey",
|
|
label: "Name",
|
|
debounce: :blur,
|
|
monospace: false
|
|
}
|
|
}
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
insert_cell_with_output(session.pid, section_id, input)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s/[data-el-outputs-container] form/)
|
|
|> render_change(%{"html_value" => "line\r\nline"})
|
|
|
|
assert %{input_infos: %{"input1" => %{value: "line\nline"}}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "form input changes are reflected only in local LV data",
|
|
%{conn: conn, session: session, test: test} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
Process.register(self(), test)
|
|
|
|
form_control = %{
|
|
type: :control,
|
|
ref: "control_ref1",
|
|
destination: test,
|
|
attrs: %{
|
|
type: :form,
|
|
fields: [
|
|
name: %{
|
|
type: :input,
|
|
ref: "input_ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{type: :text, default: "initial", label: "Name", debounce: :blur}
|
|
}
|
|
],
|
|
submit: "Send",
|
|
report_changes: %{},
|
|
reset_on_submit: []
|
|
}
|
|
}
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
insert_cell_with_output(session.pid, section_id, form_control)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s/[data-el-outputs-container] form/)
|
|
|> render_change(%{"html_value" => "sherlock"})
|
|
|
|
# The new value is on the page
|
|
assert render(view) =~ "sherlock"
|
|
# but it's not reflected in the synchronized session data
|
|
assert %{input_infos: %{"input1" => %{value: "initial"}}} = Session.get_data(session.pid)
|
|
|
|
view
|
|
|> element(~s/[data-el-outputs-container] button/, "Send")
|
|
|> render_click()
|
|
|
|
assert_receive {:event, "control_ref1", %{data: %{name: "sherlock"}, type: :submit}}
|
|
end
|
|
|
|
test "file input", %{conn: conn, session: session, test: test} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
Process.register(self(), test)
|
|
|
|
input = %{
|
|
type: :input,
|
|
ref: "ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{type: :file, default: nil, label: "File", accept: :any}
|
|
}
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
insert_cell_with_output(session.pid, section_id, input)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> file_input(~s/[data-el-outputs-container] form/, :file, [
|
|
%{
|
|
last_modified: 1_594_171_879_000,
|
|
name: "data.txt",
|
|
content: "content",
|
|
size: 7,
|
|
type: "text/plain"
|
|
}
|
|
])
|
|
|> render_upload("data.txt")
|
|
|
|
assert %{input_infos: %{"input1" => %{value: value}}} = Session.get_data(session.pid)
|
|
|
|
assert %{file_ref: file_ref, client_name: "data.txt"} = value
|
|
|
|
send(session.pid, {:runtime_file_path_request, self(), file_ref})
|
|
assert_receive {:runtime_file_path_reply, {:ok, path}}
|
|
assert File.read!(path) == "content"
|
|
end
|
|
end
|
|
|
|
describe "outputs" do
|
|
test "chunked text output update", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, terminal_text("line 1\n", true)})
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
assert render(view) =~ "line 1"
|
|
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, terminal_text("line 2\n", true)})
|
|
|
|
wait_for_session_update(session.pid)
|
|
# Render once, so that the send_update is processed
|
|
_ = render(view)
|
|
assert render(view) =~ "line 2"
|
|
end
|
|
|
|
test "frame output update", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
frame = %{type: :frame, ref: "1", outputs: [terminal_text("In frame")], placeholder: true}
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, frame})
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
assert render(view) =~ "In frame"
|
|
|
|
frame_update = %{
|
|
type: :frame_update,
|
|
ref: "1",
|
|
update: {:replace, [terminal_text("Updated frame")]}
|
|
}
|
|
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, frame_update})
|
|
|
|
wait_for_session_update(session.pid)
|
|
|
|
# Render once, so that frame send_update is processed
|
|
_ = render(view)
|
|
|
|
content = render(view)
|
|
assert content =~ "Updated frame"
|
|
refute content =~ "In frame"
|
|
end
|
|
|
|
test "frame output update when within grid", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
frame = %{type: :frame, ref: "1", outputs: [terminal_text("In frame")], placeholder: true}
|
|
grid = %{type: :grid, outputs: [frame], columns: ["Frame"], gap: 8, boxed: false}
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, grid})
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
assert render(view) =~ "In frame"
|
|
|
|
frame_update = %{
|
|
type: :frame_update,
|
|
ref: "1",
|
|
update: {:replace, [terminal_text("Updated frame")]}
|
|
}
|
|
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, frame_update})
|
|
|
|
wait_for_session_update(session.pid)
|
|
|
|
# Render once, so that frame send_update is processed
|
|
_ = render(view)
|
|
|
|
content = render(view)
|
|
assert content =~ "Updated frame"
|
|
refute content =~ "In frame"
|
|
end
|
|
|
|
test "client-specific output is sent only to one target", %{conn: conn, session: session} do
|
|
user1 = build(:user, name: "Jake Peralta")
|
|
{_, client_id} = Session.register_client(session.pid, self(), user1)
|
|
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
send(
|
|
session.pid,
|
|
{:runtime_evaluation_output_to, client_id, cell_id, terminal_text("line 1\n", true)}
|
|
)
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_output, _, ^cell_id, terminal_text("line 1\n", true)}}
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
refute render(view) =~ "line 1"
|
|
end
|
|
|
|
test "clients-only output is sent to all targets, but not reflected in session",
|
|
%{conn: conn, session: session} do
|
|
user1 = build(:user, name: "Jake Peralta")
|
|
Session.register_client(session.pid, self(), user1)
|
|
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
send(
|
|
session.pid,
|
|
{:runtime_evaluation_output_to_clients, cell_id, terminal_text("line 1\n")}
|
|
)
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_output, _, ^cell_id, terminal_text("line 1\n")}}
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
refute render(view) =~ "line 1"
|
|
end
|
|
|
|
test "shows change indicator on bound inputs",
|
|
%{conn: conn, session: session, test: test} do
|
|
section_id = insert_section(session.pid)
|
|
|
|
Process.register(self(), test)
|
|
|
|
input = %{
|
|
type: :input,
|
|
ref: "ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{type: :number, default: 1, label: "Name", debounce: :blur}
|
|
}
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
insert_cell_with_output(session.pid, section_id, input)
|
|
|
|
code = source_for_input_read(input.id)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, code)
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
assert_receive {:operation, {:add_cell_evaluation_response, _, ^cell_id, _, _}}
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
refute render(view) =~ "This input has changed."
|
|
|
|
Session.set_input_value(session.pid, input.id, 10)
|
|
wait_for_session_update(session.pid)
|
|
|
|
assert render(view) =~ "This input has changed."
|
|
end
|
|
|
|
test "frame output update with input", %{conn: conn, session: session, test: test} do
|
|
Session.subscribe(session.id)
|
|
evaluate_setup(session.pid)
|
|
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
frame = %{type: :frame, ref: "1", outputs: [], placeholder: true}
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, frame})
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
input = %{
|
|
type: :input,
|
|
ref: "ref1",
|
|
id: "input1",
|
|
destination: test,
|
|
attrs: %{type: :number, default: 1, label: "Input inside frame", debounce: :blur}
|
|
}
|
|
|
|
frame_update = %{
|
|
type: :frame_update,
|
|
ref: "1",
|
|
update: {:replace, [input]}
|
|
}
|
|
|
|
send(session.pid, {:runtime_evaluation_output, cell_id, frame_update})
|
|
|
|
wait_for_session_update(session.pid)
|
|
|
|
# Render once, so that frame send_update is processed
|
|
_ = render(view)
|
|
|
|
content = render(view)
|
|
assert content =~ "Input inside frame"
|
|
assert has_element?(view, ~s/input[value="1"]/)
|
|
end
|
|
end
|
|
|
|
describe "smart cells" do
|
|
test "shows a new cell insert button when a new smart cell kind becomes available",
|
|
%{conn: conn, session: session} do
|
|
insert_section(session.pid)
|
|
|
|
{:ok, runtime} = Runtime.Embedded.new() |> Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
refute render(view) =~ "Database connection"
|
|
|
|
send(
|
|
session.pid,
|
|
{:runtime_smart_cell_definitions,
|
|
[%{kind: "dbconn", name: "Database connection", requirement_presets: []}]}
|
|
)
|
|
|
|
wait_for_session_update(session.pid)
|
|
|
|
assert render(view) =~ "Database connection"
|
|
end
|
|
end
|
|
|
|
describe "runtime settings" do
|
|
test "connecting to elixir standalone updates connect button to reconnect",
|
|
%{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/runtime")
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
view
|
|
|> element("button", "Elixir standalone")
|
|
|> render_click()
|
|
|
|
[elixir_standalone_view] = live_children(view)
|
|
|
|
elixir_standalone_view
|
|
|> element("button", "Connect")
|
|
|> render_click()
|
|
|
|
assert_receive {:operation, {:set_runtime, _pid, %Runtime.ElixirStandalone{} = runtime}}
|
|
|
|
page = render(view)
|
|
assert page =~ Atom.to_string(runtime.node)
|
|
assert page =~ "Reconnect"
|
|
assert page =~ "Disconnect"
|
|
end
|
|
end
|
|
|
|
describe "persistence settings" do
|
|
@tag :tmp_dir
|
|
test "saving to file shows the newly created file in file selector",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/file")
|
|
|
|
path = Path.join(tmp_dir, "notebook.livemd")
|
|
|
|
view
|
|
|> element(~s{form[phx-change="set_path"]})
|
|
|> render_change(%{path: path})
|
|
|
|
view
|
|
|> element(~s{#persistence-modal button}, "Save")
|
|
|> render_click()
|
|
|
|
assert Session.get_data(session.pid).file
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/file")
|
|
|
|
assert view
|
|
|> element("button", "notebook.livemd")
|
|
|> has_element?()
|
|
|
|
view
|
|
|> element(~s{#persistence-modal button}, "Stop saving")
|
|
|> render_click()
|
|
|
|
refute Session.get_data(session.pid).file
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "changing output persistence updates data",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/file")
|
|
|
|
path = Path.join(tmp_dir, "notebook.livemd")
|
|
|
|
view
|
|
|> element(~s{form[phx-change="set_path"]})
|
|
|> render_change(%{path: path})
|
|
|
|
view
|
|
|> element(~s{form[phx-change="set_options"]})
|
|
|> render_change(%{persist_outputs: "true"})
|
|
|
|
view
|
|
|> element(~s{#persistence-modal button}, "Save")
|
|
|> render_click()
|
|
|
|
assert %{notebook: %{persist_outputs: true}} = Session.get_data(session.pid)
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "showing all files from default directory",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
old_dir = Livebook.Settings.default_dir()
|
|
new_dir = Livebook.FileSystem.File.local(tmp_dir)
|
|
|
|
Livebook.Settings.set_default_dir(new_dir)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/file")
|
|
|
|
assert render(view) =~ new_dir.path
|
|
|
|
Livebook.Settings.set_default_dir(old_dir)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/settings/file")
|
|
|
|
refute render(view) =~ new_dir.path
|
|
assert render(view) =~ old_dir.path
|
|
end
|
|
end
|
|
|
|
describe "completion" do
|
|
test "replies with nil completion reference when no runtime is started",
|
|
%{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, "Process.sleep(10)")
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("intellisense_request", %{
|
|
"cell_id" => cell_id,
|
|
"type" => "completion",
|
|
"hint" => "System.ver",
|
|
"editor_auto_completion" => false
|
|
})
|
|
|
|
assert_reply(view, %{"ref" => nil})
|
|
end
|
|
|
|
test "replies with completion reference and then sends asynchronous response",
|
|
%{conn: conn, session: session} do
|
|
section_id = insert_section(session.pid)
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, "Process.sleep(10)")
|
|
|
|
{:ok, runtime} = Runtime.Embedded.new() |> Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("intellisense_request", %{
|
|
"cell_id" => cell_id,
|
|
"type" => "completion",
|
|
"hint" => "System.ver",
|
|
"editor_auto_completion" => false
|
|
})
|
|
|
|
assert_reply(view, %{"ref" => ref})
|
|
assert ref != nil
|
|
|
|
assert_push_event(
|
|
view,
|
|
"intellisense_response",
|
|
%{
|
|
"ref" => ^ref,
|
|
"response" => %{items: [%{label: "version/0"}]}
|
|
},
|
|
1000
|
|
)
|
|
end
|
|
end
|
|
|
|
test "forking the session", %{conn: conn, session: session} do
|
|
Session.set_notebook_name(session.pid, "My notebook")
|
|
wait_for_session_update(session.pid)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert {:error, {:live_redirect, %{to: "/sessions/" <> session_id}}} =
|
|
result =
|
|
view
|
|
|> element("button", "Fork")
|
|
|> render_click()
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "My notebook - fork"
|
|
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "starring and unstarring the notebook", %{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
notebook_path = Path.join(tmp_dir, "notebook.livemd")
|
|
file = Livebook.FileSystem.File.local(notebook_path)
|
|
Session.set_file(session.pid, file)
|
|
|
|
Livebook.NotebookManager.subscribe_starred_notebooks()
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s/button[phx-click="star_notebook"]/)
|
|
|> render_click()
|
|
|
|
assert_receive {:starred_notebooks_updated, starred_notebooks}
|
|
|
|
assert Enum.any?(starred_notebooks, &(&1.file == file))
|
|
|
|
view
|
|
|> element(~s/button[phx-click="unstar_notebook"]/)
|
|
|> render_click()
|
|
|
|
assert_receive {:starred_notebooks_updated, starred_notebooks}
|
|
|
|
refute Enum.any?(starred_notebooks, &(&1.file == file))
|
|
end
|
|
|
|
describe "connected users" do
|
|
test "lists connected users", %{conn: conn, session: session} do
|
|
user1 = build(:user, name: "Jake Peralta")
|
|
|
|
client_pid =
|
|
spawn_link(fn ->
|
|
receive do
|
|
:stop -> :ok
|
|
end
|
|
end)
|
|
|
|
Session.register_client(session.pid, client_pid, user1)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert render(view) =~ "Jake Peralta"
|
|
|
|
send(client_pid, :stop)
|
|
end
|
|
|
|
test "updates users list whenever a user joins or leaves",
|
|
%{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
user1 = build(:user, name: "Jake Peralta")
|
|
|
|
client_pid =
|
|
spawn_link(fn ->
|
|
receive do
|
|
:stop -> :ok
|
|
end
|
|
end)
|
|
|
|
{_, client_id} = Session.register_client(session.pid, client_pid, user1)
|
|
|
|
assert_receive {:operation, {:client_join, ^client_id, _user}}
|
|
assert render(view) =~ "Jake Peralta"
|
|
|
|
send(client_pid, :stop)
|
|
assert_receive {:operation, {:client_leave, ^client_id}}
|
|
refute render(view) =~ "Jake Peralta"
|
|
end
|
|
|
|
test "updates users list whenever a user changes his data",
|
|
%{conn: conn, session: session} do
|
|
user1 = build(:user, name: "Jake Peralta")
|
|
|
|
client_pid =
|
|
spawn_link(fn ->
|
|
receive do
|
|
:stop -> :ok
|
|
end
|
|
end)
|
|
|
|
Session.register_client(session.pid, client_pid, user1)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
Session.subscribe(session.id)
|
|
|
|
assert render(view) =~ "Jake Peralta"
|
|
|
|
Users.broadcast_change(%{user1 | name: "Raymond Holt"})
|
|
assert_receive {:operation, {:update_user, _client_id, _user}}
|
|
|
|
refute render(view) =~ "Jake Peralta"
|
|
assert render(view) =~ "Raymond Holt"
|
|
|
|
send(client_pid, :stop)
|
|
end
|
|
end
|
|
|
|
describe "relative paths" do
|
|
test "renders an info message when the path doesn't have notebook extension",
|
|
%{conn: conn, session: session} do
|
|
session_path = "/sessions/#{session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/document.pdf")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Got unrecognised session path: document.pdf"
|
|
end
|
|
|
|
test "renders an info message when the session has neither original url nor path",
|
|
%{conn: conn, session: session} do
|
|
session_path = "/sessions/#{session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
|
|
assert render(view) =~
|
|
"Cannot resolve notebook path notebook.livemd, because the current notebook has no location"
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "renders an error message when the relative notebook does not exist",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
index_file = FileSystem.File.resolve(tmp_dir, "index.livemd")
|
|
notebook_file = FileSystem.File.resolve(tmp_dir, "notebook.livemd")
|
|
|
|
Session.set_file(session.pid, index_file)
|
|
wait_for_session_update(session.pid)
|
|
|
|
session_path = "/sessions/#{session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
|
|
assert render(view) =~
|
|
"Cannot navigate, failed to read #{notebook_file.path}, reason: no such file or directory"
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "opens a relative notebook if it exists",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
index_file = FileSystem.File.resolve(tmp_dir, "index.livemd")
|
|
notebook_file = FileSystem.File.resolve(tmp_dir, "notebook.livemd")
|
|
|
|
Session.set_file(session.pid, index_file)
|
|
wait_for_session_update(session.pid)
|
|
|
|
:ok = FileSystem.File.write(notebook_file, "# Sibling notebook")
|
|
|
|
assert {:error, {:live_redirect, %{to: new_session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Sibling notebook"
|
|
|
|
"/sessions/" <> session_id = new_session_path
|
|
{:ok, session} = Sessions.fetch_session(session_id)
|
|
data = Session.get_data(session.pid)
|
|
assert data.file == notebook_file
|
|
|
|
Session.close(session.pid)
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "if the current session has no path, forks the relative notebook",
|
|
%{conn: conn, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
index_file = FileSystem.File.resolve(tmp_dir, "index.livemd")
|
|
notebook_file = FileSystem.File.resolve(tmp_dir, "notebook.livemd")
|
|
|
|
{:ok, session} = Sessions.create_session(origin: {:file, index_file})
|
|
|
|
:ok = FileSystem.File.write(notebook_file, "# Sibling notebook")
|
|
|
|
assert {:error, {:live_redirect, %{to: new_session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Sibling notebook - fork"
|
|
|
|
"/sessions/" <> session_id = new_session_path
|
|
{:ok, new_session} = Sessions.fetch_session(session_id)
|
|
data = Session.get_data(new_session.pid)
|
|
assert data.file == nil
|
|
assert data.origin == {:file, notebook_file}
|
|
|
|
Session.close([session.pid, new_session.pid])
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "if the notebook is already open, redirects to the session",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
index_file = FileSystem.File.resolve(tmp_dir, "index.livemd")
|
|
notebook_file = FileSystem.File.resolve(tmp_dir, "notebook.livemd")
|
|
|
|
Session.set_file(session.pid, index_file)
|
|
wait_for_session_update(session.pid)
|
|
|
|
:ok = FileSystem.File.write(notebook_file, "# Sibling notebook")
|
|
|
|
assert {:error, {:live_redirect, %{to: session_path}}} =
|
|
live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
assert {:error, {:live_redirect, %{to: ^session_path}}} =
|
|
live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
"/sessions/" <> session_id = session_path
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "handles nested paths", %{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
parent_file = FileSystem.File.resolve(tmp_dir, "parent.livemd")
|
|
child_dir = FileSystem.File.resolve(tmp_dir, "dir/")
|
|
child_file = FileSystem.File.resolve(child_dir, "child.livemd")
|
|
|
|
Session.set_file(session.pid, parent_file)
|
|
wait_for_session_update(session.pid)
|
|
|
|
:ok = FileSystem.File.write(child_file, "# Child notebook")
|
|
|
|
assert {:error, {:live_redirect, %{to: "/sessions/" <> session_id}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/dir/child.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Child notebook"
|
|
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "handles parent paths", %{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
parent_file = FileSystem.File.resolve(tmp_dir, "parent.livemd")
|
|
child_dir = FileSystem.File.resolve(tmp_dir, "dir/")
|
|
child_file = FileSystem.File.resolve(child_dir, "child.livemd")
|
|
|
|
Session.set_file(session.pid, child_file)
|
|
wait_for_session_update(session.pid)
|
|
|
|
:ok = FileSystem.File.write(parent_file, "# Parent notebook")
|
|
|
|
assert {:error, {:live_redirect, %{to: "/sessions/" <> session_id}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/__parent__/parent.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Parent notebook"
|
|
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
test "resolves remote URLs", %{conn: conn} do
|
|
bypass = Bypass.open()
|
|
|
|
Bypass.expect_once(bypass, "GET", "/notebook.livemd", fn conn ->
|
|
conn
|
|
|> Plug.Conn.put_resp_content_type("text/plain")
|
|
|> Plug.Conn.resp(200, "# My notebook")
|
|
end)
|
|
|
|
index_url = bypass_url(bypass.port) <> "/index.livemd"
|
|
{:ok, session} = Sessions.create_session(origin: {:url, index_url})
|
|
|
|
assert {:error, {:live_redirect, %{to: "/sessions/" <> session_id}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "My notebook"
|
|
|
|
Session.close(session.pid)
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
test "when a remote URL cannot be loaded, attempts to resolve a flat URL", %{conn: conn} do
|
|
bypass = Bypass.open()
|
|
|
|
# Multi-level path is not available
|
|
Bypass.expect_once(bypass, "GET", "/nested/path/to/notebook.livemd", fn conn ->
|
|
Plug.Conn.resp(conn, 500, "Error")
|
|
end)
|
|
|
|
# A flat path is available
|
|
Bypass.expect_once(bypass, "GET", "/notebook.livemd", fn conn ->
|
|
conn
|
|
|> Plug.Conn.put_resp_content_type("text/plain")
|
|
|> Plug.Conn.resp(200, "# My notebook")
|
|
end)
|
|
|
|
index_url = bypass_url(bypass.port) <> "/index.livemd"
|
|
{:ok, session} = Sessions.create_session(origin: {:url, index_url})
|
|
|
|
assert {:error, {:live_redirect, %{to: "/sessions/" <> session_id}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/nested/path/to/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "My notebook"
|
|
|
|
Session.close(session.pid)
|
|
close_session_by_id(session_id)
|
|
end
|
|
|
|
test "renders an error message if relative remote notebook cannot be loaded", %{conn: conn} do
|
|
bypass = Bypass.open()
|
|
|
|
Bypass.expect_once(bypass, "GET", "/notebook.livemd", fn conn ->
|
|
Plug.Conn.resp(conn, 500, "Error")
|
|
end)
|
|
|
|
index_url = bypass_url(bypass.port) <> "/index.livemd"
|
|
|
|
{:ok, session} = Sessions.create_session(origin: {:url, index_url})
|
|
|
|
session_path = "/sessions/#{session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
assert render(view) =~ "Cannot navigate, failed to download notebook from the given URL"
|
|
|
|
Session.close(session.pid)
|
|
end
|
|
|
|
test "if the remote notebook is already imported, redirects to the session",
|
|
%{conn: conn, test: test} do
|
|
test_path = test |> to_string() |> URI.encode_www_form()
|
|
index_url = "http://example.com/#{test_path}/index.livemd"
|
|
notebook_url = "http://example.com/#{test_path}/notebook.livemd"
|
|
|
|
{:ok, index_session} = Sessions.create_session(origin: {:url, index_url})
|
|
{:ok, notebook_session} = Sessions.create_session(origin: {:url, notebook_url})
|
|
|
|
notebook_session_path = "/sessions/#{notebook_session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^notebook_session_path}}} =
|
|
live(conn, ~p"/sessions/#{index_session.id}/notebook.livemd")
|
|
|
|
Session.close([index_session.pid, notebook_session.pid])
|
|
end
|
|
|
|
test "renders an error message if there are already multiple session imported from the relative URL",
|
|
%{conn: conn, test: test} do
|
|
test_path = test |> to_string() |> URI.encode_www_form()
|
|
index_url = "http://example.com/#{test_path}/index.livemd"
|
|
notebook_url = "http://example.com/#{test_path}/notebook.livemd"
|
|
|
|
{:ok, index_session} = Sessions.create_session(origin: {:url, index_url})
|
|
{:ok, notebook_session1} = Sessions.create_session(origin: {:url, notebook_url})
|
|
{:ok, notebook_session2} = Sessions.create_session(origin: {:url, notebook_url})
|
|
|
|
index_session_path = "/sessions/#{index_session.id}"
|
|
|
|
assert {:error, {:live_redirect, %{to: ^index_session_path}}} =
|
|
result = live(conn, ~p"/sessions/#{index_session.id}/notebook.livemd")
|
|
|
|
{:ok, view, _} = follow_redirect(result, conn)
|
|
|
|
assert render(view) =~
|
|
"Cannot navigate, because multiple sessions were found for #{notebook_url}"
|
|
|
|
Session.close([index_session.pid, notebook_session1.pid, notebook_session2.pid])
|
|
end
|
|
end
|
|
|
|
describe "package search" do
|
|
test "lists search entries", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/package-search")
|
|
|
|
[search_view] = live_children(view)
|
|
|
|
# Search the predefined dependencies in the embedded runtime
|
|
search_view
|
|
|> element(~s{form[phx-change="search"]})
|
|
|> render_change(%{"search" => "ja"})
|
|
|
|
page = render(view)
|
|
assert page =~ "jason"
|
|
assert page =~ "A blazing fast JSON parser and generator in pure Elixir"
|
|
assert page =~ "1.3.0"
|
|
end
|
|
end
|
|
|
|
describe "secrets" do
|
|
setup do
|
|
{:ok, hub: Livebook.Hubs.fetch_hub!(Livebook.Hubs.Personal.id())}
|
|
end
|
|
|
|
test "adds a secret from form", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
secret = build(:secret, name: "FOO", value: "123", hub_id: nil)
|
|
|
|
view
|
|
|> element(~s{form[phx-submit="save"]})
|
|
|> render_submit(%{secret: %{name: secret.name, value: secret.value, hub_id: secret.hub_id}})
|
|
|
|
assert_session_secret(view, session.pid, secret)
|
|
end
|
|
|
|
test "adds a livebook secret from form", %{conn: conn, session: session, hub: hub} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
secret = build(:secret, name: "BAR", value: "456")
|
|
|
|
view
|
|
|> element(~s{form[phx-submit="save"]})
|
|
|> render_submit(%{
|
|
secret: %{name: secret.name, value: secret.value, hub_id: secret.hub_id}
|
|
})
|
|
|
|
assert secret in Livebook.Hubs.get_secrets(hub)
|
|
end
|
|
|
|
test "syncs secrets", %{conn: conn, session: session, hub: hub} do
|
|
session_secret = insert_secret(name: "FOO", value: "123")
|
|
secret = build(:secret, name: "FOO", value: "456")
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
|
|
view
|
|
|> element(~s{form[phx-submit="save"]})
|
|
|> render_submit(%{
|
|
secret: %{name: secret.name, value: secret.value, hub_id: secret.hub_id}
|
|
})
|
|
|
|
assert_session_secret(view, session.pid, secret)
|
|
assert secret in Livebook.Hubs.get_secrets(hub)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
Session.set_secret(session.pid, session_secret)
|
|
|
|
secret = build(:secret, name: "FOO", value: "789")
|
|
|
|
view
|
|
|> element(~s{form[phx-submit="save"]})
|
|
|> render_submit(%{
|
|
secret: %{name: secret.name, value: secret.value, hub_id: secret.hub_id}
|
|
})
|
|
|
|
assert_session_secret(view, session.pid, secret)
|
|
assert secret in Livebook.Hubs.get_secrets(hub)
|
|
end
|
|
|
|
test "never syncs secrets when updating from session",
|
|
%{conn: conn, session: session, hub: hub} do
|
|
hub_secret = insert_secret(name: "FOO", value: "123")
|
|
secret = build(:secret, name: "FOO", value: "456", hub_id: nil)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
Session.set_secret(session.pid, hub_secret)
|
|
|
|
view
|
|
|> element(~s{form[phx-submit="save"]})
|
|
|> render_submit(%{secret: %{name: secret.name, value: secret.value, hub_id: secret.hub_id}})
|
|
|
|
assert_session_secret(view, session.pid, secret)
|
|
refute secret in Livebook.Hubs.get_secrets(hub)
|
|
assert hub_secret in Livebook.Hubs.get_secrets(hub)
|
|
end
|
|
|
|
test "shows the 'Add secret' button for missing secrets", %{conn: conn, session: session} do
|
|
secret = build(:secret, name: "ANOTHER_GREAT_SECRET", value: "123456", hub_id: nil)
|
|
Session.subscribe(session.id)
|
|
section_id = insert_section(session.pid)
|
|
code = ~s{System.fetch_env!("LB_#{secret.name}")}
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
assert_receive {:operation, {:add_cell_evaluation_response, _, ^cell_id, _, _}}
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view
|
|
|> element("a", "Add secret")
|
|
|> has_element?()
|
|
end
|
|
|
|
test "adding a missing secret using 'Add secret' button",
|
|
%{conn: conn, session: session, hub: hub} do
|
|
secret = build(:secret, name: "MYUNAVAILABLESECRET", value: "123456", hub_id: nil)
|
|
|
|
# Subscribe and executes the code to trigger
|
|
# the `System.EnvError` exception and outputs the 'Add secret' button
|
|
Session.subscribe(session.id)
|
|
section_id = insert_section(session.pid)
|
|
code = ~s{System.fetch_env!("LB_#{secret.name}")}
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
assert_receive {:operation, {:add_cell_evaluation_response, _, ^cell_id, _, _}}
|
|
|
|
# Enters the session to check if the button exists
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
expected_url = ~p"/sessions/#{session.id}/secrets?secret_name=#{secret.name}"
|
|
add_secret_button = element(view, "a[href='#{expected_url}']")
|
|
assert has_element?(add_secret_button)
|
|
|
|
# Clicks the button and fills the form to create a new secret
|
|
# that prefilled the name with the received from exception.
|
|
render_click(add_secret_button)
|
|
form_element = element(view, "#secrets-modal form[phx-submit='save']")
|
|
assert has_element?(form_element)
|
|
render_submit(form_element, %{secret: %{value: secret.value, hub_id: secret.hub_id}})
|
|
|
|
# Checks if the secret isn't an app secret
|
|
refute secret in Livebook.Hubs.get_secrets(hub)
|
|
|
|
# Checks if the secret exists and is inside the session,
|
|
# then executes the code cell again and checks if the
|
|
# secret value is what we expected.
|
|
assert_session_secret(view, session.pid, secret)
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id, terminal_text(output), _}}
|
|
|
|
assert output == "\e[32m\"#{secret.value}\"\e[0m"
|
|
end
|
|
|
|
test "granting access for unavailable secret using 'Add secret' button",
|
|
%{conn: conn, session: session, hub: hub} do
|
|
secret = insert_secret(name: "UNAVAILABLESECRET", value: "123456")
|
|
|
|
# Subscribe and executes the code to trigger
|
|
# the `System.EnvError` exception and outputs the 'Add secret' button
|
|
Session.subscribe(session.id)
|
|
section_id = insert_section(session.pid)
|
|
code = ~s{System.fetch_env!("LB_#{secret.name}")}
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, code)
|
|
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
assert_receive {:operation, {:add_cell_evaluation_response, _, ^cell_id, _, _}}
|
|
|
|
# Enters the session to check if the button exists
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
expected_url = ~p"/sessions/#{session.id}/secrets?secret_name=#{secret.name}"
|
|
add_secret_button = element(view, "a[href='#{expected_url}']")
|
|
assert has_element?(add_secret_button)
|
|
|
|
# Checks if the secret is persisted
|
|
assert secret in Livebook.Hubs.get_secrets(hub)
|
|
|
|
# Clicks the button and checks if the 'Grant access' banner
|
|
# is being shown, so clicks it's button to set the app secret
|
|
# to the session, allowing the user to fetches the secret.
|
|
render_click(add_secret_button)
|
|
|
|
assert render(view) =~ "in #{hub_label(secret)}. Allow this session to access it?"
|
|
|
|
grant_access_button = element(view, "#secrets-modal button", "Grant access")
|
|
render_click(grant_access_button)
|
|
|
|
# Checks if the secret exists and is inside the session,
|
|
# then executes the code cell again and checks if the
|
|
# secret value is what we expected.
|
|
assert_session_secret(view, session.pid, secret)
|
|
Session.queue_cell_evaluation(session.pid, cell_id)
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id, terminal_text(output), _}}
|
|
|
|
assert output == "\e[32m\"#{secret.value}\"\e[0m"
|
|
end
|
|
|
|
test "reloading outdated secret value", %{conn: conn, session: session} do
|
|
hub_secret = insert_secret(name: "FOO", value: "123")
|
|
Session.set_secret(session.pid, hub_secret)
|
|
|
|
{:ok, updated_hub_secret} = Livebook.Secrets.update_secret(hub_secret, %{value: "456"})
|
|
hub = Livebook.Hubs.fetch_hub!(hub_secret.hub_id)
|
|
:ok = Livebook.Hubs.update_secret(hub, updated_hub_secret)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/secrets")
|
|
|
|
assert_session_secret(view, session.pid, hub_secret)
|
|
|
|
view
|
|
|> element(~s{button[aria-label="load latest value"]})
|
|
|> render_click()
|
|
|
|
assert_session_secret(view, session.pid, updated_hub_secret)
|
|
end
|
|
|
|
test "redirects the user to update or delete a secret",
|
|
%{conn: conn, session: session, hub: hub} do
|
|
Session.subscribe(session.id)
|
|
|
|
# creates a secret
|
|
secret_name = "SECRET_TO_BE_UPDATED_OR_DELETED"
|
|
secret_value = "123"
|
|
insert_secret(name: secret_name, value: secret_value)
|
|
|
|
# receives the operation event
|
|
assert_receive {:operation, {:sync_hub_secrets, "__server__"}}
|
|
|
|
# selects the notebook's hub with team hub id
|
|
Session.set_notebook_hub(session.pid, hub.id)
|
|
|
|
# loads the session page
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
# clicks the button to edit a secret
|
|
view
|
|
|> element("#hub-#{hub.id}-secret-#{secret_name}-edit-button")
|
|
|> render_click()
|
|
|
|
# redirects to hub page and loads the modal with
|
|
# the secret name and value filled
|
|
assert_redirect(view, ~p"/hub/#{hub.id}/secrets/edit/#{secret_name}")
|
|
{:ok, view, _} = live(conn, ~p"/hub/#{hub.id}/secrets/edit/#{secret_name}")
|
|
|
|
assert render(view) =~ "Edit secret"
|
|
|
|
# fills and submits the secrets modal form
|
|
# to update the secret on team hub page
|
|
secret_new_value = "123456"
|
|
attrs = %{secret: %{name: secret_name, value: secret_new_value}}
|
|
form = element(view, "#secrets-form")
|
|
|
|
render_change(form, attrs)
|
|
render_submit(form, attrs)
|
|
|
|
# receives the operation event
|
|
assert_receive {:operation, {:sync_hub_secrets, "__server__"}}
|
|
|
|
# validates the secret
|
|
secrets = Livebook.Hubs.get_secrets(hub)
|
|
hub_secret = Enum.find(secrets, &(&1.name == secret_name))
|
|
|
|
assert hub_secret.value == secret_new_value
|
|
refute hub_secret.value == secret_value
|
|
end
|
|
end
|
|
|
|
describe "environment variables" do
|
|
test "outputs persisted env var from ets", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
section_id = insert_section(session.pid)
|
|
|
|
cell_id =
|
|
insert_text_cell(session.pid, section_id, :code, ~s{System.get_env("MY_AWESOME_ENV")})
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id,
|
|
terminal_text("\e[35mnil\e[0m"), _}}
|
|
|
|
attrs = params_for(:env_var, name: "MY_AWESOME_ENV", value: "MyEnvVarValue")
|
|
Settings.set_env_var(attrs)
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id,
|
|
terminal_text("\e[32m\"MyEnvVarValue\"\e[0m"), _}}
|
|
|
|
Settings.set_env_var(%{attrs | value: "OTHER_VALUE"})
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id,
|
|
terminal_text("\e[32m\"OTHER_VALUE\"\e[0m"), _}}
|
|
|
|
Settings.unset_env_var("MY_AWESOME_ENV")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id,
|
|
terminal_text("\e[35mnil\e[0m"), _}}
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "outputs persisted PATH delimited with os PATH env var",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
# Start the standalone runtime, to encapsulate env var changes
|
|
{:ok, runtime} = Runtime.ElixirStandalone.new() |> Runtime.connect()
|
|
Session.set_runtime(session.pid, runtime)
|
|
|
|
separator =
|
|
case :os.type() do
|
|
{:win32, _} -> ";"
|
|
_ -> ":"
|
|
end
|
|
|
|
initial_os_path = System.get_env("PATH", "")
|
|
expected_path = initial_os_path <> separator <> tmp_dir
|
|
|
|
attrs = params_for(:env_var, name: "PATH", value: tmp_dir)
|
|
Settings.set_env_var(attrs)
|
|
|
|
Session.subscribe(session.id)
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
section_id = insert_section(session.pid)
|
|
|
|
cell_id = insert_text_cell(session.pid, section_id, :code, ~s{System.get_env("PATH")})
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id, terminal_text(output), _}}
|
|
|
|
assert output == "\e[32m\"#{String.replace(expected_path, "\\", "\\\\")}\"\e[0m"
|
|
|
|
Settings.unset_env_var("PATH")
|
|
|
|
view
|
|
|> element(~s{[data-el-session]})
|
|
|> render_hook("queue_cell_evaluation", %{"cell_id" => cell_id})
|
|
|
|
assert_receive {:operation,
|
|
{:add_cell_evaluation_response, _, ^cell_id, terminal_text(output), _}}
|
|
|
|
assert output == "\e[32m\"#{String.replace(initial_os_path, "\\", "\\\\")}\"\e[0m"
|
|
end
|
|
end
|
|
|
|
describe "file management" do
|
|
@tag :tmp_dir
|
|
test "adding :attachment file entry from storage",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
Session.subscribe(session.id)
|
|
|
|
path = Path.join(tmp_dir, "image.jpg")
|
|
File.write!(path, "content")
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/storage")
|
|
|
|
view
|
|
|> element(~s{form[phx-change="set_path"]})
|
|
|> render_change(%{path: path})
|
|
|
|
# Validations
|
|
assert view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_change(%{"data" => %{"name" => "na me", "copy" => "true"}}) =~
|
|
"should contain only alphanumeric characters, dash, underscore and dot"
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{"data" => %{"name" => "image.jpg", "copy" => "true"}})
|
|
|
|
assert_receive {:operation, {:add_file_entries, _client_id, [%{name: "image.jpg"}]}}
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :attachment, name: "image.jpg"}]}} =
|
|
Session.get_data(session.pid)
|
|
|
|
assert FileSystem.File.resolve(session.files_dir, "image.jpg") |> FileSystem.File.read() ==
|
|
{:ok, "content"}
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "adding :file file entry from storage", %{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
path = Path.join(tmp_dir, "image.jpg")
|
|
File.write!(path, "content")
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/storage")
|
|
|
|
view
|
|
|> element(~s{form[phx-change="set_path"]})
|
|
|> render_change(%{path: path})
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{"data" => %{"name" => "image.jpg", "copy" => "false"}})
|
|
|
|
assert %{
|
|
notebook: %{
|
|
file_entries: [
|
|
%{
|
|
type: :file,
|
|
name: "image.jpg",
|
|
file: %FileSystem.File{
|
|
file_system_module: Livebook.FileSystem.Local,
|
|
path: ^path
|
|
}
|
|
}
|
|
]
|
|
}
|
|
} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "adding :attachment file entry from url", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
|
|
bypass = Bypass.open()
|
|
file_url = "http://localhost:#{bypass.port}/image.jpg"
|
|
|
|
Bypass.expect_once(bypass, "GET", "/image.jpg", fn conn ->
|
|
Plug.Conn.resp(conn, 200, "content")
|
|
end)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/url")
|
|
|
|
# Validations
|
|
page =
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_change(%{"data" => %{"name" => "na me", "copy" => "true", "url" => "invalid"}})
|
|
|
|
page =~ "should contain only alphanumeric characters, dash, underscore and dot"
|
|
page =~ "must be a valid url"
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{
|
|
"data" => %{"name" => "image.jpg", "copy" => "true", "url" => file_url}
|
|
})
|
|
|
|
assert_receive {:operation, {:add_file_entries, _client_id, [%{name: "image.jpg"}]}}
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :attachment, name: "image.jpg"}]}} =
|
|
Session.get_data(session.pid)
|
|
|
|
assert FileSystem.File.resolve(session.files_dir, "image.jpg") |> FileSystem.File.read() ==
|
|
{:ok, "content"}
|
|
end
|
|
|
|
test "adding :url file entry from url", %{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/url")
|
|
|
|
url = "https://example.com/image.jpg"
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{
|
|
"data" => %{"name" => "image.jpg", "copy" => "false", "url" => url}
|
|
})
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :url, name: "image.jpg", url: ^url}]}} =
|
|
Session.get_data(session.pid)
|
|
end
|
|
|
|
test "adding :attachment file entry from upload", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/upload")
|
|
|
|
# Validations
|
|
assert view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_change(%{"data" => %{"name" => "na me"}}) =~
|
|
"should contain only alphanumeric characters, dash, underscore and dot"
|
|
|
|
assert view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_change(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
view
|
|
|> file_input(~s{#add-file-entry-form}, :file, [
|
|
%{
|
|
last_modified: 1_594_171_879_000,
|
|
name: "image.jpg",
|
|
content: "content",
|
|
size: 7,
|
|
type: "text/plain"
|
|
}
|
|
])
|
|
|> render_upload("image.jpg")
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{"data" => %{"name" => "image.jpg"}})
|
|
|
|
assert_receive {:operation, {:add_file_entries, _client_id, [%{name: "image.jpg"}]}}
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :attachment, name: "image.jpg"}]}} =
|
|
Session.get_data(session.pid)
|
|
|
|
assert FileSystem.File.resolve(session.files_dir, "image.jpg") |> FileSystem.File.read() ==
|
|
{:ok, "content"}
|
|
end
|
|
|
|
test "adding :attachment file entry from unlisted files", %{conn: conn, session: session} do
|
|
for name <- ["file1.txt", "file2.txt", "file3.txt"] do
|
|
file = FileSystem.File.resolve(session.files_dir, name)
|
|
:ok = FileSystem.File.write(file, "content")
|
|
end
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/add-file/unlisted")
|
|
|
|
page = render(view)
|
|
|
|
assert page =~ "file1.txt"
|
|
assert page =~ "file2.txt"
|
|
assert page =~ "file3.txt"
|
|
|
|
view
|
|
|> element(~s{#add-file-entry-form})
|
|
|> render_submit(%{"selected_indices" => ["0", "2"]})
|
|
|
|
assert %{
|
|
notebook: %{
|
|
file_entries: [
|
|
%{type: :attachment, name: "file1.txt"},
|
|
%{type: :attachment, name: "file3.txt"}
|
|
]
|
|
}
|
|
} = Session.get_data(session.pid)
|
|
end
|
|
|
|
test "renaming :attachment file entry", %{conn: conn, session: session} do
|
|
%{files_dir: files_dir} = session
|
|
image_file = FileSystem.File.resolve(files_dir, "image.jpg")
|
|
:ok = FileSystem.File.write(image_file, "")
|
|
Session.add_file_entries(session.pid, [%{type: :attachment, name: "image.jpg"}])
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
|
|
view
|
|
|> element(~s/[data-el-files-list] menu a/, "Rename")
|
|
|> render_click()
|
|
|
|
view
|
|
|> element(~s/#rename-file-entry-form/)
|
|
|> render_submit(%{"data" => %{"name" => "image2.jpg"}})
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :attachment, name: "image2.jpg"}]}} =
|
|
Session.get_data(session.pid)
|
|
|
|
page = view |> element(~s/[data-el-files-list]/) |> render()
|
|
assert page =~ "image2.jpg"
|
|
refute page =~ "image.jpg"
|
|
end
|
|
|
|
test "deleting :attachment file entry and removing the file from the file system",
|
|
%{conn: conn, session: session} do
|
|
%{files_dir: files_dir} = session
|
|
image_file = FileSystem.File.resolve(files_dir, "image.jpg")
|
|
:ok = FileSystem.File.write(image_file, "")
|
|
Session.add_file_entries(session.pid, [%{type: :attachment, name: "image.jpg"}])
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
|
|
view
|
|
|> element(~s/[data-el-files-list] menu button/, "Delete")
|
|
|> render_click()
|
|
|
|
render_confirm(view, delete_from_file_system: true)
|
|
|
|
assert {:ok, false} = FileSystem.File.exists?(image_file)
|
|
|
|
assert %{notebook: %{file_entries: []}} = Session.get_data(session.pid)
|
|
|
|
refute view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
end
|
|
|
|
test "deleting :attachment file entry and keeping the file in the file system",
|
|
%{conn: conn, session: session} do
|
|
%{files_dir: files_dir} = session
|
|
image_file = FileSystem.File.resolve(files_dir, "image.jpg")
|
|
:ok = FileSystem.File.write(image_file, "")
|
|
Session.add_file_entries(session.pid, [%{type: :attachment, name: "image.jpg"}])
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
|
|
view
|
|
|> element(~s/[data-el-files-list] menu button/, "Delete")
|
|
|> render_click()
|
|
|
|
render_confirm(view, delete_from_file_system: false)
|
|
|
|
assert {:ok, true} = FileSystem.File.exists?(image_file)
|
|
|
|
assert %{notebook: %{file_entries: []}} = Session.get_data(session.pid)
|
|
|
|
refute view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "transferring file entry", %{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
Session.subscribe(session.id)
|
|
|
|
tmp_dir = FileSystem.File.local(tmp_dir <> "/")
|
|
image_file = FileSystem.File.resolve(tmp_dir, "image.jpg")
|
|
:ok = FileSystem.File.write(image_file, "content")
|
|
Session.add_file_entries(session.pid, [%{type: :file, name: "image.jpg", file: image_file}])
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view |> element(~s/[data-el-files-list]/) |> render() =~ "image.jpg"
|
|
|
|
view
|
|
|> element(~s/[data-el-files-list] menu button/, "Move to attachments")
|
|
|> render_click()
|
|
|
|
assert_receive {:operation,
|
|
{:add_file_entries, _client_id, [%{type: :attachment, name: "image.jpg"}]}}
|
|
|
|
assert %{notebook: %{file_entries: [%{type: :attachment, name: "image.jpg"}]}} =
|
|
Session.get_data(session.pid)
|
|
|
|
assert {:ok, true} = FileSystem.File.exists?(image_file)
|
|
|
|
assert FileSystem.File.resolve(session.files_dir, "image.jpg") |> FileSystem.File.read() ==
|
|
{:ok, "content"}
|
|
end
|
|
|
|
test "allowing access to file entry in quarantine", %{conn: conn} do
|
|
file = Livebook.FileSystem.File.new(Livebook.FileSystem.Local.new(), p("/document.pdf"))
|
|
|
|
notebook = %{
|
|
Livebook.Notebook.new()
|
|
| file_entries: [
|
|
%{type: :file, name: "document.pdf", file: file}
|
|
],
|
|
quarantine_file_entry_names: MapSet.new(["document.pdf"])
|
|
}
|
|
|
|
{:ok, session} = Sessions.create_session(notebook: notebook)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert view
|
|
|> element(~s/[data-el-files-list]/)
|
|
|> render() =~ "Click to review access"
|
|
|
|
view
|
|
|> element(~s/[data-el-files-list] button/, "document.pdf")
|
|
|> render_click()
|
|
|
|
render_confirm(view)
|
|
|
|
refute view
|
|
|> element(~s/[data-el-files-list]/)
|
|
|> render() =~ "Click to review access"
|
|
|
|
Session.close(session.pid)
|
|
end
|
|
end
|
|
|
|
describe "apps" do
|
|
test "deploying an app", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
section_id = insert_section(session.pid)
|
|
|
|
insert_cell_with_output(
|
|
session.pid,
|
|
section_id,
|
|
terminal_text("Hello from the app!")
|
|
)
|
|
|
|
slug = Livebook.Utils.random_short_id()
|
|
|
|
Livebook.Apps.subscribe()
|
|
|
|
view
|
|
|> element(~s/[data-el-app-info] a/, "Configure")
|
|
|> render_click()
|
|
|
|
view
|
|
|> element(~s/#app-settings-modal form/)
|
|
|> render_change(%{"app_settings" => %{"slug" => slug}})
|
|
|
|
view
|
|
|> element(~s/#app-settings-modal button/, "Deploy")
|
|
|> render_click()
|
|
|
|
assert_receive {:app_created, %{slug: ^slug} = app}
|
|
|
|
assert_receive {:operation, {:set_deployed_app_slug, _client_id, ^slug}}
|
|
|
|
assert render(view) =~ "/apps/#{slug}"
|
|
|
|
{:ok, view, _} =
|
|
conn
|
|
|> live(~p"/apps/#{slug}")
|
|
|> follow_redirect(conn)
|
|
|
|
assert_receive {:app_updated,
|
|
%{slug: ^slug, sessions: [%{app_status: %{execution: :executed}}]}}
|
|
|
|
assert render(view) =~ "Hello from the app!"
|
|
|
|
Livebook.App.close(app.pid)
|
|
end
|
|
|
|
test "stopping and terminating app session", %{conn: conn, session: session} do
|
|
Session.subscribe(session.id)
|
|
|
|
slug = Livebook.Utils.random_short_id()
|
|
app_settings = %{Livebook.Notebook.AppSettings.new() | slug: slug}
|
|
Session.set_app_settings(session.pid, app_settings)
|
|
|
|
Livebook.Apps.subscribe()
|
|
Session.deploy_app(session.pid)
|
|
|
|
assert_receive {:app_created, %{slug: ^slug} = app}
|
|
|
|
assert_receive {:app_updated,
|
|
%{
|
|
slug: ^slug,
|
|
sessions: [%{app_status: %{execution: :executed}} = app_session]
|
|
}}
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
view
|
|
|> element(~s/[data-el-app-info] button[aria-label="deactivate app session"]/)
|
|
|> render_click()
|
|
|
|
assert_receive {:app_updated,
|
|
%{slug: ^slug, sessions: [%{app_status: %{lifecycle: :deactivated}}]}}
|
|
|
|
assert render(view) =~ "/apps/#{slug}/#{app_session.id}"
|
|
|
|
view
|
|
|> element(~s/[data-el-app-info] button[aria-label="terminate app session"]/)
|
|
|> render_click()
|
|
|
|
assert_receive {:app_updated, %{slug: ^slug, sessions: []}}
|
|
|
|
refute render(view) =~ "/apps/#{slug}/#{app_session.id}"
|
|
|
|
Livebook.App.close(app.pid)
|
|
end
|
|
|
|
test "shows a warning when any session secrets are defined", %{conn: conn, session: session} do
|
|
secret = build(:secret, name: "FOO", value: "456", hub_id: nil)
|
|
Session.set_secret(session.pid, secret)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}")
|
|
|
|
assert render(view) =~
|
|
"The notebook uses session secrets, but those are not available to deployed apps. Convert them to Hub secrets instead."
|
|
end
|
|
end
|
|
|
|
describe "docker deployment" do
|
|
test "instructs to choose a file when the notebook is not persisted",
|
|
%{conn: conn, session: session} do
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/app-docker")
|
|
|
|
assert render(view) =~ "To deploy this app, make sure to save the notebook first."
|
|
assert render(view) =~ ~p"/sessions/#{session.id}/settings/file"
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "instructs to change app settings when invalid",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
notebook_path = Path.join(tmp_dir, "notebook.livemd")
|
|
file = Livebook.FileSystem.File.local(notebook_path)
|
|
Session.set_file(session.pid, file)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/app-docker")
|
|
|
|
assert render(view) =~ "To deploy this app, make sure to specify valid settings."
|
|
assert render(view) =~ ~p"/sessions/#{session.id}/settings/app"
|
|
end
|
|
|
|
@tag :tmp_dir
|
|
test "shows dockerfile and allows saving it",
|
|
%{conn: conn, session: session, tmp_dir: tmp_dir} do
|
|
notebook_path = Path.join(tmp_dir, "notebook.livemd")
|
|
file = Livebook.FileSystem.File.local(notebook_path)
|
|
Session.set_file(session.pid, file)
|
|
|
|
slug = Livebook.Utils.random_short_id()
|
|
app_settings = %{Livebook.Notebook.AppSettings.new() | slug: slug}
|
|
Session.set_app_settings(session.pid, app_settings)
|
|
|
|
{:ok, view, _} = live(conn, ~p"/sessions/#{session.id}/app-docker")
|
|
|
|
assert render(view) =~ "FROM ghcr.io/livebook-dev/livebook:"
|
|
|
|
view
|
|
|> element("button", "Save alongside notebook")
|
|
|> render_click()
|
|
|
|
dockerfile_path = Path.join(tmp_dir, "Dockerfile")
|
|
|
|
assert File.read!(dockerfile_path) =~ "COPY notebook.livemd /apps"
|
|
end
|
|
end
|
|
end
|