feat(ui): add replay modal to ExchangeDetail page
Add a Replay button in the exchange header that opens a modal allowing users to re-send the exchange to a live agent. The modal pre-populates headers and body from the original exchange input, provides an agent selector filtered to live agents for the application, and supports editable header key-value rows with add/remove. Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -565,3 +565,110 @@
|
|||||||
color: var(--text-faint);
|
color: var(--text-faint);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* ==========================================================================
|
||||||
|
REPLAY MODAL
|
||||||
|
========================================================================== */
|
||||||
|
.replayWarning {
|
||||||
|
background: var(--warning-bg, #3d3520);
|
||||||
|
border: 1px solid var(--warning-border, #6b5c2a);
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
padding: 10px 14px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--warning, #e6b84f);
|
||||||
|
margin-bottom: 16px;
|
||||||
|
line-height: 1.5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayAgentRow {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayFieldLabel {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text-secondary);
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayHeadersTable {
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayHeadersHead {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr 28px;
|
||||||
|
gap: 8px;
|
||||||
|
padding-bottom: 6px;
|
||||||
|
font-size: 10px;
|
||||||
|
font-weight: 600;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayHeaderRow {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr 28px;
|
||||||
|
gap: 8px;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayRemoveBtn {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 16px;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 0;
|
||||||
|
line-height: 1;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 28px;
|
||||||
|
height: 28px;
|
||||||
|
border-radius: var(--radius-sm);
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayRemoveBtn:hover {
|
||||||
|
color: var(--error);
|
||||||
|
background: var(--error-bg);
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayAddHeader {
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
color: var(--amber);
|
||||||
|
font-size: 12px;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 4px 0;
|
||||||
|
margin-top: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayAddHeader:hover {
|
||||||
|
color: var(--amber-deep);
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayBodyArea {
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayBodyTextarea {
|
||||||
|
font-family: var(--font-mono);
|
||||||
|
font-size: 12px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.replayFooter {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 8px;
|
||||||
|
margin-top: 16px;
|
||||||
|
padding-top: 12px;
|
||||||
|
border-top: 1px solid var(--border-subtle);
|
||||||
|
}
|
||||||
|
|||||||
@@ -92,6 +92,13 @@ export default function ExchangeDetail() {
|
|||||||
const [logSearch, setLogSearch] = useState('')
|
const [logSearch, setLogSearch] = useState('')
|
||||||
const [logLevels, setLogLevels] = useState<Set<string>>(new Set())
|
const [logLevels, setLogLevels] = useState<Set<string>>(new Set())
|
||||||
|
|
||||||
|
// Replay modal state
|
||||||
|
const [replayOpen, setReplayOpen] = useState(false)
|
||||||
|
const [replayHeaders, setReplayHeaders] = useState<Array<{ key: string; value: string }>>([])
|
||||||
|
const [replayBody, setReplayBody] = useState('')
|
||||||
|
const [replayAgent, setReplayAgent] = useState('')
|
||||||
|
const [replayTab, setReplayTab] = useState('headers')
|
||||||
|
|
||||||
const procList = detail
|
const procList = detail
|
||||||
? (detail.processors?.length ? detail.processors : (detail.children ?? []))
|
? (detail.processors?.length ? detail.processors : (detail.children ?? []))
|
||||||
: []
|
: []
|
||||||
@@ -263,6 +270,42 @@ export default function ExchangeDetail() {
|
|||||||
})
|
})
|
||||||
}, [detail, app, appConfig, tracingStore, updateConfig, toast])
|
}, [detail, app, appConfig, tracingStore, updateConfig, toast])
|
||||||
|
|
||||||
|
// ── Replay ─────────────────────────────────────────────────────────────
|
||||||
|
const { data: liveAgents } = useAgents('LIVE', detail?.applicationName)
|
||||||
|
const replay = useReplayExchange()
|
||||||
|
|
||||||
|
// Pre-populate replay form when modal opens
|
||||||
|
useEffect(() => {
|
||||||
|
if (!replayOpen || !detail) return
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(detail.inputHeaders ?? '{}')
|
||||||
|
const entries = Object.entries(parsed).map(([k, v]) => ({
|
||||||
|
key: k,
|
||||||
|
value: typeof v === 'string' ? v : JSON.stringify(v),
|
||||||
|
}))
|
||||||
|
setReplayHeaders(entries.length > 0 ? entries : [{ key: '', value: '' }])
|
||||||
|
} catch {
|
||||||
|
setReplayHeaders([{ key: '', value: '' }])
|
||||||
|
}
|
||||||
|
setReplayBody(detail.inputBody ?? '')
|
||||||
|
// Default to current agent if it is live
|
||||||
|
const agentIds = (liveAgents ?? []).map((a: any) => a.id)
|
||||||
|
setReplayAgent(agentIds.includes(detail.agentId) ? detail.agentId : (agentIds[0] ?? ''))
|
||||||
|
setReplayTab('headers')
|
||||||
|
}, [replayOpen])
|
||||||
|
|
||||||
|
function handleReplay() {
|
||||||
|
const headers: Record<string, string> = {}
|
||||||
|
replayHeaders.forEach((h) => { if (h.key) headers[h.key] = h.value })
|
||||||
|
replay.mutate(
|
||||||
|
{ agentId: replayAgent, headers, body: replayBody },
|
||||||
|
{
|
||||||
|
onSuccess: () => { toast({ title: 'Replay command sent', variant: 'success' }); setReplayOpen(false) },
|
||||||
|
onError: (err) => { toast({ title: `Replay failed: ${err.message}`, variant: 'error' }) },
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
// Correlation chain
|
// Correlation chain
|
||||||
const correlatedExchanges = useMemo(() => {
|
const correlatedExchanges = useMemo(() => {
|
||||||
if (!correlationData?.data || correlationData.data.length <= 1) return []
|
if (!correlationData?.data || correlationData.data.length <= 1) return []
|
||||||
@@ -369,6 +412,9 @@ export default function ExchangeDetail() {
|
|||||||
<div className={styles.headerStatLabel}>Processors</div>
|
<div className={styles.headerStatLabel}>Processors</div>
|
||||||
<div className={styles.headerStatValue}>{countProcessors(procList)}</div>
|
<div className={styles.headerStatValue}>{countProcessors(procList)}</div>
|
||||||
</div>
|
</div>
|
||||||
|
<Button variant="primary" size="sm" onClick={() => setReplayOpen(true)}>
|
||||||
|
↻ Replay
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -694,6 +740,113 @@ export default function ExchangeDetail() {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{/* Replay Modal */}
|
||||||
|
<Modal open={replayOpen} onClose={() => setReplayOpen(false)} title="Replay Exchange" size="lg">
|
||||||
|
<div className={styles.replayWarning}>
|
||||||
|
This will re-send the exchange to a live agent. The agent will process
|
||||||
|
it as a new exchange. Use with caution in production environments.
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={styles.replayAgentRow}>
|
||||||
|
<span className={styles.replayFieldLabel}>Target Agent</span>
|
||||||
|
<Select
|
||||||
|
value={replayAgent}
|
||||||
|
onChange={(e) => setReplayAgent(e.target.value)}
|
||||||
|
options={(liveAgents ?? []).map((a: any) => ({
|
||||||
|
value: a.id,
|
||||||
|
label: a.name || a.id,
|
||||||
|
}))}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Tabs
|
||||||
|
tabs={[
|
||||||
|
{ label: 'Headers', value: 'headers', count: replayHeaders.filter((h) => h.key).length },
|
||||||
|
{ label: 'Body', value: 'body' },
|
||||||
|
]}
|
||||||
|
active={replayTab}
|
||||||
|
onChange={setReplayTab}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{replayTab === 'headers' && (
|
||||||
|
<div className={styles.replayHeadersTable}>
|
||||||
|
<div className={styles.replayHeadersHead}>
|
||||||
|
<span>Key</span>
|
||||||
|
<span>Value</span>
|
||||||
|
<span />
|
||||||
|
</div>
|
||||||
|
{replayHeaders.map((h, i) => (
|
||||||
|
<div key={i} className={styles.replayHeaderRow}>
|
||||||
|
<Input
|
||||||
|
value={h.key}
|
||||||
|
placeholder="Header name"
|
||||||
|
onChange={(e) => {
|
||||||
|
const next = [...replayHeaders]
|
||||||
|
next[i] = { ...next[i], key: e.target.value }
|
||||||
|
setReplayHeaders(next)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
value={h.value}
|
||||||
|
placeholder="Value"
|
||||||
|
onChange={(e) => {
|
||||||
|
const next = [...replayHeaders]
|
||||||
|
next[i] = { ...next[i], value: e.target.value }
|
||||||
|
setReplayHeaders(next)
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={styles.replayRemoveBtn}
|
||||||
|
onClick={() => {
|
||||||
|
const next = replayHeaders.filter((_, j) => j !== i)
|
||||||
|
setReplayHeaders(next.length > 0 ? next : [{ key: '', value: '' }])
|
||||||
|
}}
|
||||||
|
aria-label="Remove header"
|
||||||
|
>
|
||||||
|
×
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className={styles.replayAddHeader}
|
||||||
|
onClick={() => setReplayHeaders([...replayHeaders, { key: '', value: '' }])}
|
||||||
|
>
|
||||||
|
+ Add header
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{replayTab === 'body' && (
|
||||||
|
<div className={styles.replayBodyArea}>
|
||||||
|
<Textarea
|
||||||
|
className={styles.replayBodyTextarea}
|
||||||
|
value={replayBody}
|
||||||
|
onChange={(e) => setReplayBody(e.target.value)}
|
||||||
|
rows={12}
|
||||||
|
resize="vertical"
|
||||||
|
placeholder="Request body (JSON, XML, text, etc.)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className={styles.replayFooter}>
|
||||||
|
<Button variant="secondary" size="sm" onClick={() => setReplayOpen(false)}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="primary"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleReplay}
|
||||||
|
loading={replay.isPending}
|
||||||
|
disabled={!replayAgent}
|
||||||
|
>
|
||||||
|
↻ Replay
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user