Architecture Overview
n8n-as-code is a monorepo built with a modular architecture that separates concerns while maintaining tight integration between components.
🏗️ Monorepo Structure
n8n-as-code/
├── packages/
│ ├── cli/ # Command-line interface + embedded sync engine
│ ├── skills/ # AI agent integration
│ └── vscode-extension/ # VS Code extension
├── docs/ # Documentation (Docusaurus)
├── scripts/ # Build and utility scripts
└── plans/ # Architecture planning documents
📦 Package Dependencies
graph TD
A[VS Code Extension] --> B[CLI - incl. Sync Engine]
C[CLI] --> B
D[n8nac skills] --> E[n8n API]
B --> E
style A fill:#ff6b35
style C fill:#ff6b35
style D fill:#ff6b35
style B fill:#2c3e50
Dependency Flow
- Sync Engine (embedded in
n8nac,cli/src/core): Shared business logic - CLI (
n8nac): Command-line interface - VS Code Extension: Visual interface using
n8nac - Skills Library (
@n8n-as-code/skills, accessed vian8nac skills): AI integration
🧩 Sync Engine Architecture (inside cli)
3-Way Merge Architecture
The sync library implements a 3-way merge architecture that cleanly separates state observation from state mutation:
graph TD
A[Watcher] -->|observes| B[File System]
A -->|observes| C[n8n API]
A -->|emits status| D[SyncManager]
D -->|orchestrates| E[SyncEngine]
D -->|resolves| F[ResolutionManager]
E -->|reads/writes| B
E -->|API calls| C
E -->|updates| G[StateManager]
G -->|persists| H[.n8n-state.json]
Key Principles
- Separation of Concerns: Watcher observes, SyncEngine mutates
- 3-Way Comparison: Uses base-local-remote to detect conflicts
- Deterministic Detection: Only flags conflicts when both sides changed
- State Persistence:
.n8n-state.jsontracks last synced state (base)
Service Layer
// Sync services architecture
classDiagram
class Watcher {
+start()
+stop()
+scanLocalFiles()
+refreshRemoteState()
-calculateStatus()
}
class SyncEngine {
+push(workflow)
+pull(workflow)
+delete(workflow)
+finalizeSync(workflow)
}
class ResolutionManager {
+promptForConflict()
+promptForDeletion()
+resolveConflict()
}
class SyncManager {
+listWorkflows()
+fetch(workflowId)
+pull(workflowId)
+push(workflowId)
+resolveConflict()
+refreshLocalState()
+refreshRemoteState()
}
class StateManager {
+loadState()
+saveState()
+updateWorkflowState()
}
class N8nApiClient {
+getWorkflows()
+getWorkflow()
+updateWorkflow()
+createWorkflow()
}
class WorkflowSanitizer {
+sanitize()
+validate()
+sortNodes()
}
SyncManager --> Watcher
SyncManager --> SyncEngine
SyncManager --> ResolutionManager
SyncEngine --> StateManager
SyncEngine --> N8nApiClient
SyncEngine --> WorkflowSanitizer
Watcher --> StateManager
Key Components
1. Watcher (State Observation)
- Passive observer that never performs sync operations
- Watches file system for local changes (with 500ms debouncing)
- Polls n8n API for remote changes
- Calculates workflow status using 3-way comparison:
localHash- SHA-256 hash of current file contentremoteHash- SHA-256 hash of current n8n workflowlastSyncedHash- SHA-256 hash from.n8n-state.json(base)
- Emits status events:
status-changed,conflict,deletion
2. SyncEngine (State Mutation)
- Stateless I/O executor that performs actual sync operations
push()- Uploads local workflow to n8npull()- Downloads remote workflow to local filedelete()- Deletes workflow from n8n or localfinalizeSync()- Updates.n8n-state.jsonafter successful operations- Creates backups before destructive operations
- Uses WorkflowSanitizer to clean workflows before saving
3. ResolutionManager
- Dedicated service for interactive conflict and deletion resolution
- Provides CLI prompts for user decisions
- Handles "show diff" functionality
- Maintains separation between automated and user-driven actions
4. SyncManager (Orchestration)
- High-level orchestrator that coordinates components
listWorkflows()- Lists all workflows with current sync statusfetch(workflowId)- Updates remote state cache for a specific workflowpull(workflowId)- Downloads remote workflow to local filepush(workflowId)- Uploads local workflow to n8nresolveConflict()- Force-resolves conflicts keeping local or remote- Emits events:
log,conflict,change,error,connection-lost
5. StateManager
- Manages
.n8n-state.jsonfile (the "base" in 3-way merge) - Tracks
lastSyncedHashandlastSyncedAtfor each workflow - Provides atomic read/write operations
- Enables 3-way merge conflict detection
6. N8n API Client
- Communicates with n8n REST API
- Handles authentication and rate limiting
- Provides typed API responses
7. Workflow Sanitizer
- Validates workflow JSON structure
- Removes sensitive data (credentials)
- Sorts nodes and connections canonically for consistent hashing
- Ensures compatibility with n8n
5 Workflow States
Based on 3-way comparison (base vs local vs remote):
| Status | Icon | Description |
|---|---|---|
TRACKED | 📄 plain file | Both local and remote exist; either side may have changed — user syncs explicitly |
CONFLICT | 🔴 red alert | Both local and remote changed since last sync |
EXIST_ONLY_LOCALLY | 📄+ orange file-add | New workflow created locally, not yet pushed |
EXIST_ONLY_REMOTELY | ☁️ blue cloud | Workflow exists remotely, not yet pulled locally |
🔌 VS Code Extension Architecture
Extension Components
// VS Code extension architecture
classDiagram
class Extension {
+activate()
+deactivate()
}
class WorkflowTreeProvider {
+getTreeItem()
+getChildren()
+refresh()
}
class WorkflowWebview {
+render()
+update()
+handleMessage()
}
class ProxyService {
+forwardRequest()
+handleResponse()
}
Extension --> WorkflowTreeProvider
Extension --> WorkflowWebview
Extension --> ProxyService
WorkflowWebview --> ProxyService
Communication Flow
- Tree View: Displays workflows organized by instance
- Webview: Renders n8n canvas for visual editing
- Proxy Service: Bridges VS Code and n8n API
- Sync Integration: Uses Sync library for synchronization
🖥️ CLI Architecture
Command Structure
// CLI command architecture
classDiagram
class CLI {
+parseArgs()
+executeCommand()
}
class BaseCommand {
+run()
+validate()
+execute()
}
class InitCommand {
+initializeProject()
+createConfig()
}
class SyncCommand {
+syncWorkflows()
+handleConflicts()
}
class ListCommand {
+listWorkflows()
+filterByStatus()
}
CLI --> BaseCommand
BaseCommand <|-- InitCommand
BaseCommand <|-- SyncCommand
BaseCommand <|-- ListCommand
Command Processing
- Argument Parsing: Commander.js for CLI parsing
- Command Execution: Each command extends BaseCommand
- Configuration: Loads from file, env vars, or args
- Error Handling: Consistent error reporting
🤖 Skills Library Architecture (n8nac skills)
AI Integration
The @n8n-as-code/skills package is an internal library exposed publicly via the n8nac skills subcommand group. It powers AI context generation for Cursor, Cline, Copilot, and other AI tools.
// Skills Library architecture
classDiagram
class AgentCLI {
+generateContext()
+processRequest()
}
class AIContextGenerator {
+generateAgentsMD()
+generateSchema()
+generateSnippets()
}
class NodeSchemaProvider {
+getNodeSchemas()
+validateNode()
}
class SnippetGenerator {
+generateSnippets()
+formatSnippet()
}
AgentCLI --> AIContextGenerator
AgentCLI --> NodeSchemaProvider
AgentCLI --> SnippetGenerator
Context Generation
- AGENTS.md: Instructions for AI assistants
- n8n-schema.json: Validation schema
- Code Snippets: VS Code snippets for common patterns
🔄 Data Flow
Synchronization Flow
sequenceDiagram
participant User
participant VS Code
participant CLI
participant SyncEngine
participant n8n
User->>VS Code: Right-click → Push workflow
VS Code->>SyncEngine: push(workflowId)
SyncEngine->>n8n: Upload changes
n8n-->>SyncEngine: Confirm update
SyncEngine-->>VS Code: Update status (TRACKED)
User->>CLI: n8nac pull <workflowId>
CLI->>SyncEngine: pull(workflowId)
SyncEngine->>n8n: Download latest
n8n-->>SyncEngine: Return workflow
SyncEngine-->>CLI: Save locally, update .n8n-state.json
Conflict Resolution
- Detection: State Manager detects conflicting changes
- Notification: User is notified of conflict
- Resolution: Options: keep local, keep remote, or merge
- Sync: Resolved workflow is synchronized
🏭 Build System
TypeScript Configuration
- Base Config: Shared TypeScript configuration
- Package Configs: Individual package configurations
- Build Scripts: Unified build process
Testing Strategy
- Unit Tests: Jest for individual components
- Integration Tests: End-to-end workflow tests
- Mocking: Nock for HTTP requests, in-memory file system
CI/CD Pipeline
- Linting: ESLint with TypeScript support
- Testing: Jest with coverage reporting
- Building: TypeScript compilation
- Publishing: Custom commit-driven release automation
🔐 Security Architecture
Credential Management
- Never Stored: Credentials never committed to Git
- Environment Variables: API keys via env vars
- Configuration Files: Local config with gitignore
Data Sanitization
- Workflow Sanitization: Removes credentials before storage
- Validation: Schema validation for all inputs
- Error Handling: Secure error messages without sensitive data
📈 Scalability Considerations
Performance Optimizations
- Batch Operations: Bulk sync operations
- Caching: Local state caching
- Incremental Sync: Only sync changed workflows
Memory Management
- Stream Processing: Large workflow processing
- Cleanup: Proper resource disposal
- Monitoring: Memory usage tracking
🛠️ Development Workflow
Local Development
# Install dependencies
npm install
# Build all packages
npm run build
# Run tests
npm test
# Start documentation
npm run docs
Package Management
- Workspaces: npm workspaces for monorepo
- Dependencies: Shared and package-specific deps
- Versioning: Independent versioning with commit-driven release automation
📚 Related Documentation
- Sync Engine: Sync engine internals (embedded in CLI)
- Skills Library: AI integration details
- Contribution Guide: How to contribute
This architecture enables n8n-as-code to provide a seamless experience across different interfaces while maintaining a single source of truth for workflow management.