triaging-visual-review-runsद्वारा posthog
Visual Review is PostHog's screenshot-regression product: CI captures storybook + playwright screenshots, diffs them against committed baseline hashes, and gates the PR until a human approves the visible changes. A PR with visual changes carries a visual-review GitHub status check that stays red until each diffed snapshot is approved or tolerated in the VR UI .
npx skills add https://github.com/posthog/ai-plugin --skill triaging-visual-review-runsTriaging visual review runs
Visual Review is PostHog's screenshot-regression product: CI captures storybook + playwright screenshots,
diffs them against committed baseline hashes, and gates the PR until a human approves the visible changes.
A PR with visual changes carries a visual-review GitHub status check that stays red until each diffed
snapshot is approved or tolerated in the VR UI.
This skill teaches an agent how to answer the questions a human reviewer would actually ask, by chaining
the read-only VR MCP tools — instead of reaching for gh pr view and tab-hopping to the VR web UI.
When this skill applies
Trigger this skill on any of:
- A PR number, branch name, or commit SHA paired with words like visual review, VR, snapshot, screenshot, storybook diff, playwright snapshot, baseline, approve, tolerated, quarantine.
- Questions about why a PR is blocked, what visually changed, or whether a diff is real.
- "Is my run done?" / "What's left to review?" / "Has this story flaked recently?"
- A failing
visual-reviewGitHub check or a PR comment from theposthog-botmentioning visual review.
When the user asks for the rendered diff image itself, the VR web UI is faster — direct them there. This skill is for everything around the diff: status, scope, history, triage.
Tools
All read-only. None of these require write scopes; approval/toleration still happens in the web UI.
| Tool | Purpose |
|---|---|
posthog:visual-review-runs-list | List runs, filter by pr_number / commit_sha / branch / review_state. Start here. |
posthog:visual-review-runs-retrieve | Full detail for a single run (status, summary counts, supersession). |
posthog:visual-review-runs-snapshots-list | Per-snapshot results inside a run: identifier, result, diff %, classification, baseline + current artifact URLs. |
posthog:visual-review-runs-snapshot-history-list | A single story's last N runs across master/PRs — the flake check. |
posthog:visual-review-runs-counts-retrieve | Aggregate counts for queue triage (how many runs in needs_review, etc.). |
posthog:visual-review-runs-tolerated-hashes-list | Hashes the team has explicitly accepted as "known flake / acceptable variation". |
posthog:visual-review-repos-list | Repos (one per GitHub repo) — usually only one matters; useful for filtering. |
posthog:visual-review-repos-retrieve | Repo metadata: baseline file paths, PR-comment configuration. |
Vocabulary cheat sheet
These appear in tool output and matter for interpretation:
- Run
review_state:needs_review(open, awaiting human),clean(zero diffs),processing(CI still uploading),stale(a newer run on the same PR has superseded this one — checksuperseded_by_id). - Run
run_type:storybook(component snapshots) orplaywright(full-page e2e snapshots). - Snapshot
result:unchanged,changed(real diff),new(no baseline yet),removed. - Snapshot
classification_reason:tolerated_hash(matches a known-tolerated hash, no action needed),below_threshold(under the noise floor),exact(byte-identical),""(real diff requiring review). - Snapshot
review_state:pendingorapproved. - Run
summary:total / changed / new / removed / unchanged / unresolved / tolerated_matched—unresolvedis what's actually blocking review.
Workflows
"What's the VR status of this PR?"
The single most common job. Map a PR number to its run state in two calls.
posthog:visual-review-runs-list { pr_number: <n>, limit: 5 }— sort bycreated_atdesc, take the latest non-stale one.- If the run has
summary.changed > 0orsummary.unresolved > 0, drill in:posthog:visual-review-runs-snapshots-list { id: <run_id> }and report thechangedsnapshots.
Report back: PR number, run UUID, review_state, summary counts, and the _posthogUrl deep link so the
user can click straight to the diff viewer.
"Is the diff real or unrelated?"
The most useful judgment a code-aware agent can add. Combine three signals: scope match, flake history, and the actual rendered images. The agent should look at the screenshots — not just describe metadata.
-
Scope check —
git diff master...HEAD --stat(or against the PR's base branch) → list of touched paths. Cross-reference withposthog:visual-review-runs-snapshots-list { id }filtered toresult: changed→ story identifiers. Stories are namespaced like<area>-<scene>--<story>--<theme>; e.g.scenes-app-settings-user--settings-user-profile--darkmaps tofrontend/src/scenes/settings/user/.... Use this to translate story id → likely source path. -
Visual inspection — for each
changedsnapshot, the tool result containscurrent_artifact.download_urlandbaseline_artifact.download_url. These are pre-signed S3 URLs to PNG files; pull them and look:curl -s -o /tmp/vr-baseline.png "<baseline_artifact.download_url>" curl -s -o /tmp/vr-current.png "<current_artifact.download_url>"Then
Readboth files (the Read tool renders images visually) and compare. Things to call out:- The actual visible delta (text changed, button moved, layout shift, color drift, missing element).
- Whether the change is consistent with the diff_pixel_count and diff_percentage in the metadata (e.g. 54% diff but the images look near-identical → screenshot framing changed, not the UI).
- Whether the baseline and current have different dimensions (
width/heightfields). Mismatched dimensions usually mean the story rendered to a different viewport or didn't fully render before screenshot — a flake signal, not a regression.
-
Flake history — run the flake check below for any story that looks suspect.
-
Verdict — combine all three:
- Scope plausible + visible regression matches the code change → real diff, recommend approval.
- Scope mismatch + dimensions mismatch + frequent prior changes → flake, recommend tolerating the hash.
- Scope plausible + visible regression looks unintended → push a fix; do not approve.
Always include a one-line description of what you saw in the images — the user uses this to decide whether to trust your verdict without opening the VR UI themselves.
Flake check: "Has this story been changing?"
Once you have a suspect snapshot identifier:
posthog:visual-review-runs-snapshot-history-list { id: <snapshot_id> } → returns prior outcomes for the same story.
Verdicts:
- Mostly
unchangedand this run's diff is the outlier → likely a real regression caused by this PR. - Frequent
changedacross unrelated branches/master → flaky story; recommend tolerating the hash via the UI. - Recent
removedor large-jump dimension change → baseline likely stale; recommend re-baselining on master.
Triaging the queue
When the user is doing housekeeping rather than asking about a specific PR:
posthog:visual-review-runs-counts-retrieve→ total queue size.posthog:visual-review-runs-list { review_state: needs_review, limit: 50 }(paginate if needed).- Group by
branchauthor orrun_typeto surface clusters (e.g., "12 PRs blocked on the same shared component change" usually means a single underlying root cause to address). - Prefer surfacing runs whose
summary.changed > 0over runs that are onlynew—newmeans no baseline yet, which is usually trivial to approve;changedis the real review work.
Output expectations
For PR-status questions, lead with the verdict in one line, then 2-4 bullets of supporting context. Always
include the _posthogUrl deep link to the run — humans need to see the rendered images to make the call,
the agent can only describe the metadata.
For triage / aggregate questions, a short table beats prose. Group by what the user is going to act on.
What NOT to do
- Do not approve or tolerate snapshots from this skill — those endpoints are intentionally not exposed as
MCP tools yet. Direct the user to the run's
_posthogUrl. - Do not assume the failing GitHub check on a PR is unrelated to VR — if a
visual-reviewcheck is red on a PR you're working on, that's the trigger to run this skill. - Do not declare a verdict from metadata alone when
result: changed. Pull the baseline and current PNGs and look at them; metadata can only say "something changed", not whether the change is intended.