Skip to main content

Documentation Index

Fetch the complete documentation index at: https://docs.velt.dev/llms.txt

Use this file to discover all available pages before exploring further.

This is the deep-dive reference. Setup covers the happy path. This page covers everything you reach for when you need precise control over node behavior, edge routing, quorum policies, SLA breaches, and event consumption.

Node configuration

Every node has a nodeId, a type (agent or human), a config block, and an optional slaMs deadline.

Agent nodes

FieldTypeNotes
agentIdstringRequired.
promptOverridestring≤ 8000 chars.
inputMappingobjectPass step inputs to the agent.
blockingbooleanDefault false. When true, the step parks in waiting until external resolutions arrive via /steps/recordAgentResolution.
resolutionPolicyobjectRequired when blocking: true. { kind: "allResolved" | "minResolved", minCount?: integer }. minCount is required when kind === "minResolved".
agentMaxRuntimeMsinteger≤ 86400000.
requireNonEmptyOutputbooleanFail the step if the agent returns an empty output.
{
  "nodeId": "brand-check",
  "type": "agent",
  "config": {
    "agentId": "brand-agent-v1",
    "blocking": false,
    "requireNonEmptyOutput": true
  },
  "slaMs": 3600000
}

Human nodes

Exactly one of reviewers[] (preferred) or reviewerIds[] (legacy) must be provided.
FieldTypeNotes
reviewersarrayPreferred: [{ userId, mandatory }]. Must include at least one mandatory: true. UserIds must be unique.
reviewerIdsstring[]Legacy. Accepted for back-compat only.
commentBodystring≤ 8000 chars. Initial comment surfaced to reviewers.
{
  "nodeId": "human-legal",
  "type": "human",
  "config": {
    "reviewers": [{ "userId": "u_legal_01", "mandatory": true }],
    "commentBody": "Please review for legal compliance."
  }
}

Edge gating expressions

Edges can carry an optional when expression evaluated against the source step’s output. Expressions compile at write time (pure AST, no eval) and walk at runtime. No untrusted code ever runs.
{ "from": "brand-check", "to": "legal-review", "when": "output.passesBrandCheck == true" }
Supported operators: equality, comparison, boolean, regex, includes, startsWith, endsWith, length, isEmpty. Path roots:
RootResolves to
output.*The source step’s output object.
step.*The source step’s metadata (status, timing).
execution.input.*The triggerContext you passed on dispatch.
If when is omitted, the edge always fires.

SLA and breach handling

Set slaMs on any node to give the step a deadline. If the step doesn’t complete within the window, it transitions to breached and emits a step.breached event. To handle breaches, declare an outgoing edge that routes on the breached status. Otherwise the engine emits the missing-breach-edge linter rule and rejects the definition. Silent dead-ends are a bug.

Parallel groups and quorum policies

A parallel group declares a set of member nodes that conceptually run in parallel and share an approval threshold.
{
  "groupId": "parallel-review",
  "memberNodeIds": ["human-legal", "human-brand"],
  "expectedSteps": 2,
  "quorum": 2,
  "onQuorumMet": "waitAll"
}
FieldTypeRequiredDescription
groupIdstringyes1 to 64 chars. Stable identifier.
memberNodeIdsstring[]yes1 to 500 nodes. Each must be declared as a top-level node. A node may belong to at most one group.
expectedStepsintegeryes1 to 500. Total members the group expects to terminate. Must equal memberNodeIds.length.
quorumintegeryes1 to expectedSteps. The number of approvals required to fire the policy’s side effect.
onQuorumMetenumnowaitAll (default), cancelOnQuorum, or joinOnQuorum.
requiredNodeIdsstring[]noSpecific members whose approval is required for quorum to be met. Every entry must also be in memberNodeIds. length must be <= quorum.

Quorum is approval count, not completion count

A member counts as an approval when its terminal status is completed AND its output.decision === 'approve'. Rejections, failures, breaches, and cancellations contribute to the completion counter only, never the approval counter. Two consequences:
  1. Non-blocking agent members never satisfy quorum. They have no decision concept. Only place human or blocking agent nodes inside groups whose policy is cancelOnQuorum or joinOnQuorum.
  2. A reject does not block the group from rolling up to “complete”. It just keeps approvedShards from advancing. Group-completion (expectedSteps met) and group-quorum (approval threshold) are tracked separately.

onQuorumMet policies

PolicySide effect on first-time approval-quorum-metPer-member fan-out
waitAll (default)Emits group.quorum-met event only. Group is purely informational.Each member’s outgoing edges fire on its own completion. If two members both fan out to the same downstream node, you get two downstream step instances.
cancelOnQuorumEmits group.quorum-met AND cancels every sibling member step still in waiting (system-actor cancellation, audit reason group-quorum-met).Each completing member still fans out per-edge. Cancelled siblings do not fan out.
joinOnQuorumEmits group.quorum-met, cancels waiting siblings, AND fires a single group-owned downstream fan-out: one new step per shared outgoing-edge target with deterministic stepId group_<groupId>__to__<childNodeId>. The successor’s input is { groupOutputs, groupId, quorum, totalApproved }.Suppressed for group members. The group container owns fan-out, so downstream successors run exactly once.

Specific-must-approve quorum

By default, quorum is anonymous: any N approvals out of M members trigger the policy. To express “these specific members must approve”, declare them in requiredNodeIds:
{
  "groupId": "approver-group",
  "memberNodeIds": ["legal", "finance", "brand"],
  "expectedSteps": 3,
  "quorum": 2,
  "requiredNodeIds": ["legal", "finance"]
}
Quorum-met now requires both:
  1. Every nodeId in requiredNodeIds is among the approvers, AND
  2. Total approval count reaches the numeric quorum.
In the example, brand alone approving doesn’t satisfy quorum even if quorum: 2 is reached numerically. legal AND finance must both also approve. If requiredNodeIds is omitted or empty, behavior collapses back to anonymous quorum.

Linter rules

Definitions are linted at create and update time. Any rule violation is rejected with INVALID_ARGUMENT and an explicit code in the error message.
CodeMeaning
duplicate-node-idTwo nodes share the same nodeId.
dangling-edgeEdge references a from or to that isn’t declared.
cycle-detectedThe graph contains a cycle. v1 is DAG-only.
unreachable-nodeA node has no path from any root.
node-missing-configA node has no config block.
missing-breach-edgeA node has slaMs set but no outgoing edge that routes on status == 'breached'. Breaches would silently dead-end.
group-duplicate-idTwo groups share the same groupId.
group-members-emptymemberNodeIds is empty.
group-member-missingA member references an unknown node.
group-expected-steps-invalidexpectedSteps < 1.
group-quorum-invalidquorum < 1 or quorum > expectedSteps.
group-cancelonquorum-requires-quorum-lt-expectedcancelOnQuorum requires quorum < expectedSteps.
group-joinonquorum-members-must-share-successorsjoinOnQuorum requires every member to have an identical set of outgoing-edge target nodes.
group-required-not-in-membersAn entry in requiredNodeIds is not in memberNodeIds.
group-required-exceeds-quorumrequiredNodeIds.length > quorum.
group-node-in-multiple-groupsA node appears as a member of two or more groups.

Events

For receiver setup (signature verification, security rules, delivery basics), see Setup, Configure your webhook receiver.

Event reference

Externally-visible events delivered via webhook and returned from Get Execution Events:
Event type (external)Internal nameWhen emitteddata highlights
execution.dispatchedsameExecution created. First step(s) scheduled.{ definitionId, definitionVersion, rootStepIds }
execution.completedsameAll steps terminal, no unhandled failures.null
execution.failedsameAny blocking step ended in failed or breached without a recovery edge.{ failureReason }
execution.cancelledsame/executions/cancel or full-execution rollback.{ reason? }
step.awaiting-approvalstep.waitingA human or blocking-agent step entered waiting.{ waitingForReviewers, mandatoryCount, resumeKey }
step.completedsameStep transitioned to completed.For human or blocking-agent: { aggregatorStatus, nodeType, decision, aggregatorBacked }. For non-blocking agents: { agentId }.
step.failedsameStep transitioned to failed (retry budget exhausted).{ error: { code, message } }
step.breachedsameStep exceeded its configured SLA before completing.{ reason }
step.cancelledsameStep cancelled via /steps/cancel or by quorum-met side effect.{ actorId, reason }
group.quorum-metparallel-group.quorum-metA parallel group’s approval threshold was first satisfied.{ groupId, total, quorum, completedTotal, expectedSteps }
Internal-only events (step.scheduled, step.started, step.retried, step.resumed, step.response-recorded, step.overridden, parallel-group.completed, idempotency.suppressed) fill seq gaps but are filtered from external delivery. Your stream may have non-contiguous seq values.

Cancellation reasons

step.cancelled events carry a data.reason string. This is an open string set. Consumers should switch on event.type for control flow, not on data.reason.
ReasonSourceMeaning
group-quorum-metsystemCancelled by the engine when the parent group’s approval quorum was met under cancelOnQuorum or joinOnQuorum. Audit shows actorId: "system:group-quorum".
(admin-supplied)adminFree-form reason passed to /steps/cancel.

Webhook retry policy

AttemptDelay before retry
1 (initial)n/a
22 s
38 s
432 s
52 min
68 min, then dead-letter
After 5 failed retries, the payload is written to a dead-letter queue. Recover missed events via Get Execution Events with sinceSeq. At-least-once delivery. The same eventId and seq appear on retries. Make your receiver idempotent on (executionId, seq).

Errors

All errors follow the standard envelope:
{ "error": { "message": "...", "status": "INVALID_ARGUMENT", "details": {} } }

Canonical codes

CodeMeaningTypical cause
INVALID_ARGUMENTSchema or linter failure.Missing field, wrong type, value out of range, linter rule violation.
UNAUTHENTICATEDMissing or invalid x-velt-auth-token.
PERMISSION_DENIEDAuth token valid but lacks the required scope.Non-admin attempting /steps/cancel or /steps/resolve.
NOT_FOUNDTarget doc does not exist.Unknown executionId, definitionId, or stepId.
ALREADY_EXISTSConflicting create.Creating a definition with a definitionId already in use.
FAILED_PRECONDITIONOptimistic lock or state-machine violation.ifVersion mismatch on update. Cancelling a terminal step. Deleting a definition with in-flight executions.
RESOURCE_EXHAUSTEDRate limit exceeded.Per-IP or per-API-key quota.
DEADLINE_EXCEEDEDInternal timeout.Retry with idempotency.

Schema-level validation errors

messageTrigger
webhookUrl and webhookSecret must be provided togetherDispatch supplied one but not the other.
webhookUrl must use https schemeNon-HTTPS scheme.
webhookUrl host resolves to a private, loopback, or link-local addressLiteral private IP, localhost, metadata.google.internal, or *.internal.
at least one of reviewerIds or reviewers must be providedHuman node with no reviewers.
cannot set both reviewerIds and reviewers, use oneBoth populated. Pick the modern reviewers[] form.
reviewer userIds must be uniqueDuplicate userId in reviewers[].
reviewers must include at least one mandatory reviewer (allMandatoryApproved would otherwise never resolve)Every reviewer.mandatory === false.
resolutionPolicy required when blocking === trueBlocking agent node without a policy.
minCount required when kind === "minResolved"resolutionPolicy.kind = "minResolved" with no minCount.

Rate limiting

Rate limits are applied per API key, with additional per-endpoint tiers on high-volume routes. A RESOURCE_EXHAUSTED error indicates you should back off with exponential retry. Dispatch retries are safe to replay with an idempotencyKey.

Object reference

interface ExecutionView {
  executionId: string;
  status: 'pending' | 'running' | 'completed' | 'failed' | 'cancelled';
  startedAt: number;             // epoch ms
  completedAt: number | null;
  cancelledAt: number | null;
  definitionId: string;
  definitionVersion: number;
  correlationId: string;
  idempotencyKey: string;
  failureReason: { code: string; message: string } | null;
  steps: StepView[];
}

interface StepView {
  stepId: string;
  nodeId: string;
  nodeType: 'agent' | 'human';
  status: 'pending' | 'running' | 'waiting' | 'completed' | 'failed' | 'skipped' | 'cancelled' | 'breached';
  groupId: string | null;
  startedAt: number | null;
  completedAt: number | null;
  output: Record<string, unknown>;
  error: { code: string; message: string } | null;
}

interface DefinitionView {
  definitionId: string;
  name: string;
  description: string | null;
  version: number;
  scope: { level: 'apiKey' | 'organization' | 'document'; organizationId: string | null; documentId: string | null };
  nodes: NodeView[];
  edges: EdgeView[];
  groups: ParallelGroupDef[] | null;
  triggers: WorkflowTriggerConfig[] | null;
  tags: string[] | null;
  custom: Record<string, unknown> | null;
  createdAt: number;
  updatedAt: number;
  status: 'active' | 'tombstoned';
}

interface ApprovalEventView {
  eventId: string;
  seq: number;             // monotonic per-execution
  type: string;            // external event type, see Event reference
  stepId: string | null;
  timestamp: number;       // epoch ms
  correlationId: string;
  data?: Record<string, unknown>;
}
For human steps, output (after resume) includes the aggregator rollup:
{
  reviewers: Array<{ userId: string; mandatory: boolean }>;
  reviewerIds: string[];
  reviewerEmails: string[];
  commentBody: string | null;
  aggregatorStatus: 'resolved' | 'rejected';
  approveCount: number;
  rejectCount: number;
  totalResponses: number;
  mandatoryCount: number;
  mandatoryApproveCount: number;
  decision: 'approve' | 'reject';
  approved: boolean;
  resumedAt: number;
  resumeKey: string;
}
For joinOnQuorum group successor steps, input includes:
{
  groupOutputs: Record<string /* memberNodeId */, Record<string, unknown> /* member's output */>;
  groupId: string;
  quorum: number;
  totalApproved: number;
}