skills/simplify-trace/SKILL.md
Chrome DevTools performance traces are tens to hundreds of MB of JSON — far too large to read directly. This skill turns one into a few-KB markdown report that surfaces the work taking too long or happening too often.
Run the script on the trace file. It prints markdown to stdout (or --out FILE):
node skills/simplify-trace/scripts/simplify-trace.mjs <trace.json> [--top N] [--long-task-ms MS] [--window START-END] [--out report.md]
--top N — rows per table (default 25).--long-task-ms MS — long-task threshold (default 50).--window START-END — scope the whole report to a time slice (offsets in ms from trace start).--only a,b / --except a,b / --all — pick which sections to emit.--match TEXT — case-insensitive filter for rows (event names, function names, URLs).--list — print the available section keys and exit.--out FILE — write to a file instead of stdout.node --max-old-space-size=8192.Default to running with --out to a temp file for big traces, then read the report. Don't read the raw trace.
Default sections: summary, longtasks, events, frequent, functions, categories. Opt-in: timeline (per-second main-thread busy time — use to locate activity), network (resource/fetch waterfall with TTFB/duration/size), websocket (WebSocket lifecycle — /app/file doc sync). Run --list for the full set. Interrogate narrowly, e.g.:
# where is the activity? then window to it
node …/simplify-trace.mjs trace.json --only timeline
# what was the network/socket doing during the action?
node …/simplify-trace.mjs trace.json --only network,websocket --window 62000-69000 --match tldraw
The trace's metadata.startTime (ISO/UTC) anchors offset 0 to wall-clock, so trace offsets can be lined up against server logs (zero-cache, sync-worker) by timestamp. For an idle, network-quiet gap, the trace shows when but not why — add performance.mark()/console.timeStamp() in the client path and they appear on the trace timeline.
A recording of a single action (switch file, open menu) is mostly idle setup time, which dilutes the action across the whole trace. Window to the action instead:
--window START-END around that span. All tables then describe only the action.The recording artifact CpuProfiler::StartProfiling (the profiler turning on, ~50–60ms) is excluded from the long-task table, including when it's nested inside a RunTask. If long tasks shows "None", the action genuinely has no single blocking task — look at aggregate self time, GC, and animation-loop events instead.
file:line. Synthetic (idle)/(program) frames are excluded here (idle is in the header).r, Tg) come with a file:line — use it to locate the source.The script only summarizes; it does not modify the trace.