defmodule LivebookWeb.SessionControllerTest do use LivebookWeb.ConnCase, async: true require Phoenix.LiveViewTest alias Livebook.{Sessions, Session, Notebook, FileSystem} describe "show_file" do test "returns not found when the given session does not exist", %{conn: conn} do id = Livebook.Utils.random_node_aware_id() conn = get(conn, ~p"/sessions/#{id}/files/image.jpg") assert conn.status == 404 assert conn.resp_body == "Not found" end test "returns the file when it does exist", %{conn: conn} do {:ok, session} = Sessions.create_session() :ok = FileSystem.File.resolve(session.files_dir, "test.jpg") |> FileSystem.File.write("") Session.add_file_entries(session.pid, [%{type: :attachment, name: "test.jpg"}]) conn = get(conn, ~p"/sessions/#{session.id}/files/test.jpg") assert conn.status == 200 assert get_resp_header(conn, "content-type") == ["image/jpeg"] Session.close(session.pid) end test "returns not found when file entry with this name does not exist", %{conn: conn} do {:ok, session} = Sessions.create_session() :ok = FileSystem.File.resolve(session.files_dir, "test.jpg") |> FileSystem.File.write("") conn = get(conn, ~p"/sessions/#{session.id}/files/test.jpg") assert conn.status == 404 assert conn.resp_body == "Not found" Session.close(session.pid) end test "returns not found when the given file does not exist", %{conn: conn} do {:ok, session} = Sessions.create_session() Session.add_file_entries(session.pid, [%{type: :attachment, name: "test.jpg"}]) conn = get(conn, ~p"/sessions/#{session.id}/files/test.jpg") assert conn.status == 404 assert conn.resp_body == "No such file or directory" Session.close(session.pid) end test "returns not found when file entry with this name is not an attachment", %{conn: conn} do {:ok, session} = Sessions.create_session() :ok = FileSystem.File.resolve(session.files_dir, "test.jpg") |> FileSystem.File.write("") Session.add_file_entries(session.pid, [ %{type: :url, name: "test.jpg", url: "https://example.com"} ]) conn = get(conn, ~p"/sessions/#{session.id}/files/test.jpg") assert conn.status == 404 assert conn.resp_body == "Not found" Session.close(session.pid) end end describe "download_file" do test "returns not found when the given session does not exist", %{conn: conn} do id = Livebook.Utils.random_node_aware_id() conn = get(conn, ~p"/sessions/#{id}/download/files/data.csv") assert conn.status == 404 assert conn.resp_body == "Not found" end test "returns not found when file entry with this name does not exist", %{conn: conn} do {:ok, session} = Sessions.create_session() conn = get(conn, ~p"/sessions/#{session.id}/download/files/data.csv") assert conn.status == 404 assert conn.resp_body == "Not found" Session.close(session.pid) end test "returns the file contents when it does exist", %{conn: conn} do {:ok, session} = Sessions.create_session() :ok = FileSystem.File.resolve(session.files_dir, "data.csv") |> FileSystem.File.write("hello") Session.add_file_entries(session.pid, [%{type: :attachment, name: "data.csv"}]) conn = get(conn, ~p"/sessions/#{session.id}/download/files/data.csv") assert conn.status == 200 assert get_resp_header(conn, "content-disposition") == [~s/attachment; filename="data.csv"/] assert get_resp_header(conn, "content-type") == ["text/csv"] assert conn.resp_body == "hello" Session.close(session.pid) end test "downloads remote file to cache and returns the file contents", %{conn: conn} do bypass = Bypass.open() url = "http://localhost:#{bypass.port}/data.csv" Bypass.expect_once(bypass, "GET", "/data.csv", fn conn -> Plug.Conn.resp(conn, 200, "hello") end) {:ok, session} = Sessions.create_session() Session.add_file_entries(session.pid, [%{type: :url, name: "data.csv", url: url}]) conn = get(conn, ~p"/sessions/#{session.id}/download/files/data.csv") assert conn.status == 200 assert get_resp_header(conn, "content-disposition") == [~s/attachment; filename="data.csv"/] assert get_resp_header(conn, "content-type") == ["text/csv"] assert conn.resp_body == "hello" Session.close(session.pid) end end describe "download_source" do test "returns not found when the given session does not exist", %{conn: conn} do id = Livebook.Utils.random_node_aware_id() conn = get(conn, ~p"/sessions/#{id}/download/export/livemd") assert conn.status == 404 assert conn.resp_body == "Not found" end test "returns bad request when given an invalid format", %{conn: conn} do {:ok, session} = Sessions.create_session() conn = get(conn, ~p"/sessions/#{session.id}/download/export/invalid") assert conn.status == 400 assert conn.resp_body == "Invalid format, supported formats: livemd, exs" Session.close(session.pid) end test "handles live markdown notebook source", %{conn: conn} do {:ok, session} = Sessions.create_session() conn = get(conn, ~p"/sessions/#{session.id}/download/export/livemd") assert conn.status == 200 assert get_resp_header(conn, "content-disposition") == [~s/attachment; filename="untitled_notebook.livemd"/] assert get_resp_header(conn, "content-type") == ["text/plain"] assert conn.resp_body == """ # Untitled notebook ## Section ```elixir ``` """ Session.close(session.pid) end test "includes output in markdown when include_outputs parameter is set", %{conn: conn} do notebook = %{ Notebook.new() | name: "My Notebook", sections: [ %{ Notebook.Section.new() | name: "Section 1", cells: [ %{ Notebook.Cell.new(:code) | source: """ IO.puts("hey")\ """, outputs: [{0, %{type: :terminal_text, text: "hey", chunk: true}}] } ] } ] } {:ok, session} = Sessions.create_session(notebook: notebook) conn = get(conn, ~p"/sessions/#{session.id}/download/export/livemd?include_outputs=true") assert conn.status == 200 assert get_resp_header(conn, "content-disposition") == [~s/attachment; filename="my_notebook.livemd"/] assert get_resp_header(conn, "content-type") == ["text/plain"] assert conn.resp_body == """ # My Notebook ## Section 1 ```elixir IO.puts("hey") ``` ``` hey ``` """ Session.close(session.pid) end test "handles elixir notebook source", %{conn: conn} do {:ok, session} = Sessions.create_session() conn = get(conn, ~p"/sessions/#{session.id}/download/export/exs") assert conn.status == 200 assert get_resp_header(conn, "content-disposition") == [~s/attachment; filename="untitled_notebook.exs"/] assert get_resp_header(conn, "content-type") == ["text/plain"] assert conn.resp_body == """ # Run as: iex --dot-iex path/to/notebook.exs # Title: Untitled notebook # ── Section ── """ Session.close(session.pid) end end describe "show_asset" do test "fetches assets and redirects to the session-less path", %{conn: conn} do %{notebook: notebook, hash: hash} = notebook_with_js_output() conn = start_session_and_request_asset(conn, notebook, hash) assert redirected_to(conn, 301) == ~p"/public/sessions/assets/#{hash}/main.js" {:ok, asset_path} = Session.local_asset_path(hash, "main.js") assert File.exists?(asset_path) end test "fetches assets and redirects even on empty asset directories", %{conn: conn} do %{notebook: notebook, hash: hash} = notebook_with_js_output() assets_path = Session.local_assets_path(hash) File.mkdir_p!(assets_path) conn = start_session_and_request_asset(conn, notebook, hash) assert redirected_to(conn, 301) == ~p"/public/sessions/assets/#{hash}/main.js" assert File.exists?(Path.join(assets_path, "main.js")) end test "skips the session if assets are in cache", %{conn: conn} do %{notebook: notebook, hash: hash} = notebook_with_js_output() # Fetch the assets for the first time start_session_and_request_asset(conn, notebook, hash) # Use nonexistent session, so any communication would fail random_session_id = Livebook.Utils.random_node_aware_id() conn = get(conn, ~p"/public/sessions/#{random_session_id}/assets/#{hash}/main.js") assert redirected_to(conn, 301) == ~p"/public/sessions/assets/#{hash}/main.js" end end describe "show_cached_asset" do test "returns not found when no matching assets are in the cache", %{conn: conn} do %{notebook: _notebook, hash: hash} = notebook_with_js_output() conn = get(conn, ~p"/public/sessions/assets/#{hash}/main.js") assert conn.status == 404 assert conn.resp_body == "Not found" end test "returns the requests asset if available in cache", %{conn: conn} do %{notebook: notebook, hash: hash} = notebook_with_js_output() # Fetch the assets for the first time start_session_and_request_asset(conn, notebook, hash) conn = get(conn, ~p"/public/sessions/assets/#{hash}/main.js") assert conn.status == 200 assert "export function init(" <> _ = conn.resp_body end test "supports gzip compression", %{conn: conn} do %{notebook: notebook, hash: hash} = notebook_with_js_output() start_session_and_request_asset(conn, notebook, hash) conn = conn |> put_req_header("accept-encoding", "gzip") |> get(~p"/public/sessions/assets/#{hash}/main.js") assert conn.status == 200 assert "export function init(" <> _ = :zlib.gunzip(conn.resp_body) end end describe "show_audio_image" do @tag :tmp_dir test "given :wav input returns the audio binary", %{conn: conn, tmp_dir: tmp_dir} do {session, input_id} = start_session_with_audio_input(:wav, "wav content", tmp_dir) {:ok, view, _} = Phoenix.LiveViewTest.live(conn, ~p"/sessions/#{session.id}") token = LivebookWeb.SessionHelpers.generate_input_token(view.pid, input_id) conn = get(conn, ~p"/sessions/audio-input/#{token}") assert conn.status == 200 assert conn.resp_body == "wav content" assert get_resp_header(conn, "accept-ranges") == ["bytes"] Session.close(session.pid) end @tag :tmp_dir test "given :wav input supports range requests", %{conn: conn, tmp_dir: tmp_dir} do {session, input_id} = start_session_with_audio_input(:wav, "wav content", tmp_dir) {:ok, view, _} = Phoenix.LiveViewTest.live(conn, ~p"/sessions/#{session.id}") token = LivebookWeb.SessionHelpers.generate_input_token(view.pid, input_id) conn = conn |> put_req_header("range", "bytes=4-") |> get(~p"/sessions/audio-input/#{token}") assert conn.status == 206 assert conn.resp_body == "content" assert get_resp_header(conn, "content-range") == ["bytes 4-10/11"] Session.close(session.pid) end @tag :tmp_dir test "given :pcm_f32 input returns a WAV binary", %{conn: conn, tmp_dir: tmp_dir} do {session, input_id} = start_session_with_audio_input(:pcm_f32, "pcm content", tmp_dir) {:ok, view, _} = Phoenix.LiveViewTest.live(conn, ~p"/sessions/#{session.id}") token = LivebookWeb.SessionHelpers.generate_input_token(view.pid, input_id) conn = get(conn, ~p"/sessions/audio-input/#{token}") assert conn.status == 200 assert <<_header::44-binary, "pcm content">> = conn.resp_body assert get_resp_header(conn, "accept-ranges") == ["bytes"] Session.close(session.pid) end @tag :tmp_dir test "given :pcm_f32 input supports range requests", %{conn: conn, tmp_dir: tmp_dir} do {session, input_id} = start_session_with_audio_input(:pcm_f32, "pcm content", tmp_dir) {:ok, view, _} = Phoenix.LiveViewTest.live(conn, ~p"/sessions/#{session.id}") token = LivebookWeb.SessionHelpers.generate_input_token(view.pid, input_id) conn = conn |> put_req_header("range", "bytes=48-") |> get(~p"/sessions/audio-input/#{token}") assert conn.status == 206 assert conn.resp_body == "content" assert get_resp_header(conn, "content-range") == ["bytes 48-54/55"] Session.close(session.pid) end end describe "show_input_image" do @tag :tmp_dir test "returns the image binary", %{conn: conn, tmp_dir: tmp_dir} do {session, input_id} = start_session_with_image_input(:rgb, "rgb content", tmp_dir) {:ok, view, _} = Phoenix.LiveViewTest.live(conn, ~p"/sessions/#{session.id}") token = LivebookWeb.SessionHelpers.generate_input_token(view.pid, input_id) conn = get(conn, ~p"/sessions/image-input/#{token}") assert conn.status == 200 assert conn.resp_body == "rgb content" Session.close(session.pid) end end defp start_session_and_request_asset(conn, notebook, hash) do {:ok, session} = Sessions.create_session(notebook: notebook) # We need runtime in place to actually copy the archive {:ok, runtime} = Livebook.Runtime.Embedded.new() |> Livebook.Runtime.connect() Session.set_runtime(session.pid, runtime) conn = get(conn, ~p"/public/sessions/#{session.id}/assets/#{hash}/main.js") Session.close(session.pid) conn end defp notebook_with_js_output() do archive_path = Path.expand("../../support/assets.tar.gz", __DIR__) hash = "test-" <> Livebook.Utils.random_id() assets_info = %{archive_path: archive_path, hash: hash, js_path: "main.js"} output = %{type: :js, js_view: %{assets: assets_info}, export: nil} notebook = %{ Notebook.new() | sections: [ %{ Notebook.Section.new() | cells: [%{Notebook.Cell.new(:code) | outputs: [{0, output}]}] } ] } %{notebook: notebook, hash: hash} end defp start_session_with_audio_input(format, binary, tmp_dir) do input = %{ type: :input, ref: "ref", id: "input1", destination: :noop, attrs: %{type: :audio, default: nil, label: "Audio", format: format, sampling_rate: 16_000} } cell = %{Notebook.Cell.new(:code) | outputs: [{1, input}]} notebook = %{Notebook.new() | sections: [%{Notebook.Section.new() | cells: [cell]}]} {:ok, session} = Sessions.create_session(notebook: notebook) source_path = Path.join(tmp_dir, "audio.bin") File.write!(source_path, binary) {:ok, file_ref} = Session.register_file(session.pid, source_path, "key") Session.set_input_value(session.pid, "input1", %{ file_ref: file_ref, sampling_rate: 16_000, num_channels: 1, format: format }) {session, input.id} end defp start_session_with_image_input(format, binary, tmp_dir) do input = %{ type: :input, ref: "ref", id: "input1", destination: :noop, attrs: %{type: :image, default: nil, label: "Image", format: :rgb, size: nil, fit: :contain} } cell = %{Notebook.Cell.new(:code) | outputs: [{1, input}]} notebook = %{Notebook.new() | sections: [%{Notebook.Section.new() | cells: [cell]}]} {:ok, session} = Sessions.create_session(notebook: notebook) source_path = Path.join(tmp_dir, "image.bin") File.write!(source_path, binary) {:ok, file_ref} = Session.register_file(session.pid, source_path, "key") Session.set_input_value(session.pid, "input1", %{ file_ref: file_ref, height: 300, width: 300, format: format }) {session, input.id} end end