modulex-js
v0.1.0
Published
Official ModuleX JavaScript/TypeScript SDK — AI workflow orchestration platform
Maintainers
Readme
modulex-js
Official JavaScript/TypeScript SDK for the ModuleX AI workflow orchestration platform.
- Zero runtime dependencies
- Full TypeScript support with exported types
- Dual ESM + CommonJS build
- SSE streaming for real-time workflow events
- Automatic retries with exponential backoff
- 122 API endpoints covered
Installation
npm install modulex-js
# or
pnpm add modulex-js
# or
yarn add modulex-jsQuick Start
import { Modulex } from 'modulex-js';
const client = new Modulex({
apiKey: 'mx_live_...',
organizationId: 'your-org-id',
});
// Get current user
const me = await client.auth.me();
console.log(me.email);
// List workflows
const { workflows } = await client.workflows.list({ status: 'active' });JavaScript (CommonJS)
const { Modulex } = require('modulex-js');
const client = new Modulex({
apiKey: process.env.MODULEX_API_KEY,
organizationId: process.env.MODULEX_ORG_ID,
});Authentication
Create an API key from the ModuleX dashboard. API keys use the mx_live_ prefix.
const client = new Modulex({
apiKey: process.env.MODULEX_API_KEY,
});The organizationId can be set at the client level (applied to all requests) or overridden per-request:
// Client-level default
const client = new Modulex({
apiKey: 'mx_live_...',
organizationId: 'default-org-id',
});
// Per-request override
const workflows = await client.workflows.list({}, {
organizationId: 'other-org-id',
});Configuration
const client = new Modulex({
apiKey: 'mx_live_...', // Required
organizationId: 'org-uuid', // Default org for all requests
baseUrl: 'https://api.modulex.dev', // API base URL (default)
timeout: 30_000, // Request timeout in ms (default: 30000)
maxRetries: 3, // Retry count for transient errors (default: 3)
fetch: customFetch, // Custom fetch implementation
});Resources
Workflows
// List workflows
const result = await client.workflows.list({ status: 'active', page: 1, pageSize: 20 });
// Auto-paginate all workflows
for await (const workflow of client.workflows.listAll({ status: 'active' })) {
console.log(workflow.name);
}
// Get a single workflow
const workflow = await client.workflows.get('workflow-uuid');
// Create a workflow
const created = await client.workflows.create({
workflowSchema: { metadata: { name: 'My Workflow', version: '1.0' }, config: {}, state_schema: { fields: {} }, nodes: [], edges: [], entry_point: 'start' },
name: 'My Workflow',
status: 'draft',
});
// Update / Delete
await client.workflows.update('workflow-uuid', { name: 'New Name', status: 'active' });
await client.workflows.delete('workflow-uuid');Workflow Execution
// Run a workflow
const run = await client.executions.run({
workflowId: 'workflow-uuid',
input: { messages: [{ role: 'user', content: 'Hello' }] },
stream: true,
});
// Get execution state
const state = await client.executions.getState(run.thread_id);
// Resume after interrupt
await client.executions.resume({
workflowId: 'workflow-uuid',
runId: run.run_id,
resumeValue: 'user input',
});
// Cancel execution
await client.executions.cancel(run.run_id, { reason: 'No longer needed' });SSE Streaming
const run = await client.executions.run({
workflowId: 'workflow-uuid',
input: { messages: [{ role: 'user', content: 'Hello' }] },
stream: true,
});
for await (const event of client.executions.listen(run.run_id)) {
switch (event.event) {
case 'node_update':
console.log(`Node ${event.data.node_id}: ${event.data.status}`);
break;
case 'done':
console.log(`Completed in ${event.data.total_execution_time_ms}ms`);
break;
case 'error':
console.error(event.data.error_message);
break;
}
}Credentials
// List credentials
const creds = await client.credentials.list({ integrationName: 'openai' });
// Create a credential
await client.credentials.create({
integrationName: 'openai',
authData: { api_key: 'sk-...' },
displayName: 'My OpenAI Key',
});
// Test a credential
const result = await client.credentials.test('credential-uuid');
console.log(result.is_valid);Knowledge Bases
// Create a knowledge base
const kb = await client.knowledge.create({
name: 'Docs',
embeddingConfig: { provider: 'openai', model: 'text-embedding-3-small', dimension: 1536 },
});
// Upload a document
const doc = await client.knowledge.uploadDocument(kb.id, {
file: new Blob([buffer], { type: 'application/pdf' }),
filename: 'guide.pdf',
});
// Search
const results = await client.knowledge.search(kb.id, {
query: 'How does X work?',
topK: 5,
});Schedules
// Create a schedule
const schedule = await client.schedules.create({
workflowId: 'workflow-uuid',
name: 'Daily Report',
scheduleType: 'cron',
cronExpression: '0 9 * * 1-5',
timezone: 'America/New_York',
});
// Pause / Resume
await client.schedules.pause(schedule.id);
await client.schedules.resume(schedule.id);Templates
const templates = await client.templates.list();
const template = await client.templates.get('template-uuid');
const used = await client.templates.use('template-uuid');Deployments
const deployment = await client.deployments.create('workflow-uuid', {
deploymentNote: 'v2.0 release',
});
await client.deployments.activate('workflow-uuid', deployment.id);Composer
const session = await client.composer.chat({
workflowId: 'workflow-uuid',
message: 'Add an LLM node that summarizes the input',
});
for await (const event of client.composer.listen(session.composer_chat_id, session.run_id)) {
console.log(event.event, event.data);
}
await client.composer.save(session.composer_chat_id);Dashboard & Analytics
const logs = await client.dashboard.logs({ limit: 50, category: 'CREDENTIALS' });
const overview = await client.dashboard.analyticsOverview();
const users = await client.dashboard.users({ search: 'john' });Subscriptions
const plans = await client.subscriptions.organizationPlans();
const billing = await client.subscriptions.billing();
const checkout = await client.subscriptions.checkoutLink({ planId: 'plan-uuid', interval: 'month' });System
const health = await client.system.health();
const timezones = await client.system.timezones();Error Handling
import {
Modulex,
NotFoundError,
RateLimitError,
AuthenticationError,
ValidationError,
} from 'modulex-js';
try {
const workflow = await client.workflows.get('invalid-id');
} catch (error) {
if (error instanceof NotFoundError) {
console.log('Workflow not found');
} else if (error instanceof RateLimitError) {
console.log(`Rate limited. Retry after ${error.retryAfter}s`);
} else if (error instanceof AuthenticationError) {
console.log('Invalid API key');
} else if (error instanceof ValidationError) {
console.log('Validation error:', error.body);
}
}The SDK automatically retries requests on transient errors (429, 500, 502, 503) with exponential backoff.
Cancellation
Use AbortController to cancel any request or stream:
const controller = new AbortController();
// Cancel after 10 seconds
setTimeout(() => controller.abort(), 10_000);
for await (const event of client.executions.listen(runId, {
signal: controller.signal,
})) {
console.log(event);
}TypeScript
The SDK is written in TypeScript and exports all types:
import type {
WorkflowDefinition,
NodeDefinition,
WorkflowRunParams,
WorkflowSSEEvent,
SSEEvent,
} from 'modulex-js';Browser Support
The SDK works in modern browsers that support fetch, ReadableStream, and AbortController. No Node.js-specific APIs are required for the core SDK.
Requirements
- Node.js 18+ or modern browser
- TypeScript 5.0+ (for type definitions)
Documentation
Full API documentation is available at docs.modulex.dev.
Contributing
Contributions are welcome! Please see CONTRIBUTING.md for development setup, coding guidelines, and the PR process.
Security
To report security vulnerabilities, please see SECURITY.md. Do not use public GitHub issues for security reports.
