@schoolai/shipyard 3.2.0 → 3.2.1-rc.20260421.0
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/dist/{chunk-FD2QK7IN.js → chunk-DKMDBOFU.js} +35 -2
- package/dist/chunk-DKMDBOFU.js.map +1 -0
- package/dist/{chunk-5SJBSLGT.js → chunk-FMMRZTOF.js} +2 -2
- package/dist/{chunk-5SJBSLGT.js.map → chunk-FMMRZTOF.js.map} +1 -1
- package/dist/{chunk-CANDZLMB.js → chunk-JSUYB74F.js} +2 -2
- package/dist/index.js +4 -4
- package/dist/{login-GRM7QIPC.js → login-KSI4GTLM.js} +3 -3
- package/dist/{roi-3Y7MWLSW.js → roi-YQ6OLVHX.js} +8 -5
- package/dist/roi-YQ6OLVHX.js.map +1 -0
- package/dist/{serve-3EFFP3PN.js → serve-P5WC5JIT.js} +1478 -222
- package/dist/{serve-3EFFP3PN.js.map → serve-P5WC5JIT.js.map} +1 -1
- package/dist/{start-24JXLIQJ.js → start-2K7HFXHV.js} +4 -4
- package/package.json +1 -1
- package/dist/chunk-FD2QK7IN.js.map +0 -1
- package/dist/roi-3Y7MWLSW.js.map +0 -1
- /package/dist/{chunk-CANDZLMB.js.map → chunk-JSUYB74F.js.map} +0 -0
- /package/dist/{login-GRM7QIPC.js.map → login-KSI4GTLM.js.map} +0 -0
- /package/dist/{start-24JXLIQJ.js.map → start-2K7HFXHV.js.map} +0 -0
|
@@ -4,7 +4,7 @@ import {
|
|
|
4
4
|
ROI_EVENT_TYPE,
|
|
5
5
|
RoiEventSchema,
|
|
6
6
|
assertNeverEvent
|
|
7
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-DKMDBOFU.js";
|
|
8
8
|
import "./chunk-EHQITHQX.js";
|
|
9
9
|
import {
|
|
10
10
|
logger
|
|
@@ -25,7 +25,8 @@ function emptyContext(taskId, userId, records) {
|
|
|
25
25
|
started: null,
|
|
26
26
|
ended: null,
|
|
27
27
|
commits: [],
|
|
28
|
-
pr: null
|
|
28
|
+
pr: null,
|
|
29
|
+
merged: null
|
|
29
30
|
};
|
|
30
31
|
}
|
|
31
32
|
function applyEvent(ctx, event) {
|
|
@@ -42,6 +43,9 @@ function applyEvent(ctx, event) {
|
|
|
42
43
|
case "pr_attributed":
|
|
43
44
|
if (!ctx.pr) ctx.pr = event;
|
|
44
45
|
break;
|
|
46
|
+
case "pr_merged":
|
|
47
|
+
if (!ctx.merged || event.timestamp < ctx.merged.timestamp) ctx.merged = event;
|
|
48
|
+
break;
|
|
45
49
|
default:
|
|
46
50
|
assertNeverEvent(event);
|
|
47
51
|
}
|
|
@@ -136,8 +140,7 @@ function computeFunnel(ctxs) {
|
|
|
136
140
|
if (ctx.started || ctx.record?.taskStartedAt != null) tasksStarted++;
|
|
137
141
|
if (ctx.commits.length > 0) producedCommits++;
|
|
138
142
|
if (ctx.pr || ctx.record?.prUrl) openedPr++;
|
|
139
|
-
if (ctx.
|
|
140
|
-
merged++;
|
|
143
|
+
if (ctx.merged != null || ctx.record?.mergedAt != null) merged++;
|
|
141
144
|
if (ctx.record?.abandonedAt != null) abandoned++;
|
|
142
145
|
}
|
|
143
146
|
return {
|
|
@@ -586,4 +589,4 @@ function renderReport(report, format) {
|
|
|
586
589
|
export {
|
|
587
590
|
roiCommand
|
|
588
591
|
};
|
|
589
|
-
//# sourceMappingURL=roi-
|
|
592
|
+
//# sourceMappingURL=roi-YQ6OLVHX.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/shared/commands/roi.ts","../../../packages/roi-aggregator/src/joins/task-context.ts","../../../packages/roi-aggregator/src/metrics/cost-per-pr.ts","../../../packages/roi-aggregator/src/metrics/efficiency-curve.ts","../../../packages/roi-aggregator/src/metrics/funnel.ts","../../../packages/roi-aggregator/src/metrics/leverage.ts","../../../packages/roi-aggregator/src/metrics/per-engineer.ts","../../../packages/roi-aggregator/src/reports/build-report.ts","../../../packages/roi-aggregator/src/reports/csv.ts","../../../packages/roi-aggregator/src/reports/html.ts","../../../packages/roi-aggregator/src/reports/json.ts","../../../packages/roi-aggregator/src/sources/d1.ts","../../../packages/roi-aggregator/src/sources/tasks-json.ts"],"sourcesContent":["import { promises as fs } from 'node:fs';\nimport { parseArgs } from 'node:util';\nimport {\n buildRoiReport,\n buildTaskContexts,\n defaultTasksJsonPath,\n fetchRoiEvents,\n readTasksJson,\n renderCsvReport,\n renderHtmlReport,\n renderJsonReport,\n} from '@shipyard/roi-aggregator';\nimport { logger } from '../logger.js';\n\nexport interface RoiCommandArgs {\n since?: string;\n until?: string;\n format: 'json' | 'html' | 'csv';\n output?: string;\n workerUrl: string;\n token: string;\n tasksJsonPath?: string;\n}\n\nfunction parseIsoToMs(raw: string | undefined): number | undefined {\n if (!raw) return undefined;\n const ms = Date.parse(raw);\n return Number.isFinite(ms) ? ms : undefined;\n}\n\nexport async function roiCommand(argv: string[] = process.argv.slice(3)): Promise<void> {\n const { values } = parseArgs({\n args: argv,\n options: {\n since: { type: 'string' },\n until: { type: 'string' },\n format: { type: 'string', default: 'json' },\n output: { type: 'string', short: 'o' },\n 'worker-url': { type: 'string' },\n token: { type: 'string' },\n 'tasks-json': { type: 'string' },\n help: { type: 'boolean', short: 'h' },\n },\n strict: true,\n allowPositionals: false,\n });\n\n if (values.help) {\n process.stdout.write(\n [\n 'Usage: shipyard roi [options]',\n '',\n ' --since DATE Start of window (ISO date). Default: 30 days ago.',\n ' --until DATE End of window (ISO date). Default: now.',\n ' --format FORMAT Output format: json, html, csv. Default: json.',\n ' --output FILE, -o Write to file instead of stdout.',\n ' --worker-url URL Metrics worker base URL. Env: SHIPYARD_METRICS_URL.',\n ' --token TOKEN JWT bearer token. Env: SHIPYARD_USER_TOKEN.',\n ' --tasks-json PATH Override path to local tasks.json.',\n ' -h, --help Show this help.',\n '',\n ].join('\\n')\n );\n return;\n }\n\n const format = normalizeFormat(values.format);\n if (!format) {\n logger.error('Invalid --format. Expected one of: json, html, csv.');\n process.exit(1);\n }\n\n const workerUrl = values['worker-url'] ?? process.env.SHIPYARD_METRICS_URL;\n const token = values.token ?? process.env.SHIPYARD_USER_TOKEN;\n if (!workerUrl || !token) {\n logger.error(\n 'Missing required --worker-url / SHIPYARD_METRICS_URL and --token / SHIPYARD_USER_TOKEN.'\n );\n process.exit(1);\n }\n\n const until = parseIsoToMs(values.until) ?? Date.now();\n const since = parseIsoToMs(values.since) ?? until - 30 * 24 * 60 * 60 * 1000;\n const sinceIso = new Date(since).toISOString();\n const untilIso = new Date(until).toISOString();\n\n const [{ roiEvents }, tasks] = await Promise.all([\n fetchRoiEvents({ workerUrl, authToken: token, since, until }),\n readTasksJson(values['tasks-json'] ?? defaultTasksJsonPath()),\n ]);\n\n const ctxs = buildTaskContexts(roiEvents, tasks.tasks);\n const report = buildRoiReport(ctxs, { since: sinceIso, until: untilIso });\n\n const rendered = renderReport(report, format);\n if (values.output) {\n await fs.writeFile(values.output, rendered, 'utf8');\n logger.info({ output: values.output, format }, 'roi report written');\n } else {\n process.stdout.write(rendered);\n if (!rendered.endsWith('\\n')) process.stdout.write('\\n');\n }\n}\n\nfunction normalizeFormat(raw: string | undefined): 'json' | 'html' | 'csv' | null {\n if (raw === 'json' || raw === 'html' || raw === 'csv') return raw;\n return null;\n}\n\nfunction renderReport(\n report: Parameters<typeof renderJsonReport>[0],\n format: 'json' | 'html' | 'csv'\n): string {\n switch (format) {\n case 'json':\n return renderJsonReport(report);\n case 'html':\n return renderHtmlReport(report);\n case 'csv':\n return renderCsvReport(report);\n default: {\n const _exhaustive: never = format;\n throw new Error(`unhandled format: ${String(_exhaustive)}`);\n }\n }\n}\n","import {\n assertNeverEvent,\n type CommitAttributedEvent,\n type PrAttributedEvent,\n type PrMergedEvent,\n type RoiEvent,\n type TaskEndedEvent,\n type TaskStartedEvent,\n} from '../schemas/events';\nimport type { RawTaskRecord } from '../sources/tasks-json';\n\nexport interface TaskContext {\n taskId: string;\n userId: string;\n record: RawTaskRecord | null;\n started: TaskStartedEvent | null;\n ended: TaskEndedEvent | null;\n commits: CommitAttributedEvent[];\n pr: PrAttributedEvent | null;\n merged: PrMergedEvent | null;\n}\n\nfunction emptyContext(\n taskId: string,\n userId: string,\n records: Record<string, RawTaskRecord>\n): TaskContext {\n return {\n taskId,\n userId,\n record: records[taskId] ?? null,\n started: null,\n ended: null,\n commits: [],\n pr: null,\n merged: null,\n };\n}\n\nfunction applyEvent(ctx: TaskContext, event: RoiEvent): void {\n switch (event.kind) {\n case 'task_started':\n if (!ctx.started || event.timestamp < ctx.started.timestamp) ctx.started = event;\n break;\n case 'task_ended':\n if (!ctx.ended || event.timestamp > ctx.ended.timestamp) ctx.ended = event;\n break;\n case 'commit_attributed':\n ctx.commits.push(event);\n break;\n case 'pr_attributed':\n if (!ctx.pr) ctx.pr = event;\n break;\n case 'pr_merged':\n if (!ctx.merged || event.timestamp < ctx.merged.timestamp) ctx.merged = event;\n break;\n default:\n assertNeverEvent(event);\n }\n}\n\nexport function buildTaskContexts(\n events: RoiEvent[],\n records: Record<string, RawTaskRecord>\n): TaskContext[] {\n const byTask = new Map<string, TaskContext>();\n\n for (const event of events) {\n let ctx = byTask.get(event.taskId);\n if (!ctx) {\n ctx = emptyContext(event.taskId, event.userId, records);\n byTask.set(event.taskId, ctx);\n }\n applyEvent(ctx, event);\n }\n\n for (const record of Object.values(records)) {\n if (!byTask.has(record.taskId)) {\n byTask.set(record.taskId, emptyContext(record.taskId, '', records));\n }\n }\n\n return [...byTask.values()];\n}\n","import type { TaskContext } from '../joins/task-context';\n\nexport function computeCostPerMergedPr(ctxs: TaskContext[]): number {\n let totalCost = 0;\n let mergedPrs = 0;\n for (const ctx of ctxs) {\n const hasMergedPr =\n ctx.record?.status === 'completed' && (ctx.pr != null || ctx.record?.prUrl != null);\n if (!hasMergedPr) continue;\n mergedPrs += 1;\n const cost = ctx.ended?.totalCostUsd ?? ctx.record?.totalCostUsd ?? sumCommitCosts(ctx);\n totalCost += cost;\n }\n return mergedPrs === 0 ? 0 : Number((totalCost / mergedPrs).toFixed(4));\n}\n\nfunction sumCommitCosts(ctx: TaskContext): number {\n return ctx.commits.reduce((sum, c) => sum + c.costUsd, 0);\n}\n","import type { TaskContext } from '../joins/task-context';\nimport type { TimeSeriesMetric, TimeSeriesPoint } from '../schemas/report';\n\n/** ISO 8601 week number — week containing the first Thursday of the year is week 1. */\nfunction isoWeek(ms: number): string {\n const d = new Date(ms);\n const thursday = new Date(d);\n thursday.setUTCDate(d.getUTCDate() + 4 - (d.getUTCDay() || 7));\n const yearStart = Date.UTC(thursday.getUTCFullYear(), 0, 1);\n const weekNo = Math.ceil(((thursday.getTime() - yearStart) / 86_400_000 + 1) / 7);\n return `${thursday.getUTCFullYear()}-W${String(weekNo).padStart(2, '0')}`;\n}\n\nexport function computeTokensPerMergedPrWeekly(ctxs: TaskContext[]): TimeSeriesMetric {\n const buckets = new Map<string, { tokens: number; prs: number }>();\n for (const ctx of ctxs) {\n const merged =\n ctx.record?.status === 'completed' && (ctx.pr != null || ctx.record?.prUrl != null);\n if (!merged) continue;\n const ts = ctx.ended?.timestamp ?? ctx.record?.updatedAt;\n if (ts == null) continue;\n const bucket = isoWeek(ts);\n const cur = buckets.get(bucket) ?? { tokens: 0, prs: 0 };\n cur.tokens += ctx.ended?.totalOutputTokens ?? ctx.record?.totalOutputTokens ?? 0;\n cur.prs += 1;\n buckets.set(bucket, cur);\n }\n const series: TimeSeriesPoint[] = [...buckets.entries()]\n .sort(([a], [b]) => a.localeCompare(b))\n .map(([week, { tokens, prs }]) => ({\n bucket: week,\n value: prs === 0 ? 0 : Math.round(tokens / prs),\n denominator: prs,\n }));\n return { metric: 'tokens_per_pr', unit: 'tokens', bucketSize: 'week', series };\n}\n\nexport function computeConcurrentTasksWeekly(ctxs: TaskContext[]): TimeSeriesMetric {\n const buckets = new Map<string, { peak: number }>();\n for (const ctx of ctxs) {\n const ts = ctx.started?.timestamp ?? ctx.record?.createdAt;\n if (ts == null) continue;\n const active = ctx.started?.activeTaskCount ?? 0;\n const bucket = isoWeek(ts);\n const cur = buckets.get(bucket) ?? { peak: 0 };\n cur.peak = Math.max(cur.peak, active);\n buckets.set(bucket, cur);\n }\n const series: TimeSeriesPoint[] = [...buckets.entries()]\n .sort(([a], [b]) => a.localeCompare(b))\n .map(([week, { peak }]) => ({ bucket: week, value: peak }));\n return { metric: 'concurrent_tasks', unit: 'count', bucketSize: 'week', series };\n}\n","import type { TaskContext } from '../joins/task-context';\nimport type { TaskOutcomeFunnel } from '../schemas/report';\n\nexport function computeFunnel(ctxs: TaskContext[]): TaskOutcomeFunnel {\n let tasksStarted = 0;\n let producedCommits = 0;\n let openedPr = 0;\n let merged = 0;\n let abandoned = 0;\n\n for (const ctx of ctxs) {\n if (ctx.started || ctx.record?.taskStartedAt != null) tasksStarted++;\n if (ctx.commits.length > 0) producedCommits++;\n if (ctx.pr || ctx.record?.prUrl) openedPr++;\n if (ctx.merged != null || ctx.record?.mergedAt != null) merged++;\n if (ctx.record?.abandonedAt != null) abandoned++;\n }\n\n return {\n tasksStarted,\n producedCommits,\n openedPr,\n merged,\n abandoned,\n revertedWithin30d: 0,\n };\n}\n","import type { TaskContext } from '../joins/task-context';\n\nexport const HUMAN_BASELINE_LEVERAGE = 1.0;\n\nexport interface LeverageSummary {\n peakByUser: Record<string, number>;\n avgPeak: number;\n humanBaseline: number;\n}\n\nexport function computePeakLeverage(ctxs: TaskContext[]): LeverageSummary {\n const peakByUser: Record<string, number> = {};\n for (const ctx of ctxs) {\n const user = ctx.userId || 'unknown';\n const count = ctx.started?.activeTaskCount ?? 0;\n peakByUser[user] = Math.max(peakByUser[user] ?? 0, count);\n }\n const values = Object.values(peakByUser);\n const avgPeak =\n values.length === 0\n ? 0\n : Number((values.reduce((a, b) => a + b, 0) / values.length).toFixed(2));\n return { peakByUser, avgPeak, humanBaseline: HUMAN_BASELINE_LEVERAGE };\n}\n","import type { TaskContext } from '../joins/task-context';\nimport type { EngineerScorecardRow } from '../schemas/report';\n\nfunction classifySignal(sampleSize: number, peakParallel: number): EngineerScorecardRow['signal'] {\n if (sampleSize < 3) return 'insufficient-data';\n if (sampleSize < 10) return 'modest';\n if (sampleSize < 20) return peakParallel >= 3 ? 'strong' : 'modest';\n if (peakParallel >= 5) return 'outlier-high';\n if (peakParallel >= 3) return 'strong';\n return 'modest';\n}\n\nfunction engineerRow(userId: string, ctxs: TaskContext[]): EngineerScorecardRow {\n const mergedPrs = ctxs.filter(\n (c) => c.record?.status === 'completed' && (c.pr || c.record?.prUrl)\n );\n const peakParallel = ctxs.reduce((max, c) => Math.max(max, c.started?.activeTaskCount ?? 0), 0);\n const sampleSize = ctxs.length;\n const costPerPr =\n mergedPrs.length === 0\n ? 0\n : Number(\n (\n mergedPrs.reduce((sum, c) => sum + (c.ended?.totalCostUsd ?? 0), 0) / mergedPrs.length\n ).toFixed(4)\n );\n return {\n userId,\n peakParallel,\n costPerPrUsd: costPerPr,\n revertRate: 0,\n sampleSize,\n signal: classifySignal(sampleSize, peakParallel),\n };\n}\n\nexport function computeEngineerScorecard(ctxs: TaskContext[]): EngineerScorecardRow[] {\n const byUser = new Map<string, TaskContext[]>();\n for (const ctx of ctxs) {\n const user = ctx.userId || 'unknown';\n const list = byUser.get(user) ?? [];\n list.push(ctx);\n byUser.set(user, list);\n }\n\n const rows: EngineerScorecardRow[] = [];\n for (const [userId, userCtxs] of byUser.entries()) {\n rows.push(engineerRow(userId, userCtxs));\n }\n return rows.sort((a, b) => b.sampleSize - a.sampleSize);\n}\n","import type { TaskContext } from '../joins/task-context';\nimport { computeCostPerMergedPr } from '../metrics/cost-per-pr';\nimport {\n computeConcurrentTasksWeekly,\n computeTokensPerMergedPrWeekly,\n} from '../metrics/efficiency-curve';\nimport { computeFunnel } from '../metrics/funnel';\nimport { computePeakLeverage } from '../metrics/leverage';\nimport { computeEngineerScorecard } from '../metrics/per-engineer';\nimport { REPORT_SCHEMA_VERSION, type RoiReport, type RoiWarning } from '../schemas/report';\n\nexport interface BuildReportOptions {\n since: string;\n until: string;\n}\n\nexport function buildRoiReport(ctxs: TaskContext[], opts: BuildReportOptions): RoiReport {\n const warnings: RoiWarning[] = [];\n if (ctxs.length === 0) {\n warnings.push({\n code: 'no_data',\n message:\n 'No ROI events found in the selected window. Ensure the daemon is running and events are being ingested.',\n context: {},\n });\n }\n\n const leverage = computePeakLeverage(ctxs);\n const funnel = computeFunnel(ctxs);\n const costPerMergedPr = computeCostPerMergedPr(ctxs);\n const byEngineer = computeEngineerScorecard(ctxs);\n const totalCost = ctxs.reduce((sum, c) => sum + (c.ended?.totalCostUsd ?? 0), 0);\n const totalTokens = ctxs.reduce((sum, c) => sum + (c.ended?.totalOutputTokens ?? 0), 0);\n const engineersActive = new Set(\n ctxs.map((c) => c.userId || 'unknown').filter((u) => u !== 'unknown')\n ).size;\n\n for (const row of byEngineer) {\n if (row.sampleSize < 20) {\n warnings.push({\n code: 'small_sample',\n message: `Engineer ${row.userId} has only ${row.sampleSize} tasks in window. Deltas not statistically significant.`,\n context: { userId: row.userId, sampleSize: row.sampleSize },\n });\n }\n }\n\n return {\n schemaVersion: REPORT_SCHEMA_VERSION,\n generatedAt: new Date().toISOString(),\n window: { since: opts.since, until: opts.until },\n totals: {\n tasksStarted: funnel.tasksStarted,\n tasksAbandoned: funnel.abandoned,\n tasksShipped: funnel.merged,\n costUsd: Number(totalCost.toFixed(4)),\n tokens: totalTokens,\n engineersActive,\n },\n kpis: {\n costPerMergedPr,\n peakLeverageAvg: leverage.avgPeak,\n peakLeverageHumanBaseline: leverage.humanBaseline,\n revertRateShipyard: 0,\n },\n series: {\n tokensPerPrWeekly: computeTokensPerMergedPrWeekly(ctxs),\n concurrentTasksWeekly: computeConcurrentTasksWeekly(ctxs),\n },\n funnel,\n byEngineer,\n facets: [],\n warnings,\n };\n}\n","import type { RoiReport } from '../schemas/report';\n\nfunction csvEscape(value: string | number): string {\n const s = String(value);\n if (s.includes(',') || s.includes('\"') || s.includes('\\n')) {\n return `\"${s.replace(/\"/g, '\"\"')}\"`;\n }\n return s;\n}\n\nexport function renderCsvReport(report: RoiReport): string {\n const lines: string[] = [];\n lines.push('section,key,value');\n lines.push(`totals,tasksStarted,${report.totals.tasksStarted}`);\n lines.push(`totals,tasksAbandoned,${report.totals.tasksAbandoned}`);\n lines.push(`totals,tasksShipped,${report.totals.tasksShipped}`);\n lines.push(`totals,costUsd,${report.totals.costUsd}`);\n lines.push(`totals,tokens,${report.totals.tokens}`);\n lines.push(`totals,engineersActive,${report.totals.engineersActive}`);\n lines.push(`kpi,costPerMergedPr,${report.kpis.costPerMergedPr}`);\n lines.push(`kpi,peakLeverageAvg,${report.kpis.peakLeverageAvg}`);\n lines.push(`funnel,tasksStarted,${report.funnel.tasksStarted}`);\n lines.push(`funnel,producedCommits,${report.funnel.producedCommits}`);\n lines.push(`funnel,openedPr,${report.funnel.openedPr}`);\n lines.push(`funnel,merged,${report.funnel.merged}`);\n lines.push(`funnel,abandoned,${report.funnel.abandoned}`);\n\n lines.push('');\n lines.push('engineer,sampleSize,peakParallel,costPerPrUsd,revertRate,signal');\n for (const row of report.byEngineer) {\n lines.push(\n [\n csvEscape(row.userId),\n row.sampleSize,\n row.peakParallel,\n row.costPerPrUsd,\n row.revertRate,\n row.signal,\n ].join(',')\n );\n }\n\n return `${lines.join('\\n')}\\n`;\n}\n","import type { RoiReport } from '../schemas/report';\n\nexport function renderHtmlReport(report: RoiReport): string {\n const data = JSON.stringify(report).replace(/</g, '\\\\u003c');\n return `<!DOCTYPE html>\n<html lang=\"en\">\n<head>\n<meta charset=\"utf-8\">\n<title>Shipyard ROI — ${report.window.since} → ${report.window.until}</title>\n<style>\n :root { color-scheme: light dark; --text:#e7e5e4; --muted:#a8a29e; --bg:#0c0a09; --surface:#1c1917; --border:#292524; --ok:#1d9e75; --warn:#f0a030; --danger:#e24b4a; }\n @media (prefers-color-scheme: light) { :root { --text:#1c1917; --muted:#57534e; --bg:#fafaf9; --surface:#ffffff; --border:#e7e5e4; } }\n body { font-family: system-ui, sans-serif; background: var(--bg); color: var(--text); margin: 0; padding: 32px; max-width: 1100px; margin-left: auto; margin-right: auto; line-height: 1.5; }\n h1 { font-size: 22px; font-weight: 500; margin: 0 0 4px 0; }\n .subtitle { font-size: 12px; color: var(--muted); margin-bottom: 24px; font-family: ui-monospace, monospace; }\n .kpi-grid { display:grid; grid-template-columns:repeat(auto-fit,minmax(180px,1fr)); gap:12px; margin-bottom:20px; }\n .kpi { background: var(--surface); padding: 16px; border-radius: 12px; border: 1px solid var(--border); }\n .kpi .label { font-size: 11px; text-transform: uppercase; letter-spacing: 1px; color: var(--muted); }\n .kpi .value { font-size: 28px; font-weight: 500; margin-top: 4px; }\n .kpi .baseline { font-size: 11px; color: var(--muted); margin-top: 4px; font-family: ui-monospace, monospace; }\n .panel { background: var(--surface); padding: 14px 16px; border-radius: 12px; border: 1px solid var(--border); margin-bottom: 16px; }\n .panel h2 { font-size: 14px; font-weight: 500; margin: 0 0 12px 0; }\n table { width: 100%; border-collapse: collapse; font-size: 13px; }\n th { text-align: left; font-size: 11px; text-transform: uppercase; letter-spacing: 1px; color: var(--muted); border-bottom: 1px solid var(--border); padding: 8px 10px; font-weight: 500; }\n td { padding: 8px 10px; border-bottom: 1px solid var(--border); font-variant-numeric: tabular-nums; }\n .funnel-row { display: grid; grid-template-columns: 1fr 80px; gap: 8px; align-items: center; margin-bottom: 6px; }\n .funnel-bar { height: 10px; background: color-mix(in srgb, var(--ok) 25%, transparent); border-radius: 3px; overflow: hidden; }\n .funnel-fill { height: 100%; background: var(--ok); }\n .funnel-fill.revert { background: var(--danger); }\n .funnel-label { font-size: 12px; color: var(--muted); }\n .funnel-count { font-family: ui-monospace, monospace; font-size: 12px; text-align: right; }\n .warn { background: color-mix(in srgb, var(--warn) 12%, transparent); color: var(--warn); padding: 10px 12px; border-radius: 8px; font-size: 12px; margin-bottom: 8px; }\n .chart-wrap { position: relative; height: 240px; }\n</style>\n</head>\n<body>\n <h1>Shipyard ROI — ${escapeHtml(report.window.since)} → ${escapeHtml(report.window.until)}</h1>\n <div class=\"subtitle\">Generated ${escapeHtml(report.generatedAt)} · ${report.totals.engineersActive} engineer(s) · schema v${report.schemaVersion}</div>\n\n <div class=\"kpi-grid\">\n <div class=\"kpi\"><div class=\"label\">Cost / merged PR</div><div class=\"value\">$${report.kpis.costPerMergedPr.toFixed(2)}</div></div>\n <div class=\"kpi\"><div class=\"label\">Peak leverage</div><div class=\"value\">${report.kpis.peakLeverageAvg.toFixed(1)}×</div><div class=\"baseline\">vs ${report.kpis.peakLeverageHumanBaseline.toFixed(1)}× human baseline</div></div>\n <div class=\"kpi\"><div class=\"label\">Tasks shipped</div><div class=\"value\">${report.totals.tasksShipped}</div></div>\n <div class=\"kpi\"><div class=\"label\">Token spend</div><div class=\"value\">$${report.totals.costUsd.toFixed(2)}</div></div>\n </div>\n\n <div class=\"panel\">\n <h2>Task outcome funnel</h2>\n ${renderFunnel(report)}\n </div>\n\n <div class=\"panel\">\n <h2>Tokens per merged PR (weekly)</h2>\n <div class=\"chart-wrap\"><canvas id=\"chart-tokens\"></canvas></div>\n </div>\n\n <div class=\"panel\">\n <h2>Peak concurrent tasks (weekly)</h2>\n <div class=\"chart-wrap\"><canvas id=\"chart-concurrent\"></canvas></div>\n </div>\n\n <div class=\"panel\">\n <h2>Per-engineer scorecard</h2>\n <table>\n <thead><tr><th>Engineer</th><th>Tasks</th><th>Peak ‖</th><th>$ / PR</th><th>Signal</th></tr></thead>\n <tbody>\n ${report.byEngineer\n .map(\n (row) =>\n `<tr><td>${escapeHtml(row.userId)}</td><td>${row.sampleSize}</td><td>${row.peakParallel}</td><td>$${row.costPerPrUsd.toFixed(2)}</td><td>${escapeHtml(row.signal)}</td></tr>`\n )\n .join('')}\n </tbody>\n </table>\n </div>\n\n ${\n report.warnings.length > 0\n ? `<div class=\"panel\"><h2>Warnings</h2>${report.warnings.map((w) => `<div class=\"warn\">${escapeHtml(w.message)}</div>`).join('')}</div>`\n : ''\n }\n\n<script src=\"https://cdn.jsdelivr.net/npm/chart.js@4.4.7/dist/chart.umd.min.js\" crossorigin=\"anonymous\"></script>\n<script>\n const report = ${data};\n const textColor = getComputedStyle(document.documentElement).getPropertyValue('--text').trim();\n const gridColor = getComputedStyle(document.documentElement).getPropertyValue('--border').trim();\n Chart.defaults.color = textColor;\n\n const tokensEl = document.getElementById('chart-tokens');\n if (tokensEl) {\n new Chart(tokensEl, {\n type: 'line',\n data: {\n labels: report.series.tokensPerPrWeekly.series.map(p => p.bucket),\n datasets: [{ label: 'tokens / PR', data: report.series.tokensPerPrWeekly.series.map(p => p.value), borderColor: '#1d9e75', tension: 0.3 }]\n },\n options: { responsive: true, maintainAspectRatio: false, plugins: { legend: { display: false } }, scales: { x: { grid: { color: gridColor } }, y: { grid: { color: gridColor } } } }\n });\n }\n\n const concurrentEl = document.getElementById('chart-concurrent');\n if (concurrentEl) {\n const buckets = report.series.concurrentTasksWeekly.series.map(p => p.bucket);\n const baselineValue = report.kpis.peakLeverageHumanBaseline;\n new Chart(concurrentEl, {\n type: 'line',\n data: {\n labels: buckets,\n datasets: [\n { label: 'peak concurrent', data: report.series.concurrentTasksWeekly.series.map(p => p.value), borderColor: '#0f6e56', backgroundColor: 'rgba(15,110,86,0.18)', fill: true, tension: 0.3 },\n { label: 'human baseline', data: buckets.map(() => baselineValue), borderColor: '#888780', borderDash: [4, 4], borderWidth: 1, pointRadius: 0, fill: false }\n ]\n },\n options: { responsive: true, maintainAspectRatio: false, plugins: { legend: { display: true, labels: { boxWidth: 12, font: { size: 11 } } } }, scales: { x: { grid: { color: gridColor } }, y: { grid: { color: gridColor }, beginAtZero: true } } }\n });\n }\n</script>\n</body>\n</html>\n`;\n}\n\nfunction renderFunnel(report: RoiReport): string {\n const stages = [\n { label: 'Tasks started', count: report.funnel.tasksStarted },\n { label: 'Produced commits', count: report.funnel.producedCommits },\n { label: 'Opened PR', count: report.funnel.openedPr },\n { label: 'Merged', count: report.funnel.merged },\n { label: 'Abandoned', count: report.funnel.abandoned, revert: true },\n ];\n const max = Math.max(1, ...stages.map((s) => s.count));\n return stages\n .map(\n (s) =>\n `<div class=\"funnel-row\"><div><div class=\"funnel-label\">${escapeHtml(s.label)}</div><div class=\"funnel-bar\"><div class=\"funnel-fill ${s.revert ? 'revert' : ''}\" style=\"width: ${Math.round((s.count / max) * 100)}%\"></div></div></div><div class=\"funnel-count\">${s.count}</div></div>`\n )\n .join('');\n}\n\nfunction escapeHtml(value: string): string {\n return String(value)\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"');\n}\n","import type { RoiReport } from '../schemas/report';\n\nexport function renderJsonReport(report: RoiReport): string {\n return JSON.stringify(report, null, 2);\n}\n","import { ROI_EVENT_TYPE, type RoiEvent, RoiEventSchema } from '../schemas/events';\n\nexport interface FetchEventsOptions {\n workerUrl: string;\n authToken: string;\n since?: number;\n until?: number;\n limit?: number;\n fetchImpl?: typeof fetch;\n}\n\nexport interface RawQueryEvent {\n eventType: string;\n taskId: string | null;\n userId: string;\n clientTimestamp: number;\n serverTimestamp: number;\n payload: unknown;\n}\n\nexport interface FetchedEvents {\n events: RawQueryEvent[];\n roiEvents: RoiEvent[];\n}\n\nexport async function fetchRoiEvents(opts: FetchEventsOptions): Promise<FetchedEvents> {\n const fetchImpl = opts.fetchImpl ?? fetch;\n const baseUrl = opts.workerUrl.replace(/\\/$/, '');\n const qs = new URLSearchParams({ eventType: ROI_EVENT_TYPE });\n if (opts.since != null) qs.set('since', String(opts.since));\n if (opts.until != null) qs.set('until', String(opts.until));\n if (opts.limit != null) qs.set('limit', String(opts.limit));\n\n const response = await fetchImpl(`${baseUrl}/query/events?${qs.toString()}`, {\n method: 'GET',\n headers: { Authorization: `Bearer ${opts.authToken}` },\n });\n if (!response.ok) {\n throw new Error(`query/events responded ${response.status}: ${await response.text()}`);\n }\n // eslint-disable-next-line no-restricted-syntax -- fetch response.json() is untyped at the boundary\n const body = (await response.json()) as { events: RawQueryEvent[] };\n const events = body.events ?? [];\n const roiEvents: RoiEvent[] = [];\n for (const row of events) {\n if (row.eventType !== ROI_EVENT_TYPE) continue;\n const parsed = RoiEventSchema.safeParse(row.payload);\n if (parsed.success) roiEvents.push(parsed.data);\n }\n return { events, roiEvents };\n}\n","import { promises as fs } from 'node:fs';\nimport { homedir } from 'node:os';\nimport { join } from 'node:path';\n\nexport interface RawTaskRecord {\n taskId: string;\n channelId: string;\n title: string;\n status: string;\n mode?: string;\n createdAt: number;\n updatedAt: number;\n taskStartedAt: number | null;\n cwd: string;\n totalCostUsd?: number;\n totalOutputTokens?: number;\n taskType?: string;\n prUrl?: string;\n abandonedAt?: number | null;\n /** ROI: canonical \"shipped\" signal — stamped by the 3-tier merge detector. */\n mergedAt?: number | null;\n}\n\nexport interface TasksJsonShape {\n schemaVersion?: number;\n tasks: Record<string, RawTaskRecord>;\n}\n\nexport function defaultTasksJsonPath(): string {\n return join(homedir(), '.shipyard', 'data', 'tasks.json');\n}\n\nexport async function readTasksJson(\n filePath: string = defaultTasksJsonPath()\n): Promise<TasksJsonShape> {\n let raw: string;\n try {\n raw = await fs.readFile(filePath, 'utf8');\n } catch {\n return { tasks: {} };\n }\n try {\n // eslint-disable-next-line no-restricted-syntax -- JSON.parse result is validated by field access below\n const parsed = JSON.parse(raw) as TasksJsonShape;\n return { schemaVersion: parsed.schemaVersion, tasks: parsed.tasks ?? {} };\n } catch {\n process.stderr.write(\n `[roi-aggregator] warning: ${filePath} exists but contains malformed JSON — local task context will be empty\\n`\n );\n return { tasks: {} };\n }\n}\n"],"mappings":"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","names":["fs","fs"]}
|