defmodule LivebookWeb.JSOutputChannelTest do use LivebookWeb.ChannelCase, async: true setup do session_id = Livebook.Utils.random_node_aware_id() {:ok, _, socket} = LivebookWeb.Socket |> socket() |> subscribe_and_join(LivebookWeb.JSOutputChannel, "js_output", %{ "session_id" => session_id }) %{socket: socket} end test "loads initial data from the widget server and pushes to the client", %{socket: socket} do push(socket, "connect", %{"session_token" => session_token(), "ref" => "1"}) assert_receive {:connect, from, %{}} send(from, {:connect_reply, [1, 2, 3], %{ref: "1"}}) assert_push "init:1", %{"root" => [nil, [1, 2, 3]]} end test "sends client events to the corresponding widget server", %{socket: socket} do push(socket, "connect", %{"session_token" => session_token(), "ref" => "1"}) assert_receive {:connect, from, %{}} send(from, {:connect_reply, [1, 2, 3], %{ref: "1"}}) push(socket, "event", %{"root" => [["ping", "1"], [1, 2, 3]]}) assert_receive {:event, "ping", [1, 2, 3], %{origin: _origin}} end describe "binary payload" do test "initial data", %{socket: socket} do push(socket, "connect", %{"session_token" => session_token(), "ref" => "1"}) assert_receive {:connect, from, %{}} payload = {:binary, %{message: "hey"}, <<1, 2, 3>>} send(from, {:connect_reply, payload, %{ref: "1"}}) assert_push "init:1", {:binary, <<24::size(32), "[null,{\"message\":\"hey\"}]", 1, 2, 3>>} end test "form client to server", %{socket: socket} do push(socket, "connect", %{"session_token" => session_token(), "ref" => "1"}) assert_receive {:connect, from, %{}} send(from, {:connect_reply, [1, 2, 3], %{ref: "1"}}) raw = {:binary, <<32::size(32), "[[\"ping\",\"1\"],{\"message\":\"hey\"}]", 1, 2, 3>>} push(socket, "event", raw) payload = {:binary, %{"message" => "hey"}, <<1, 2, 3>>} assert_receive {:event, "ping", ^payload, %{origin: _origin}} end end defp session_token() do Phoenix.Token.sign(LivebookWeb.Endpoint, "js output", %{pid: self()}) end end