@opensassi/opencode 0.1.3 → 0.1.5

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.
Files changed (39) hide show
  1. package/AGENTS.md +3 -2
  2. package/dashboard/dashboard.e2e.test.ts +247 -0
  3. package/dashboard/dist/index.d.ts +9 -0
  4. package/dashboard/dist/index.js +36 -0
  5. package/dashboard/dist/routes/api.d.ts +2 -0
  6. package/dashboard/dist/routes/api.js +215 -0
  7. package/dashboard/dist/services/cache.d.ts +13 -0
  8. package/dashboard/dist/services/cache.js +29 -0
  9. package/dashboard/dist/services/experiments.d.ts +11 -0
  10. package/dashboard/dist/services/experiments.js +108 -0
  11. package/dashboard/dist/services/git.d.ts +12 -0
  12. package/dashboard/dist/services/git.js +149 -0
  13. package/dashboard/dist/services/sessions.d.ts +25 -0
  14. package/dashboard/dist/services/sessions.js +208 -0
  15. package/dashboard/dist/services/specs.d.ts +9 -0
  16. package/dashboard/dist/services/specs.js +102 -0
  17. package/dashboard/dist/types.d.ts +173 -0
  18. package/dashboard/dist/types.js +1 -0
  19. package/dashboard/opencode.e2e.test.ts +100 -0
  20. package/dashboard/playwright.config.ts +11 -0
  21. package/dashboard/public/app.js +961 -0
  22. package/dashboard/public/index.html +29 -0
  23. package/dashboard/public/style.css +231 -0
  24. package/dashboard/src/index.ts +53 -0
  25. package/dashboard/src/routes/api.ts +235 -0
  26. package/dashboard/src/services/cache.ts +38 -0
  27. package/dashboard/src/services/experiments.ts +117 -0
  28. package/dashboard/src/services/git.ts +139 -0
  29. package/dashboard/src/services/sessions.ts +216 -0
  30. package/dashboard/src/services/specs.ts +95 -0
  31. package/dashboard/src/types.ts +168 -0
  32. package/dashboard/technical-specification.md +414 -0
  33. package/dashboard/test-api.sh +127 -0
  34. package/dashboard/tsconfig.json +16 -0
  35. package/lib/util/paths.js +9 -1
  36. package/package.json +10 -1
  37. package/scripts/dashboard.js +17 -0
  38. package/scripts/generate-daily-summaries.js +190 -0
  39. package/skills/opensassi/SKILL.md +8 -5
@@ -0,0 +1,173 @@
1
+ export interface SubjectArea {
2
+ name: string;
3
+ prompter_time_hours: number;
4
+ sme_time_hours: number;
5
+ ai_multiplier: number;
6
+ }
7
+ export interface SessionEntry {
8
+ session_id: string;
9
+ duration_minutes: number;
10
+ prompter_time_minutes: number;
11
+ sme_time_minutes: number;
12
+ top_component_summary: string;
13
+ tags: string[];
14
+ human_confidence: 'high' | 'medium' | 'low';
15
+ }
16
+ export interface DailyMetadata {
17
+ generated_at: string;
18
+ audited: boolean;
19
+ audit_note: string;
20
+ }
21
+ export interface NormalizedDay {
22
+ date: string;
23
+ metadata?: DailyMetadata;
24
+ total_prompter_time_hours: number;
25
+ total_sme_time_hours: number;
26
+ ai_multiplier: number;
27
+ total_sessions: number;
28
+ top_subject_areas: SubjectArea[];
29
+ session_breakdown: SessionEntry[];
30
+ }
31
+ export interface FormatA {
32
+ dashboard: {
33
+ metadata: DailyMetadata;
34
+ daily_summary: {
35
+ date: string;
36
+ total_prompter_time_hours: number;
37
+ total_sme_time_hours: number;
38
+ ai_multiplier: number;
39
+ total_sessions: number;
40
+ top_subject_areas: SubjectArea[];
41
+ };
42
+ session_breakdown: SessionEntry[];
43
+ };
44
+ }
45
+ export interface FormatB {
46
+ date: string;
47
+ total_prompter_time_hours: number;
48
+ total_sme_time_hours: number;
49
+ ai_multiplier: number;
50
+ total_sessions: number;
51
+ top_subject_areas: SubjectArea[];
52
+ session_breakdown: SessionEntry[];
53
+ }
54
+ export interface SessionInfo {
55
+ id: string;
56
+ slug: string;
57
+ title: string;
58
+ agent: string;
59
+ model: {
60
+ id: string;
61
+ providerID: string;
62
+ };
63
+ summary: {
64
+ additions: number;
65
+ deletions: number;
66
+ files: number;
67
+ };
68
+ time: {
69
+ created: number;
70
+ updated: number;
71
+ };
72
+ }
73
+ export interface MessageInfo {
74
+ role: string;
75
+ time: {
76
+ created: number;
77
+ };
78
+ agent: string;
79
+ model: {
80
+ providerID: string;
81
+ modelID: string;
82
+ };
83
+ summary: {
84
+ diffs: Array<{
85
+ path: string;
86
+ type: string;
87
+ lines: Record<string, number>;
88
+ }>;
89
+ };
90
+ id: string;
91
+ }
92
+ export interface MessagePart {
93
+ type: string;
94
+ text?: string;
95
+ }
96
+ export interface SessionMessage {
97
+ info: MessageInfo;
98
+ parts: MessagePart[];
99
+ }
100
+ export interface SessionDetail {
101
+ info: SessionInfo;
102
+ messages: SessionMessage[];
103
+ }
104
+ export interface GitLogEntry {
105
+ commit: string;
106
+ author: string;
107
+ date: string;
108
+ message: string;
109
+ files_changed: number;
110
+ insertions: number;
111
+ deletions: number;
112
+ }
113
+ export interface GitStats {
114
+ total_commits: number;
115
+ total_files_changed: number;
116
+ total_insertions: number;
117
+ total_deletions: number;
118
+ per_date: Record<string, {
119
+ commits: number;
120
+ insertions: number;
121
+ deletions: number;
122
+ }>;
123
+ }
124
+ export interface SearchResult {
125
+ session_id: string;
126
+ date: string;
127
+ summary: string;
128
+ tags: string[];
129
+ match_type: 'summary' | 'tag' | 'transcript';
130
+ match_snippet: string;
131
+ }
132
+ export interface CrossDayStats {
133
+ total_days: number;
134
+ total_sessions: number;
135
+ total_prompter_time_hours: number;
136
+ total_sme_time_hours: number;
137
+ avg_multiplier: number;
138
+ per_day: NormalizedDay[];
139
+ }
140
+ export interface SpecNode {
141
+ name: string;
142
+ path: string;
143
+ isDir: boolean;
144
+ children?: SpecNode[];
145
+ }
146
+ export interface HealthStatus {
147
+ status: 'ok' | 'error';
148
+ days_count: number;
149
+ sessions_count: number;
150
+ sessions_path: string;
151
+ }
152
+ export interface ExperimentEntry {
153
+ date: string;
154
+ directory: string;
155
+ description: string;
156
+ outcome: string;
157
+ agent: string;
158
+ }
159
+ export interface ExperimentFile {
160
+ path: string;
161
+ name: string;
162
+ size: number;
163
+ }
164
+ export interface ExperimentSubdir {
165
+ name: string;
166
+ files: ExperimentFile[];
167
+ }
168
+ export interface ExperimentDetail {
169
+ entry: ExperimentEntry;
170
+ readme: string | null;
171
+ subdirs: ExperimentSubdir[];
172
+ allFiles: ExperimentFile[];
173
+ }
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,100 @@
1
+ import { test, expect } from '@playwright/test';
2
+ import { execSync, spawn, type ChildProcess } from 'node:child_process';
3
+ import { resolve } from 'node:path';
4
+
5
+ const ROOT = resolve(import.meta.dirname ?? '.', '..');
6
+ let server: ChildProcess;
7
+ const PORT = 3098;
8
+
9
+ test.beforeAll(() => {
10
+ server = spawn(process.execPath, [resolve(ROOT, 'scripts/dashboard.js'), '--port', String(PORT)], {
11
+ cwd: ROOT,
12
+ stdio: 'pipe',
13
+ });
14
+ const maxWait = 15000;
15
+ const start = Date.now();
16
+ while (Date.now() - start < maxWait) {
17
+ try {
18
+ execSync(`curl -sf http://127.0.0.1:${PORT}/api/health`, { stdio: 'ignore' });
19
+ break;
20
+ } catch {
21
+ execSync('sleep 0.3');
22
+ }
23
+ }
24
+ });
25
+
26
+ test.afterAll(() => {
27
+ if (server) server.kill();
28
+ });
29
+
30
+ test.describe('opencode dashboard CLI', () => {
31
+ test('npx command starts server and health returns ok', async () => {
32
+ const res = await fetch(`http://127.0.0.1:${PORT}/api/health`);
33
+ expect(res.ok).toBe(true);
34
+ const body = await res.json();
35
+ expect(body.status).toBe('ok');
36
+ expect(body.days_count).toBeGreaterThan(0);
37
+ });
38
+
39
+ test('api/days returns opencode dates', async () => {
40
+ const res = await fetch(`http://127.0.0.1:${PORT}/api/days`);
41
+ expect(res.ok).toBe(true);
42
+ const body = await res.json();
43
+ expect(body.days.length).toBeGreaterThanOrEqual(4);
44
+ expect(body.days).toContain('2026-05-16');
45
+ });
46
+
47
+ test('api/days/latest returns latest day', async () => {
48
+ const res = await fetch(`http://127.0.0.1:${PORT}/api/days/latest`);
49
+ expect(res.ok).toBe(true);
50
+ const body = await res.json();
51
+ expect(body.date).toBeTruthy();
52
+ expect(body.total_sessions).toBeGreaterThan(0);
53
+ });
54
+
55
+ test('api/sessions returns sessions list', async () => {
56
+ const res = await fetch(`http://127.0.0.1:${PORT}/api/sessions`);
57
+ expect(res.ok).toBe(true);
58
+ const body = await res.json();
59
+ expect(body.total).toBeGreaterThan(0);
60
+ expect(body.sessions.length).toBeGreaterThan(0);
61
+ expect(body.sessions[0].entry).toHaveProperty('session_id');
62
+ });
63
+
64
+ test('api/stats returns aggregated stats', async () => {
65
+ const res = await fetch(`http://127.0.0.1:${PORT}/api/stats`);
66
+ expect(res.ok).toBe(true);
67
+ const body = await res.json();
68
+ expect(body.total_days).toBeGreaterThan(0);
69
+ expect(body.total_sessions).toBeGreaterThan(0);
70
+ expect(body.per_day.length).toBeGreaterThan(0);
71
+ });
72
+
73
+ test('frontend serves index.html', async ({ page }) => {
74
+ await page.goto(`http://127.0.0.1:${PORT}/`);
75
+ await expect(page.locator('nav')).toBeVisible();
76
+ await expect(page.locator('.nav-brand')).toHaveText('opencode');
77
+ });
78
+
79
+ test('overview page shows stats', async ({ page }) => {
80
+ await page.goto(`http://127.0.0.1:${PORT}/#/`);
81
+ await expect(page.getByRole('heading', { name: 'Overview' })).toBeVisible();
82
+ await expect(page.locator('.stats-grid')).toBeVisible();
83
+ });
84
+
85
+ test('daily page lists dates', async ({ page }) => {
86
+ await page.goto(`http://127.0.0.1:${PORT}/#/daily`);
87
+ await expect(page.locator('.page-title')).toHaveText('Daily Reports');
88
+ const links = page.locator('a[href^="#/daily/"]');
89
+ const count = await links.count();
90
+ expect(count).toBeGreaterThanOrEqual(4);
91
+ });
92
+
93
+ test('sessions page shows session cards', async ({ page }) => {
94
+ await page.goto(`http://127.0.0.1:${PORT}/#/sessions`);
95
+ await expect(page.locator('.page-title')).toHaveText(/All Sessions/);
96
+ await page.waitForSelector('.session-card');
97
+ const cards = await page.locator('.session-card').count();
98
+ expect(cards).toBeGreaterThan(0);
99
+ });
100
+ });
@@ -0,0 +1,11 @@
1
+ import { defineConfig } from '@playwright/test';
2
+
3
+ export default defineConfig({
4
+ testDir: '.',
5
+ testMatch: '**/*.e2e.test.ts',
6
+ timeout: 30000,
7
+ retries: 0,
8
+ use: {
9
+ headless: true,
10
+ },
11
+ });