The Exchange
The data envelope that flows through every capability.
What is an exchange?
Every piece of data that moves through a capability is wrapped in an exchange. When a source produces data, it becomes an exchange. Every operation receives that exchange and passes it along. The destination receives it last.
An exchange has two parts:
body-- the main payload. This is your data: an object, a string, a number, whatever your capability is working with.headers-- metadata about the exchange. Timestamps, IDs, adapter-specific context, and anything you want to carry alongside the data without putting it in the body.
{
"id": "a3f4e1b2-9c6d-4e8a-b1f3-2d7c0e5a9f12",
"body": {
"to": "alice@example.com",
"subject": "Your order is confirmed"
},
"headers": {
"routecraft.correlation_id": "req-abc-123",
"routecraft.route": "send-confirmation"
}
}
Body
The body is what your operations act on. .transform(), .filter(), and .process() all receive the current body (or the full exchange) and return something new.
craft()
.id('greet')
.from(simple({ name: 'Alice' }))
.transform((body) => `Hello, ${body.name}!`) // body is { name: 'Alice' }
.to(log()) // body is now 'Hello, Alice!'
The body type flows through the DSL. TypeScript tracks what shape the body is at each step, giving you full type safety throughout the pipeline.
Headers
Headers travel alongside the body without being part of it. They are useful for metadata you want available throughout the pipeline but do not want polluting the body.
Set a header with .header():
craft()
.id('process-order')
.from(simple({ orderId: '123', amount: 49.99 }))
.header('x-tenant', 'acme-corp')
.header('x-priority', (exchange) => exchange.body.amount > 100 ? 'high' : 'normal')
.process((exchange) => {
const tenant = exchange.headers['x-tenant'] // 'acme-corp'
const priority = exchange.headers['x-priority'] // 'normal'
return exchange
})
.to(log())
Headers can be static values or derived from the exchange at runtime.
Built-in headers
RouteCraft sets a number of routecraft.* headers automatically on every exchange:
| Header | Description |
|---|---|
routecraft.exchange_id | Unique ID for this exchange |
routecraft.correlation_id | Shared ID across split/tap branches for tracing |
routecraft.route | ID of the capability that produced this exchange |
routecraft.context_id | ID of the running context |
These are useful for logging, debugging, and correlating exchanges across capability chains.
Body vs full exchange access
Most operations give you a choice: work with just the body, or the full exchange.
Body only with .transform():
.transform((body) => body.toUpperCase())
Full exchange with .process():
.process((exchange) => {
const tenantId = exchange.headers['x-tenant']
return { ...exchange, body: { ...exchange.body, tenantId } }
})
Full exchange with .filter():
.filter((exchange) => exchange.headers['x-priority'] === 'high')
Use .transform() when you only need the data. Use .process() or .filter() when you need headers, correlation IDs, or the context.
Exchange in taps
When you .tap(), the tap receives a deep copy of the exchange with a new ID. The correlation ID is preserved so you can trace the tap back to its parent exchange. The main pipeline continues immediately without waiting for the tap.
craft()
.id('order-pipeline')
.from(source)
.tap((exchange) => {
// exchange.headers['routecraft.correlation_id'] links back to the parent
auditLog.write(exchange)
})
.to(destination)
Related
Exchange headers reference
Full list of built-in routecraft.* headers.