@clearedby/sdk
A tiny, dependency-free client for the ClearedBy gate. Call gate() before a consequential agent action — policy clears it instantly, rejects it, or holds it for a human. wait() blocks until a held item resolves; guard() wraps a function so it only runs once cleared.
Works anywhere fetch exists (Node 18+, edge, browser).
Install
npm i @clearedby/sdkQuick start
import { ClearedBy, RejectedError } from '@clearedby/sdk'
const cb = new ClearedBy({ apiKey: process.env.CLEAREDBY_API_KEY! })
// Only issue the refund if the gate clears it (now or after a human approves).
try {
await cb.guard(
{ action: 'refund.create', params: { amount: 250, order: 'SO-441', currency: 'GBP' } },
async () => {
await shopify.refund('SO-441', 250)
},
)
} catch (err) {
if (err instanceof RejectedError) {
// a policy rule or a reviewer said no
console.log('refund blocked:', err.result.reason)
} else throw err
}Blocking, fire-and-forget, or poll
When an action is held for a human, you choose how to handle the wait:
Block — simplest; best for short, in-session decisions. guard() (above) or gate() + wait():
const r = await cb.gate(input)
const final = await cb.wait(r.id) // resolves when a human decides (or times out)Fire-and-forget — submit and return immediately; ClearedBy POSTs the verdict to your URL when a human decides. Best for long / out-of-hours reviews, so you never hold a request open:
const r = await cb.gate({ ...input, callbackUrl: 'https://your-app.com/hooks/clearedby' })
// r.status is 'pending' — return now. Later, ClearedBy POSTs to your hook (with retries):
// { id, status: 'cleared' | 'rejected' | 'sent_back', reason, attestation: { seq, hash } }
// Branch on status to act — your endpoint is where the work happens.Verify the callback is genuinely from ClearedBy: the gate response's r.resume.signing_secret is the HMAC key, and each POST carries X-ClearedBy-Signature: sha256=hmac(secret, rawBody) — recompute it over the raw request body and compare.
Poll — gate(input), then call status(id) whenever you like.
The send-back (revise) loop
A reviewer can return an action to revise instead of approving or rejecting it. guard() throws SentBackError (and wait()/status() report status: 'sent_back'). Catch it, regenerate, and resubmit() against the same lineage:
import { SentBackError } from '@clearedby/sdk'
let input = { action: 'refund.create', params: { amount: 250, order: 'SO-441' } }
for (;;) {
try {
await cb.guard(input, () => shopify.refund('SO-441', input.params.amount))
break
} catch (e) {
if (e instanceof SentBackError) {
input = revise(input, e.result.reason) // your regeneration, using the reviewer's note
input.parentItemId = e.result.id // keep the whole revise chain on one lineage
continue
}
throw e
}
}Proof cases & evidence
A gate call isn't just "may I do this" — the agent submits the case for it, and the reviewer judges that case. Add an optional proof to any gate() / resubmit():
await cb.gate({
action: 'refund.create',
params: { amount: 420, order: 'SO-118', currency: 'GBP' },
proof: {
reason: 'Customer reports the item arrived damaged; delivery confirmed.',
confidence: 0.9,
recommended_outcome: 'approve_refund',
risk_flags: ['full_order_value'],
evidence: [
{ type: 'threshold', value: { label: 'Refund vs order total', value: 420, limit: 400, unit: '£' } },
{ type: 'entity', value: { name: 'a.popov@example.com', subtitle: 'Order SO-118',
fields: [{ label: 'Orders', value: 14 }, { label: 'Prior refunds', value: 1 }],
badges: [{ label: 'damaged item', tone: 'warn' }] } },
{ type: 'timeline', value: [{ label: 'Delivered', at: 'Jun 9' }, { label: 'Damage reported', at: 'Jun 10' }] },
{ type: 'source', value: { title: 'Support ticket #4821', snippet: '…arrived cracked…', url: 'https://…' } },
{ type: 'conversation', value: { title: 'Support thread #4821', messages: [
{ from: 'customer', text: 'The lamp arrived cracked.', at: 'Jun 10' },
{ from: 'agent', text: 'So sorry — could you send a photo?', at: 'Jun 10' },
{ from: 'customer', text: 'Here it is:', image: 'https://…/damage.jpg', at: 'Jun 10' } ] } },
],
},
})The reviewer sees a Proof Case panel — recommended outcome, confidence, reason, risk flags, and a friendly card per evidence item. Recognised types (threshold, entity, timeline, table, media, source, conversation) render as cards — conversation is a chat/support thread shown as a message-bubble transcript; any other type falls back to a generic key/value card, so you can always invent your own. Set an optional width: 'half' | 'full' on any card to control layout — compact cards (threshold, entity) pair side-by-side by default. The typed shapes (ThresholdEvidence, EntityEvidence, …) are exported — annotate a value (or use satisfies Evidence) to get them checked. Every card type and field is catalogued in EVIDENCE.md.
ClearedBy presents your case to a human and pins it, tamper-evident, to their decision — it does not verify the truth of evidence. The cards are your assertions; the reviewer rules on them. Set verified: true on a source only when it's independently checkable (e.g. a signed object). proof is optional and fully backward-compatible.
API
new ClearedBy({ apiKey, baseUrl?, fetch? })
baseUrl defaults to https://app.clearedby.com. Pass fetch to inject an implementation.
gate(input) → GateResult
input: { action, params?, context?, proof?, policy?, mode?, callbackUrl?, timeout?, parentItemId? }. Returns { id, status: 'cleared' | 'rejected' | 'pending' | 'sent_back', … }. Omit policy to use the org default. proof is the agent's case for the action (see Proof cases & evidence). In shadow mode the call never blocks and returns { status: 'cleared', shadow: true, would: { verdict, rule } }.
status(id) → GateResult
Current state of a gated item.
wait(id, { timeoutMs?, pollMs? }) → GateResult
Blocks until the item is cleared/rejected or the timeout elapses (throws ClearedByError with status 408 on timeout).
guard(input, fn) → fn's return
Gates, then runs fn only if cleared (immediately, or after wait resolves a held item). Throws RejectedError if rejected, SentBackError if a reviewer sends it back to revise, ClearedByError on timeout. In shadow mode fn always runs.
resubmit(parentItemId, input) → GateResult
Re-open a sent-back action with revised input, linked to the original so the whole revise chain stays one auditable lineage. (Same as gate({ ...input, parentItemId }).)
complete(id, { status, executedParams?, externalRef?, evidence?, result? }) → recorded
Proof-of-execution: after you actually carry out a cleared action, record that you did it (and what you did) onto the attestation chain. status is 'done' | 'failed' | 'partial'.
Errors
ClearedByError— non-2xx from the API (.status,.code,.body) or awaittimeout.RejectedError— the action was rejected (.resultholds the gate result + reason).SentBackError— a reviewer returned the action to revise (.result.reasonis the note,.result.idis the parent toresubmit()against).
Every cleared/rejected decision is signed into your org's tamper-evident attestation chain — see the dashboard ledger.