@omiron33/omi-neuron-web
v0.2.29
Published
A drop-in Next.js library for data analysis and 3D visualization with OpenAI-powered insights
Maintainers
Readme
omi-neuron-web
Agent Implementation Guide (exhaustive)
This README is written for AI agents and engineers who need a complete, precise view of the repo and how to implement or extend it. It enumerates every public entry point, type, class, configuration field, and runtime behavior that the library exposes today.
If you are working inside this repo, also read AGENTS.md for execution rules, plan order, and task tracking.
What this repo provides
- A TypeScript library for graph data analysis plus 3D visualization (Three.js) with a Next.js-friendly API.
- A CLI that scaffolds configuration, Docker Postgres (pgvector), and Next.js route handlers.
- A Node/Edge/Cluster data model, plus analysis and embeddings pipelines.
- A React provider and hooks that wrap the API client.
- Static mode (
StaticDataProvider) for authored graphs without API/database dependency. - Status-based node coloring for workflow visualization (draft/active/complete/blocked/archived).
- Static cluster rendering with convex hull boundaries around explicit node groups.
Package entry points
The published package exposes these entry points (see package.json exports):
@omiron33/omi-neuron-web(root) - types, schemas, analysis, storage, config helpers, visualization component, React hooks/provider.@omiron33/omi-neuron-web/api- API route factory, repositories, query builder, middleware, API types.@omiron33/omi-neuron-web/visualization- NeuronWeb renderer, NeuronWebExplorer wrapper, theme/layout utilities, performance + story helpers.@omiron33/omi-neuron-web/migration- migration list and runner.@omiron33/omi-neuron-web/cli- CLI entry (also exposed via theomi-neuronbinary).
Repository map (what to edit)
src/core/types/*- Type system. If you add fields, update Zod schemas and migrations.src/core/schemas/*- Zod runtime validation. Keep in sync with types.src/core/analysis/*- Embeddings, clustering, relationship inference, scoring, pipeline.src/core/events/*- Event bus and event types.src/storage/*- Database, Docker manager, query helpers, migrations.src/api/*- Route factory, repositories, query builder, middleware.src/visualization/*- Three.js scene, renderers, interactions, animations, layout.src/react/*- Provider, context, API client, hooks.src/cli/*- CLI commands and templates.docs/*- Additional docs (older, not as exhaustive as this README).
Agent workflow (plan-driven repo)
This repo is designed for plan-driven execution (see AGENTS.md for the full rules). Quick workflow:
- Read
plans/next-step.jsonto find the current task. - Read the corresponding task file under
tasks/and follow itsdepends_onconstraints. - Implement the requirements and deliverables exactly (avoid scope creep).
- Validate before marking complete:
pnpm typecheckpnpm lintpnpm testpnpm build
- Update task frontmatter (
status: completed) and advanceplans/next-step.json.
Root exports (from src/index.ts)
// Core types + schemas
export * from './core/types';
export * from './core/schemas';
export * from './core/events';
// Analysis engines + pipeline composition
export { DataProcessor } from './core/analysis/data-processor';
export { EmbeddingsService } from './core/analysis/embeddings-service';
export { ClusteringEngine } from './core/analysis/clustering-engine';
export { RelationshipEngine } from './core/analysis/relationship-engine';
export { ScoringEngine } from './core/analysis/scoring-engine';
export { AnalysisPipeline } from './core/analysis/pipeline';
export * from './core/analysis/steps';
// Extensibility + local-first stores + ingestion
export * from './core/providers';
export * from './core/store';
export * from './core/ingestion';
// Storage helpers
export * from './storage';
// Config helper
export * from './config';
// Visualization
export { NeuronWeb, NeuronWebExplorer } from './visualization';
export type {
NeuronWebProps,
NeuronWebExplorerProps,
NeuronWebExplorerFilters,
NeuronWebExplorerResolvedFilters,
NeuronWebTheme,
NeuronLayoutOptions,
NeuronLayoutMode,
TreeLayoutOptions,
DensityOptions,
DensityMode,
ClickCardOptions,
ClickZoomOptions,
CardsMode,
} from './visualization';
// React integration
export * from './react/hooks';
export { NeuronWebProvider } from './react/NeuronWebProvider';
// Static mode (no API/database required)
export { StaticDataProvider } from './react/StaticDataProvider';
export type { StaticDataProviderProps } from './react/StaticDataProvider';
export { StaticModeError } from './react/static/in-memory-api-client';
// Version constant
export { VERSION } from './version';VERSION is defined in src/version.ts and is kept in sync with package.json on tag-based releases
(see .github/workflows/publish-on-tag.yml).
Configuration: NeuronConfig and NeuronSettings
These types live in src/core/types/settings.ts and are validated by Zod schemas in src/core/schemas/settings.ts.
NeuronConfig (full configuration)
export interface NeuronConfig extends NeuronSettings {
openai: {
apiKey: string;
organization?: string;
requestsPerMinute?: number;
maxRetries?: number;
};
database: {
mode: 'docker' | 'external';
port: number;
containerName?: string;
image?: string;
user?: string;
password?: string;
database?: string;
url?: string;
pool?: {
min: number;
max: number;
idleTimeoutMs: number;
connectionTimeoutMs: number;
};
resources?: {
memoryLimit: string;
cpuLimit?: string;
};
};
api: {
basePath: string;
enableCors: boolean;
rateLimit?: {
windowMs: number;
max: number;
};
};
logging: {
level: 'debug' | 'info' | 'warn' | 'error';
prettyPrint: boolean;
};
}NeuronSettings (runtime settings persisted in DB)
export interface NeuronSettings {
instance: {
name: string;
version: string;
repoName: string;
};
visualization: VisualizationSettings;
analysis: AnalysisSettings;
nodeTypes: NodeTypeConfig[];
domains: DomainConfig[];
relationshipTypes: RelationshipTypeConfig[];
}Default settings (exact defaults)
From DEFAULT_VISUALIZATION_SETTINGS and DEFAULT_ANALYSIS_SETTINGS in src/core/types/settings.ts:
export const DEFAULT_VISUALIZATION_SETTINGS = {
domainColors: {},
defaultDomainColor: '#c0c5ff',
edgeColor: '#4d4d55',
edgeActiveColor: '#c6d4ff',
backgroundColor: '#020314',
defaultCameraPosition: [4, 8, 20],
defaultCameraTarget: [0, 0, 0],
minZoomDistance: 4,
maxZoomDistance: 42,
enableStarfield: true,
starfieldCount: 1200,
labelDistance: 26,
maxVisibleLabels: 50,
performanceMode: 'auto',
nodeCountThreshold: 120,
pixelRatioCap: 2,
enableAnimations: true,
focusTweenDuration: 800,
filterTransitionDuration: 650,
enableHover: true,
enableClick: true,
enableDoubleClick: true,
enablePan: true,
enableZoom: true,
enableRotate: true,
};
export const DEFAULT_ANALYSIS_SETTINGS = {
embeddingModel: 'text-embedding-3-small',
embeddingDimensions: 1536,
embeddingBatchSize: 20,
embeddingCacheTTL: 86400,
clusteringAlgorithm: 'kmeans',
defaultClusterCount: 8,
minClusterSize: 3,
clusterSimilarityThreshold: 0.75,
relationshipInferenceModel: 'gpt-4o-mini',
relationshipMinConfidence: 0.7,
relationshipMaxPerNode: 10,
openaiRateLimit: 60,
maxConcurrentAnalysis: 5,
};Full config example (neuron.config.ts)
The CLI (omi-neuron init) writes a full config file. A complete config looks like:
import { defineNeuronConfig, DEFAULT_ANALYSIS_SETTINGS, DEFAULT_VISUALIZATION_SETTINGS, VERSION } from '@omiron33/omi-neuron-web';
export default defineNeuronConfig({
instance: {
name: 'my-instance',
version: VERSION,
repoName: 'my-repo',
},
visualization: DEFAULT_VISUALIZATION_SETTINGS,
analysis: DEFAULT_ANALYSIS_SETTINGS,
nodeTypes: [],
domains: [],
relationshipTypes: [],
openai: {
apiKey: process.env.OPENAI_API_KEY ?? '',
organization: undefined,
requestsPerMinute: undefined,
maxRetries: 3,
},
database: {
mode: 'docker',
port: 5433,
containerName: undefined,
image: 'pgvector/pgvector:pg16',
user: 'neuron',
password: 'neuron_dev',
database: 'neuron_web',
url: undefined,
pool: {
min: 0,
max: 10,
idleTimeoutMs: 30000,
connectionTimeoutMs: 5000,
},
resources: {
memoryLimit: '1g',
cpuLimit: undefined,
},
},
storage: {
mode: 'postgres', // or 'memory' | 'file' for local-first prototyping
},
api: {
basePath: '/api/neuron',
enableCors: false,
rateLimit: undefined,
},
logging: {
level: 'info',
prettyPrint: true,
},
});Environment variables
OPENAI_API_KEY- required for embeddings and relationship inference.DATABASE_URL- optional; if set, the Database factory uses it.DB_HOST,DB_PORT,DB_USER,DB_PASSWORD,DB_NAME- optional percreateDatabaseFromEnv().- Local-first
storage.mode = 'memory' | 'file'does not require a database connection for core CRUD + graph fetch.
- Local-first
Core data model (Types)
Nodes
NeuronNode fields (see src/core/types/node.ts):
- identity:
id,slug,label,nodeType,domain,createdAt,updatedAt - content:
summary,description,content - metadata:
metadata: Record<string, unknown> - analysis fields:
embedding,embeddingModel,embeddingGeneratedAt,clusterId,clusterSimilarity - relationship counts:
connectionCount,inboundCount,outboundCount - status:
analysisStatus,analysisError - workflow status:
status?: NodeStatus(for status-based coloring) - visualization hints:
tier,visualPriority,positionOverride
NodeStatus type (for workflow-based coloring):
export type NodeStatus = 'default' | 'draft' | 'active' | 'complete' | 'blocked' | 'archived';When a node has a status set, it takes priority over domain-based coloring in visualization.
Input types:
NeuronNodeCreate(minimal required fields:label; optionalslug,nodeType,domain,summary,description,content,metadata,tier)NeuronNodeUpdate(partial updates)NeuronNodeBatchCreate(batch create with options)
Visualization type:
NeuronVisualNodehasid,slug,label,domain,tier,metadata,ref,connectionCount,position.
Edges
NeuronEdge fields (see src/core/types/edge.ts):
- identity:
id,fromNodeId,toNodeId,relationshipType - strength/confidence:
strength,confidence - evidence:
EdgeEvidence[] - metadata:
label,description,metadata,source,sourceModel,bidirectional, timestamps
Input types: NeuronEdgeCreate, NeuronEdgeUpdate.
Visualization type: NeuronVisualEdge uses from and to slugs.
Clusters
NeuronCluster fields (see src/core/types/cluster.ts):
- identity:
id,label,clusterType - centroid:
centroid: number[] - stats:
memberCount,avgSimilarity,cohesion - metadata:
description,keywords,metadata
NeuronVisualCluster (for static/authored cluster visualization):
export interface NeuronVisualCluster {
id: string;
label: string;
nodeIds: string[]; // explicit node membership
color?: string; // optional cluster color
position?: { x: number; y: number; z: number }; // optional fixed center
metadata?: Record<string, unknown>;
}Visual clusters render as convex hull boundaries around their member nodes with labels at the centroid.
Analysis
AnalysisRun, AnalysisRequest, AnalysisResponse, AnalysisPipelineConfig are in src/core/types/analysis.ts.
Events
Event types and payloads are defined in src/core/types/events.ts.
Zod schemas
All runtime validation lives in src/core/schemas/*.
Key exports:
- Nodes:
nodeTierSchema,nodeCreateSchema,nodeUpdateSchema,nodeBatchCreateSchema,nodeFilterSchema - Edges:
edgeEvidenceSchema,edgeCreateSchema,edgeUpdateSchema,edgeFilterSchema - Clusters:
clusteringConfigSchema,clusterCreateSchema,clusterUpdateSchema - Analysis:
analysisOptionsSchema,analysisRequestSchema - Settings:
nodeTypeConfigSchema,domainConfigSchema,relationshipTypeConfigSchema,visualizationSettingsSchema,analysisSettingsSchema,neuronSettingsSchema,neuronSettingsUpdateSchema,neuronConfigSchema - API:
listNodesParamsSchema,listEdgesParamsSchema,getGraphParamsSchema,expandGraphRequestSchema,findPathRequestSchema,semanticSearchRequestSchema,findSimilarRequestSchema
Storage layer
Database
Class: Database in src/storage/database.ts.
Constructor:
new Database({
connectionString?: string,
host?: string,
port?: number,
user?: string,
password?: string,
database?: string,
pool?: { min; max; idleTimeoutMs; connectionTimeoutMs },
ssl?: boolean | object,
slowQueryThresholdMs?: number,
});Methods:
connect()/disconnect()isConnected()query<T>(sql, params)queryOne<T>(sql, params)execute(sql, params)transaction(fn)tableExists(tableName)getPoolStats()
DockerManager
Class: DockerManager in src/storage/docker-manager.ts.
Constructor config:
new DockerManager({
repoName: string,
port: number,
containerName?: string,
image?: string,
user?: string,
password?: string,
database?: string,
memoryLimit?: string,
});Methods:
start({ forceRecreate?, waitForReady?, timeoutMs? })stop({ removeVolumes?, timeoutMs? })ensureRunning({ waitForReady? })isRunning()healthCheck()->{ containerRunning, databaseReady, connectionString, details? }getStats()-> container statsgetConnectionString()getLogs({ tail?, since? })execSql(sql)updatePort(newPort)
MigrationRunner
Class: MigrationRunner in src/storage/migrations/runner.ts.
Methods:
getStatus()-> applied/pendinggetPending()getApplied()up({ to? })down({ to?, count? })reset()dryRun('up' | 'down')
Database schema (migrations)
Defined in src/storage/migrations/*.ts:
nodestable:slug,label,node_type,domain,summary,description,content,metadata,tier,visual_priority,position_override,connection_count,inbound_count,outbound_count,analysis_status,analysis_error, timestamps.edgestable:from_node_id,to_node_id,relationship_type,strength,confidence,evidence,label,description,metadata,source,source_model,bidirectional, timestamps.settingstable: JSONB settings forvisualization,analysis,node_types,domains,relationship_types.clusterstable andcluster_membershipstable.analysis_runstable for pipeline jobs.- Extensions:
uuid-ossp,vector(pgvector).
Analysis engines
DataProcessor
File: src/core/analysis/data-processor.ts
- Converts arbitrary input records to
NeuronNodeCreate. - Options:
skipDuplicates,updateOnConflict,defaultNodeType,defaultDomain,contentFields,labelField,slugField,metadataFields. - Key methods:
processItem,processBatch,processJSON,processCSV,generateSlug,extractContent,detectDuplicates.
EmbeddingsService
File: src/core/analysis/embeddings-service.ts
Constructor:
new EmbeddingsService({
openaiApiKey: string,
model: 'text-embedding-ada-002' | 'text-embedding-3-small' | 'text-embedding-3-large',
dimensions?: number,
batchSize: number,
rateLimit: number,
cacheTTL: number,
maxRetries: number,
}, db)Methods:
generateEmbedding(text)generateBatchEmbeddings(texts)embedNode(nodeId)embedNodes(nodeIds)getCachedEmbedding(nodeId)cacheEmbedding(nodeId, embedding, model)invalidateCache(nodeIds?)countTokens(text)estimateCost(nodeCount)
ClusteringEngine
File: src/core/analysis/clustering-engine.ts
Methods:
clusterNodes(config)->{ clusters, unassigned }recluster(config)assignToCluster(nodeId)findBestCluster(embedding)recomputeCentroid(clusterId)/recomputeAllCentroids()generateClusterLabel(clusterId)/generateAllLabels()calculateSilhouetteScore(clusters)calculateCohesion(cluster)
RelationshipEngine
File: src/core/analysis/relationship-engine.ts
Constructor config:
{
model: string;
minConfidence: number;
maxPerNode: number;
similarityThreshold: number;
includeExisting: boolean;
batchSize: number;
rateLimit: number;
}Methods:
inferForNode(nodeId)inferForNodes(nodeIds)inferAll()findCandidates(nodeId)validateRelationship(rel)createEdgesFromInferences(inferences, autoApprove?)
ScoringEngine
File: src/core/analysis/scoring-engine.ts
- Constructor accepts optional config
{ similarityWeight, connectionWeight, recencyWeight, domainBoost }. - Key methods:
cosineSimilarity,semanticSimilarity,scoreForQuery,findSimilar,calculateNodeImportance,rankAllNodes,scoreRelevance.
AnalysisPipeline
File: src/core/analysis/pipeline.ts
Constructor: (db, embeddings, clustering, relationships, events).
Methods:
runFull(options)runEmbeddings(options)runClustering(options)runRelationships(options)getJob(jobId)/listJobs({ status?, limit? })cancelJob(jobId)/isRunning()
PipelineOptions fields include nodeIds, forceRecompute, skipEmbeddings, embeddingModel, skipClustering, clusterCount, clusteringAlgorithm, skipRelationships, relationshipThreshold, maxRelationshipsPerNode, onProgress, webhookUrl.
API layer
createNeuronRoutes
Entry point: createNeuronRoutes(config, options?) from @omiron33/omi-neuron-web/api.
Optional options:
store?: GraphStore(inject a custom store / backend)embeddingProvider?: EmbeddingProvider(override search embeddings provider)requestContext?: RequestContextOptions(scope + auth context extraction)auth?: AuthGuardOptions(optionalauthorize(request, context)hook)bodySizeLimit?: BodySizeLimitOptions(optional request payload guard)rateLimit?: { limiter, keyFn?, windowMs?, max? }(hook-based rate limiting; requires a consumer-provided limiter)
Returns:
{
nodes: createNodesRoutes(config),
edges: createEdgesRoutes(config),
graph: createGraphRoutes(config),
analyze: createAnalyzeRoutes(config),
suggestions: createSuggestionsRoutes(config),
settings: createSettingsRoutes(config),
search: createSearchRoutes(config),
health: createHealthRoutes(config),
}Route behaviors
All handlers accept a Fetch API Request and return Response (Next.js route handlers are compatible).
Nodes:
GET /nodes-> list nodes with pagination. UseslistNodesParamsSchema.POST /nodes-> batch create. UsesnodeBatchCreateSchema.PATCH /nodes/:id-> update. UsesnodeUpdateSchema.DELETE /nodes/:id-> delete.
Edges:
GET /edges-> list edges. UseslistEdgesParamsSchema.POST /edges-> batch create. UsesedgeCreateSchema.PATCH /edges/:id-> update.DELETE /edges/:id-> delete.
Graph:
GET /graph-> returnsNeuronVisualNodeandNeuronVisualEdgearrays.POST /graph/expand-> expansion by depth.POST /graph/path-> find paths.
Analyze:
POST /analyze-> runs analysis pipeline (embeddings, clustering, relationships, or full).GET /analyze/:jobId-> fetch job status.GET /analyze/:jobId/stream-> SSE stream of job progress events (Phase 7E).POST /analyze/:jobId/cancel-> cancel job.GET /analyze/history-> list recent jobs.
Suggestions (Phase 7E):
GET /suggestions-> list suggested edges (review queue).POST /suggestions/:id/approve-> approve a suggestion and create an edge.POST /suggestions/approve-> bulk approve by ids.POST /suggestions/:id/reject-> reject a suggestion.POST /suggestions/reject-> bulk reject by ids.
Settings:
GET /settings-> fetch settings from DB.PATCH /settings-> update settings.POST /settings/reset-> reset entire or specific sections.
Search:
POST /search-> semantic search.POST /search/similar-> find similar to node.
Health:
GET /health-> basic DB connectivity.
API client
NeuronApiClient in src/react/api-client.ts wraps these endpoints; see that file for exact method names and payloads.
Middleware
From src/api/middleware/* (exported in @omiron33/omi-neuron-web/api):
withNeuronMiddleware(handler, { cors?, logging? })withErrorHandler(handler)andhandleError(error)withLogging(handler, { enabled?, logger? })withTiming(handler)withCors({ origins? })withValidation(schema, source)withRequestContext(handler, options?)andbuildRequestContext(request, options?)withAuthGuard(handler, { authorize? })withBodySizeLimit(handler, { maxBytes? })withRateLimit(handler, { windowMs, max, keyFn?, limiter? })
React integration
NeuronWebProvider
Wrap your app and provide configuration. The provider loads settings from /api/neuron/settings via the API client.
import { NeuronWebProvider } from '@omiron33/omi-neuron-web';
export default function RootLayout({ children }) {
return (
<NeuronWebProvider
config={{
// Client-safe: the base path for your server routes.
apiBasePath: '/api/neuron',
// Optional tenant scope. When set, the API client sends `x-neuron-scope`.
scope: 'default',
settings: {},
onEvent: (event) => {},
onError: (error) => {},
}}
>
{children}
</NeuronWebProvider>
);
}Notes:
- Keep secrets server-side. Do not pass
OPENAI_API_KEY(or database URLs) into client components. - Configure OpenAI + database access in your Next.js route handlers (see
docs/secure-nextjs-setup.md).
StaticDataProvider (Standalone Mode)
Use StaticDataProvider for static/authored graphs that don't require API or database connectivity. Perfect for:
- Quest/narrative authoring tools
- Documentation/knowledge graphs
- Workflow/pipeline visualizers
- Game skill trees
- Org charts and hierarchies
import { StaticDataProvider, NeuronWeb } from '@omiron33/omi-neuron-web';
const nodes = [
{ id: 'q001', label: 'Start Quest', domain: 'quest', status: 'complete' },
{ id: 'q002', label: 'Find Artifact', domain: 'quest', status: 'active' },
{ id: 'q003', label: 'Return Home', domain: 'quest', status: 'draft' },
];
const edges = [
{ id: 'e1', from: 'q001', to: 'q002', relationshipType: 'leads_to' },
{ id: 'e2', from: 'q002', to: 'q003', relationshipType: 'leads_to' },
];
const clusters = [
{ id: 'act1', label: 'Act 1', nodeIds: ['q001', 'q002'], color: '#4a5568' },
{ id: 'act2', label: 'Act 2', nodeIds: ['q003'], color: '#2d3748' },
];
export default function QuestGraph() {
return (
<StaticDataProvider
nodes={nodes}
edges={edges}
clusters={clusters}
mutableMode={false}
>
<NeuronWeb graphData={{ nodes, edges, clusters }} />
</StaticDataProvider>
);
}StaticDataProviderProps:
interface StaticDataProviderProps {
children: React.ReactNode;
nodes: (NeuronNode | NeuronVisualNode)[];
edges: (NeuronEdge | NeuronVisualEdge)[];
clusters?: NeuronVisualCluster[];
settings?: Partial<NeuronSettings>;
mutableMode?: boolean; // if true, mutations modify in-memory store
onEvent?: (event: NeuronEvent) => void;
onError?: (error: Error, context: ErrorContext) => void;
}Behavior:
- All hooks (
useNeuronNodes,useNeuronGraph, etc.) work unchanged - Analysis and search methods throw
StaticModeError(not available in static mode) - Context provides
isStaticMode: trueflag - Data updates via
updateData()on the internal client or by passing new props
Status-based coloring
Nodes with a status field render with status colors that override domain colors:
const DEFAULT_STATUS_COLORS = {
default: '#c0c5ff', // Same as defaultDomainColor
draft: '#9ca3af', // Gray
active: '#4ade80', // Green
complete: '#60a5fa', // Blue
blocked: '#f87171', // Red
archived: '#6b7280', // Dark gray
};Override via theme:
<NeuronWeb
graphData={graphData}
theme={{
colors: {
statusColors: {
default: '#c0c5ff',
draft: '#888888',
active: '#00ff00',
complete: '#0088ff',
blocked: '#ff0000',
archived: '#444444',
},
},
}}
/>Static clusters (convex hull visualization)
Pass clusters in graphData to render convex hull boundaries around node groups:
<NeuronWeb
graphData={{
nodes,
edges,
clusters: [
{ id: 'cluster1', label: 'Group A', nodeIds: ['n1', 'n2', 'n3'], color: '#4a5568' },
{ id: 'cluster2', label: 'Group B', nodeIds: ['n4', 'n5'], color: '#2d3748' },
],
}}
/>Cluster rendering:
- Computes 2D convex hull from projected node positions (Graham scan algorithm)
- Renders semi-transparent mesh fill with border outline
- Displays label at cluster centroid
- Updates when node positions change (e.g., ambient motion)
- Requires 3+ nodes to show hull; fewer shows only the label
Hooks
useNeuronContext()-> access provider context.useNeuronGraph({ domains?, nodeTypes?, minEdgeStrength?, autoRefresh?, refreshInterval? })-> fetch and explore graph.useNeuronNodes({ initialFilters?, pageSize? })-> CRUD nodes and pagination.useNeuronAnalysis()-> start/cancel/check analysis jobs.useNeuronJobStream({ jobId, transport?, pollIntervalMs?, scope? })-> stream analysis progress (SSE with polling fallback).useNeuronSettings()-> update or reset settings.useNeuronSearch()-> semantic search and find similar nodes.useNeuronSuggestions({ status?, relationshipType?, minConfidence?, page?, limit? })-> list/approve/reject suggested edges.useNeuronEvents()-> subscribe/emit events.
Also available:
useNodeEvents(handlers)useAnalysisEvents(handlers)
Visualization (NeuronWeb)
Optional explorer wrapper (Phase 7F)
If you want a minimal “app-ready” wrapper around the headless renderer, use NeuronWebExplorer:
- controlled mode: pass
graphDatadirectly - hook-driven mode: omit
graphData(requiresNeuronWebProvider) - slot-based UI:
renderToolbar,renderLegend,renderSelectionPanel
Docs and demo:
docs/visualization/explorer.mddocs/visualization/story-tooling.mddocs/visualization/performance-budgets.mddocs/visualization/density-strategy.mdexamples/basic-usage/app/explorer/page.tsx
Component props (full list)
NeuronWebProps from src/visualization/types.ts:
export interface NeuronWebProps {
graphData: {
nodes: NeuronVisualNode[];
edges: NeuronVisualEdge[];
storyBeats?: NeuronStoryBeat[];
clusters?: NeuronVisualCluster[];
};
fullHeight?: boolean;
isFullScreen?: boolean;
className?: string;
style?: React.CSSProperties;
isLoading?: boolean;
error?: string | null;
selectedNode?: NeuronNode | null;
focusNodeSlug?: string | null;
onFocusConsumed?: () => void;
visibleNodeSlugs?: string[] | null;
onNodeClick?: (node: NeuronNode) => void;
onNodeDoubleClick?: (node: NeuronNode) => void;
onNodeHover?: (node: NeuronNode | null) => void;
onNodeFocused?: (node: NeuronNode) => void;
onEdgeClick?: (edge: NeuronEdge) => void;
onBackgroundClick?: () => void;
onCameraChange?: (position: [number, number, number]) => void;
activeStoryBeatId?: string | null;
storyBeatStepDurationMs?: number;
onStoryBeatComplete?: (beat: NeuronStoryBeat) => void;
studyPathRequest?: StudyPathRequest | null;
onStudyPathComplete?: () => void;
layout?: NeuronLayoutOptions;
cameraFit?: CameraFitOptions;
cardsMode?: CardsMode;
clickCard?: ClickCardOptions;
clickZoom?: ClickZoomOptions;
theme?: NeuronWebThemeOverride;
domainColors?: Record<string, string>;
renderNodeHover?: (node: NeuronVisualNode) => React.ReactNode;
hoverCard?: HoverCardOptions;
renderNodeDetail?: (node: NeuronNode) => React.ReactNode;
renderEmptyState?: () => React.ReactNode;
renderLoadingState?: () => React.ReactNode;
performanceMode?: 'auto' | 'normal' | 'degraded' | 'fallback';
rendering?: RenderingOptions;
density?: DensityOptions;
ariaLabel?: string;
}Props currently used inside NeuronWeb:
- Used:
graphData,className,style,fullHeight,isFullScreen,isLoading,error,renderEmptyState,renderLoadingState,ariaLabel,theme,layout,renderNodeHover,renderNodeDetail,hoverCard,clickCard,clickZoom,cardsMode,onNodeHover,onNodeClick,onNodeDoubleClick,onNodeFocused,onBackgroundClick,performanceMode,focusNodeSlug,onFocusConsumed,visibleNodeSlugs,selectedNode,onEdgeClick,onCameraChange,domainColors,studyPathRequest,onStudyPathComplete,activeStoryBeatId,storyBeatStepDurationMs,onStoryBeatComplete,graphData.storyBeats. - Used:
cameraFit(auto-fit bounds to a viewport fraction). - Reserved: none (all props are live).
Visualization abilities (agent quick reference)
- Controlled selection: pass
selectedNodeto mirror selection state; background clicks only callonBackgroundClick(clearing selection is up to the consumer when controlled). - Node focus:
focusNodeSlugtriggers camera focus + selection;onFocusConsumedfires after the focus is applied. - Edge clicks:
onEdgeClick(edge)fires when the user clicks a rendered edge line. - Camera updates:
onCameraChange(position)fires on OrbitControls changes (RAF-throttled) with the camera position. - Story beats: provide
graphData.storyBeatsand setactiveStoryBeatIdto play;storyBeatStepDurationMsoverrides per-step duration;onStoryBeatCompletefires when playback ends. IfstudyPathRequestis set, it takes precedence. - Domain colors:
domainColorsmerges into the theme'scolors.domainColors(overrides defaults per domain key). - Visibility filtering:
visibleNodeSlugs = nullshows all nodes;[]hides all nodes; explicit slugs filter the view.
Rendering presets (Phase 4C)
Use rendering.preset for a presets-first way to opt into richer visuals/animation (with automatic gating in
degraded and fallback modes):
import { NeuronWeb, DEFAULT_RENDERING_OPTIONS } from '@omiron33/omi-neuron-web/visualization';
<NeuronWeb
graphData={{ nodes, edges }}
rendering={{
...DEFAULT_RENDERING_OPTIONS,
preset: 'subtle', // or 'minimal' | 'cinematic'
}}
/>More detail:
docs/component-props.md(consumer-facing prop guide)docs/visualization/rendering-options-v1.mdanddocs/visualization/animation-profiles.md
Layout modes
NeuronLayoutMode = 'auto' | 'positioned' | 'fuzzy' | 'atlas' | 'tree'.
applyFuzzyLayout() behavior:
positioned: return nodes as-is (use positions from DB or API).auto: if any node lacks aposition, runatlaslayout; otherwise return nodes as-is.atlas: sphere layout with built-in override map (ATLAS_POSITION_OVERRIDESinsrc/visualization/layouts/fuzzy-layout.ts).fuzzy: deterministic pseudo-random scatter based onseedand node key.tree: hierarchical tree layout constrained to 2D plane (z=0), positions nodes based on parent/child relationships from edges.
Layout options:
{
mode?: NeuronLayoutMode;
radius?: number; // base radius for layout
insightRadius?: number; // smaller radius for insight nodes
jitter?: number; // random jitter
zSpread?: number; // z-axis spread
seed?: string; // deterministic seed
spread?: number; // global scale factor
overrides?: Record<string, [number, number, number]>; // slug/id -> position
tree?: TreeLayoutOptions; // tree-specific options (used when mode is 'tree')
}Tree layout (storyboarding)
The 'tree' layout mode renders nodes in a hierarchical tree structure, ideal for storyboarding or visualizing parent/child relationships. Nodes are constrained to a 2D plane (z=0).
interface TreeLayoutOptions {
horizontalSpacing?: number; // spacing between siblings (default: 3)
verticalSpacing?: number; // spacing between levels (default: 4)
rootNodeId?: string; // force a specific root node (slug or id)
direction?: 'down' | 'up' | 'left' | 'right'; // tree growth direction (default: 'down')
}Direction options:
'down': root at top, children below (default)'up': root at bottom, children above'right': root on left, children to the right'left': root on right, children to the left
Example: Vertical tree (top-down)
<NeuronWeb
graphData={{
nodes: [
{ id: 'root', label: 'Root', slug: 'root' },
{ id: 'a', label: 'Child A', slug: 'child-a' },
{ id: 'b', label: 'Child B', slug: 'child-b' },
{ id: 'a1', label: 'Grandchild A1', slug: 'grandchild-a1' },
],
edges: [
{ id: 'e1', from: 'root', to: 'child-a', relationshipType: 'parent_of', strength: 1 },
{ id: 'e2', from: 'root', to: 'child-b', relationshipType: 'parent_of', strength: 1 },
{ id: 'e3', from: 'child-a', to: 'grandchild-a1', relationshipType: 'parent_of', strength: 1 },
],
}}
layout={{
mode: 'tree',
tree: {
direction: 'down',
horizontalSpacing: 4,
verticalSpacing: 5,
},
}}
/>Example: Horizontal tree (left-to-right)
<NeuronWeb
graphData={graphData}
layout={{
mode: 'tree',
tree: {
direction: 'right',
horizontalSpacing: 3,
verticalSpacing: 4,
},
}}
/>The tree layout automatically:
- Identifies root nodes (nodes with no incoming edges)
- Positions children centered below/beside their parent
- Handles multiple disconnected trees (multiple roots)
- Places orphan nodes (not connected to any tree) in a separate row
Camera auto-fit (center-third framing)
Use cameraFit to center and zoom so all nodes sit inside a specific viewport fraction (default: 0.33 = center third).
When isFullScreen is true and cameraFit.enabled is not specified, auto-fit is enabled by default.
Auto-fit pauses whenever a node is selected/focused (click or programmatic focus) so the camera
doesn't snap back to the full graph. Clear selection (background click) to allow future fits.
<NeuronWeb
graphData={graphData}
isFullScreen
cameraFit={{
enabled: true,
mode: 'once', // or 'onChange' to refit when node positions change
viewportFraction: 0.33, // center-third framing
padding: 0.15, // 15% padding around bounds
}}
/>Technochristian (www.technochristianity.com)-style settings (agent example)
Use this preset to match the Technochristian (www.technochristianity.com) look/behavior in a consuming app without introducing any new props. This combines fullscreen, center-third framing, stable atlas layout, click-to-zoom, and restrained visuals:

<NeuronWeb
graphData={graphData}
isFullScreen
cameraFit={{
enabled: true,
mode: 'once',
viewportFraction: 0.33,
padding: 0.15,
}}
layout={{
mode: 'atlas',
spread: 1.15,
jitter: 0.15,
zSpread: 1.1,
}}
clickZoom={{ enabled: true, offset: [0, 2, 8] }}
clickCard={{ enabled: true, width: 320, offset: [24, 24] }}
hoverCard={{ enabled: false }}
cardsMode="click"
theme={{
colors: {
background: '#0b0f1f',
edgeDefault: 'rgba(130, 150, 190, 0.4)',
edgeActive: 'rgba(160, 190, 255, 0.7)',
labelText: 'rgba(230, 235, 245, 0.92)',
labelBackground: 'rgba(20, 24, 40, 0.8)',
},
effects: {
starfieldEnabled: true,
glowEnabled: true,
glowIntensity: 0.55,
ambientMotionEnabled: true,
ambientMotionSpeed: 0.25,
ambientMotionAmplitude: 0.3,
edgeFlowEnabled: true,
edgeFlowSpeed: 0.6,
fogEnabled: true,
fogColor: '#0b0f1f',
fogNear: 20,
fogFar: 80,
},
animation: {
focusDuration: 900,
transitionDuration: 600,
hoverScale: 1.05,
selectedScale: 1.08,
selectionPulseScale: 1.12,
selectionPulseDuration: 1200,
},
}}
/>Programmatic focus (focusNodeSlug)
Use focusNodeSlug to drive selection + camera focus from outside the component
(mirrors Technochristian (www.technochristianity.com)’s “focus a node when something else happens” behavior).
Behavior:
- Looks up by slug, with id fallback.
- Sets selection, pulses the node, and emphasizes connected edges.
- If
clickZoom.enabledis true (default), the camera tween runs. clickZoom.distancelocks the camera distance for focus;clickZoom.offsetuses a fixed offset.- Fires
onNodeFocused(node)after the focus tween. - Calls
onFocusConsumed()so the parent can clear the request.
<NeuronWeb
graphData={graphData}
focusNodeSlug={focusSlug}
onFocusConsumed={() => setFocusSlug(null)}
onNodeFocused={(node) => console.log('focused', node.slug)}
clickZoom={{ enabled: true }}
/>Filtered views (visibleNodeSlugs)
visibleNodeSlugs limits the graph to a subset of nodes (and their edges).
This is the mechanism Technochristian (www.technochristianity.com) uses for filtered views and decluttering.
Semantics:
nullorundefined→ show all nodes/edges.[](empty array) → show none.- Filters nodes by slug or id; edges are kept only if both endpoints are visible.
storyBeats(if present) are filtered to beats with ≥ 2 visible nodeIds.- A subtle fade/scale/blur transition is applied on each filter change.
- If the currently selected node disappears, selection is cleared.
// Show only a curated slice
<NeuronWeb
graphData={graphData}
visibleNodeSlugs={['uap', 'neph', 'jude6']}
/>;
// Show everything (default)
<NeuronWeb graphData={graphData} visibleNodeSlugs={null} />;
// Hide everything
<NeuronWeb graphData={graphData} visibleNodeSlugs={[]} />;Study path playback (follow the path between nodes)
Use studyPathRequest to step through an ordered list of nodes. Each step:
- selects the node
- tweens the camera to the node (if
clickZoom.enabled) - highlights the edge between the current and next step
<NeuronWeb
graphData={graphData}
studyPathRequest={{
steps: [
{ nodeSlug: 'uap', label: 'Start' },
{ nodeSlug: 'neph', label: 'Next' },
{ nodeSlug: 'jude6', label: 'Finish' },
],
stepDurationMs: 4200,
}}
onStudyPathComplete={() => console.log('study path done')}
/>Fallback form (two-step path):
<NeuronWeb
graphData={graphData}
studyPathRequest={{ fromNodeId: 'uap', toNodeId: 'neph' }}
/>Click cards + click zoom
Enable a persistent card on click and optional zoom-to-node behavior. You can lock the focus distance or offset to mimic Technochristian-style snaps:
<NeuronWeb
graphData={graphData}
clickCard={{ enabled: true, width: 320, offset: [24, 24] }}
clickZoom={{ enabled: true, offset: [0, 2, 8] }}
/>Orbit pivot behavior:
- On pointer down, the orbit target shifts to the cursor (or the node under it), so rotating after a focus doesn’t stay locked to the previously focused node.
Card mode (global override)
cardsMode lets you force card behavior irrespective of hoverCard.enabled or clickCard.enabled
(when cardsMode is set, it wins).
<NeuronWeb graphData={graphData} cardsMode="none" />
<NeuronWeb graphData={graphData} cardsMode="hover" />
<NeuronWeb graphData={graphData} cardsMode="click" />
<NeuronWeb graphData={graphData} cardsMode="both" />Disable click cards or zoom:
<NeuronWeb graphData={graphData} clickCard={{ enabled: false }} clickZoom={{ enabled: false }} />To disable in fullscreen:
<NeuronWeb graphData={graphData} isFullScreen cameraFit={{ enabled: false }} />Theme
NeuronWebTheme defines colors, typography, effects, and animation. Defaults are in src/visualization/constants.ts.
Key behavior in NeuronWeb:
- Performance mode is auto-detected unless explicitly set.
- Hover cards are disabled in
fallbackmode. - Labels and starfield density are reduced in
degraded/fallback.
Fullscreen vs element rendering
isFullScreen: truemakes the componentposition: fixedwithwidth: 100vwandheight: 100vh.- Default (not full screen) uses
width: 100%andheight: 100%. The parent element must define a height. fullHeight: truesetsmin-height: 100vhwhen not full screen.
Visualization internals (agent map)
Key internal modules:
src/visualization/scene/scene-manager.ts- sets up Three.js scene, camera, renderer, lights, starfield, labels.src/visualization/scene/node-renderer.ts- renders nodes, labels, hover/selection states, status-based coloring.src/visualization/scene/edge-renderer.ts- renders edges, flow animations.src/visualization/scene/cluster-renderer.ts- renders convex hull boundaries around static clusters.src/visualization/interactions/interaction-manager.ts- hover/click/double-click handling.src/visualization/animations/animation-controller.ts- camera focus/transition animations.src/visualization/layouts/fuzzy-layout.ts- layout generation (fuzzy, atlas, positioned modes).src/visualization/layouts/tree-layout.ts- hierarchical tree layout for storyboarding.
Events
Event system in src/core/events/event-bus.ts with types in src/core/types/events.ts.
- Subscribe with
EventBus.subscribe(type, handler)orsubscribeAll(handler). - Emit with
EventBus.emit(createEvent(type, payload, source)). useNeuronEvents()provides React helpers to subscribe/emit inside components.
API types (request/response)
All API request/response types are defined in src/core/types/api.ts. If you need to implement a server or client manually, this file is the contract.
CLI reference (omi-neuron)
The CLI is defined in src/cli/index.ts and subcommands in src/cli/commands/*.
Commands:
omi-neuron init- scaffold config, docker compose, env file, and Next.js route.--name,--port,--storage,--file-path,--persist-interval,--skip-docker,--skip-api,--skip-config,--app-dir,--force
omi-neuron db up|down|migrate|status|reset|seedup:--port,--force-recreate,--waitdown:--remove-volumesmigrate:--status,--rollback <n>,--to <version>,--dry-runstatus:--json,--verbosereset:--confirmseed:--file,--clear,--count
omi-neuron analyze embeddings|cluster|relationships|fullembeddings:--node-ids,--forcecluster:--count,--algorithmrelationships:--threshold,--dry-runfull:--force
omi-neuron ingest init|markdown|github|rss|notion- ingest external sources into the graph.init: scaffold a connector source JSONmarkdown:--source,--path,--limit,--since,--dry-rungithub:--source,--repo,--token,--state,--limit,--since,--dry-runrss:--source,--url,--limit,--since,--dry-runnotion:--source,--path,--limit,--since,--dry-run
omi-neuron validate- checks config, docker, DB, and OpenAI env.omi-neuron config get|set|list|reset- manageneuron.config.tsvalues.
Known gaps and reserved fields
- Several
NeuronWebPropsare defined but not yet used internally (see list above). VisualizationSettingsvsNeuronWebTheme: the component uses theme overrides, not settings. If you want settings to drive visuals, wire them inNeuronWeb.VERSIONis updated on tag-based releases (see below). In non-release branches, treat it as informational.
Build and test
pnpm install
pnpm build
pnpm typecheck
pnpm lint
pnpm testRelease via tags (automation)
This repo publishes on tag push. The tag version is authoritative and must be a patch bump.
Workflow:
- Create and push
vX.Y.Z(same major/minor aspackage.json, patch +1). - GitHub Actions updates
package.json+src/version.ts, commits tomain, builds, and publishes.
Helper skill (repo-local):
skills/tagged-npm-release/scripts/create_patch_tag.shAuthentication:
- Preferred: npm Trusted Publishing (OIDC). Configure the GitHub repo/workflow in npm package settings. No secret required.
- Legacy fallback: set
NPM_TOKENin GitHub repo secrets and remove--provenancefrom the workflow.
License
MIT
