chain-insights 0.2.32 → 0.3.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +40 -14
- package/dist/cases-Cp9DUbEV.mjs +6 -0
- package/dist/{cases-c0iV-XLI.cjs → cases-sTY5aXav.cjs} +3 -3
- package/dist/cli.cjs +122 -66
- package/dist/cli.mjs +122 -66
- package/dist/cli.mjs.map +1 -1
- package/dist/{viz-Da9YWN_I.cjs → data-extractor-Cavd7wHk.cjs} +11 -34
- package/dist/{viz-DkJyqlUu.mjs → data-extractor-DZUJu1Bz.mjs} +3 -32
- package/dist/data-extractor-DZUJu1Bz.mjs.map +1 -0
- package/dist/{dossier-Br62hCG7.cjs → dossier-BXy57V4-.cjs} +13 -1
- package/dist/{dossier-Bl0NkJKC.mjs → dossier-Bjpcbcxa.mjs} +4 -2
- package/dist/{dossier-Bl0NkJKC.mjs.map → dossier-Bjpcbcxa.mjs.map} +1 -1
- package/dist/export-BqTCO9lP.mjs +591 -0
- package/dist/export-BqTCO9lP.mjs.map +1 -0
- package/dist/export-DsXgtCwO.cjs +592 -0
- package/dist/index.cjs +1 -1
- package/dist/index.mjs +1 -1
- package/dist/{init-DBC9Ml33.mjs → init-DLBL_nVG.mjs} +27 -1
- package/dist/{init-DBC9Ml33.mjs.map → init-DLBL_nVG.mjs.map} +1 -1
- package/dist/{init-CFaUWgjK.cjs → init-zqbd7i-_.cjs} +26 -0
- package/dist/mcp-proxy.cjs +215 -77
- package/dist/mcp-proxy.d.cts.map +1 -1
- package/dist/mcp-proxy.d.mts.map +1 -1
- package/dist/mcp-proxy.mjs +215 -77
- package/dist/mcp-proxy.mjs.map +1 -1
- package/dist/{public-tools-BwguvIsf.cjs → public-tools-BvMb3H2P.cjs} +701 -1479
- package/dist/{public-tools-DoRNhMn9.mjs → public-tools-wJoAFDFa.mjs} +700 -1479
- package/dist/public-tools-wJoAFDFa.mjs.map +1 -0
- package/dist/{resolver-D7VBb0uB.mjs → resolver-2jXNtWQO.mjs} +12 -29
- package/dist/resolver-2jXNtWQO.mjs.map +1 -0
- package/dist/{resolver-BUU7ZgW-.cjs → resolver-CZdQwKvh.cjs} +11 -28
- package/dist/{runner-BCDeBYsR.cjs → runner-BhZ4lnF1.cjs} +2 -2
- package/dist/{runner-CTFK0Qcg.mjs → runner-DIJSbkjc.mjs} +3 -3
- package/dist/{runner-CTFK0Qcg.mjs.map → runner-DIJSbkjc.mjs.map} +1 -1
- package/dist/{selector-CTUiQrzI.mjs → selector-CF2o5gxN.mjs} +2 -2
- package/dist/{selector-CTUiQrzI.mjs.map → selector-CF2o5gxN.mjs.map} +1 -1
- package/dist/{selector-DBS2jYH4.cjs → selector-DfAMZEC9.cjs} +1 -1
- package/dist/{session-DwyikazY.cjs → session-BT7VpbAd.cjs} +13 -1
- package/dist/{session-Bha3zFrx.mjs → session-DROyhebe.mjs} +4 -2
- package/dist/{session-Bha3zFrx.mjs.map → session-DROyhebe.mjs.map} +1 -1
- package/dist/{store-BT2SCcQr.mjs → store-CTtqQtaE.mjs} +10 -4
- package/dist/{store-BT2SCcQr.mjs.map → store-CTtqQtaE.mjs.map} +1 -1
- package/dist/{store-DogLawSj.cjs → store-CqPfs47P.cjs} +37 -7
- package/dist/{tool-visibility-BHRFLXuU.mjs → tool-visibility-BpyZHRBi.mjs} +4 -2
- package/dist/tool-visibility-BpyZHRBi.mjs.map +1 -0
- package/dist/{tool-visibility-iAVQV3t0.cjs → tool-visibility-Buq7YdUZ.cjs} +3 -1
- package/dist/viz-5y24S5X1.mjs +35 -0
- package/dist/viz-5y24S5X1.mjs.map +1 -0
- package/dist/viz-Dqp3C5kb.cjs +44 -0
- package/docs/contributing.md +3 -2
- package/docs/graph-tools.md +125 -117
- package/docs/investigation-workspaces.md +14 -0
- package/docs/mcp-proxy.md +15 -2
- package/package.json +1 -1
- package/skills/chain-insights-cypher/SKILL.md +6 -0
- package/skills/chain-insights-developer-experience/SKILL.md +26 -6
- package/skills/chain-insights-investigation/SKILL.md +64 -48
- package/skills/chain-insights-trace-funds/SKILL.md +80 -197
- package/skills/test-chain-insights-graphrag-mcp/SKILL.md +1 -1
- package/skills/test-chain-insights-graphrag-mcp/scripts/run-uat.sh +4 -4
- package/dist/cases-qjPtbnUd.mjs +0 -6
- package/dist/public-tools-DoRNhMn9.mjs.map +0 -1
- package/dist/resolver-D7VBb0uB.mjs.map +0 -1
- package/dist/tool-visibility-BHRFLXuU.mjs.map +0 -1
- package/dist/viz-DkJyqlUu.mjs.map +0 -1
package/dist/cli.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"cli.mjs","names":[],"sources":["../src/cli.ts"],"sourcesContent":["import { Command, Option } from 'commander'\nimport { execFileSync } from 'node:child_process'\nimport { fileURLToPath } from 'node:url'\nimport path from 'node:path'\nimport { PACKAGE_INFO, PACKAGE_VERSION } from './version.js'\n\n// Resolve bin/install.cjs relative to this file's location in dist/\nconst __dirname = path.dirname(fileURLToPath(import.meta.url))\nconst installerPath = path.resolve(__dirname, '..', 'bin', 'install.cjs')\n\nconst program = new Command()\n\nprogram\n .name('chain-insights')\n .description('AML investigation toolkit for blockchain analysis')\n .version(PACKAGE_INFO.version)\n .option('--claude', 'Install Claude Code skills globally to ~/.claude/skills/')\n .option('--codex', 'Install Codex skills globally to ~/.codex/skills/ and register MCP')\n .option('--hermes', 'Install Hermes skills globally to ~/.hermes/skills/chain-insights/ and register MCP')\n\n// Handle installer flags when invoked with no subcommand (bare `chain-insights --claude`)\nconst rawArgs = process.argv.slice(2)\nconst installerFlags = rawArgs.filter(a => a === '--claude' || a === '--codex' || a === '--hermes')\nif (installerFlags.length > 0 && !rawArgs.some(a => !a.startsWith('-'))) {\n try {\n execFileSync(process.execPath, [installerPath, ...installerFlags], { stdio: 'inherit' })\n } catch (err) {\n console.error('Installation failed:', (err as Error).message)\n process.exit(1)\n }\n process.exit(0)\n}\n\nif (rawArgs[0] === 'mcp' && rawArgs[1] === 'trace-funds') {\n console.error(\"error: unknown command 'trace-funds'\")\n process.exit(1)\n}\n\nasync function resolveCaseSelector(input: string): Promise<string> {\n const { resolveCaseSelector } = await import('./cases/selector.js')\n return resolveCaseSelector(input)\n}\n\nasync function scopeCasesToInvocationDir(): Promise<void> {\n if (process.env['CHAIN_INSIGHTS_CASES_ROOT']?.trim()) return\n const { activeCasesRoot } = await import('./workspace/active.js')\n process.env['CHAIN_INSIGHTS_CASES_ROOT'] = activeCasesRoot()\n}\n\nasync function showCaseContext(caseSelector: string): Promise<void> {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const ctx = await CaseStore.loadContext(caseId)\n console.log(`\\n=== Case: ${ctx.case.id} ===`)\n console.log(`Name: ${ctx.case.name}`)\n console.log(`Status: ${ctx.case.status}`)\n console.log(`Tags: ${ctx.case.tags.join(', ') || 'none'}`)\n console.log(`Evidence files: ${ctx.evidenceCount}`)\n console.log(`Dossiers: ${ctx.dossierSummaries.length}`)\n if (ctx.lastSession) {\n console.log(`\\n--- Last Session (${ctx.lastSession.sessionId}) ---`)\n console.log(ctx.lastSession.body.slice(0, 500))\n } else {\n console.log('\\nNo previous sessions.')\n }\n if (ctx.dossierSummaries.length > 0) {\n console.log('\\n--- Entity Dossiers ---')\n for (const d of ctx.dossierSummaries) {\n console.log(` ${d.address} [${d.type}] tags: ${d.riskTags || 'none'}`)\n }\n }\n}\n\nfunction optionalNumber(value: string | undefined): number | undefined {\n if (value === undefined) return undefined\n const parsed = Number(value)\n if (!Number.isFinite(parsed)) throw new Error(`Invalid number: ${value}`)\n return parsed\n}\n\nfunction optionalNumberArg(value: unknown, name: string): number | undefined {\n if (value === undefined) return undefined\n if (typeof value === 'number' && Number.isFinite(value)) return value\n if (typeof value === 'string') return optionalNumber(value)\n throw new Error(`Invalid number for ${name}: ${String(value)}`)\n}\n\ntype ScamTopologyActivityPolicyModeArg = 'node_relative_only' | 'global_incident_only'\n\nfunction optionalScamTopologyActivityPolicy(value: unknown): ScamTopologyActivityPolicyModeArg | undefined {\n if (value === undefined || value === null || value === '') return undefined\n if (value === 'node_relative_only' || value === 'global_incident_only') return value\n throw new Error('activity_policy must be one of: node_relative_only, global_incident_only')\n}\n\nasync function withGraphMcpClient<T>(name: string, fn: (client: import('@modelcontextprotocol/sdk/client/index.js').Client, config: Awaited<ReturnType<typeof import('./config/index.js').loadConfig>>) => Promise<T>): Promise<T> {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n const { createConfiguredGraphMcpFetch, resolveGraphMcpEndpoint } = await import('./mcp/client.js')\n const paymentFetch = await createConfiguredGraphMcpFetch(config)\n const { Client } = await import('@modelcontextprotocol/sdk/client/index.js')\n const { StreamableHTTPClientTransport } = await import('@modelcontextprotocol/sdk/client/streamableHttp.js')\n const client = new Client({ name, version: PACKAGE_VERSION })\n await client.connect(new StreamableHTTPClientTransport(new URL(resolveGraphMcpEndpoint(config)), { fetch: paymentFetch }))\n try {\n return await fn(client, config)\n } finally {\n await client.close()\n }\n}\n\nfunction printMcpTextContent(result: { content?: Array<{ type: string; text?: string }> }): void {\n for (const item of result.content ?? []) {\n if (item.type === 'text') console.log(item.text)\n }\n}\n\nasync function printNetworkCapabilities(opts: { json?: boolean }): Promise<void> {\n const { loadConfig } = await import('./config/index.js')\n const { fetchNetworkCapabilities, formatNetworkCapabilities } = await import('./mcp/capabilities.js')\n const document = await fetchNetworkCapabilities(await loadConfig())\n if (opts.json) {\n console.log(JSON.stringify(document, null, 2))\n } else {\n console.log(formatNetworkCapabilities(document))\n }\n}\n\nprogram\n .command('networks')\n .alias('network')\n .description('List supported graph networks, capability layers, retention, and freshness')\n .option('--json', 'Print raw capability JSON')\n .action(async (opts: { json?: boolean }) => {\n try {\n await printNetworkCapabilities(opts)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('serve')\n .description('Start local visualization server')\n .option('-p, --port <number>', 'Port to bind (default: 4321)', '4321')\n .action(async (opts: { port: string }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n const workspaceRoot = requireWorkspaceRoot()\n const { startServer } = await import('./server/index.js')\n console.log(`Workspace: ${workspaceRoot}`)\n startServer(parseInt(opts.port, 10))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('status')\n .description('Show toolkit status and configuration')\n .action(async () => {\n const { loadConfig } = await import('./config/index.js')\n const { findActiveWorkspace, activeDataDir } = await import('./workspace/active.js')\n const config = await loadConfig()\n const workspace = findActiveWorkspace()\n const graphMcpStatus = config.graphMcpMode === 'debug' && config.graphMcpAuthToken?.trim()\n ? 'bearer access mode'\n : `${config.graphMcpMode} mode`\n console.log('Config: ', activeDataDir(config.dataDir))\n if (workspace) console.log('Workspace:', workspace.root)\n console.log('Server: ', `http://127.0.0.1:${config.serverPort}`)\n console.log('Graph MCP:', graphMcpStatus)\n console.log('Graph endpoint:', config.graphMcpEndpoint)\n })\n\nprogram\n .command('debug')\n .description('Configure Graph MCP debug mode')\n .addCommand(\n new Command('on')\n .description('Enable Graph MCP debug mode without x402 payments')\n .requiredOption('--token <token>', 'Debug bearer token')\n .option('--endpoint <url>', 'Graph MCP endpoint')\n .action(async (opts: { token: string; endpoint?: string }) => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({\n graphMcpMode: 'debug',\n graphMcpAuthToken: opts.token,\n ...(opts.endpoint ? { graphMcpEndpoint: opts.endpoint } : {}),\n })\n console.log('Graph MCP debug mode enabled')\n if (opts.endpoint) console.log(`Graph endpoint: ${opts.endpoint}`)\n console.log('Payments: disabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('off')\n .description('Disable Graph MCP debug mode and use paid x402 calls')\n .action(async () => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({ graphMcpMode: 'paid', graphMcpAuthToken: '' })\n console.log('Graph MCP debug mode disabled')\n console.log('Payments: enabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('status')\n .description('Show Graph MCP payment/debug mode')\n .action(async () => {\n try {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n console.log(`Graph MCP mode: ${config.graphMcpMode}`)\n console.log(`Graph endpoint: ${config.graphMcpEndpoint}`)\n console.log(`Debug token: ${config.graphMcpAuthToken?.trim() ? 'configured' : 'not configured'}`)\n console.log(`Payments: ${config.graphMcpMode === 'debug' ? 'disabled' : 'enabled'}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('access-key')\n .description('Configure Graph MCP test access key mode')\n .addCommand(\n new Command('set')\n .description('Use a Graph MCP test access key without x402 payments')\n .argument('<key>', 'Test access key')\n .option('--endpoint <url>', 'Graph MCP endpoint')\n .action(async (key: string, opts: { endpoint?: string }) => {\n try {\n const normalizedKey = key.trim()\n if (!normalizedKey) throw new Error('Test access key is required')\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({\n graphMcpMode: 'debug',\n graphMcpAuthToken: normalizedKey,\n ...(opts.endpoint ? { graphMcpEndpoint: opts.endpoint } : {}),\n })\n console.log('Graph MCP test access key configured')\n if (opts.endpoint) console.log(`Graph endpoint: ${opts.endpoint}`)\n console.log('Payments: disabled when the server accepts this key')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('clear')\n .description('Remove the Graph MCP test access key and use paid x402 calls')\n .action(async () => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({ graphMcpMode: 'paid', graphMcpAuthToken: '' })\n console.log('Graph MCP test access key cleared')\n console.log('Payments: enabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('status')\n .description('Show Graph MCP test access key status')\n .action(async () => {\n try {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n console.log(`Graph endpoint: ${config.graphMcpEndpoint}`)\n console.log(`Access key: ${config.graphMcpAuthToken?.trim() ? 'configured' : 'not configured'}`)\n console.log(`Payments: ${config.graphMcpAuthToken?.trim() ? 'disabled when accepted by server' : 'enabled'}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('init')\n .description('Initialize an investigation workspace')\n .argument('[dir]', 'Workspace directory to initialize', '.')\n .option('--force', 'Overwrite existing workspace files')\n .action(async (dir: string, opts: { force?: boolean }) => {\n try {\n const { initWorkspace } = await import('./workspace/init.js')\n const result = await initWorkspace({ targetDir: dir, force: opts.force })\n console.log(`Workspace initialized: ${result.workspaceRoot}`)\n console.log(`Files written: ${result.filesWritten.length}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('setup')\n .description('Configure external MCP clients')\n .addCommand(\n new Command('claude-desktop')\n .alias('claude')\n .description('Install or update the Claude Desktop MCP server entry')\n .option('--config <path>', 'Path to claude_desktop_config.json')\n .option('--dry-run', 'Print the intended change without writing files')\n .action(async (opts: { config?: string; dryRun?: boolean }) => {\n try {\n const { setupClaudeDesktop } = await import('./claude-desktop/setup.js')\n const result = await setupClaudeDesktop({\n configPath: opts.config,\n dryRun: opts.dryRun,\n })\n\n console.log(`Claude Desktop config: ${result.configPath}`)\n console.log('MCP server: chain-insights')\n console.log(`Command: ${result.command}`)\n console.log(`Args: ${result.args.join(' ')}`)\n if (result.dryRun) {\n console.log(`Dry run: ${result.changed ? 'would update config' : 'already up to date'}`)\n } else if (result.changed) {\n console.log(`Updated: yes`)\n if (result.backupPath) console.log(`Backup: ${result.backupPath}`)\n } else {\n console.log('Updated: already up to date')\n }\n console.log('Reload required: quit and reopen Claude Desktop; it does not hot-reload MCP config.')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('config')\n .description('Read or write configuration values')\n .addCommand(\n new Command('get')\n .argument('<key>', 'Config key to read')\n .action(async (key: string) => {\n const { loadConfig } = await import('./config/index.js')\n const { CONFIG_KEYS } = await import('./config/schema.js')\n if (!CONFIG_KEYS.includes(key as typeof CONFIG_KEYS[number])) {\n console.error(`Unknown config key: ${key}`)\n process.exit(1)\n }\n const config = await loadConfig()\n const value = (config as Record<string, unknown>)[key]\n console.log(value ?? '')\n })\n )\n .addCommand(\n new Command('set')\n .argument('<key>', 'Config key to write')\n .argument('<value>', 'Value to set')\n .action(async (key: string, value: string) => {\n // D-01: walletPrivateKey is intercepted before saveConfig — the raw private key\n // must NEVER be written to config.json.\n if (key === 'walletPrivateKey') {\n try {\n const { setWalletPrivateKey } = await import('./wallet/index.js')\n const address = await setWalletPrivateKey(value)\n console.log('Wallet private key encrypted and stored in ~/.chain-insights/wallet.json')\n console.log(`Wallet address: ${address}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n return // MUST return — walletPrivateKey must never reach saveConfig or config.json\n }\n const { loadConfig, saveConfig } = await import('./config/index.js')\n const { CONFIG_KEYS, DEFAULT_CONFIG } = await import('./config/schema.js')\n const current = await loadConfig()\n if (!CONFIG_KEYS.includes(key as typeof CONFIG_KEYS[number])) {\n console.error(`Unknown config key: ${key}`)\n process.exit(1)\n }\n const existing = (current as Record<string, unknown>)[key]\n const defaultValue = (DEFAULT_CONFIG as Record<string, unknown>)[key]\n const coerced = typeof existing === 'number' || typeof defaultValue === 'number' ? Number(value) : value\n await saveConfig({ [key]: coerced } as Parameters<typeof saveConfig>[0])\n const displayed = key.toLowerCase().includes('token') ? '[redacted]' : coerced\n console.log(`Set ${key} = ${displayed}`)\n })\n )\n\nprogram\n .command('wallet')\n .description('Manage the local Base USDC payment wallet')\n .addCommand(\n new Command('import')\n .description('Import a Base payment wallet')\n .argument('<private-key>', '0x-prefixed EVM private key')\n .action(async (privateKey: string) => {\n try {\n const { setWalletPrivateKey } = await import('./wallet/index.js')\n const address = await setWalletPrivateKey(privateKey)\n console.log(`Wallet imported: ${address}`)\n console.log('Next: run `chain-insights wallet ready`')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('address')\n .description('Print the local payment wallet address')\n .action(async () => {\n try {\n const { getWalletAccount } = await import('./wallet/tools.js')\n const account = await getWalletAccount()\n console.log(account.address)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('balance')\n .description('Show the local payment wallet Base USDC balance')\n .action(async () => {\n try {\n const { getWalletBalanceText } = await import('./wallet/tools.js')\n console.log(await getWalletBalanceText())\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('ready')\n .description('Check and prepare the wallet for paid GraphRAG MCP calls')\n .option('--check-only', 'Only check readiness; do not submit the one-time payment setup')\n .addOption(new Option('--no-approve', 'Deprecated alias for --check-only').hideHelp())\n .option('--payment-usdc <amount>', 'USDC setup cap to prepare for paid calls', '1')\n .addOption(new Option('--approval-usdc <amount>', 'Deprecated alias for --payment-usdc').hideHelp())\n .option('--json', 'Print machine-readable readiness metadata')\n .action(async (opts: { checkOnly?: boolean; approve?: boolean; paymentUsdc?: string; approvalUsdc?: string; json?: boolean }) => {\n try {\n const { formatWalletReadiness, parsePaymentApprovalUnits, prepareWalletForPaidCalls } = await import('./wallet/tools.js')\n const minimumApprovalUnits = parsePaymentApprovalUnits(opts.paymentUsdc ?? opts.approvalUsdc ?? '1')\n const result = await prepareWalletForPaidCalls({\n minimumApprovalUnits,\n approve: opts.checkOnly ? false : opts.approve !== false,\n })\n\n if (opts.json) {\n console.log(JSON.stringify(result, (_key, value) => (\n typeof value === 'bigint' ? value.toString() : value\n ), 2))\n return\n }\n\n console.log(formatWalletReadiness(result.readiness, result.approval))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('topup')\n .description('Open a local browser page to top up the payment wallet')\n .option('--no-open', 'Print the top-up URL without opening a browser')\n .option('--json', 'Print machine-readable top-up metadata')\n .action(async (opts: { open?: boolean; json?: boolean }) => {\n try {\n const { buildTopupInfo, getWalletAccount } = await import('./wallet/tools.js')\n const { startTopupServer } = await import('./wallet/topup-server.js')\n const account = await getWalletAccount()\n const url = await startTopupServer(account)\n const info = buildTopupInfo(account.address, url)\n\n if (opts.json) {\n console.log(JSON.stringify(info, null, 2))\n } else {\n console.log(`Top-up URL: ${url}`)\n console.log(`Wallet: ${account.address}`)\n console.log('Network: Base')\n console.log('Token: USDC')\n console.log('Press Ctrl+C to stop the top-up server.')\n }\n\n if (opts.open !== false) {\n const open = (await import('open')).default\n await open(url)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('mcp')\n .description('Interact with the Chain Insights MCP endpoint')\n .allowExcessArguments(false)\n .addCommand(\n new Command('networks')\n .description('List supported graph networks, capability layers, retention, and freshness')\n .option('--json', 'Print raw capability JSON')\n .action(async (opts: { json?: boolean }) => {\n try {\n await printNetworkCapabilities(opts)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('tools')\n .description('List available MCP tools (cached 24h)')\n .option('--refresh', 'Force refresh schema cache')\n .action(async (opts: { refresh?: boolean }) => {\n try {\n const { loadSchema, saveSchema } = await import('./mcp/schema-cache.js')\n const { formatToolsTable } = await import('./mcp/format.js')\n const { visibleRemoteTools } = await import('./mcp/tool-visibility.js')\n const { loadConfig } = await import('./config/index.js')\n const { createConfiguredGraphMcpFetch, resolveGraphMcpEndpoint } = await import('./mcp/client.js')\n const config = await loadConfig()\n const graphMcpEndpoint = resolveGraphMcpEndpoint(config)\n let tools = opts.refresh ? null : await loadSchema(graphMcpEndpoint)\n if (!tools) {\n const paymentFetch = await createConfiguredGraphMcpFetch(config)\n const { Client } = await import('@modelcontextprotocol/sdk/client/index.js')\n const { StreamableHTTPClientTransport } = await import('@modelcontextprotocol/sdk/client/streamableHttp.js')\n const client = new Client({ name: 'chain-insights-cli', version: PACKAGE_VERSION })\n await client.connect(new StreamableHTTPClientTransport(new URL(graphMcpEndpoint), { fetch: paymentFetch }))\n try {\n const result = await client.listTools()\n tools = result.tools as Array<{ name: string; description?: string }>\n await saveSchema(tools, graphMcpEndpoint)\n } finally {\n await client.close()\n }\n }\n console.log(formatToolsTable(visibleRemoteTools(tools)))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('address-risk')\n .description('Screen an address for risk, exchange behavior, and optional compare_address connection risk')\n .requiredOption('--address <address>', 'Full blockchain address to screen')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--compare-address <address>', 'Optional second address for connection-risk compare mode')\n .option('--remote', 'Force remote MCP tool call instead of local Chain Insights recipe')\n .action(async (opts: { address: string; network: string; compareAddress?: string; remote?: boolean }) => {\n try {\n await withGraphMcpClient('chain-insights-cli-address-risk', async (client) => {\n if (opts.remote) {\n const result = await client.callTool({\n name: 'address_risk',\n arguments: {\n address: opts.address,\n network: opts.network,\n ...(opts.compareAddress ? { compare_address: opts.compareAddress } : {}),\n },\n })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n return\n }\n const { addressRisk } = await import('./investigation/public-tools.js')\n const result = await addressRisk(client, {\n address: opts.address,\n network: opts.network,\n compareAddress: opts.compareAddress,\n })\n console.log(result.summaryText)\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('track-funds')\n .description('Trace trusted/victim addresses and optional known untrusted/scammer addresses')\n .requiredOption('--trusted-addresses <addresses>', 'Comma-separated full trusted/victim addresses, max 5')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--untrusted-addresses <addresses>', 'Comma-separated full known untrusted/scammer addresses, max 5')\n .option('--case <id>', 'Case ID to attach compact evidence pointers')\n .option('--max-hops <number>', 'Maximum trace hops, 1-5')\n .option('--per-address-limit <number>', 'Maximum exchange paths/results per address, 1-10')\n .option('--min-amount-sum <number>', 'Minimum r.amount_sum for traced edges')\n .option('--remote', 'Force remote MCP tool call instead of local Chain Insights recipe')\n .action(async (opts: {\n trustedAddresses: string\n network: string\n untrustedAddresses?: string\n case?: string\n maxHops?: string\n perAddressLimit?: string\n minAmountSum?: string\n remote?: boolean\n }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n requireWorkspaceRoot()\n await withGraphMcpClient('chain-insights-cli-track-funds', async (client, config) => {\n if (opts.remote) {\n const result = await client.callTool({\n name: 'track_funds',\n arguments: {\n trusted_addresses: opts.trustedAddresses,\n network: opts.network,\n ...(opts.untrustedAddresses ? { untrusted_addresses: opts.untrustedAddresses } : {}),\n },\n })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n return\n }\n const { trackFunds } = await import('./investigation/public-tools.js')\n const caseId = opts.case ? await resolveCaseSelector(opts.case) : undefined\n const result = await trackFunds(client, config, {\n trustedAddresses: opts.trustedAddresses,\n untrustedAddresses: opts.untrustedAddresses,\n network: opts.network,\n caseId,\n maxHops: optionalNumber(opts.maxHops),\n perAddressLimit: optionalNumber(opts.perAddressLimit),\n minAmountSum: optionalNumber(opts.minAmountSum),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('scam-topology')\n .description('Build victim-incident scam topology and ML-ready scam labels')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .requiredOption('--victim-address <address>', 'Full victim/source address that anchors the incident')\n .requiredOption('--incident-timestamp-ms <milliseconds>', 'Earliest known incident transfer timestamp in milliseconds')\n .option('--max-hops <number>', 'Maximum trace hops, default 16, max 64')\n .option('--activity-policy <mode>', 'Traversal activity policy: node_relative_only or global_incident_only', 'node_relative_only')\n .option('--case <id>', 'Case ID to attach compact evidence pointers')\n .action(async (opts: {\n network: string\n victimAddress: string\n incidentTimestampMs: string\n maxHops?: string\n activityPolicy?: string\n case?: string\n }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n requireWorkspaceRoot()\n await withGraphMcpClient('chain-insights-cli-scam-topology', async (client, config) => {\n const { scamTopology } = await import('./investigation/public-tools.js')\n const incidentTimestampMs = optionalNumber(opts.incidentTimestampMs)\n if (incidentTimestampMs === undefined) throw new Error('incident-timestamp-ms is required')\n const caseId = opts.case ? await resolveCaseSelector(opts.case) : undefined\n const result = await scamTopology(client, config, {\n victimAddress: opts.victimAddress,\n network: opts.network,\n maxHops: optionalNumber(opts.maxHops),\n incidentTimestampMs,\n activityPolicyMode: optionalScamTopologyActivityPolicy(opts.activityPolicy),\n caseId,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('stake-insights')\n .description('Explain Bittensor staking behavior around an address, coldkey, or hotkey')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--address <address>', 'Full Bittensor address to inspect as either coldkey or hotkey')\n .option('--coldkey <address>', 'Full Bittensor coldkey address to inspect')\n .option('--hotkey <address>', 'Full Bittensor hotkey address to inspect')\n .option('--netuid <number>', 'Optional subnet netuid filter')\n .option('--start-timestamp-ms <milliseconds>', 'Optional inclusive lower activity timestamp bound')\n .option('--end-timestamp-ms <milliseconds>', 'Optional inclusive upper activity timestamp bound')\n .option('--start-block <number>', 'Optional start block. Current stake graph parity may require timestamp windows instead.')\n .option('--end-block <number>', 'Optional end block. Current stake graph parity may require timestamp windows instead.')\n .option('--depth <number>', 'Optional expansion depth limit, default 1, max 3')\n .action(async (opts: {\n network: string\n address?: string\n coldkey?: string\n hotkey?: string\n netuid?: string\n startTimestampMs?: string\n endTimestampMs?: string\n startBlock?: string\n endBlock?: string\n depth?: string\n }) => {\n try {\n await withGraphMcpClient('chain-insights-cli-stake-insights', async (client) => {\n const { stakeInsights } = await import('./investigation/public-tools.js')\n const result = await stakeInsights(client, {\n network: opts.network,\n address: opts.address,\n coldkey: opts.coldkey,\n hotkey: opts.hotkey,\n netuid: optionalNumber(opts.netuid),\n startTimestampMs: optionalNumber(opts.startTimestampMs),\n endTimestampMs: optionalNumber(opts.endTimestampMs),\n startBlock: optionalNumber(opts.startBlock),\n endBlock: optionalNumber(opts.endBlock),\n depth: optionalNumber(opts.depth),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('call')\n .description('Call an MCP tool directly (debug)')\n .argument('<tool>', 'Tool name to call')\n .argument('[args...]', 'Key=value arguments (e.g. address=0x1234 chain=ethereum)')\n .action(async (tool: string, rawArgs: string[]) => {\n try {\n const { parseMcpCallArgs } = await import('./mcp/call-args.js')\n const { assertPublicMcpToolName } = await import('./mcp/tool-visibility.js')\n const args = parseMcpCallArgs(rawArgs)\n assertPublicMcpToolName(tool)\n await withGraphMcpClient('chain-insights-cli-call', async (client, config) => {\n if (tool === 'address_risk') {\n const { addressRisk } = await import('./investigation/public-tools.js')\n const result = await addressRisk(client, {\n address: String(args['address'] ?? ''),\n network: String(args['network'] ?? ''),\n compareAddress: args['compare_address'] === undefined ? undefined : String(args['compare_address']),\n })\n console.log(result.summaryText)\n return\n }\n if (tool === 'track_funds') {\n const { trackFunds } = await import('./investigation/public-tools.js')\n const result = await trackFunds(client, config, {\n trustedAddresses: args['trusted_addresses'] as string | string[] | undefined ?? '',\n untrustedAddresses: args['untrusted_addresses'] as string | string[] | undefined,\n network: String(args['network'] ?? ''),\n caseId: args['case_id'] === undefined ? undefined : String(args['case_id']),\n maxHops: typeof args['max_hops'] === 'number' ? args['max_hops'] : undefined,\n perAddressLimit: typeof args['per_address_limit'] === 'number' ? args['per_address_limit'] : undefined,\n minAmountSum: typeof args['min_amount_sum'] === 'number' ? args['min_amount_sum'] : undefined,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n if (tool === 'scam_topology') {\n const { scamTopology } = await import('./investigation/public-tools.js')\n const victimAddress = String(args['victim_address'] ?? '').trim()\n if (!victimAddress) throw new Error('victim_address is required')\n const incidentTimestampMs = optionalNumberArg(args['incident_timestamp_ms'], 'incident_timestamp_ms')\n if (incidentTimestampMs === undefined) throw new Error('incident_timestamp_ms is required')\n const result = await scamTopology(client, config, {\n victimAddress,\n network: String(args['network'] ?? ''),\n caseId: args['case_id'] === undefined ? undefined : String(args['case_id']),\n maxHops: typeof args['max_hops'] === 'number' ? args['max_hops'] : undefined,\n incidentTimestampMs,\n activityPolicyMode: optionalScamTopologyActivityPolicy(args['activity_policy']),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n if (tool === 'stake_insights') {\n const { stakeInsights } = await import('./investigation/public-tools.js')\n const result = await stakeInsights(client, {\n network: String(args['network'] ?? ''),\n address: args['address'] === undefined ? undefined : String(args['address']),\n coldkey: args['coldkey'] === undefined ? undefined : String(args['coldkey']),\n hotkey: args['hotkey'] === undefined ? undefined : String(args['hotkey']),\n netuid: optionalNumberArg(args['netuid'], 'netuid'),\n startTimestampMs: optionalNumberArg(args['start_timestamp_ms'], 'start_timestamp_ms'),\n endTimestampMs: optionalNumberArg(args['end_timestamp_ms'], 'end_timestamp_ms'),\n startBlock: optionalNumberArg(args['start_block'], 'start_block'),\n endBlock: optionalNumberArg(args['end_block'], 'end_block'),\n depth: optionalNumberArg(args['depth'] ?? args['max_hops'], 'depth'),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n const result = await client.callTool({ name: tool, arguments: args })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nconst caseCommand = new Command('case')\n .description('Manage investigation cases')\n .hook('preAction', async () => {\n await scopeCasesToInvocationDir()\n })\n .addCommand(\n new Command('open')\n .description('Open a new investigation case')\n .argument('<name>', 'Case name (e.g. \"Tornado Mixer Investigation\")')\n .option('--tags <tags>', 'Comma-separated tags (e.g. aml,mixer,defi)', '')\n .option('--description <desc>', 'Brief description of the investigation', '')\n .action(async (name: string, opts: { tags: string; description: string }) => {\n try {\n if (/^[1-9]\\d*$/.test(name.trim())) {\n throw new Error('Numeric case names look like list selectors. Use a descriptive case name, e.g. `cia case open \"Tracking stolen funds from <address>\"`.')\n }\n const { CaseStore } = await import('./cases/index.js')\n const tags = opts.tags ? opts.tags.split(',').map(t => t.trim()).filter(Boolean) : []\n const c = await CaseStore.create({ name, tags, description: opts.description })\n const { casesRoot } = await import('./cases/store.js')\n console.log(`Case opened: ${c.id}`)\n console.log(`Directory: ${path.join(casesRoot(), c.id)}/`)\n console.log(`Status: ${c.status}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('activate')\n .description('Activate a case (set status to active)')\n .argument('<case-id>', 'Case ID to activate')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'active')\n console.log(`Case ${c.id} is now: active`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('suspend')\n .description('Suspend a case (set status to suspended)')\n .argument('<case-id>', 'Case ID to suspend')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'suspended')\n console.log(`Case ${c.id} is now: suspended`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('close')\n .description('Close a case permanently')\n .argument('<case-id>', 'Case ID to close')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'closed')\n console.log(`Case ${c.id} is now: closed`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('list')\n .description('List all investigation cases')\n .option('--status <status>', 'Filter by status (open|active|suspended|closed)')\n .action(async (opts: { status?: string }) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const cases = await CaseStore.list()\n const filtered = opts.status ? cases.filter(c => c.status === opts.status) : cases\n if (filtered.length === 0) {\n console.log('No cases found.')\n return\n }\n for (const [index, c] of filtered.entries()) {\n console.log(`${index + 1}. ${c.id} [${c.status}] ${c.name}`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('evidence')\n .description('Manage case evidence')\n .addCommand(\n new Command('add')\n .description('Add evidence to a case from an MCP query result')\n .argument('<case-id>', 'Case ID to add evidence to')\n .option('--source <tool>', 'MCP tool name that produced this evidence', 'manual')\n .option('--content <text>', 'Evidence content (MCP response or notes)', '')\n .option('--query-params <params>', 'Query parameters used (e.g. address=0x1234)', '')\n .action(async (caseSelector: string, opts: { source: string; content: string; queryParams: string }) => {\n try {\n const { EvidenceStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const result = await EvidenceStore.append(caseId, {\n source: opts.source,\n content: opts.content,\n queryParams: opts.queryParams,\n })\n console.log(`Evidence saved: ${result.filename}`)\n console.log(`SHA-256: ${result.sha256}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('verify')\n .description('Verify evidence manifest integrity for a case')\n .argument('<case-id>', 'Case ID to verify')\n .action(async (caseSelector: string) => {\n try {\n const { EvidenceStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const result = await EvidenceStore.verifyManifest(caseId)\n if (result.ok) {\n console.log(`Manifest OK — ${result.count} evidence file(s) verified`)\n } else {\n console.error(`Manifest FAILED — tampered files: ${(result.tampered ?? []).join(', ')}`)\n process.exit(1)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('dossier')\n .description('Manage entity dossiers for a case')\n .addCommand(\n new Command('update')\n .description('Append a finding to an entity dossier')\n .argument('<case-id>', 'Case ID')\n .argument('<address>', 'Entity address or identifier')\n .option('--finding <text>', 'Finding to append to the dossier', '')\n .option('--type <type>', 'Entity type (eoa|contract|exchange|mixer|unknown)', 'unknown')\n .action(async (caseSelector: string, address: string, opts: { finding: string; type: string }) => {\n try {\n const { DossierStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const validTypes = ['eoa', 'contract', 'exchange', 'mixer', 'unknown'] as const\n const entityType = validTypes.includes(opts.type as typeof validTypes[number])\n ? (opts.type as typeof validTypes[number])\n : 'unknown'\n await DossierStore.appendFinding(caseId, address, opts.finding, entityType)\n console.log(`Dossier updated for ${address}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('session')\n .description('Manage investigation sessions')\n .addCommand(\n new Command('start')\n .description('Start a new investigation session for a case')\n .argument('<case-id>', 'Case ID')\n .argument('[title...]', 'Optional session title')\n .action(async (caseSelector: string, titleParts: string[]) => {\n try {\n const { SessionStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const title = titleParts.join(' ').trim()\n const s = await SessionStore.start(caseId, title ? { title } : {})\n console.log(`Session started: ${s.sessionId}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('end')\n .description('End the current session with findings and next steps')\n .argument('<case-id>', 'Case ID')\n .option('--findings <text>', 'Key findings from this session', '')\n .option('--next-steps <text>', 'Next steps for the investigation', '')\n .action(async (caseSelector: string, opts: { findings: string; nextSteps: string }) => {\n try {\n const { SessionStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n await SessionStore.end(caseId, { findings: opts.findings, nextSteps: opts.nextSteps })\n await SessionStore.archiveOldSessions(caseId)\n console.log(`Session ended for case ${caseId}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('show')\n .description('Show saved case context')\n .argument('<case-id>', 'Case ID or case list number to show')\n .action(async (caseSelector: string) => {\n try {\n await showCaseContext(caseSelector)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram.addCommand(caseCommand)\n\nprogram\n .command('playbook')\n .description('Run and manage investigation playbooks')\n .addCommand(\n new Command('run')\n .description('Execute a playbook by name')\n .argument('<name>', 'Playbook name (e.g. trace-funds, risk-check, entity-profile)')\n .option('--case <id>', 'Case ID to attach evidence to (auto-created if omitted)')\n .option('--from <n>', 'Resume from step N (1-based)', '1')\n .option('--dry-run', 'Show steps without executing')\n .option('-p, --param <kv...>', 'Parameters as key=value pairs (repeatable, e.g. -p address=0x1 -p hops=3)')\n .action(async (name: string, opts: { case?: string; from: string; dryRun?: boolean; param?: string[] }) => {\n try {\n // 1. Parse --param key=value pairs into Record<string,string> — split on first '=' only (T-05-06)\n const resolvedParams: Record<string, string> = {}\n for (const kv of (opts.param ?? [])) {\n const eq = kv.indexOf('=')\n if (eq === -1) {\n console.error(`Invalid param format: \"${kv}\". Use key=value`)\n process.exit(1)\n }\n const key = kv.slice(0, eq)\n if (!key) {\n console.error(`Invalid param format: \"${kv}\". Key must be non-empty`)\n process.exit(1)\n }\n resolvedParams[key] = kv.slice(eq + 1)\n }\n // 2. Resolve playbook content (user-dir first, built-in fallback)\n const { resolvePlaybookContent } = await import('./playbooks/resolver.js')\n const markdown = await resolvePlaybookContent(name)\n // 3. Parse markdown → PlaybookDefinition\n const { PlaybookParser } = await import('./playbooks/parser.js')\n const definition = PlaybookParser.parse(markdown, resolvedParams)\n // 4. Validate required params are provided\n for (const spec of definition.params) {\n if (spec.required && !resolvedParams[spec.name] && !spec.default) {\n console.error(`Missing required param: ${spec.name}. Pass with: -p ${spec.name}=<value>`)\n process.exit(1)\n }\n }\n // 5. Validate --from value\n const fromN = parseInt(opts.from, 10)\n if (isNaN(fromN) || fromN < 1) {\n console.error(`Invalid --from value: \"${opts.from}\". Must be a positive integer.`)\n process.exit(1)\n }\n // 6. Run\n const { PlaybookRunner } = await import('./playbooks/runner.js')\n await PlaybookRunner.run(definition, {\n caseId: opts.case,\n from: fromN,\n dryRun: opts.dryRun,\n params: resolvedParams,\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('list')\n .description('List available playbooks (built-in and user-defined)')\n .action(async () => {\n try {\n const { listPlaybooks } = await import('./playbooks/resolver.js')\n const playbooks = await listPlaybooks()\n if (playbooks.length === 0) { console.log('No playbooks found.'); return }\n for (const p of playbooks) {\n console.log(` ${p.name.padEnd(20)} [${p.source}]`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('show')\n .description('Show steps for a playbook without executing')\n .argument('<name>', 'Playbook name')\n .action(async (name: string) => {\n try {\n const { resolvePlaybookContent } = await import('./playbooks/resolver.js')\n const { PlaybookParser } = await import('./playbooks/parser.js')\n const markdown = await resolvePlaybookContent(name)\n const definition = PlaybookParser.parse(markdown, {})\n console.log(`Playbook: ${definition.name} v${definition.version}`)\n console.log(`${definition.description}\\n`)\n console.log(`Parameters:`)\n for (const p of definition.params) {\n const req = p.required ? '(required)' : `(optional, default: ${p.default ?? 'none'})`\n console.log(` ${p.name}: ${p.type} ${req}`)\n }\n console.log(`\\nSteps:`)\n for (const step of definition.steps) {\n console.log(` ${step.index}. ${step.label} → tool: ${step.tool}`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('viz')\n .description('Generate money flow visualization')\n .argument('[case-id]', 'Case ID to visualize')\n .option('--data <file>', 'Raw transaction JSON file for ad-hoc visualization')\n .option('-p, --port <number>', 'Server port', '4321')\n .action(async (caseId: string | undefined, opts: { data?: string; port: string }) => {\n try {\n if (!caseId && !opts.data) {\n console.error('Provide either a case ID or --data <file.json>')\n process.exit(1)\n }\n const { generateVisualization } = await import('./viz/index.js')\n const result = await generateVisualization({ caseId, dataFile: opts.data })\n const { startServer } = await import('./server/index.js')\n const port = parseInt(opts.port, 10)\n startServer(port)\n const url = `http://127.0.0.1:${port}/viz/${result.vizId}`\n console.log(`Visualization: ${url}`)\n const open = (await import('open')).default\n await open(url)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram.parse(process.argv)\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"cli.mjs","names":[],"sources":["../src/cli.ts"],"sourcesContent":["import { Command, Option } from 'commander'\nimport { execFileSync } from 'node:child_process'\nimport { fileURLToPath } from 'node:url'\nimport path from 'node:path'\nimport { PACKAGE_INFO, PACKAGE_VERSION } from './version.js'\n\n// Resolve bin/install.cjs relative to this file's location in dist/\nconst __dirname = path.dirname(fileURLToPath(import.meta.url))\nconst installerPath = path.resolve(__dirname, '..', 'bin', 'install.cjs')\n\nconst program = new Command()\n\nprogram\n .name('chain-insights')\n .description('AML investigation toolkit for blockchain analysis')\n .version(PACKAGE_INFO.version)\n .option('--claude', 'Install Claude Code skills globally to ~/.claude/skills/')\n .option('--codex', 'Install Codex skills globally to ~/.codex/skills/ and register MCP')\n .option('--hermes', 'Install Hermes skills globally to ~/.hermes/skills/chain-insights/ and register MCP')\n\n// Handle installer flags when invoked with no subcommand (bare `chain-insights --claude`)\nconst rawArgs = process.argv.slice(2)\nconst installerFlags = rawArgs.filter(a => a === '--claude' || a === '--codex' || a === '--hermes')\nif (installerFlags.length > 0 && !rawArgs.some(a => !a.startsWith('-'))) {\n try {\n execFileSync(process.execPath, [installerPath, ...installerFlags], { stdio: 'inherit' })\n } catch (err) {\n console.error('Installation failed:', (err as Error).message)\n process.exit(1)\n }\n process.exit(0)\n}\n\nif (rawArgs[0] === 'mcp' && ['trace-funds', 'track-funds', 'scam-topology'].includes(rawArgs[1] ?? '')) {\n console.error(`error: unknown command '${rawArgs[1]}'`)\n process.exit(1)\n}\n\nasync function resolveCaseSelector(input: string): Promise<string> {\n const { resolveCaseSelector } = await import('./cases/selector.js')\n return resolveCaseSelector(input)\n}\n\nasync function scopeCasesToInvocationDir(): Promise<void> {\n if (process.env['CHAIN_INSIGHTS_CASES_ROOT']?.trim()) return\n const { activeCasesRoot } = await import('./workspace/active.js')\n process.env['CHAIN_INSIGHTS_CASES_ROOT'] = activeCasesRoot()\n}\n\nasync function showCaseContext(caseSelector: string): Promise<void> {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const ctx = await CaseStore.loadContext(caseId)\n console.log(`\\n=== Case: ${ctx.case.id} ===`)\n console.log(`Name: ${ctx.case.name}`)\n console.log(`Status: ${ctx.case.status}`)\n console.log(`Tags: ${ctx.case.tags.join(', ') || 'none'}`)\n console.log(`Evidence files: ${ctx.evidenceCount}`)\n console.log(`Dossiers: ${ctx.dossierSummaries.length}`)\n if (ctx.lastSession) {\n console.log(`\\n--- Last Session (${ctx.lastSession.sessionId}) ---`)\n console.log(ctx.lastSession.body.slice(0, 500))\n } else {\n console.log('\\nNo previous sessions.')\n }\n if (ctx.dossierSummaries.length > 0) {\n console.log('\\n--- Entity Dossiers ---')\n for (const d of ctx.dossierSummaries) {\n console.log(` ${d.address} [${d.type}] tags: ${d.riskTags || 'none'}`)\n }\n }\n}\n\nfunction optionalNumber(value: string | undefined): number | undefined {\n if (value === undefined) return undefined\n const parsed = Number(value)\n if (!Number.isFinite(parsed)) throw new Error(`Invalid number: ${value}`)\n return parsed\n}\n\nfunction optionalNumberArg(value: unknown, name: string): number | undefined {\n if (value === undefined) return undefined\n if (typeof value === 'number' && Number.isFinite(value)) return value\n if (typeof value === 'string') return optionalNumber(value)\n throw new Error(`Invalid number for ${name}: ${String(value)}`)\n}\n\nasync function withGraphMcpClient<T>(name: string, fn: (client: import('@modelcontextprotocol/sdk/client/index.js').Client, config: Awaited<ReturnType<typeof import('./config/index.js').loadConfig>>) => Promise<T>): Promise<T> {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n const { createConfiguredGraphMcpFetch, resolveGraphMcpEndpoint } = await import('./mcp/client.js')\n const paymentFetch = await createConfiguredGraphMcpFetch(config)\n const { Client } = await import('@modelcontextprotocol/sdk/client/index.js')\n const { StreamableHTTPClientTransport } = await import('@modelcontextprotocol/sdk/client/streamableHttp.js')\n const client = new Client({ name, version: PACKAGE_VERSION })\n await client.connect(new StreamableHTTPClientTransport(new URL(resolveGraphMcpEndpoint(config)), { fetch: paymentFetch }))\n try {\n return await fn(client, config)\n } finally {\n await client.close()\n }\n}\n\nfunction printMcpTextContent(result: { content?: Array<{ type: string; text?: string }> }): void {\n for (const item of result.content ?? []) {\n if (item.type === 'text') console.log(item.text)\n }\n}\n\nasync function printNetworkCapabilities(opts: { json?: boolean }): Promise<void> {\n const { loadConfig } = await import('./config/index.js')\n const { fetchNetworkCapabilities, formatNetworkCapabilities } = await import('./mcp/capabilities.js')\n const document = await fetchNetworkCapabilities(await loadConfig())\n if (opts.json) {\n console.log(JSON.stringify(document, null, 2))\n } else {\n console.log(formatNetworkCapabilities(document))\n }\n}\n\nprogram\n .command('networks')\n .alias('network')\n .description('List supported graph networks, capability layers, retention, and freshness')\n .option('--json', 'Print raw capability JSON')\n .action(async (opts: { json?: boolean }) => {\n try {\n await printNetworkCapabilities(opts)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('serve')\n .description('Start local visualization server')\n .option('-p, --port <number>', 'Port to bind (default: 4321)', '4321')\n .action(async (opts: { port: string }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n const workspaceRoot = requireWorkspaceRoot()\n const { startServer } = await import('./server/index.js')\n console.log(`Workspace: ${workspaceRoot}`)\n startServer(parseInt(opts.port, 10))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('status')\n .description('Show toolkit status and configuration')\n .action(async () => {\n const { loadConfig } = await import('./config/index.js')\n const { findActiveWorkspace, activeDataDir } = await import('./workspace/active.js')\n const config = await loadConfig()\n const workspace = findActiveWorkspace()\n const graphMcpStatus = config.graphMcpMode === 'debug' && config.graphMcpAuthToken?.trim()\n ? 'bearer access mode'\n : `${config.graphMcpMode} mode`\n console.log('Config: ', activeDataDir(config.dataDir))\n if (workspace) console.log('Workspace:', workspace.root)\n console.log('Server: ', `http://127.0.0.1:${config.serverPort}`)\n console.log('Graph MCP:', graphMcpStatus)\n console.log('Graph endpoint:', config.graphMcpEndpoint)\n })\n\nprogram\n .command('debug')\n .description('Configure Graph MCP debug mode')\n .addCommand(\n new Command('on')\n .description('Enable Graph MCP debug mode without x402 payments')\n .requiredOption('--token <token>', 'Debug bearer token')\n .option('--endpoint <url>', 'Graph MCP endpoint')\n .action(async (opts: { token: string; endpoint?: string }) => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({\n graphMcpMode: 'debug',\n graphMcpAuthToken: opts.token,\n ...(opts.endpoint ? { graphMcpEndpoint: opts.endpoint } : {}),\n })\n console.log('Graph MCP debug mode enabled')\n if (opts.endpoint) console.log(`Graph endpoint: ${opts.endpoint}`)\n console.log('Payments: disabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('off')\n .description('Disable Graph MCP debug mode and use paid x402 calls')\n .action(async () => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({ graphMcpMode: 'paid', graphMcpAuthToken: '' })\n console.log('Graph MCP debug mode disabled')\n console.log('Payments: enabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('status')\n .description('Show Graph MCP payment/debug mode')\n .action(async () => {\n try {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n console.log(`Graph MCP mode: ${config.graphMcpMode}`)\n console.log(`Graph endpoint: ${config.graphMcpEndpoint}`)\n console.log(`Debug token: ${config.graphMcpAuthToken?.trim() ? 'configured' : 'not configured'}`)\n console.log(`Payments: ${config.graphMcpMode === 'debug' ? 'disabled' : 'enabled'}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('access-key')\n .description('Configure Graph MCP test access key mode')\n .addCommand(\n new Command('set')\n .description('Use a Graph MCP test access key without x402 payments')\n .argument('<key>', 'Test access key')\n .option('--endpoint <url>', 'Graph MCP endpoint')\n .action(async (key: string, opts: { endpoint?: string }) => {\n try {\n const normalizedKey = key.trim()\n if (!normalizedKey) throw new Error('Test access key is required')\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({\n graphMcpMode: 'debug',\n graphMcpAuthToken: normalizedKey,\n ...(opts.endpoint ? { graphMcpEndpoint: opts.endpoint } : {}),\n })\n console.log('Graph MCP test access key configured')\n if (opts.endpoint) console.log(`Graph endpoint: ${opts.endpoint}`)\n console.log('Payments: disabled when the server accepts this key')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('clear')\n .description('Remove the Graph MCP test access key and use paid x402 calls')\n .action(async () => {\n try {\n const { saveConfig } = await import('./config/index.js')\n await saveConfig({ graphMcpMode: 'paid', graphMcpAuthToken: '' })\n console.log('Graph MCP test access key cleared')\n console.log('Payments: enabled for Graph MCP calls')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('status')\n .description('Show Graph MCP test access key status')\n .action(async () => {\n try {\n const { loadConfig } = await import('./config/index.js')\n const config = await loadConfig()\n console.log(`Graph endpoint: ${config.graphMcpEndpoint}`)\n console.log(`Access key: ${config.graphMcpAuthToken?.trim() ? 'configured' : 'not configured'}`)\n console.log(`Payments: ${config.graphMcpAuthToken?.trim() ? 'disabled when accepted by server' : 'enabled'}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('init')\n .description('Initialize an investigation workspace')\n .argument('[dir]', 'Workspace directory to initialize', '.')\n .option('--force', 'Overwrite existing workspace files')\n .action(async (dir: string, opts: { force?: boolean }) => {\n try {\n const { initWorkspace } = await import('./workspace/init.js')\n const result = await initWorkspace({ targetDir: dir, force: opts.force })\n console.log(`Workspace initialized: ${result.workspaceRoot}`)\n console.log(`Files written: ${result.filesWritten.length}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram\n .command('setup')\n .description('Configure external MCP clients')\n .addCommand(\n new Command('claude-desktop')\n .alias('claude')\n .description('Install or update the Claude Desktop MCP server entry')\n .option('--config <path>', 'Path to claude_desktop_config.json')\n .option('--dry-run', 'Print the intended change without writing files')\n .action(async (opts: { config?: string; dryRun?: boolean }) => {\n try {\n const { setupClaudeDesktop } = await import('./claude-desktop/setup.js')\n const result = await setupClaudeDesktop({\n configPath: opts.config,\n dryRun: opts.dryRun,\n })\n\n console.log(`Claude Desktop config: ${result.configPath}`)\n console.log('MCP server: chain-insights')\n console.log(`Command: ${result.command}`)\n console.log(`Args: ${result.args.join(' ')}`)\n if (result.dryRun) {\n console.log(`Dry run: ${result.changed ? 'would update config' : 'already up to date'}`)\n } else if (result.changed) {\n console.log(`Updated: yes`)\n if (result.backupPath) console.log(`Backup: ${result.backupPath}`)\n } else {\n console.log('Updated: already up to date')\n }\n console.log('Reload required: quit and reopen Claude Desktop; it does not hot-reload MCP config.')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('config')\n .description('Read or write configuration values')\n .addCommand(\n new Command('get')\n .argument('<key>', 'Config key to read')\n .action(async (key: string) => {\n const { loadConfig } = await import('./config/index.js')\n const { CONFIG_KEYS } = await import('./config/schema.js')\n if (!CONFIG_KEYS.includes(key as typeof CONFIG_KEYS[number])) {\n console.error(`Unknown config key: ${key}`)\n process.exit(1)\n }\n const config = await loadConfig()\n const value = (config as Record<string, unknown>)[key]\n console.log(value ?? '')\n })\n )\n .addCommand(\n new Command('set')\n .argument('<key>', 'Config key to write')\n .argument('<value>', 'Value to set')\n .action(async (key: string, value: string) => {\n // D-01: walletPrivateKey is intercepted before saveConfig — the raw private key\n // must NEVER be written to config.json.\n if (key === 'walletPrivateKey') {\n try {\n const { setWalletPrivateKey } = await import('./wallet/index.js')\n const address = await setWalletPrivateKey(value)\n console.log('Wallet private key encrypted and stored in ~/.chain-insights/wallet.json')\n console.log(`Wallet address: ${address}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n return // MUST return — walletPrivateKey must never reach saveConfig or config.json\n }\n const { loadConfig, saveConfig } = await import('./config/index.js')\n const { CONFIG_KEYS, DEFAULT_CONFIG } = await import('./config/schema.js')\n const current = await loadConfig()\n if (!CONFIG_KEYS.includes(key as typeof CONFIG_KEYS[number])) {\n console.error(`Unknown config key: ${key}`)\n process.exit(1)\n }\n const existing = (current as Record<string, unknown>)[key]\n const defaultValue = (DEFAULT_CONFIG as Record<string, unknown>)[key]\n const coerced = typeof existing === 'number' || typeof defaultValue === 'number' ? Number(value) : value\n await saveConfig({ [key]: coerced } as Parameters<typeof saveConfig>[0])\n const displayed = key.toLowerCase().includes('token') ? '[redacted]' : coerced\n console.log(`Set ${key} = ${displayed}`)\n })\n )\n\nprogram\n .command('wallet')\n .description('Manage the local Base USDC payment wallet')\n .addCommand(\n new Command('import')\n .description('Import a Base payment wallet')\n .argument('<private-key>', '0x-prefixed EVM private key')\n .action(async (privateKey: string) => {\n try {\n const { setWalletPrivateKey } = await import('./wallet/index.js')\n const address = await setWalletPrivateKey(privateKey)\n console.log(`Wallet imported: ${address}`)\n console.log('Next: run `chain-insights wallet ready`')\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('address')\n .description('Print the local payment wallet address')\n .action(async () => {\n try {\n const { getWalletAccount } = await import('./wallet/tools.js')\n const account = await getWalletAccount()\n console.log(account.address)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('balance')\n .description('Show the local payment wallet Base USDC balance')\n .action(async () => {\n try {\n const { getWalletBalanceText } = await import('./wallet/tools.js')\n console.log(await getWalletBalanceText())\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('ready')\n .description('Check and prepare the wallet for paid GraphRAG MCP calls')\n .option('--check-only', 'Only check readiness; do not submit the one-time payment setup')\n .addOption(new Option('--no-approve', 'Deprecated alias for --check-only').hideHelp())\n .option('--payment-usdc <amount>', 'USDC setup cap to prepare for paid calls', '1')\n .addOption(new Option('--approval-usdc <amount>', 'Deprecated alias for --payment-usdc').hideHelp())\n .option('--json', 'Print machine-readable readiness metadata')\n .action(async (opts: { checkOnly?: boolean; approve?: boolean; paymentUsdc?: string; approvalUsdc?: string; json?: boolean }) => {\n try {\n const { formatWalletReadiness, parsePaymentApprovalUnits, prepareWalletForPaidCalls } = await import('./wallet/tools.js')\n const minimumApprovalUnits = parsePaymentApprovalUnits(opts.paymentUsdc ?? opts.approvalUsdc ?? '1')\n const result = await prepareWalletForPaidCalls({\n minimumApprovalUnits,\n approve: opts.checkOnly ? false : opts.approve !== false,\n })\n\n if (opts.json) {\n console.log(JSON.stringify(result, (_key, value) => (\n typeof value === 'bigint' ? value.toString() : value\n ), 2))\n return\n }\n\n console.log(formatWalletReadiness(result.readiness, result.approval))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('topup')\n .description('Open a local browser page to top up the payment wallet')\n .option('--no-open', 'Print the top-up URL without opening a browser')\n .option('--json', 'Print machine-readable top-up metadata')\n .action(async (opts: { open?: boolean; json?: boolean }) => {\n try {\n const { buildTopupInfo, getWalletAccount } = await import('./wallet/tools.js')\n const { startTopupServer } = await import('./wallet/topup-server.js')\n const account = await getWalletAccount()\n const url = await startTopupServer(account)\n const info = buildTopupInfo(account.address, url)\n\n if (opts.json) {\n console.log(JSON.stringify(info, null, 2))\n } else {\n console.log(`Top-up URL: ${url}`)\n console.log(`Wallet: ${account.address}`)\n console.log('Network: Base')\n console.log('Token: USDC')\n console.log('Press Ctrl+C to stop the top-up server.')\n }\n\n if (opts.open !== false) {\n const open = (await import('open')).default\n await open(url)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('mcp')\n .description('Interact with the Chain Insights MCP endpoint')\n .allowExcessArguments(false)\n .addCommand(\n new Command('networks')\n .description('List supported graph networks, capability layers, retention, and freshness')\n .option('--json', 'Print raw capability JSON')\n .action(async (opts: { json?: boolean }) => {\n try {\n await printNetworkCapabilities(opts)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('tools')\n .description('List available MCP tools (cached 24h)')\n .option('--refresh', 'Force refresh schema cache')\n .action(async (opts: { refresh?: boolean }) => {\n try {\n const { loadSchema, saveSchema } = await import('./mcp/schema-cache.js')\n const { formatToolsTable } = await import('./mcp/format.js')\n const { visibleRemoteTools } = await import('./mcp/tool-visibility.js')\n const { loadConfig } = await import('./config/index.js')\n const { createConfiguredGraphMcpFetch, resolveGraphMcpEndpoint } = await import('./mcp/client.js')\n const config = await loadConfig()\n const graphMcpEndpoint = resolveGraphMcpEndpoint(config)\n let tools = opts.refresh ? null : await loadSchema(graphMcpEndpoint)\n if (!tools) {\n const paymentFetch = await createConfiguredGraphMcpFetch(config)\n const { Client } = await import('@modelcontextprotocol/sdk/client/index.js')\n const { StreamableHTTPClientTransport } = await import('@modelcontextprotocol/sdk/client/streamableHttp.js')\n const client = new Client({ name: 'chain-insights-cli', version: PACKAGE_VERSION })\n await client.connect(new StreamableHTTPClientTransport(new URL(graphMcpEndpoint), { fetch: paymentFetch }))\n try {\n const result = await client.listTools()\n tools = result.tools as Array<{ name: string; description?: string }>\n await saveSchema(tools, graphMcpEndpoint)\n } finally {\n await client.close()\n }\n }\n console.log(formatToolsTable(visibleRemoteTools(tools)))\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('address-risk')\n .description('Screen an address for risk, exchange behavior, and optional compare_address connection risk')\n .requiredOption('--address <address>', 'Full blockchain address to screen')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--compare-address <address>', 'Optional second address for connection-risk compare mode')\n .option('--remote', 'Force remote MCP tool call instead of local Chain Insights recipe')\n .action(async (opts: { address: string; network: string; compareAddress?: string; remote?: boolean }) => {\n try {\n await withGraphMcpClient('chain-insights-cli-address-risk', async (client) => {\n if (opts.remote) {\n const result = await client.callTool({\n name: 'address_risk',\n arguments: {\n address: opts.address,\n network: opts.network,\n ...(opts.compareAddress ? { compare_address: opts.compareAddress } : {}),\n },\n })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n return\n }\n const { addressRisk } = await import('./investigation/public-tools.js')\n const result = await addressRisk(client, {\n address: opts.address,\n network: opts.network,\n compareAddress: opts.compareAddress,\n })\n console.log(result.summaryText)\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('trace-victim-funds')\n .description('Trace victim/source addresses forward to exchange deposit candidates')\n .requiredOption('--victim-addresses <addresses>', 'Comma-separated full victim/source addresses, max 5')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--known-suspect-addresses <addresses>', 'Optional known suspect addresses for context only, max 5')\n .option('--case <id>', 'Case ID to attach compact evidence pointers')\n .option('--incident-timestamp-ms <milliseconds>', 'Optional incident timestamp in milliseconds')\n .option('--max-hops <number>', 'Maximum trace hops, 1-5')\n .option('--per-address-limit <number>', 'Maximum exchange paths/results per address, 1-10')\n .option('--min-amount-sum <number>', 'Minimum r.amount_sum for traced edges')\n .option('--remote', 'Force remote MCP tool call instead of local Chain Insights recipe')\n .action(async (opts: {\n victimAddresses: string\n network: string\n knownSuspectAddresses?: string\n case?: string\n incidentTimestampMs?: string\n maxHops?: string\n perAddressLimit?: string\n minAmountSum?: string\n remote?: boolean\n }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n requireWorkspaceRoot()\n await withGraphMcpClient('chain-insights-cli-trace-victim-funds', async (client, config) => {\n if (opts.remote) {\n const result = await client.callTool({\n name: 'trace_victim_funds',\n arguments: {\n victim_addresses: opts.victimAddresses,\n network: opts.network,\n ...(opts.knownSuspectAddresses ? { known_suspect_addresses: opts.knownSuspectAddresses } : {}),\n },\n })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n return\n }\n const { traceVictimFunds } = await import('./investigation/public-tools.js')\n const caseId = opts.case ? await resolveCaseSelector(opts.case) : undefined\n const result = await traceVictimFunds(client, config, {\n victimAddresses: opts.victimAddresses,\n knownSuspectAddresses: opts.knownSuspectAddresses,\n network: opts.network,\n caseId,\n incidentTimestampMs: optionalNumber(opts.incidentTimestampMs),\n maxHops: optionalNumber(opts.maxHops),\n perAddressLimit: optionalNumber(opts.perAddressLimit),\n minAmountSum: optionalNumber(opts.minAmountSum),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('trace-suspect-funds')\n .description('Trace suspected scammer, mule, operator, or laundering-ring addresses forward to cashout topology')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .requiredOption('--suspect-addresses <addresses>', 'Comma-separated full suspect-controlled addresses, max 5')\n .option('--incident-timestamp-ms <milliseconds>', 'Optional incident timestamp in milliseconds')\n .option('--max-hops <number>', 'Maximum trace hops, default 3, max 5')\n .option('--per-address-limit <number>', 'Maximum exchange paths/results per address, 1-10')\n .option('--min-amount-sum <number>', 'Minimum r.amount_sum for traced edges')\n .option('--case <id>', 'Case ID to attach compact evidence pointers')\n .action(async (opts: {\n network: string\n suspectAddresses: string\n incidentTimestampMs?: string\n maxHops?: string\n perAddressLimit?: string\n minAmountSum?: string\n case?: string\n }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n requireWorkspaceRoot()\n await withGraphMcpClient('chain-insights-cli-trace-suspect-funds', async (client, config) => {\n const { traceSuspectFunds } = await import('./investigation/public-tools.js')\n const caseId = opts.case ? await resolveCaseSelector(opts.case) : undefined\n const result = await traceSuspectFunds(client, config, {\n suspectAddresses: opts.suspectAddresses,\n network: opts.network,\n maxHops: optionalNumber(opts.maxHops),\n perAddressLimit: optionalNumber(opts.perAddressLimit),\n minAmountSum: optionalNumber(opts.minAmountSum),\n incidentTimestampMs: optionalNumber(opts.incidentTimestampMs),\n caseId,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('trace-deposit-sources')\n .description('Trace backward from suspected deposit/cashout addresses to upstream sources and convergence')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .requiredOption('--deposit-addresses <addresses>', 'Comma-separated full suspected deposit/cashout addresses, max 5')\n .option('--max-hops <number>', 'Maximum reverse traceback hops, default 2, max 5')\n .option('--case <id>', 'Case ID to attach compact evidence pointers')\n .action(async (opts: {\n network: string\n depositAddresses: string\n maxHops?: string\n case?: string\n }) => {\n try {\n const { requireWorkspaceRoot } = await import('./workspace/output-root.js')\n requireWorkspaceRoot()\n await withGraphMcpClient('chain-insights-cli-trace-deposit-sources', async (client, config) => {\n const { traceDepositSources } = await import('./investigation/public-tools.js')\n const caseId = opts.case ? await resolveCaseSelector(opts.case) : undefined\n const result = await traceDepositSources(client, config, {\n depositAddresses: opts.depositAddresses,\n network: opts.network,\n maxHops: optionalNumber(opts.maxHops),\n caseId,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('stake-insights')\n .description('Explain Bittensor staking behavior around an address, coldkey, or hotkey')\n .requiredOption('--network <network>', 'Network to query. Run `cia mcp networks` for supported networks.')\n .option('--address <address>', 'Full Bittensor address to inspect as either coldkey or hotkey')\n .option('--coldkey <address>', 'Full Bittensor coldkey address to inspect')\n .option('--hotkey <address>', 'Full Bittensor hotkey address to inspect')\n .option('--netuid <number>', 'Optional subnet netuid filter')\n .option('--start-timestamp-ms <milliseconds>', 'Optional inclusive lower activity timestamp bound')\n .option('--end-timestamp-ms <milliseconds>', 'Optional inclusive upper activity timestamp bound')\n .option('--start-block <number>', 'Optional start block. Current stake graph parity may require timestamp windows instead.')\n .option('--end-block <number>', 'Optional end block. Current stake graph parity may require timestamp windows instead.')\n .option('--depth <number>', 'Optional expansion depth limit, default 1, max 3')\n .action(async (opts: {\n network: string\n address?: string\n coldkey?: string\n hotkey?: string\n netuid?: string\n startTimestampMs?: string\n endTimestampMs?: string\n startBlock?: string\n endBlock?: string\n depth?: string\n }) => {\n try {\n await withGraphMcpClient('chain-insights-cli-stake-insights', async (client) => {\n const { stakeInsights } = await import('./investigation/public-tools.js')\n const result = await stakeInsights(client, {\n network: opts.network,\n address: opts.address,\n coldkey: opts.coldkey,\n hotkey: opts.hotkey,\n netuid: optionalNumber(opts.netuid),\n startTimestampMs: optionalNumber(opts.startTimestampMs),\n endTimestampMs: optionalNumber(opts.endTimestampMs),\n startBlock: optionalNumber(opts.startBlock),\n endBlock: optionalNumber(opts.endBlock),\n depth: optionalNumber(opts.depth),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('call')\n .description('Call an MCP tool directly (debug)')\n .argument('<tool>', 'Tool name to call')\n .argument('[args...]', 'Key=value arguments (e.g. address=0x1234 chain=ethereum)')\n .action(async (tool: string, rawArgs: string[]) => {\n try {\n const { parseMcpCallArgs } = await import('./mcp/call-args.js')\n const { assertPublicMcpToolName } = await import('./mcp/tool-visibility.js')\n const args = parseMcpCallArgs(rawArgs)\n assertPublicMcpToolName(tool)\n await withGraphMcpClient('chain-insights-cli-call', async (client, config) => {\n if (tool === 'address_risk') {\n const { addressRisk } = await import('./investigation/public-tools.js')\n const result = await addressRisk(client, {\n address: String(args['address'] ?? ''),\n network: String(args['network'] ?? ''),\n compareAddress: args['compare_address'] === undefined ? undefined : String(args['compare_address']),\n })\n console.log(result.summaryText)\n return\n }\n if (tool === 'trace_victim_funds') {\n const { traceVictimFunds } = await import('./investigation/public-tools.js')\n const result = await traceVictimFunds(client, config, {\n victimAddresses: args['victim_addresses'] as string | string[] | undefined ?? '',\n knownSuspectAddresses: args['known_suspect_addresses'] as string | string[] | undefined,\n network: String(args['network'] ?? ''),\n caseId: args['case_id'] === undefined ? undefined : String(args['case_id']),\n incidentTimestampMs: optionalNumberArg(args['incident_timestamp_ms'], 'incident_timestamp_ms'),\n maxHops: typeof args['max_hops'] === 'number' ? args['max_hops'] : undefined,\n perAddressLimit: typeof args['per_address_limit'] === 'number' ? args['per_address_limit'] : undefined,\n minAmountSum: typeof args['min_amount_sum'] === 'number' ? args['min_amount_sum'] : undefined,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n if (tool === 'trace_suspect_funds') {\n const { traceSuspectFunds } = await import('./investigation/public-tools.js')\n const result = await traceSuspectFunds(client, config, {\n suspectAddresses: args['suspect_addresses'] as string | string[] | undefined ?? '',\n network: String(args['network'] ?? ''),\n caseId: args['case_id'] === undefined ? undefined : String(args['case_id']),\n maxHops: typeof args['max_hops'] === 'number' ? args['max_hops'] : undefined,\n perAddressLimit: typeof args['per_address_limit'] === 'number' ? args['per_address_limit'] : undefined,\n minAmountSum: typeof args['min_amount_sum'] === 'number' ? args['min_amount_sum'] : undefined,\n incidentTimestampMs: optionalNumberArg(args['incident_timestamp_ms'], 'incident_timestamp_ms'),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n if (tool === 'trace_deposit_sources') {\n const { traceDepositSources } = await import('./investigation/public-tools.js')\n const result = await traceDepositSources(client, config, {\n depositAddresses: args['deposit_addresses'] as string | string[] | undefined ?? '',\n network: String(args['network'] ?? ''),\n caseId: args['case_id'] === undefined ? undefined : String(args['case_id']),\n maxHops: typeof args['max_hops'] === 'number' ? args['max_hops'] : undefined,\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n if (tool === 'stake_insights') {\n const { stakeInsights } = await import('./investigation/public-tools.js')\n const result = await stakeInsights(client, {\n network: String(args['network'] ?? ''),\n address: args['address'] === undefined ? undefined : String(args['address']),\n coldkey: args['coldkey'] === undefined ? undefined : String(args['coldkey']),\n hotkey: args['hotkey'] === undefined ? undefined : String(args['hotkey']),\n netuid: optionalNumberArg(args['netuid'], 'netuid'),\n startTimestampMs: optionalNumberArg(args['start_timestamp_ms'], 'start_timestamp_ms'),\n endTimestampMs: optionalNumberArg(args['end_timestamp_ms'], 'end_timestamp_ms'),\n startBlock: optionalNumberArg(args['start_block'], 'start_block'),\n endBlock: optionalNumberArg(args['end_block'], 'end_block'),\n depth: optionalNumberArg(args['depth'] ?? args['max_hops'], 'depth'),\n })\n console.log(result.summaryText)\n console.log(JSON.stringify(result.structuredContent, null, 2))\n return\n }\n const result = await client.callTool({ name: tool, arguments: args })\n printMcpTextContent(result as { content?: Array<{ type: string; text?: string }> })\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nconst caseCommand = new Command('case')\n .description('Manage investigation cases')\n .hook('preAction', async () => {\n await scopeCasesToInvocationDir()\n })\n .addCommand(\n new Command('open')\n .description('Open a new investigation case')\n .argument('<name>', 'Case name (e.g. \"Tornado Mixer Investigation\")')\n .option('--tags <tags>', 'Comma-separated tags (e.g. aml,mixer,defi)', '')\n .option('--description <desc>', 'Brief description of the investigation', '')\n .action(async (name: string, opts: { tags: string; description: string }) => {\n try {\n if (/^[1-9]\\d*$/.test(name.trim())) {\n throw new Error('Numeric case names look like list selectors. Use a descriptive case name, e.g. `cia case open \"Tracking stolen funds from <address>\"`.')\n }\n const { CaseStore } = await import('./cases/index.js')\n const tags = opts.tags ? opts.tags.split(',').map(t => t.trim()).filter(Boolean) : []\n const c = await CaseStore.create({ name, tags, description: opts.description })\n const { casesRoot } = await import('./cases/store.js')\n console.log(`Case opened: ${c.id}`)\n console.log(`Directory: ${path.join(casesRoot(), c.id)}/`)\n console.log(`Status: ${c.status}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('activate')\n .description('Activate a case (set status to active)')\n .argument('<case-id>', 'Case ID to activate')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'active')\n console.log(`Case ${c.id} is now: active`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('suspend')\n .description('Suspend a case (set status to suspended)')\n .argument('<case-id>', 'Case ID to suspend')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'suspended')\n console.log(`Case ${c.id} is now: suspended`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('close')\n .description('Close a case permanently')\n .argument('<case-id>', 'Case ID to close')\n .action(async (caseSelector: string) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const c = await CaseStore.setStatus(caseId, 'closed')\n console.log(`Case ${c.id} is now: closed`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('list')\n .description('List all investigation cases')\n .option('--status <status>', 'Filter by status (open|active|suspended|closed)')\n .action(async (opts: { status?: string }) => {\n try {\n const { CaseStore } = await import('./cases/index.js')\n const cases = await CaseStore.list()\n const filtered = opts.status ? cases.filter(c => c.status === opts.status) : cases\n if (filtered.length === 0) {\n console.log('No cases found.')\n return\n }\n for (const [index, c] of filtered.entries()) {\n console.log(`${index + 1}. ${c.id} [${c.status}] ${c.name}`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('evidence')\n .description('Manage case evidence')\n .addCommand(\n new Command('add')\n .description('Add evidence to a case from an MCP query result')\n .argument('<case-id>', 'Case ID to add evidence to')\n .option('--source <tool>', 'MCP tool name that produced this evidence', 'manual')\n .option('--content <text>', 'Evidence content (MCP response or notes)', '')\n .option('--query-params <params>', 'Query parameters used (e.g. address=0x1234)', '')\n .action(async (caseSelector: string, opts: { source: string; content: string; queryParams: string }) => {\n try {\n const { EvidenceStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const result = await EvidenceStore.append(caseId, {\n source: opts.source,\n content: opts.content,\n queryParams: opts.queryParams,\n })\n console.log(`Evidence saved: ${result.filename}`)\n console.log(`SHA-256: ${result.sha256}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('verify')\n .description('Verify evidence manifest integrity for a case')\n .argument('<case-id>', 'Case ID to verify')\n .action(async (caseSelector: string) => {\n try {\n const { EvidenceStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const result = await EvidenceStore.verifyManifest(caseId)\n if (result.ok) {\n console.log(`Manifest OK — ${result.count} evidence file(s) verified`)\n } else {\n console.error(`Manifest FAILED — tampered files: ${(result.tampered ?? []).join(', ')}`)\n process.exit(1)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('dossier')\n .description('Manage entity dossiers for a case')\n .addCommand(\n new Command('update')\n .description('Append a finding to an entity dossier')\n .argument('<case-id>', 'Case ID')\n .argument('<address>', 'Entity address or identifier')\n .option('--finding <text>', 'Finding to append to the dossier', '')\n .option('--type <type>', 'Entity type (eoa|contract|exchange|mixer|unknown)', 'unknown')\n .action(async (caseSelector: string, address: string, opts: { finding: string; type: string }) => {\n try {\n const { DossierStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const validTypes = ['eoa', 'contract', 'exchange', 'mixer', 'unknown'] as const\n const entityType = validTypes.includes(opts.type as typeof validTypes[number])\n ? (opts.type as typeof validTypes[number])\n : 'unknown'\n await DossierStore.appendFinding(caseId, address, opts.finding, entityType)\n console.log(`Dossier updated for ${address}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('session')\n .description('Manage investigation sessions')\n .addCommand(\n new Command('start')\n .description('Start a new investigation session for a case')\n .argument('<case-id>', 'Case ID')\n .argument('[title...]', 'Optional session title')\n .action(async (caseSelector: string, titleParts: string[]) => {\n try {\n const { SessionStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n const title = titleParts.join(' ').trim()\n const s = await SessionStore.start(caseId, title ? { title } : {})\n console.log(`Session started: ${s.sessionId}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('end')\n .description('End the current session with findings and next steps')\n .argument('<case-id>', 'Case ID')\n .option('--findings <text>', 'Key findings from this session', '')\n .option('--next-steps <text>', 'Next steps for the investigation', '')\n .action(async (caseSelector: string, opts: { findings: string; nextSteps: string }) => {\n try {\n const { SessionStore } = await import('./cases/index.js')\n const caseId = await resolveCaseSelector(caseSelector)\n await SessionStore.end(caseId, { findings: opts.findings, nextSteps: opts.nextSteps })\n await SessionStore.archiveOldSessions(caseId)\n console.log(`Session ended for case ${caseId}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n )\n .addCommand(\n new Command('export')\n .description('Export a case for Obsidian, LLMWiki, and agents')\n .argument('<case-id>', 'Case ID or case list number to export')\n .option('--target <target>', 'Export target: obsidian-llmwiki', 'obsidian-llmwiki')\n .option('--mode <mode>', 'Redaction mode: private|partner|public', 'private')\n .option('--out <directory>', 'Output directory. Defaults to published/<case-slug>')\n .action(async (caseSelector: string, opts: { target: string; mode: string; out?: string }) => {\n try {\n const target = opts.target === 'obsidian-llmwiki' ? opts.target : undefined\n const mode = ['private', 'partner', 'public'].includes(opts.mode)\n ? (opts.mode as 'private' | 'partner' | 'public')\n : undefined\n if (!target) throw new Error(`Unsupported export target: ${opts.target}`)\n if (!mode) throw new Error(`Unsupported export mode: ${opts.mode}`)\n\n const caseId = await resolveCaseSelector(caseSelector)\n const { exportCase } = await import('./export/index.js')\n const result = await exportCase({\n caseId,\n target,\n mode,\n outputDir: opts.out,\n })\n console.log(`Case exported: ${result.outputDir}`)\n console.log(`Manifest: ${result.manifestPath}`)\n console.log(`Files: ${result.fileCount}`)\n console.log(`Open first: ${result.nextFile}`)\n for (const warning of result.warnings) console.warn(`Warning: ${warning}`)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('show')\n .description('Show saved case context')\n .argument('<case-id>', 'Case ID or case list number to show')\n .action(async (caseSelector: string) => {\n try {\n await showCaseContext(caseSelector)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram.addCommand(caseCommand)\n\nprogram\n .command('playbook')\n .description('Run and manage investigation playbooks')\n .addCommand(\n new Command('run')\n .description('Execute a playbook by name')\n .argument('<name>', 'Playbook name (e.g. trace-funds, risk-check, entity-profile)')\n .option('--case <id>', 'Case ID to attach evidence to (auto-created if omitted)')\n .option('--from <n>', 'Resume from step N (1-based)', '1')\n .option('--dry-run', 'Show steps without executing')\n .option('-p, --param <kv...>', 'Parameters as key=value pairs (repeatable, e.g. -p address=0x1 -p hops=3)')\n .action(async (name: string, opts: { case?: string; from: string; dryRun?: boolean; param?: string[] }) => {\n try {\n // 1. Parse --param key=value pairs into Record<string,string> — split on first '=' only (T-05-06)\n const resolvedParams: Record<string, string> = {}\n for (const kv of (opts.param ?? [])) {\n const eq = kv.indexOf('=')\n if (eq === -1) {\n console.error(`Invalid param format: \"${kv}\". Use key=value`)\n process.exit(1)\n }\n const key = kv.slice(0, eq)\n if (!key) {\n console.error(`Invalid param format: \"${kv}\". Key must be non-empty`)\n process.exit(1)\n }\n resolvedParams[key] = kv.slice(eq + 1)\n }\n // 2. Resolve playbook content (user-dir first, built-in fallback)\n const { resolvePlaybookContent } = await import('./playbooks/resolver.js')\n const markdown = await resolvePlaybookContent(name)\n // 3. Parse markdown → PlaybookDefinition\n const { PlaybookParser } = await import('./playbooks/parser.js')\n const definition = PlaybookParser.parse(markdown, resolvedParams)\n // 4. Validate required params are provided\n for (const spec of definition.params) {\n if (spec.required && !resolvedParams[spec.name] && !spec.default) {\n console.error(`Missing required param: ${spec.name}. Pass with: -p ${spec.name}=<value>`)\n process.exit(1)\n }\n }\n // 5. Validate --from value\n const fromN = parseInt(opts.from, 10)\n if (isNaN(fromN) || fromN < 1) {\n console.error(`Invalid --from value: \"${opts.from}\". Must be a positive integer.`)\n process.exit(1)\n }\n // 6. Run\n const { PlaybookRunner } = await import('./playbooks/runner.js')\n await PlaybookRunner.run(definition, {\n caseId: opts.case,\n from: fromN,\n dryRun: opts.dryRun,\n params: resolvedParams,\n })\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('list')\n .description('List available playbooks (built-in and user-defined)')\n .action(async () => {\n try {\n const { listPlaybooks } = await import('./playbooks/resolver.js')\n const playbooks = await listPlaybooks()\n if (playbooks.length === 0) { console.log('No playbooks found.'); return }\n for (const p of playbooks) {\n console.log(` ${p.name.padEnd(20)} [${p.source}]`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n .addCommand(\n new Command('show')\n .description('Show steps for a playbook without executing')\n .argument('<name>', 'Playbook name')\n .action(async (name: string) => {\n try {\n const { resolvePlaybookContent } = await import('./playbooks/resolver.js')\n const { PlaybookParser } = await import('./playbooks/parser.js')\n const markdown = await resolvePlaybookContent(name)\n const definition = PlaybookParser.parse(markdown, {})\n console.log(`Playbook: ${definition.name} v${definition.version}`)\n console.log(`${definition.description}\\n`)\n console.log(`Parameters:`)\n for (const p of definition.params) {\n const req = p.required ? '(required)' : `(optional, default: ${p.default ?? 'none'})`\n console.log(` ${p.name}: ${p.type} ${req}`)\n }\n console.log(`\\nSteps:`)\n for (const step of definition.steps) {\n console.log(` ${step.index}. ${step.label} → tool: ${step.tool}`)\n }\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n )\n\nprogram\n .command('viz')\n .description('Generate money flow visualization')\n .argument('[case-id]', 'Case ID to visualize')\n .option('--data <file>', 'Raw transaction JSON file for ad-hoc visualization')\n .option('-p, --port <number>', 'Server port', '4321')\n .action(async (caseId: string | undefined, opts: { data?: string; port: string }) => {\n try {\n if (!caseId && !opts.data) {\n console.error('Provide either a case ID or --data <file.json>')\n process.exit(1)\n }\n const { generateVisualization } = await import('./viz/index.js')\n const result = await generateVisualization({ caseId, dataFile: opts.data })\n const { startServer } = await import('./server/index.js')\n const port = parseInt(opts.port, 10)\n startServer(port)\n const url = `http://127.0.0.1:${port}/viz/${result.vizId}`\n console.log(`Visualization: ${url}`)\n const open = (await import('open')).default\n await open(url)\n } catch (err) {\n console.error((err as Error).message)\n process.exit(1)\n }\n })\n\nprogram.parse(process.argv)\n"],"mappings":"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"}
|
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
const require_chunk = require("./chunk-DakpK96I.cjs");
|
|
2
|
-
const require_html_generator = require("./html-generator-Bx3UcLTB.cjs");
|
|
3
2
|
const require_frontmatter = require("./frontmatter-Dvqa5HX6.cjs");
|
|
4
3
|
const require_active = require("./active-BVr55kvW.cjs");
|
|
5
4
|
let node_path = require("node:path");
|
|
@@ -301,7 +300,7 @@ async function extractGraphFromCase(caseId) {
|
|
|
301
300
|
}
|
|
302
301
|
allEdges = aggregateEdges(allEdges);
|
|
303
302
|
try {
|
|
304
|
-
const { DossierStore } = await Promise.resolve().then(() => require("./cases-
|
|
303
|
+
const { DossierStore } = await Promise.resolve().then(() => require("./cases-sTY5aXav.cjs"));
|
|
305
304
|
const dossiers = await DossierStore.listSummaries(caseId);
|
|
306
305
|
const dossierMap = /* @__PURE__ */ new Map();
|
|
307
306
|
for (const d of dossiers) dossierMap.set(d.address, d.type);
|
|
@@ -334,43 +333,21 @@ async function extractGraphFromCase(caseId) {
|
|
|
334
333
|
});
|
|
335
334
|
}
|
|
336
335
|
//#endregion
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
try {
|
|
345
|
-
parsed = JSON.parse(content);
|
|
346
|
-
} catch {
|
|
347
|
-
throw new Error("Invalid transaction data. The input file must contain a JSON array of transaction objects with `from`, `to`, and `value` fields.");
|
|
348
|
-
}
|
|
349
|
-
const { extractGraphFromJson } = await Promise.resolve().then(() => data_extractor_exports);
|
|
350
|
-
rawData = extractGraphFromJson(parsed);
|
|
351
|
-
} else if (opts.caseId) {
|
|
352
|
-
const { extractGraphFromCase } = await Promise.resolve().then(() => data_extractor_exports);
|
|
353
|
-
const extracted = await extractGraphFromCase(opts.caseId);
|
|
354
|
-
if (extracted.nodes.length === 0) throw new Error("No Transaction Data. This case has no evidence with transaction data. Add evidence using `chain-insights evidence add` or provide a JSON file with `chain-insights viz --data <file.json>`.");
|
|
355
|
-
rawData = extracted;
|
|
356
|
-
} else throw new Error("Provide either a case ID or --data <file.json>");
|
|
357
|
-
const data = truncateGraph(rawData);
|
|
358
|
-
const vizId = opts.caseId ? `${opts.caseId}_${Date.now()}` : `adhoc_${Date.now()}`;
|
|
359
|
-
return {
|
|
360
|
-
vizId,
|
|
361
|
-
htmlPath: await require_html_generator.writeVizHtml(vizId, require_html_generator.generateHtml(data, data.metadata.caseId ? `${data.metadata.caseId} - Money Flow` : "Ad-hoc Visualization"), opts.caseId)
|
|
362
|
-
};
|
|
363
|
-
}
|
|
364
|
-
//#endregion
|
|
365
|
-
Object.defineProperty(exports, "generateVisualization", {
|
|
336
|
+
Object.defineProperty(exports, "data_extractor_exports", {
|
|
337
|
+
enumerable: true,
|
|
338
|
+
get: function() {
|
|
339
|
+
return data_extractor_exports;
|
|
340
|
+
}
|
|
341
|
+
});
|
|
342
|
+
Object.defineProperty(exports, "extractGraphFromCase", {
|
|
366
343
|
enumerable: true,
|
|
367
344
|
get: function() {
|
|
368
|
-
return
|
|
345
|
+
return extractGraphFromCase;
|
|
369
346
|
}
|
|
370
347
|
});
|
|
371
|
-
Object.defineProperty(exports, "
|
|
348
|
+
Object.defineProperty(exports, "truncateGraph", {
|
|
372
349
|
enumerable: true,
|
|
373
350
|
get: function() {
|
|
374
|
-
return
|
|
351
|
+
return truncateGraph;
|
|
375
352
|
}
|
|
376
353
|
});
|
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
import { t as __exportAll } from "./rolldown-runtime-D7D4PA-g.mjs";
|
|
2
|
-
import { r as writeVizHtml, t as generateHtml } from "./html-generator-AowOmzyi.mjs";
|
|
3
2
|
import { t as parseFrontmatter } from "./frontmatter-D0ccQnUM.mjs";
|
|
4
3
|
import { t as activeCasesRoot } from "./active-ByNgjuAg.mjs";
|
|
5
4
|
import path from "node:path";
|
|
@@ -299,7 +298,7 @@ async function extractGraphFromCase(caseId) {
|
|
|
299
298
|
}
|
|
300
299
|
allEdges = aggregateEdges(allEdges);
|
|
301
300
|
try {
|
|
302
|
-
const { DossierStore } = await import("./cases-
|
|
301
|
+
const { DossierStore } = await import("./cases-Cp9DUbEV.mjs");
|
|
303
302
|
const dossiers = await DossierStore.listSummaries(caseId);
|
|
304
303
|
const dossierMap = /* @__PURE__ */ new Map();
|
|
305
304
|
for (const d of dossiers) dossierMap.set(d.address, d.type);
|
|
@@ -332,34 +331,6 @@ async function extractGraphFromCase(caseId) {
|
|
|
332
331
|
});
|
|
333
332
|
}
|
|
334
333
|
//#endregion
|
|
335
|
-
|
|
336
|
-
var viz_exports = /* @__PURE__ */ __exportAll({ generateVisualization: () => generateVisualization });
|
|
337
|
-
async function generateVisualization(opts) {
|
|
338
|
-
let rawData;
|
|
339
|
-
if (opts.dataFile) {
|
|
340
|
-
const content = await readFile(opts.dataFile, "utf-8");
|
|
341
|
-
let parsed;
|
|
342
|
-
try {
|
|
343
|
-
parsed = JSON.parse(content);
|
|
344
|
-
} catch {
|
|
345
|
-
throw new Error("Invalid transaction data. The input file must contain a JSON array of transaction objects with `from`, `to`, and `value` fields.");
|
|
346
|
-
}
|
|
347
|
-
const { extractGraphFromJson } = await Promise.resolve().then(() => data_extractor_exports);
|
|
348
|
-
rawData = extractGraphFromJson(parsed);
|
|
349
|
-
} else if (opts.caseId) {
|
|
350
|
-
const { extractGraphFromCase } = await Promise.resolve().then(() => data_extractor_exports);
|
|
351
|
-
const extracted = await extractGraphFromCase(opts.caseId);
|
|
352
|
-
if (extracted.nodes.length === 0) throw new Error("No Transaction Data. This case has no evidence with transaction data. Add evidence using `chain-insights evidence add` or provide a JSON file with `chain-insights viz --data <file.json>`.");
|
|
353
|
-
rawData = extracted;
|
|
354
|
-
} else throw new Error("Provide either a case ID or --data <file.json>");
|
|
355
|
-
const data = truncateGraph(rawData);
|
|
356
|
-
const vizId = opts.caseId ? `${opts.caseId}_${Date.now()}` : `adhoc_${Date.now()}`;
|
|
357
|
-
return {
|
|
358
|
-
vizId,
|
|
359
|
-
htmlPath: await writeVizHtml(vizId, generateHtml(data, data.metadata.caseId ? `${data.metadata.caseId} - Money Flow` : "Ad-hoc Visualization"), opts.caseId)
|
|
360
|
-
};
|
|
361
|
-
}
|
|
362
|
-
//#endregion
|
|
363
|
-
export { viz_exports as n, generateVisualization as t };
|
|
334
|
+
export { extractGraphFromCase as n, truncateGraph as r, data_extractor_exports as t };
|
|
364
335
|
|
|
365
|
-
//# sourceMappingURL=
|
|
336
|
+
//# sourceMappingURL=data-extractor-DZUJu1Bz.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"data-extractor-DZUJu1Bz.mjs","names":[],"sources":["../src/viz/graph-model.ts","../src/viz/data-extractor.ts"],"sourcesContent":["import * as z from 'zod'\n\nexport const EntityType = z.enum(['eoa', 'contract', 'exchange', 'mixer', 'unknown'])\nexport type EntityType = z.infer<typeof EntityType>\n\nexport const RiskLevel = z.enum(['low', 'medium', 'high', 'critical', 'unknown'])\nexport type RiskLevel = z.infer<typeof RiskLevel>\n\nexport const GraphNode = z.object({\n id: z.string().min(1),\n label: z.string().optional(),\n entityType: EntityType.default('unknown'),\n riskLevel: RiskLevel.default('unknown'),\n totalIn: z.number().default(0),\n totalOut: z.number().default(0),\n txCount: z.number().int().default(0),\n firstSeen: z.string().optional(),\n lastSeen: z.string().optional(),\n})\nexport type GraphNode = z.infer<typeof GraphNode>\n\nexport const GraphEdge = z.object({\n source: z.string().min(1),\n target: z.string().min(1),\n value: z.number(),\n txHash: z.string().optional(),\n blockNumber: z.number().int().optional(),\n timestamp: z.string().optional(),\n})\nexport type GraphEdge = z.infer<typeof GraphEdge>\n\nexport const GraphData = z.object({\n nodes: z.array(GraphNode),\n edges: z.array(GraphEdge),\n metadata: z.object({\n caseId: z.string().optional(),\n title: z.string().default('Money Flow'),\n generatedAt: z.string(),\n truncated: z.boolean().default(false),\n totalNodes: z.number().int().optional(),\n hiddenNodes: z.number().int().optional(),\n }),\n})\nexport type GraphData = z.infer<typeof GraphData>\n\nconst MAX_NODES = 100\n\nexport function truncateGraph(data: GraphData): GraphData {\n if (data.nodes.length <= MAX_NODES) return data\n const sorted = [...data.nodes].sort((a, b) => (b.totalIn + b.totalOut) - (a.totalIn + a.totalOut))\n const kept = sorted.slice(0, MAX_NODES)\n const keptIds = new Set(kept.map(n => n.id))\n const filteredEdges = data.edges.filter(e => keptIds.has(e.source) && keptIds.has(e.target))\n return {\n nodes: kept,\n edges: filteredEdges,\n metadata: {\n ...data.metadata,\n truncated: true,\n totalNodes: data.nodes.length,\n hiddenNodes: data.nodes.length - MAX_NODES,\n },\n }\n}\n","import { readFile, readdir } from 'node:fs/promises'\nimport path from 'node:path'\nimport { GraphData, GraphNode, GraphEdge } from './graph-model.js'\nimport { parseFrontmatter } from '../cases/frontmatter.js'\nimport { activeCasesRoot } from '../workspace/active.js'\n\nfunction caseDir(caseId: string): string {\n if (/[/\\\\]|^\\.\\.?$/.test(caseId)) throw new Error(`Invalid case ID: ${caseId}`)\n return path.join(activeCasesRoot(), caseId)\n}\n\n/**\n * Extracts all items from ```json code blocks in a markdown string.\n * If the parsed value is an array, spreads all items.\n * If the parsed value has 'nodes' and 'edges', wraps it as a single item.\n * Returns empty array if no JSON blocks found or parsing fails.\n */\nexport function parseEvidenceJson(markdown: string): unknown[] {\n const results: unknown[] = []\n const re = /```json\\s*\\n([\\s\\S]*?)```/g\n let match: RegExpExecArray | null\n while ((match = re.exec(markdown)) !== null) {\n const raw = match[1]\n if (!raw) continue\n try {\n const parsed: unknown = JSON.parse(raw)\n if (Array.isArray(parsed)) {\n results.push(...parsed)\n } else if (\n parsed !== null &&\n typeof parsed === 'object' &&\n 'nodes' in (parsed as object) &&\n 'edges' in (parsed as object)\n ) {\n // Full GraphData object embedded in evidence\n results.push(parsed)\n }\n // Ignore other object types\n } catch {\n // Malformed JSON in evidence — skip gracefully (T-04-06: no crash)\n }\n }\n if (results.length > 0) return results\n\n const rawJson = extractEmbeddedJson(markdown)\n if (rawJson) {\n try {\n const parsed: unknown = JSON.parse(rawJson)\n if (Array.isArray(parsed)) return parsed\n if (parsed !== null && typeof parsed === 'object') return [parsed]\n } catch {\n // Ignore non-JSON evidence bodies.\n }\n }\n return results\n}\n\nfunction extractEmbeddedJson(text: string): string | null {\n const trimmed = text.trim()\n const start = [...trimmed]\n .map((char, index) => (char === '{' || char === '[' ? index : -1))\n .find(index => index >= 0)\n if (start === undefined) return null\n return trimmed.slice(start)\n}\n\ntype SimpleTx = { from: string; to: string; value: number; txHash?: string; blockNumber?: number; timestamp?: string }\n\ntype CompactEvidence = {\n schema?: string\n outgoing_flows?: Array<{\n src?: string\n dst?: string\n amount_sum?: number\n tx_count?: number\n first_tx_id?: string\n }>\n}\n\nfunction isSimpleTx(item: unknown): item is SimpleTx {\n return (\n item !== null &&\n typeof item === 'object' &&\n 'from' in (item as object) &&\n 'to' in (item as object) &&\n 'value' in (item as object)\n )\n}\n\nfunction isGraphDataLike(input: unknown): input is { nodes: unknown[]; edges: unknown[] } {\n return (\n input !== null &&\n typeof input === 'object' &&\n Array.isArray((input as Record<string, unknown>)['nodes']) &&\n Array.isArray((input as Record<string, unknown>)['edges'])\n )\n}\n\nfunction compactEvidenceToSimpleTxs(item: unknown): SimpleTx[] {\n const compact = item as CompactEvidence\n if (\n !compact ||\n typeof compact !== 'object' ||\n compact.schema !== 'chain-insights.compact_evidence.v1' ||\n !Array.isArray(compact.outgoing_flows)\n ) {\n return []\n }\n\n return compact.outgoing_flows\n .filter(flow => typeof flow.src === 'string' && typeof flow.dst === 'string' && typeof flow.amount_sum === 'number')\n .map(flow => ({\n from: flow.src!,\n to: flow.dst!,\n value: flow.amount_sum!,\n txHash: flow.first_tx_id,\n }))\n}\n\n/**\n * Converts simple [{from, to, value}] transaction arrays into graph nodes.\n * Computes totalIn, totalOut, txCount per node from edges.\n */\nfunction buildGraphFromSimpleTxs(items: SimpleTx[]): { nodes: GraphNode[]; edges: GraphEdge[] } {\n // Build edges first\n const edges: GraphEdge[] = items.map(tx => GraphEdge.parse({\n source: tx.from,\n target: tx.to,\n value: tx.value,\n txHash: tx.txHash,\n blockNumber: tx.blockNumber,\n timestamp: tx.timestamp,\n }))\n\n // Collect unique addresses\n const addresses = new Set<string>()\n for (const tx of items) {\n addresses.add(tx.from)\n addresses.add(tx.to)\n }\n\n // Compute totals per address\n const totals: Record<string, { totalIn: number; totalOut: number; txCount: number }> = {}\n for (const addr of addresses) {\n totals[addr] = { totalIn: 0, totalOut: 0, txCount: 0 }\n }\n for (const tx of items) {\n const out = totals[tx.from]\n const inp = totals[tx.to]\n if (out) {\n out.totalOut += tx.value\n out.txCount += 1\n }\n if (inp) {\n inp.totalIn += tx.value\n inp.txCount += 1\n }\n }\n\n const nodes: GraphNode[] = [...addresses].map(addr =>\n GraphNode.parse({\n id: addr,\n entityType: 'unknown',\n riskLevel: 'unknown',\n totalIn: totals[addr]?.totalIn ?? 0,\n totalOut: totals[addr]?.totalOut ?? 0,\n txCount: totals[addr]?.txCount ?? 0,\n })\n )\n\n return { nodes, edges }\n}\n\n/**\n * Handles two input formats:\n * 1. Full GraphData object (has nodes + edges) — parse with Zod\n * 2. Array of {from, to, value, ...} transaction objects — auto-derive nodes\n *\n * Throws \"Invalid transaction data\" for any other input.\n */\nexport function extractGraphFromJson(input: unknown): GraphData {\n if (isGraphDataLike(input)) {\n // Full GraphData: parse and validate through Zod\n return GraphData.parse(input)\n }\n\n if (Array.isArray(input)) {\n const simpleTxs: SimpleTx[] = []\n for (const item of input) {\n if (isSimpleTx(item)) {\n simpleTxs.push(item)\n } else {\n simpleTxs.push(...compactEvidenceToSimpleTxs(item))\n }\n }\n const { nodes, edges } = buildGraphFromSimpleTxs(simpleTxs)\n return GraphData.parse({\n nodes,\n edges,\n metadata: {\n title: 'Money Flow',\n generatedAt: new Date().toISOString(),\n },\n })\n }\n\n throw new Error(\n 'Invalid transaction data. The input file must contain a JSON array of transaction objects with `from`, `to`, and `value` fields.'\n )\n}\n\n/**\n * Merges two sets of nodes, deduplicating by id.\n * For duplicate nodes, sums totalIn, totalOut, txCount; keeps earliest firstSeen, latest lastSeen.\n */\nfunction mergeNodes(existing: GraphNode[], incoming: GraphNode[]): GraphNode[] {\n const map = new Map<string, GraphNode>()\n for (const node of existing) {\n map.set(node.id, { ...node })\n }\n for (const node of incoming) {\n const prev = map.get(node.id)\n if (prev) {\n map.set(node.id, {\n ...prev,\n totalIn: prev.totalIn + node.totalIn,\n totalOut: prev.totalOut + node.totalOut,\n txCount: prev.txCount + node.txCount,\n firstSeen: pickEarlier(prev.firstSeen, node.firstSeen),\n lastSeen: pickLater(prev.lastSeen, node.lastSeen),\n })\n } else {\n map.set(node.id, { ...node })\n }\n }\n return [...map.values()]\n}\n\nfunction pickEarlier(a: string | undefined, b: string | undefined): string | undefined {\n if (!a) return b\n if (!b) return a\n return a < b ? a : b\n}\n\nfunction pickLater(a: string | undefined, b: string | undefined): string | undefined {\n if (!a) return b\n if (!b) return a\n return a > b ? a : b\n}\n\n/**\n * Aggregates edges: for duplicate (source, target) pairs, sums value; keeps last txHash/timestamp.\n */\nfunction aggregateEdges(edges: GraphEdge[]): GraphEdge[] {\n const map = new Map<string, GraphEdge>()\n for (const edge of edges) {\n const key = `${edge.source}::${edge.target}`\n const prev = map.get(key)\n if (prev) {\n map.set(key, { ...edge, value: prev.value + edge.value })\n } else {\n map.set(key, { ...edge })\n }\n }\n return [...map.values()]\n}\n\n/**\n * Reads evidence files from a case directory, extracts JSON transaction data\n * from markdown code blocks, enriches nodes with entity types from dossiers,\n * and returns a merged GraphData.\n */\nexport async function extractGraphFromCase(caseId: string): Promise<GraphData> {\n const evidenceDir = path.join(caseDir(caseId), 'evidence')\n\n let files: string[] = []\n try {\n const all = await readdir(evidenceDir)\n files = all.filter(f => f.endsWith('.md'))\n } catch {\n // Evidence directory missing — return empty graph (graceful, not an error)\n return GraphData.parse({\n nodes: [],\n edges: [],\n metadata: {\n caseId,\n title: `${caseId} - Money Flow`,\n generatedAt: new Date().toISOString(),\n },\n })\n }\n\n let allNodes: GraphNode[] = []\n let allEdges: GraphEdge[] = []\n\n for (const file of files) {\n const raw = await readFile(path.join(evidenceDir, file), 'utf-8')\n const { body } = parseFrontmatter(raw)\n const items = parseEvidenceJson(body)\n if (items.length === 0) continue\n\n // Check if items contain a full GraphData object\n const graphDataItems = items.filter(item => isGraphDataLike(item))\n const simpleTxItems = items.flatMap(item => {\n if (isSimpleTx(item)) return [item]\n return compactEvidenceToSimpleTxs(item)\n }) as SimpleTx[]\n\n if (graphDataItems.length > 0) {\n // Merge full GraphData objects\n for (const gd of graphDataItems) {\n try {\n const parsed = GraphData.parse(gd)\n allNodes = mergeNodes(allNodes, parsed.nodes)\n allEdges = [...allEdges, ...parsed.edges]\n } catch {\n // Invalid GraphData in evidence — skip (T-04-06)\n }\n }\n // Also process any simple tx items alongside\n if (simpleTxItems.length > 0) {\n const { nodes, edges } = buildGraphFromSimpleTxs(simpleTxItems)\n allNodes = mergeNodes(allNodes, nodes)\n allEdges = [...allEdges, ...edges]\n }\n } else if (simpleTxItems.length > 0) {\n const { nodes, edges } = buildGraphFromSimpleTxs(simpleTxItems)\n allNodes = mergeNodes(allNodes, nodes)\n allEdges = [...allEdges, ...edges]\n }\n }\n\n // Aggregate edges with same source-target pair\n allEdges = aggregateEdges(allEdges)\n\n // Enrich nodes with entity types from dossiers\n try {\n const { DossierStore } = await import('../cases/index.js')\n const dossiers = await DossierStore.listSummaries(caseId)\n const dossierMap = new Map<string, string>()\n for (const d of dossiers) {\n dossierMap.set(d.address, d.type)\n }\n allNodes = allNodes.map(node => {\n const dossierType = dossierMap.get(node.id)\n if (dossierType && dossierType !== 'unknown') {\n // Validate against EntityType enum — default to 'unknown' if invalid (T-04-07)\n const validTypes = ['eoa', 'contract', 'exchange', 'mixer', 'unknown'] as const\n type ValidType = (typeof validTypes)[number]\n const entityType: ValidType = validTypes.includes(dossierType as ValidType)\n ? (dossierType as ValidType)\n : 'unknown'\n return { ...node, entityType }\n }\n return node\n })\n } catch {\n // Dossier enrichment is best-effort — don't fail extraction if dossiers unavailable\n }\n\n return GraphData.parse({\n nodes: allNodes,\n edges: allEdges,\n metadata: {\n caseId,\n title: `${caseId} - Money Flow`,\n generatedAt: new Date().toISOString(),\n },\n })\n}\n\nexport const DataExtractor = {\n extractGraphFromCase,\n extractGraphFromJson,\n parseEvidenceJson,\n}\n"],"mappings":"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"}
|