TimeTravel is a record & replay debugger for Phoenix LiveView applications.
TimeTravel allows you to record the state of your LiveView as you interact with the page and replay the interactions while viewing the state of the socket's assigns at any given point in time.
By attaching to the Telemetry events Phoenix LiveView emits we are able to construct a timeline of events & state to rewind and replay as many times as you like in order to track down bugs in the socket's state.
TimeTravel was inspired by Elm Reactor and the rr-project
See the Time Travel Demo Repo for a working example
See a short example video here:
Screen.Recording.2022-12-26.at.6.53.58.PM.mov
The package can be installed by adding time_travel
to your list of dependencies in mix.exs
:
def deps do
[
{:time_travel, "~> 0.3"}
]
end
-
Download and install the Chrome Extension
-
Add
:time_travel
as an Elixir dependency inmix.exs
(see above) -
(If you have already set up Phoenix Channels) Add the
lvdbg
channel specification inyour_app_web/channels/user_socket.ex
:
## Channels
channel "lvdbg:*", TimeTravel.LiveViewDebugChannel
- If you have not set up a Phoenix socket, run
mix phx.gen.socket User
- Make sure you have a socket configuration set up in
lib/your_app_web/endpoint.ex
:
socket "/socket", YourAppWeb.UserSocket,
websocket: true,
longpoll: false
- Import time travel and declare the socket in
app.js
(before you declare the liveSocket):
import {Socket} from "phoenix"
import {TimeTravel} from "time_travel"
let timeTravel = new TimeTravel(Socket);
let csrfToken = document.querySelector("meta[name='csrf-token']").getAttribute("content")
let liveSocket = new LiveSocket("/live", Socket, {params: {_csrf_token: csrfToken}})
- Attach to the telemetry handlers in the init callback in
lib/your_app_web/telemetry.ex
:
# telemetry.ex
# init callback
:ok =
:telemetry.attach_many(
"live-view-handler",
[
[:phoenix, :live_view, :mount, :stop],
[:phoenix, :live_view, :handle_event, :stop],
[:phoenix, :live_view, :handle_params, :stop],
[:phoenix, :live_component, :update, :stop],
[:phoenix, :live_component, :handle_event, :stop]
],
&TimeTravel.TelemetryHandler.live_view_event_handler/4,
%{}
)
- Finally,
use TimeTravel
in thelive_view
andlive_component
definitions inmy_app_web.ex
:
def live_view do
quote do
use Phoenix.LiveView,
layout: {TimeTravelDemoWeb.LayoutView, "live.html"}
# Import TimeTravel handle_cast callbacks for each LiveView
use TimeTravel, :live_view
# ...
end
end
def live_component do
quote do
use Phoenix.LiveComponent
use TimeTravel, :live_component
# ...
end
end
For a full example see the Time Travel Demo Repo
- With the LiveView open, Right click > Inspect
- Press the arrows >> for more options and select LiveView DevTools
- Interact with your LiveView (LiveComponents not supported at this time)
- Drag the slider back and forth to replace the socket assigns
- If you run out of chrome storage, press the "Clear Storage" button (The Extension & Library is memory-heavy at this time)