New Research: Supply Chain Attack on Axios Pulls Malicious Dependency from npm.Details →
Socket
Book a DemoSign in
Socket

zv1

Package Overview
Dependencies
Maintainers
1
Versions
19
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

zv1

The zv1 SDK for making processing flows created via zv1.ai, by ZeroWidth

latest
Source
npmnpm
Version
1.0.1
Version published
Weekly downloads
21
-81.42%
Maintainers
1
Weekly downloads
 
Created
Source

zv1

A Node.js implementation of ZeroWidth's zv1 framework for executing AI and automation workflows through a visual node-based interface. Design flows on zv1.ai, export as JSON, and execute with precision and control.

Apache 2.0 License

Table of Contents

Overview

The zv1 Flow Engine enables you to:

  • Execute complex AI and automation workflows
  • Connect various node types (data processing, AI models, tools, testing utilities)
  • Handle asynchronous operations with precision
  • Monitor and debug flow execution with detailed timelines
  • Integrate with external systems and APIs
  • Test error handling and recovery scenarios
  • Manage costs and resource usage

Installation

npm install zv1

Quick Start

import zv1 from 'zv1';

// Create engine instance by passing the location of your configured flow
const engine = await zv1.create('./path/to/myflow.zv1', {
  keys: {
    openrouter: process.env.OPENROUTER_API_KEY
  }
});

// Run the flow
const result = await engine.run({
  chat: [
    {
      role: 'user',
      content: "Hello, world!"
    }
  ]
});

Flow File Formats

The zv1 engine supports two flow file formats:

The new .zv1 format is a ZIP-based archive that supports hierarchical imports and modular flow design:

myflow.zv1
├── orchestration.json          # Main flow definition
├── imports/                    # Optional imports folder
│   └── a1b2c3d4-e5f6-7890-abcd-ef1234567890/   # Import folder (importId only)
│       ├── orchestration.json  # Import's flow definition
│       ├── README.md           # Optional documentation
│       └── imports/            # Optional nested imports
│           └── b2c3d4e5-f6g7-8901-bcde-f12345678901/
│               └── orchestration.json

Benefits:

  • Modular Design: Break complex flows into reusable components
  • Version Control: Track different versions of imports with snapshots
  • Hierarchical Imports: Support unlimited nesting depth
  • Developer Friendly: Can be renamed to .zip and explored manually
  • Backward Compatible: Legacy JSON flows still work

Import Naming Convention:

  • New Format: {importId} (simplified)
  • Example: a1b2c3d4-e5f6-7890-abcd-ef1234567890
  • Legacy Format: {displayName}.{snapshot}.{uniqueId} (backward compatible)
  • Version Info: Stored in import's orchestration.json metadata
  • Requested Range: Stored in main orchestration.json imports object

Legacy JSON Format

The legacy format is a single JSON file with an optional imports array:

{
  "nodes": [...],
  "links": [...],
  "imports": [
    {
      "id": "import-123",
      "display_name": "My Import",
      "nodes": [...],
      "links": [...]
    }
  ]
}

Import Resolution

The new .zv1 format supports two import declaration styles:

1. Imports Object (Recommended):

{
  "metadata": {
    "name": "My Flow",
    "version": "1.0.0"
  },
  "nodes": [...],
  "links": [...],
  "imports": {
    "a1b2c3d4-e5f6-7890-abcd-ef1234567890": "^1.2.3",
    "b2c3d4e5-f6g7-8901-bcde-f12345678901": "~1.0.0",
    "c3d4e5f6-g7h8-9012-cdef-123456789012": "2.1.0",
    "d4e5f6g7-h8i9-0123-defg-234567890123": "latest"
  }
}

Version Range Syntax:

  • Exact: "1.2.3" - Use exactly this version
  • Caret: "^1.2.3" - Compatible with 1.x.x (allows 1.3.0, 1.9.9, but not 2.0.0)
  • Tilde: "~1.2.3" - Compatible with 1.2.x (allows 1.2.4, 1.2.9, but not 1.3.0)
  • Greater: ">=1.2.3" - Version 1.2.3 or higher
  • Latest: "latest" - Use the highest available version
  • Stable: "stable" - Use the latest stable version

Import ID Format:

  • Auto-generated UUIDs (e.g., "a1b2c3d4-e5f6-7890-abcd-ef1234567890")
  • System maps to folder names like uuid.1.2.3 (actual resolved version)
  • Tracks both requested range and resolved version

2. Legacy Imports Array (Backward Compatible):

{
  "nodes": [...],
  "links": [...],
  "imports": [
    {
      "id": "import-123",
      "display_name": "My Import",
      "nodes": [...],
      "links": [...]
    }
  ]
}

Migration: Legacy JSON files are automatically converted to the new format when loaded.

Core Concepts

Nodes are the building blocks of your flow, connected by links that define data flow.

Node Types

  • Input Nodes

    • input-data: Structured data entry point
    • input-chat: Chat message arrays
    • input-prompt: String prompts
    {
      "id": "input1",
      "type": "input-data",
      "settings": {
        "key": "userInput",
        "default_value": "Hello"
      }
    }
    
  • Output Nodes

    • output-data: Return structured data
    • output-chat: Return chat messages
    {
      "id": "output1",
      "type": "output-data",
      "settings": {
        "key": "result"
      }
    }
    
  • Constant Nodes

    • Provide fixed values
    • No input connections required
    • Process first in execution order
  • Testing Nodes

    • throw-error: Always throws a custom error message
    • null-bomb: Randomly returns null/undefined based on probability
    • self-healing-error: Fails initially but succeeds on retry
    • random-error: Randomly throws errors based on probability

Flow Processing

The engine follows a specific order of operations:

  • Initialization

    • Use zv1.create() to asynchronously load node definitions and custom types
    • Validate flow structure
    • Setup execution environment
    • Initialize ErrorManager
  • Execution Order

    Constant Nodes → Input Nodes → Processing Nodes → Output/Terminal Nodes
    
  • Terminal Node Handling When no output nodes exist, the engine returns:

    {
      partial: true,
      message: "Flow completed without output nodes",
      terminal_nodes: [
        {
          node_id: "node1",
          type: "processor",
          outputs: { ... }
        }
      ]
    }
    

Input/Output System

The engine supports various data types and structures:

// Input format
const inputs = {
  data: {
    key1: "value1",
    key2: 42
  },
  chat: [
    { role: "user", content: "Hello" }
  ],
  prompt: "Generate a story about..."
};

// Output format
const outputs = {
  data: result,
  chat: responseMessages
};

API Keys & Security

The engine supports secure API key management for nodes that require external service authentication.

Configuration

const engine = await zv1.create(flow, {
  keys: {
    openrouter: "sk-...",  // OpenRouter API key
  }
});

Node Key Requirements

Nodes specify their key requirements in their configuration. All LLMs are configured by default to use OpenRouter, but this can be overridden.

{
  "config": {
    "needs_key_from": ["openai"],
    "display_name": "GPT-4 Node",
    "description": "Processes text with GPT-4"
  }
}

The engine validates key availability before execution.

Event Handlers

Monitor and extend flow execution with event handlers:

const engine = await zv1.create(flow, {
  onNodeStart: async ({ nodeId, nodeType, timestamp, inputs, settings }) => {
    console.log(`Node ${nodeId} starting execution`);
    // Example: Track metrics in your own system
    // await yourMetricsService.recordNodeStart(nodeId, timestamp);
  },
  
  onNodeComplete: async ({ nodeId, nodeType, timestamp, inputs, outputs, settings }) => {
    console.log(`Node ${nodeId} completed execution`);
    // Example: Calculate and track execution duration
    // const duration = timestamp - startTime;
    // await yourMetricsService.recordNodeComplete(nodeId, duration);
  },

  onNodeUpdate: async ({ nodeId, nodeType, timestamp, data }) => {
    console.log(`Node ${nodeId} sent update:`, data);
    // Example: Handle real-time updates from nodes
  },
  
  onError: async (errorEvent) => {
    console.log('Error occurred:', errorEvent);
    // Example: Send to your error tracking service
    // await yourErrorTracker.recordError(errorEvent);
  }
});

Common use cases:

  • Performance monitoring
  • Progress tracking
  • Streaming LLM tokens via onNodeUpdate
  • Debug logging
  • Resource management
  • External system integration
  • Error tracking and alerting

Error Handling

The engine includes a comprehensive ErrorManager for centralized error handling and detailed error reporting.

Error Types

  • node: Errors occurring during node execution
  • flow: Errors related to flow structure or validation
  • system: System-level errors
  • validation: Input/output validation errors
  • timeout: Execution timeout errors
  • resource: Resource allocation or access errors

Error Context

When verbose mode is enabled, errors include rich execution context:

{
  errorType: "node",
  message: "Node execution failed: Custom error message",
  executionId: "uuid-1234",
  nodeId: "node-1",
  nodeType: "custom-node",
  errorDetails: {
    timeline: [...],
    nodeCount: 5,
    nodesExecuted: 3,
    cost_summary: { total: 0.05, itemized: [...] }
  }
}

Error Recovery

The engine supports both fatal and recoverable errors:

// Fatal errors stop execution
if (errorType === 'system') {
  throw new Error('Critical system failure');
}

// Recoverable errors allow retry logic
if (errorType === 'node' && retryCount < maxRetries) {
  // Retry node execution
}

Resource Management

Cleanup

When using knowledge databases or complex flows with imports, it's important to clean up resources:

const engine = await zv1.create('./myflow.zv1', config);

try {
  const result = await engine.run(inputs);
  console.log(result);
} finally {
  // Always clean up to free memory and remove temporary files
  await engine.cleanup();
}

What cleanup does:

  • Closes all SQLite database connections
  • Removes temporary knowledge database files
  • Clears internal caches and timelines
  • Recursively cleans up imported engines

When to call cleanup:

  • After completing a flow execution
  • Before creating a new engine instance
  • In error handling blocks
  • When the application is shutting down

Advanced Features

Plugin System

Support for LLM plugins and tools:

// Tool schema definition
{
  "name": "calculator",
  "description": "Performs calculations",
  "parameters": {
    "type": "object",
    "properties": {
      "operation": { "type": "string" },
      "numbers": { "type": "array" }
    }
  }
}

MCP (Model Context Protocol) Support

Integrate with external tools via MCP:

const engine = await zv1.create(flow, {
  mcp: {
    tools: [
      {
        name: "getTime",
        description: "Get current time",
        url: "http://localhost:3000/mcp"
      }
    ]
  }
});

Custom Node Types

Create custom nodes by implementing:

  • Configuration file (nodeType.config.json)
  • Process function (nodeType.process.js)

Cost Tracking

Monitor resource usage with built-in cost tracking:

const result = await engine.run(inputs);
console.log('Total cost:', result.cost_summary.total);
console.log('Itemized costs:', result.cost_summary.itemized);

Testing & Development

To prevent duplication and stay organized, shared internal dependencies for each SDK are not kept in each language specific directory. Use the sync_sdks.py script at the root of this directory to pull in the /nodes, /types, and /tests to each SDK for development. Packaged bundles are shipped with this step already completed and this is not necessary for basic use OR if this package was installed via npm/yarn. You will only need to run the sync script if you are accessing this by checking out this repo directly.

python sync_sdks.py

Run the comprehensive test suites:

# Test individual nodes using their <node>.tests.json configuration
npm run test-nodes

# Test complete flows stored within /tests/flows
npm run test-flows

Advanced Node Testing Options

The node testing script supports several command-line options for efficient development and debugging:

# Test all nodes (default behavior)
node tests/test.all-nodes.js

# Test a specific node only
node tests/test.all-nodes.js --node absolute
node tests/test.all-nodes.js --node anthropic-claude-3-5-sonnet

# Resume testing from a specific node (useful after failures)
node tests/test.all-nodes.js --start-from array-map

# Show help and usage information
node tests/test.all-nodes.js --help

# Legacy support: positional argument for single node
node tests/test.all-nodes.js absolute

Use Cases:

  • Single Node Testing: Quickly test a specific node during development with --node
  • Resume Testing: Continue from where a test run failed using --start-from
  • Debugging: Focus on problematic nodes without running the entire suite
  • Development: Test only the nodes you're actively working on

Error Handling:

  • Invalid node names display available options
  • Conflicting options (e.g., --node and --start-from) are rejected
  • Nodes are processed in alphabetical order for consistent --start-from behavior

Test Flow Format

Create test flows in tests/flows/ directory:

{
  "flow": {
    "nodes": [...],
    "links": [...]
  },
  "inputs": { ... },
  "expected": { ... },
  "expectedSchema": { ... },
  "expectedError": {
    "type": "node",
    "message": "Expected error message",
    "nodeId": "node-1",
    "nodeType": "custom-node"
  }
}

Testing Utilities

The engine includes specialized testing nodes:

  • throw-error: Test error handling scenarios
  • null-bomb: Test null/undefined handling
  • self-healing-error: Test retry and recovery logic
  • random-error: Test probabilistic error conditions

API Reference

zv1 Class

class zv1 {
  
  /**
   * Create a new zv1 instance (recommended)
   * @param {string|Object|Buffer} flow - File path (.zv1 or .json), flow definition object, or ZIP data (Buffer)
   * @param {Object} config - Configuration options and context for the engine
   * @returns {Promise<zv1>} Fully initialized zv1 instance
   */
  static async create(flow, config)
  
  /**
   * Run the flow and return the final output
   * @param {Object} inputs - Data to inject into input nodes
   * @param {number} timeout - Maximum execution time in milliseconds (default: 60000)
   * @returns {Promise<Object>} The final output from output nodes
   */
  async run(inputs, timeout = 60000)
  
  /**
   * Clean up resources including knowledge databases and temporary files
   * Call this when the engine is no longer needed to free up memory
   * @returns {Promise<void>}
   */
  async cleanup()
  
}

Creating Instances

Recommended: Use the static create method

// Load from .zv1 file (new format)
const engine = await zv1.create('./myflow.zv1', config);

// Load from legacy JSON file
const engine = await zv1.create('./legacy.json', config);

// Load from flow object
const engine = await zv1.create(flowObject, config);

// Load from ZIP data in memory (Buffer)
const zipBuffer = fs.readFileSync('./myflow.zv1');
const engine = await zv1.create(zipBuffer, config);

Important: Clean up resources when done

// Always call cleanup when the engine is no longer needed
await engine.cleanup();

The static create method:

  • Auto-detects format: Supports .zv1 files, .json files, flow objects, and ZIP data (Buffer)
  • Handles imports: Loads hierarchical imports automatically
  • Validates structure: Ensures flow integrity before execution
  • Loads dependencies: Node definitions, custom types, and integrations
  • Memory efficient: Can load .zv1 files directly from memory without writing to disk
  • Returns ready instance: Fully initialized and ready to run

Configuration Options

The config object supports the following properties:

const config = {
  
  // Optional: API keys for external services
  keys: {
    openai: "sk-...",
    gemini: "...",
    // ... other service keys
  },
  
  // Optional: Node execution event handlers
  onNodeStart: async (event) => {
    // event: { nodeId, nodeType, timestamp, inputs, settings }
  },
  
  onNodeComplete: async (event) => {
    // event: { nodeId, nodeType, timestamp, inputs, outputs, settings }
  },

  onNodeUpdated: async (event) => {
    // event: { nodeId, nodeType, timestamp, data }
  },
  
  // Optional: Error event handler
  onError: async (errorEvent) => {
    // errorEvent: { errorType, message, executionId, nodeId?, nodeType?, errorDetails?, originalError? }
  },
  
  // Optional: Maximum number of plugin calls per LLM node to prevent runaway loops (default: 10)
  maxPluginCalls: 10,
  
  // Optional: Execution ID for tracking (auto-generated if not provided)
  executionId: "custom-execution-id",

  // Optional: Enable debug logging
  debug: true

};

Execution Results

When you run a flow, the engine returns detailed execution information:

const result = await engine.run(inputs);

// Example result structure:
{
  // Flow outputs (from output nodes)
  outputs: {
    data: "processed result",
    chat: [{ role: "assistant", content: "response" }]
  },
  
  // Execution timeline with detailed node information
  timeline: [
    {
      nodeId: "node-1",
      nodeType: "input-data", 
      inputs: { /* node inputs */ },
      outputs: { /* node outputs */ },
      settings: { /* node settings */ },
      startTime: "2024-01-01T10:00:00.000Z",
      endTime: "2024-01-01T10:00:00.100Z", 
      durationMs: 100,
      status: "success" // or "error"
      // errorMessage: "error details" (if status is "error")
    }
    // ... more timeline entries
  ],
  
  // Cost tracking information
  cost_summary: {
    total: 0.05,
    itemized: [
      {
        node_id: "llm-1",
        node_type: "openai-gpt-4",
        total: 0.05,
        itemized: { /* detailed cost breakdown */ }
      }
    ]
  },
  
  // Any missing input values for partial executions
  inputsMissingValues: [],
  
  // Completion message, which is also used to communicate partial runs 
  message: "Completed."
}

Examples

Basic Data Flow

{
  "nodes": [
    {
      "id": "input1",
      "type": "input-data",
      "settings": { "key": "text" }
    },
    {
      "id": "process1",
      "type": "text-transform",
      "settings": { "operation": "uppercase" }
    },
    {
      "id": "output1",
      "type": "output-data"
    }
  ],
  "links": [
    {
      "from": { "node_id": "input1", "port_name": "value" },
      "to": { "node_id": "process1", "port_name": "input" }
    },
    {
      "from": { "node_id": "process1", "port_name": "output" },
      "to": { "node_id": "output1", "port_name": "value" }
    }
  ]
}

Error Testing Flow

{
  "flow": {
    "nodes": [
      {
        "id": "input1",
        "type": "input-data",
        "settings": { "key": "test-value" }
      },
      {
        "id": "error-test",
        "type": "self-healing-error",
        "settings": {}
      }
    ],
    "links": [
      {
        "from": { "node_id": "input1", "port_name": "value" },
        "to": { "node_id": "error-test", "port_name": "value" }
      }
    ]
  },
  "inputs": { "test-value": "Recovery Test" },
  "expectedError": {
    "type": "node",
    "message": "Self-healing error: failing 1 time(s) before success",
    "nodeId": "error-test",
    "nodeType": "self-healing-error"
  }
}

License

Apache 2.0 © ZeroWidth

This engine is part of the zv1 platform. Visit our documentation for more information about the visual Workbench and other platform features.

FAQs

Package last updated on 24 Mar 2026

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts