Promptfoo search-params

URL search param and hash state management. Use when adding or modifying URL search params, working with useSearchParams, setSearchParams, useSearchParamState, or navigate() with query strings or hash fragments, or fixing browser back/forward button issues.

install
source · Clone the upstream repo
git clone https://github.com/promptfoo/promptfoo
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/promptfoo/promptfoo "$T" && mkdir -p ~/.claude/skills && cp -r "$T/.claude/skills/search-params" ~/.claude/skills/promptfoo-promptfoo-search-params && rm -rf "$T"
manifest: .claude/skills/search-params/skill.md
source content

URL Search Param State Management

Decision Framework

When updating the URL (search params or hash), choose between replace and push based on whether the change represents in-page state or a user-navigable step:

Change typeExamplesHistory behavior
In-page stateFilters, sort, pagination, tab switches, search queries
replace
— don't pollute history
Navigable stepWizard progression, multi-step forms
push
— back button should return to previous step
Unsure?Ask the developer before choosing

Why this matters: Pushing in-page state changes clutters the browser history. Users clicking "back" expect to leave the page, not undo a filter toggle. This is the #1 cause of "back button is broken" bugs.

Correct Patterns

Single search param — use
useSearchParamState
(preferred)

This hook validates with Zod and always uses

replace: true
internally, so you get correct history behavior for free.

import { useSearchParamState } from '@app/hooks/useSearchParamState';
import { z } from 'zod';

const TabSchema = z.enum(['overview', 'details', 'settings']);
const [activeTab, setActiveTab] = useSearchParamState('tab', TabSchema, 'overview');

Key file:

src/app/src/hooks/useSearchParamState.ts

Multiple search params — use
setSearchParams
with
replace: true

When updating multiple params at once, use

setSearchParams
directly but always pass
{ replace: true }
for in-page state:

const [searchParams, setSearchParams] = useSearchParams();

// Updating filters (in-page state → replace)
setSearchParams(
  (params) => {
    params.set('status', 'active');
    params.set('sort', 'name');
    return params;
  },
  { replace: true },
);

Hash-based navigation —
navigate()
with replace or push

For wizard/multi-step flows where back button should traverse steps, use push (the default):

// Wizard step navigation — push so back button works between steps
// See: src/app/src/pages/redteam/setup/page.tsx
const updateHash = (newStep: string) => {
  navigate(`#${newStep}`); // push (default) — intentional
};

For hash changes that represent in-page state, use replace:

// Tab switch on a detail page — replace to avoid history clutter
navigate(`#${section}`, { replace: true });

URL normalization after save

When the URL needs to be updated to include a new ID after a create/save operation (not a user action), use replace:

// After first save, update URL to include new ID without adding history entry
navigate(`/evals/${newConfigId}`, { replace: true });

Anti-Patterns

Pushing in-page state changes (breaks back button)

// WRONG — every filter change adds a history entry
setSearchParams((params) => {
  params.set('filter', value);
  return params;
});

// WRONG — navigate without replace for state change
navigate(`?tab=${newTab}`);

Using raw
useSearchParams
for a single param without validation

// WRONG — no validation, easy to forget { replace: true }
const [searchParams, setSearchParams] = useSearchParams();
const tab = searchParams.get('tab');
const setTab = (v: string) => {
  setSearchParams((p) => {
    p.set('tab', v);
    return p;
  });
};

// RIGHT — use the hook instead
const [tab, setTab] = useSearchParamState('tab', TabSchema, 'overview');

Using empty strings instead of null

// WRONG — useSearchParamState will throw an invariant error
setTab('');

// RIGHT — use null to clear a param
setTab(null);

Key Files

  • src/app/src/hooks/useSearchParamState.ts
    — primary hook (uses replace internally)
  • src/app/src/pages/eval/components/ResultsView.tsx
    — example of correct
    { replace: true }
    usage
  • src/app/src/pages/redteam/setup/page.tsx
    — example of intentional push for wizard steps