walkeros-understanding-flow
// Use when learning walkerOS architecture, understanding data flow, or designing composable event pipelines. Covers Source→Collector→Destination pattern and separation of concerns.
Understanding walkerOS Flow
Overview
walkerOS follows a Source → Collector → Destination(s) architecture for composable, modular event processing.
Core principle: Separation of concerns. Each component has one job. Components are composable and replaceable.
The Flow Pattern
Sources → [Pre-Transformers] → Collector → [Post-Transformers] → Destinations
(Capture) (source.next) (Processing) (dest.before) (Delivery)
- Browser DOM - Validation - Validation - Google Analytics
- DataLayer - Enrichment - Enrichment - Meta Pixel
- Server HTTP - Redaction - Consent check - Custom API
- Cloud Functions - Routing - Data Warehouse
Key Concepts
Composability
A Flow combines components. You can:
- Use multiple sources feeding one collector
- Route events to multiple destinations
- Swap components without changing others
The Flow Type
See packages/core/src/types/flow.ts for the canonical interface.
// Conceptual structure (see source for full type)
interface Flow {
stores?: Record<string, Store>;
sources?: Record<string, Source>;
transformers?: Record<string, Transformer>;
destinations?: Record<string, Destination>;
collector?: Collector.InitConfig;
}
Universal Push Interface
All components communicate via push functions:
| Component | Push Signature | Purpose |
|---|---|---|
| Source | push(input) → events | Capture external data |
| Collector | push(event) → void | Process and route |
| Destination | push(event, context) → void | Transform and deliver |
The elb() function is an alias for collector.push - used for component
wiring.
startFlow Helper
See packages/collector/src/flow.ts for
the startFlow function.
import { startFlow } from '@walkeros/collector';
const { collector, elb } = await startFlow({
stores: {
/* key-value storage, init first, destroy last */
},
sources: {
/* ... */
},
transformers: {
/* ... */
},
destinations: {
/* ... */
},
});
Stores
Stores are the 4th component type — passive key-value infrastructure that other
components consume via env.
- Referenced via
$store:storeIdinenvvalues (bundled mode) or passed directly as store instances (integrated mode) - Init first, destroy last — stores are available before any source, transformer, or destination starts, and outlive them on shutdown
- No chains — stores have no
nextorbefore. They don't participate in the event pipeline. Components access them through theirenv. - Implementations:
@walkeros/store-memory(sync, LRU),@walkeros/server-store-fs(async, filesystem),@walkeros/server-store-s3(async, S3-compatible)
{
"stores": {
"cache": { "package": "@walkeros/store-memory" }
},
"transformers": {
"cacheResponse": {
"package": "@walkeros/server-transformer-cache",
"env": { "store": "$store:cache" }
}
}
}
See walkeros-understanding-stores for the full store interface and lifecycle.
Separation of Concerns
| Concern | Handled By | NOT Handled By |
|---|---|---|
| Event capture | Sources | Collector, Destinations |
| Event structure | Event model | Components |
| Consent checking | Collector | Sources, Destinations |
| Transformation | Mapping system | Raw push calls |
| Delivery | Destinations | Sources, Collector |
Transformer Chains
Transformers run at two points in the pipeline, configured via next and
before:
Pre-Collector Chain
Runs after source captures event, before collector processing:
Bundled mode (flow.json):
{
"sources": {
"browser": {
"package": "@walkeros/web-source-browser",
"next": "validate"
}
},
"transformers": {
"validate": {
"package": "@walkeros/transformer-validator",
"next": "enrich"
},
"enrich": {
"package": "@walkeros/transformer-enricher"
}
}
}
Integrated mode (TypeScript):
sources: {
browser: {
code: sourceBrowser,
next: 'validate'
}
},
transformers: {
validate: {
code: transformerValidator,
config: { next: 'enrich' }
},
enrich: {
code: transformerEnrich
}
}
Note: In flow.json, next is at the reference level. The CLI bundler
automatically moves it into config.next for runtime - you don't need to handle
this yourself.
Post-Collector Chain
Runs after collector enrichment, before destination receives event:
Bundled mode (flow.json):
{
"destinations": {
"gtag": {
"package": "@walkeros/web-destination-gtag",
"before": "redact"
}
},
"transformers": {
"redact": {
"package": "@walkeros/transformer-redact"
}
}
}
Integrated mode (TypeScript):
destinations: {
gtag: {
code: destinationGtag,
before: 'redact'
}
},
transformers: {
redact: {
code: transformerRedact
}
}
Chain Resolution
source.next→ starts pre-collector chaintransformer.next(flow.json) ortransformer.config.next(runtime) → links transformersdestination.before→ starts post-collector chain per destination
Step Examples
Each step in a flow (source, transformer, destination) can ship step
examples -- structured { in, out } pairs that define expected input/output
behavior.
The Three Type Zones
Steps sit at boundaries between arbitrary formats and walkerOS events:
- Source: arbitrary
in(HTTP request, DOM event) → walkerOS eventout - Transformer: walkerOS event
in→ walkerOS eventout(orfalse) - Destination: walkerOS event
in→ arbitraryout(vendor API call)
See using-step-examples for the full ASCII diagram and detailed explanation.
Example: Step with Examples in Flow Config
{
"destinations": {
"gtag": {
"package": "@walkeros/web-destination-gtag",
"config": { "measurementId": "G-XXXXXX" },
"examples": {
"purchase": {
"in": {
"name": "order complete",
"data": { "id": "ORD-123", "total": 149.97 }
},
"out": [
"event",
"purchase",
{ "transaction_id": "ORD-123", "value": 149.97 }
]
}
}
}
}
}
Step examples enable it.each testing, CLI simulation with --example, and
deep validation with --deep. See
using-step-examples for the complete
lifecycle.
Flow Graph Connection Rules
This section defines which components can connect to which, and how chains are resolved at runtime. Use it as the canonical reference for building flow graphs, validating configurations, and rendering UI visualizations.
Valid connection matrix
| From | To | Via Field | Valid? |
|---|---|---|---|
| Source | Transformer | source.next | Yes |
| Source | Collector | (implicit, no next) | Yes |
| Source | Source | — | No |
| Source | Destination | — | No |
| Transformer | Transformer | transformer.next | Yes |
| Transformer | Collector | (implicit, pre-chain ends) | Yes |
| Transformer | Destination | (implicit, post-chain ends) | Yes |
| Collector | Destination | (implicit, no before) | Yes |
| Collector | Transformer | destination.before | Yes (post-chain) |
| Destination | anything | — | No (terminal) |
| Collector | Source | — | No |
Pre-transformer chains (source.next)
- Entry:
source.next: "transformerId"orsource.next: ["t1", "t2"] - Chaining:
transformer.next: "nextId"walks forward; array stops walking - Exit: chain ends, event reaches collector
- Multiple sources can reference the same transformer (fan-in)
- No
next= source connects directly to collector
Post-transformer chains (destination.before)
- Entry:
destination.before: "transformerId"ordestination.before: ["t1", "t2"] - Same
transformer.nextchain-walking logic as pre-chains - Exit: chain ends, event reaches destination
- Multiple destinations can share the same transformer
- No
before= collector connects directly to destination
Chain resolution algorithm (walkChain)
See packages/collector/src/transformer.ts for the implementation.
- String start: walks
transformer.nextlinks until chain ends - Array start: uses array as-is (explicit chain, no walking)
- Array
nextinside chain: appends array elements and stops walking - Circular references: detected via visited set, silently breaks loop
- Non-existent transformer ID: chain ends (no error, event proceeds without transformation)
Router fan-out
Router transformers use branch(event, next) from @walkeros/core to redirect
to different chains dynamically.
- Each route's
nexttarget creates a potential edge viawalkChain - First matching route wins; no match = passthrough
- Branch to non-existent transformer = event dropped (silent, logged as info)
See packages/core/src/branch.ts for the
branch() helper.
Transformer sharing
A single transformer can appear in both pre-chains (source.next) and
post-chains (destination.before). The same transformer pool is shared; role
depends on which chain references it.
Deferred activation (require)
source.config.require: ["consent"]— source deferred until "consent" event firesdestination.config.require: ["user"]— destination deferred until "user" event fires- Multiple requirements: all must be fulfilled (each fires independently)
Mapping and consent gating
- Source-level:
source.config.mappingandsource.config.consent— applied before pre-chain; blocks event entirely - Destination-level:
destination.config.mappinganddestination.config.consent— applied after post-chain; skips only that destination, queues denied events
Canvas rendering rules (for UI graph visualization)
- Shared transformers (pre+post): duplicate visually with a link indicator; editing one updates the other
- Router fan-out: keep graph planar; trace edges individually
- Orphan transformers (not in any chain): render grey/muted; gain color when connected
- Diamond patterns (fan-in + fan-out): expected and valid
- Overlapping
destination.beforechains: intentional (e.g., shared validator for monitoring)
Related Skills
- walkeros-understanding-events - Event model
- walkeros-understanding-sources - Source interface
- walkeros-understanding-destinations - Destination interface
Package READMEs:
- packages/collector/README.md - Collector details
Source Files:
- packages/collector/src/flow.ts - startFlow implementation
Documentation:
- Website: Flow - Flow concept
- Website: Collector - Collector docs