Skip to main content

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

  1. Sync Engine (embedded in n8nac, cli/src/core): Shared business logic
  2. CLI (n8nac): Command-line interface
  3. VS Code Extension: Visual interface using n8nac
  4. Skills Library (@n8n-as-code/skills, accessed via n8nac 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

  1. Separation of Concerns: Watcher observes, SyncEngine mutates
  2. 3-Way Comparison: Uses base-local-remote to detect conflicts
  3. Deterministic Detection: Only flags conflicts when both sides changed
  4. State Persistence: .n8n-state.json tracks 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 content
    • remoteHash - SHA-256 hash of current n8n workflow
    • lastSyncedHash - 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 n8n
  • pull() - Downloads remote workflow to local file
  • delete() - Deletes workflow from n8n or local
  • finalizeSync() - Updates .n8n-state.json after 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 status
  • fetch(workflowId) - Updates remote state cache for a specific workflow
  • pull(workflowId) - Downloads remote workflow to local file
  • push(workflowId) - Uploads local workflow to n8n
  • resolveConflict() - Force-resolves conflicts keeping local or remote
  • Emits events: log, conflict, change, error, connection-lost

5. StateManager

  • Manages .n8n-state.json file (the "base" in 3-way merge)
  • Tracks lastSyncedHash and lastSyncedAt for 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):

StatusIconDescription
TRACKED📄 plain fileBoth local and remote exist; either side may have changed — user syncs explicitly
CONFLICT🔴 red alertBoth local and remote changed since last sync
EXIST_ONLY_LOCALLY📄+ orange file-addNew workflow created locally, not yet pushed
EXIST_ONLY_REMOTELY☁️ blue cloudWorkflow 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

  1. Tree View: Displays workflows organized by instance
  2. Webview: Renders n8n canvas for visual editing
  3. Proxy Service: Bridges VS Code and n8n API
  4. 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

  1. Argument Parsing: Commander.js for CLI parsing
  2. Command Execution: Each command extends BaseCommand
  3. Configuration: Loads from file, env vars, or args
  4. 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

  1. AGENTS.md: Instructions for AI assistants
  2. n8n-schema.json: Validation schema
  3. 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

  1. Detection: State Manager detects conflicting changes
  2. Notification: User is notified of conflict
  3. Resolution: Options: keep local, keep remote, or merge
  4. 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

  1. Linting: ESLint with TypeScript support
  2. Testing: Jest with coverage reporting
  3. Building: TypeScript compilation
  4. 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

This architecture enables n8n-as-code to provide a seamless experience across different interfaces while maintaining a single source of truth for workflow management.