3ac07a4718
* chore: initialize TypeScript configuration and build setup - Add tsconfig.json for root and mcp-server with strict type checking - Install typescript and @types/node as devDependencies - Add npm build script for TypeScript compilation - Update main entrypoint to compiled dist/shannon.js - Update Dockerfile to build TypeScript before running - Configure output directory and module resolution for Node.js * refactor: migrate codebase from JavaScript to TypeScript - Convert all 37 JavaScript files to TypeScript (.js -> .ts) - Add type definitions in src/types/ for agents, config, errors, session - Update mcp-server with proper TypeScript types - Move entry point from shannon.mjs to src/shannon.ts - Update tsconfig.json with rootDir: "./src" for cleaner dist output - Update Dockerfile to build TypeScript before runtime - Update package.json paths to use compiled dist/shannon.js No runtime behavior changes - pure type safety migration. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * docs: update CLI references from ./shannon.mjs to shannon - Update help text in src/cli/ui.ts - Update usage examples in src/cli/command-handler.ts - Update setup message in src/shannon.ts - Update CLAUDE.md documentation with TypeScript file structure - Replace all ./shannon.mjs references with shannon command 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * chore: remove unnecessary eslint-disable comments ESLint is not configured in this project, making these comments redundant. 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
161 lines
4.4 KiB
TypeScript
161 lines
4.4 KiB
TypeScript
// Copyright (C) 2025 Keygraph, Inc.
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License version 3
|
|
// as published by the Free Software Foundation.
|
|
|
|
import chalk from 'chalk';
|
|
import { formatDuration } from '../audit/utils.js';
|
|
|
|
// Timing utilities
|
|
|
|
export class Timer {
|
|
name: string;
|
|
startTime: number;
|
|
endTime: number | null = null;
|
|
|
|
constructor(name: string) {
|
|
this.name = name;
|
|
this.startTime = Date.now();
|
|
}
|
|
|
|
stop(): number {
|
|
this.endTime = Date.now();
|
|
return this.duration();
|
|
}
|
|
|
|
duration(): number {
|
|
const end = this.endTime || Date.now();
|
|
return end - this.startTime;
|
|
}
|
|
}
|
|
|
|
interface TimingResultsPhases {
|
|
[key: string]: number;
|
|
}
|
|
|
|
interface TimingResultsCommands {
|
|
[key: string]: number;
|
|
}
|
|
|
|
interface TimingResultsAgents {
|
|
[key: string]: number;
|
|
}
|
|
|
|
interface TimingResults {
|
|
total: Timer | null;
|
|
phases: TimingResultsPhases;
|
|
commands: TimingResultsCommands;
|
|
agents: TimingResultsAgents;
|
|
}
|
|
|
|
interface CostResultsAgents {
|
|
[key: string]: number;
|
|
}
|
|
|
|
interface CostResults {
|
|
agents: CostResultsAgents;
|
|
total: number;
|
|
}
|
|
|
|
// Global timing and cost tracker
|
|
export const timingResults: TimingResults = {
|
|
total: null,
|
|
phases: {},
|
|
commands: {},
|
|
agents: {},
|
|
};
|
|
|
|
export const costResults: CostResults = {
|
|
agents: {},
|
|
total: 0,
|
|
};
|
|
|
|
// Function to display comprehensive timing summary
|
|
export const displayTimingSummary = (): void => {
|
|
if (!timingResults.total) {
|
|
console.log(chalk.yellow('No timing data available'));
|
|
return;
|
|
}
|
|
|
|
const totalDuration = timingResults.total.stop();
|
|
|
|
console.log(chalk.cyan.bold('\n⏱️ TIMING SUMMARY'));
|
|
console.log(chalk.gray('─'.repeat(60)));
|
|
|
|
// Total execution time
|
|
console.log(chalk.cyan(`📊 Total Execution Time: ${formatDuration(totalDuration)}`));
|
|
console.log();
|
|
|
|
// Phase breakdown
|
|
if (Object.keys(timingResults.phases).length > 0) {
|
|
console.log(chalk.yellow.bold('🔍 Phase Breakdown:'));
|
|
let phaseTotal = 0;
|
|
for (const [phase, duration] of Object.entries(timingResults.phases)) {
|
|
const percentage = ((duration / totalDuration) * 100).toFixed(1);
|
|
console.log(
|
|
chalk.yellow(` ${phase.padEnd(20)} ${formatDuration(duration).padStart(8)} (${percentage}%)`)
|
|
);
|
|
phaseTotal += duration;
|
|
}
|
|
console.log(
|
|
chalk.gray(
|
|
` ${'Phases Total'.padEnd(20)} ${formatDuration(phaseTotal).padStart(8)} (${((phaseTotal / totalDuration) * 100).toFixed(1)}%)`
|
|
)
|
|
);
|
|
console.log();
|
|
}
|
|
|
|
// Command breakdown
|
|
if (Object.keys(timingResults.commands).length > 0) {
|
|
console.log(chalk.blue.bold('🖥️ Command Breakdown:'));
|
|
let commandTotal = 0;
|
|
for (const [command, duration] of Object.entries(timingResults.commands)) {
|
|
const percentage = ((duration / totalDuration) * 100).toFixed(1);
|
|
console.log(
|
|
chalk.blue(` ${command.padEnd(20)} ${formatDuration(duration).padStart(8)} (${percentage}%)`)
|
|
);
|
|
commandTotal += duration;
|
|
}
|
|
console.log(
|
|
chalk.gray(
|
|
` ${'Commands Total'.padEnd(20)} ${formatDuration(commandTotal).padStart(8)} (${((commandTotal / totalDuration) * 100).toFixed(1)}%)`
|
|
)
|
|
);
|
|
console.log();
|
|
}
|
|
|
|
// Agent breakdown
|
|
if (Object.keys(timingResults.agents).length > 0) {
|
|
console.log(chalk.magenta.bold('🤖 Agent Breakdown:'));
|
|
let agentTotal = 0;
|
|
for (const [agent, duration] of Object.entries(timingResults.agents)) {
|
|
const percentage = ((duration / totalDuration) * 100).toFixed(1);
|
|
const displayName = agent.replace(/-/g, ' ');
|
|
console.log(
|
|
chalk.magenta(
|
|
` ${displayName.padEnd(20)} ${formatDuration(duration).padStart(8)} (${percentage}%)`
|
|
)
|
|
);
|
|
agentTotal += duration;
|
|
}
|
|
console.log(
|
|
chalk.gray(
|
|
` ${'Agents Total'.padEnd(20)} ${formatDuration(agentTotal).padStart(8)} (${((agentTotal / totalDuration) * 100).toFixed(1)}%)`
|
|
)
|
|
);
|
|
}
|
|
|
|
// Cost breakdown
|
|
if (Object.keys(costResults.agents).length > 0) {
|
|
console.log(chalk.green.bold('\n💰 Cost Breakdown:'));
|
|
for (const [agent, cost] of Object.entries(costResults.agents)) {
|
|
const displayName = agent.replace(/-/g, ' ');
|
|
console.log(chalk.green(` ${displayName.padEnd(20)} $${cost.toFixed(4).padStart(8)}`));
|
|
}
|
|
console.log(chalk.gray(` ${'Total Cost'.padEnd(20)} $${costResults.total.toFixed(4).padStart(8)}`));
|
|
}
|
|
|
|
console.log(chalk.gray('─'.repeat(60)));
|
|
};
|