@teamlens/web 0.1.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.
@@ -0,0 +1,2 @@
1
+ export declare function startDashboard(repoPath: string, port?: number): Promise<void>;
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAIA,wBAAsB,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,SAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CA6BjF"}
package/dist/index.js ADDED
@@ -0,0 +1,29 @@
1
+ import { createServer } from 'node:http';
2
+ import { TeamLens } from '@teamlens/core';
3
+ import { createApp } from './server.js';
4
+ export async function startDashboard(repoPath, port = 3847) {
5
+ const tl = await TeamLens.create(repoPath);
6
+ const app = createApp(tl);
7
+ const server = createServer(async (req, res) => {
8
+ try {
9
+ const url = new URL(req.url ?? '/', `http://localhost:${port}`);
10
+ const response = await app.fetch(new Request(url.toString(), {
11
+ method: req.method,
12
+ headers: Object.fromEntries(Object.entries(req.headers).filter(([, v]) => v !== undefined).map(([k, v]) => [k, Array.isArray(v) ? v.join(', ') : v])),
13
+ }));
14
+ res.writeHead(response.status, Object.fromEntries(response.headers.entries()));
15
+ const body = await response.arrayBuffer();
16
+ res.end(Buffer.from(body));
17
+ }
18
+ catch (err) {
19
+ res.writeHead(500);
20
+ res.end('Internal Server Error');
21
+ }
22
+ });
23
+ server.listen(port, () => {
24
+ console.log(` TeamLens Dashboard: http://localhost:${port}`);
25
+ });
26
+ process.on('SIGINT', () => { tl.close(); server.close(); process.exit(0); });
27
+ process.on('SIGTERM', () => { tl.close(); server.close(); process.exit(0); });
28
+ }
29
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAExC,MAAM,CAAC,KAAK,UAAU,cAAc,CAAC,QAAgB,EAAE,IAAI,GAAG,IAAI;IAChE,MAAM,EAAE,GAAG,MAAM,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;IAC3C,MAAM,GAAG,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC;IAE1B,MAAM,MAAM,GAAG,YAAY,CAAC,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;QAC7C,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,GAAG,EAAE,oBAAoB,IAAI,EAAE,CAAC,CAAC;YAChE,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,KAAK,CAAC,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,EAAE,EAAE;gBAC3D,MAAM,EAAE,GAAG,CAAC,MAAM;gBAClB,OAAO,EAAE,MAAM,CAAC,WAAW,CACzB,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAE,CAAC,CAAC,CAC1H;aACF,CAAC,CAAC,CAAC;YAEJ,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC,WAAW,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;YAC/E,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,WAAW,EAAE,CAAC;YAC1C,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;QAC7B,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;YACnB,GAAG,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAC;QACnC,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,EAAE;QACvB,OAAO,CAAC,GAAG,CAAC,0CAA0C,IAAI,EAAE,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;IAEH,OAAO,CAAC,EAAE,CAAC,QAAQ,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC7E,OAAO,CAAC,EAAE,CAAC,SAAS,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,KAAK,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;AAChF,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function analyticsRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=analytics.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"analytics.d.ts","sourceRoot":"","sources":["../../src/routes/analytics.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,cAAc,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CAgDjD"}
@@ -0,0 +1,43 @@
1
+ import { Hono } from 'hono';
2
+ export function analyticsRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/analytics', (c) => {
5
+ const days = Number(c.req.query('days') ?? '30');
6
+ const analytics = tl.analytics.getTeamAnalytics(days);
7
+ const totalSessions = tl.db.getTotalSessionCount();
8
+ const totalDuration = tl.db.getTotalSessionDuration();
9
+ const avgSessionDuration = totalSessions > 0 ? Math.round(totalDuration / totalSessions) : 0;
10
+ const memoryCount = tl.db.getMemoryCount();
11
+ const totalInsights = memoryCount.team;
12
+ const roi = tl.analytics.getRoiMetrics();
13
+ const reuseRate = totalInsights > 0 ? roi.knowledgeReuseCount / totalInsights : 0;
14
+ return c.json({
15
+ totalSessions: analytics.overview?.totalSessions ?? totalSessions,
16
+ insightsCreated: analytics.overview?.totalInsights ?? totalInsights,
17
+ activeContributors: analytics.overview?.activeDevelopers ?? tl.db.getDistinctDevelopers().length,
18
+ avgInsightsPerSession: analytics.overview?.avgInsightsPerSession ?? 0,
19
+ avgSessionDuration,
20
+ categoryBreakdown: analytics.insightsByType ?? {},
21
+ reuseRate,
22
+ filesTouched: 0,
23
+ });
24
+ });
25
+ route.get('/roi', (c) => {
26
+ const roi = tl.analytics.getRoiMetrics();
27
+ const memoryCount = tl.db.getMemoryCount();
28
+ return c.json({
29
+ ...roi,
30
+ timeSavedSeconds: Math.round(roi.estimatedHoursSaved * 3600),
31
+ totalReuses: roi.knowledgeReuseCount,
32
+ activeMemories: memoryCount.team,
33
+ uniqueInsights: memoryCount.team,
34
+ });
35
+ });
36
+ route.get('/trends', (c) => {
37
+ const days = Number(c.req.query('days') ?? '30');
38
+ const trends = tl.analytics.getUsageTrends(days);
39
+ return c.json({ trends });
40
+ });
41
+ return route;
42
+ }
43
+ //# sourceMappingURL=analytics.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"analytics.js","sourceRoot":"","sources":["../../src/routes/analytics.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,cAAc,CAAC,EAAY;IACzC,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE;QAC5B,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,CAAC;QACjD,MAAM,SAAS,GAAG,EAAE,CAAC,SAAS,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;QAEtD,MAAM,aAAa,GAAG,EAAE,CAAC,EAAE,CAAC,oBAAoB,EAAE,CAAC;QACnD,MAAM,aAAa,GAAG,EAAE,CAAC,EAAE,CAAC,uBAAuB,EAAE,CAAC;QACtD,MAAM,kBAAkB,GAAG,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAE7F,MAAM,WAAW,GAAG,EAAE,CAAC,EAAE,CAAC,cAAc,EAAE,CAAC;QAC3C,MAAM,aAAa,GAAG,WAAW,CAAC,IAAI,CAAC;QACvC,MAAM,GAAG,GAAG,EAAE,CAAC,SAAS,CAAC,aAAa,EAAE,CAAC;QACzC,MAAM,SAAS,GAAG,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,mBAAmB,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QAElF,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,aAAa,EAAE,SAAS,CAAC,QAAQ,EAAE,aAAa,IAAI,aAAa;YACjE,eAAe,EAAE,SAAS,CAAC,QAAQ,EAAE,aAAa,IAAI,aAAa;YACnE,kBAAkB,EAAE,SAAS,CAAC,QAAQ,EAAE,gBAAgB,IAAI,EAAE,CAAC,EAAE,CAAC,qBAAqB,EAAE,CAAC,MAAM;YAChG,qBAAqB,EAAE,SAAS,CAAC,QAAQ,EAAE,qBAAqB,IAAI,CAAC;YACrE,kBAAkB;YAClB,iBAAiB,EAAE,SAAS,CAAC,cAAc,IAAI,EAAE;YACjD,SAAS;YACT,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,EAAE;QACtB,MAAM,GAAG,GAAG,EAAE,CAAC,SAAS,CAAC,aAAa,EAAE,CAAC;QACzC,MAAM,WAAW,GAAG,EAAE,CAAC,EAAE,CAAC,cAAc,EAAE,CAAC;QAE3C,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,GAAG,GAAG;YACN,gBAAgB,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,mBAAmB,GAAG,IAAI,CAAC;YAC5D,WAAW,EAAE,GAAG,CAAC,mBAAmB;YACpC,cAAc,EAAE,WAAW,CAAC,IAAI;YAChC,cAAc,EAAE,WAAW,CAAC,IAAI;SACjC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,EAAE;QACzB,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,CAAC;QACjD,MAAM,MAAM,GAAG,EAAE,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QACjD,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IAC5B,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function contributorsRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=contributors.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"contributors.d.ts","sourceRoot":"","sources":["../../src/routes/contributors.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,iBAAiB,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CAoBpD"}
@@ -0,0 +1,19 @@
1
+ import { Hono } from 'hono';
2
+ export function contributorsRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/contributors', (c) => {
5
+ const limit = Number(c.req.query('limit') ?? '20');
6
+ const contributors = tl.analytics.getContributorLeaderboard(limit);
7
+ const enriched = contributors.map((entry) => {
8
+ const sessions = tl.db.getSessionsByDeveloper(entry.developer);
9
+ return {
10
+ ...entry,
11
+ sessionCount: sessions.length,
12
+ lastActiveAt: sessions[0]?.startedAt || null,
13
+ };
14
+ });
15
+ return c.json({ contributors: enriched });
16
+ });
17
+ return route;
18
+ }
19
+ //# sourceMappingURL=contributors.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"contributors.js","sourceRoot":"","sources":["../../src/routes/contributors.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,iBAAiB,CAAC,EAAY;IAC5C,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,eAAe,EAAE,CAAC,CAAC,EAAE,EAAE;QAC/B,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QACnD,MAAM,YAAY,GAAG,EAAE,CAAC,SAAS,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;QAEnE,MAAM,QAAQ,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC,KAAU,EAAE,EAAE;YAC/C,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,CAAC,sBAAsB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC/D,OAAO;gBACL,GAAG,KAAK;gBACR,YAAY,EAAE,QAAQ,CAAC,MAAM;gBAC7B,YAAY,EAAE,QAAQ,CAAC,CAAC,CAAC,EAAE,SAAS,IAAI,IAAI;aAC7C,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,YAAY,EAAE,QAAQ,EAAE,CAAC,CAAC;IAC5C,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function hotFilesRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=hot-files.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hot-files.d.ts","sourceRoot":"","sources":["../../src/routes/hot-files.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,aAAa,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CAUhD"}
@@ -0,0 +1,11 @@
1
+ import { Hono } from 'hono';
2
+ export function hotFilesRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/hot-files', (c) => {
5
+ const limit = Number(c.req.query('limit') ?? '20');
6
+ const hotFiles = tl.analytics.getHotFiles(limit);
7
+ return c.json({ hotFiles });
8
+ });
9
+ return route;
10
+ }
11
+ //# sourceMappingURL=hot-files.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hot-files.js","sourceRoot":"","sources":["../../src/routes/hot-files.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,aAAa,CAAC,EAAY;IACxC,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC,CAAC,EAAE,EAAE;QAC5B,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QACnD,MAAM,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,WAAW,CAAC,KAAK,CAAC,CAAC;QACjD,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAC;IAC9B,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function insightsRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=insights.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"insights.d.ts","sourceRoot":"","sources":["../../src/routes/insights.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,aAAa,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CAoChD"}
@@ -0,0 +1,34 @@
1
+ import { Hono } from 'hono';
2
+ export function insightsRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/insights', (c) => {
5
+ const limit = Number(c.req.query('limit') ?? '50');
6
+ const typeFilter = c.req.query('type');
7
+ const authorFilter = c.req.query('author');
8
+ const fileFilter = c.req.query('file');
9
+ let insights = tl.db.getRecentInsights(limit);
10
+ if (typeFilter) {
11
+ insights = insights.filter(m => m.category === typeFilter);
12
+ }
13
+ if (authorFilter) {
14
+ insights = insights.filter(m => m.author === authorFilter);
15
+ }
16
+ if (fileFilter) {
17
+ insights = insights.filter(m => m.relatedFiles.some(f => f.includes(fileFilter)));
18
+ }
19
+ return c.json({
20
+ insights: insights.map(m => ({
21
+ id: m.id,
22
+ content: m.content,
23
+ category: m.category,
24
+ author: m.author,
25
+ relatedFiles: m.relatedFiles,
26
+ tags: m.tags,
27
+ reuseCount: m.reuseCount,
28
+ createdAt: m.createdAt,
29
+ })),
30
+ });
31
+ });
32
+ return route;
33
+ }
34
+ //# sourceMappingURL=insights.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"insights.js","sourceRoot":"","sources":["../../src/routes/insights.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,aAAa,CAAC,EAAY;IACxC,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;QAC3B,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QACnD,MAAM,UAAU,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACvC,MAAM,YAAY,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;QAC3C,MAAM,UAAU,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAEvC,IAAI,QAAQ,GAAG,EAAE,CAAC,EAAE,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QAE9C,IAAI,UAAU,EAAE,CAAC;YACf,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,KAAK,UAAU,CAAC,CAAC;QAC7D,CAAC;QACD,IAAI,YAAY,EAAE,CAAC;YACjB,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,YAAY,CAAC,CAAC;QAC7D,CAAC;QACD,IAAI,UAAU,EAAE,CAAC;YACf,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QACpF,CAAC;QAED,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,QAAQ,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAC3B,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,QAAQ,EAAE,CAAC,CAAC,QAAQ;gBACpB,MAAM,EAAE,CAAC,CAAC,MAAM;gBAChB,YAAY,EAAE,CAAC,CAAC,YAAY;gBAC5B,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,SAAS,EAAE,CAAC,CAAC,SAAS;aACvB,CAAC,CAAC;SACJ,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function overviewRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=overview.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"overview.d.ts","sourceRoot":"","sources":["../../src/routes/overview.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,aAAa,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CAuChD"}
@@ -0,0 +1,37 @@
1
+ import { Hono } from 'hono';
2
+ export function overviewRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/overview', (c) => {
5
+ const overview = tl.analytics.getOverview();
6
+ const roi = tl.analytics.getRoiMetrics();
7
+ const totalSessions = tl.db.getTotalSessionCount();
8
+ const totalContributors = tl.db.getDistinctDevelopers().length;
9
+ const memoryCount = tl.db.getMemoryCount();
10
+ const totalInsights = memoryCount.team;
11
+ // Count active sessions
12
+ const allSessions = tl.db.getAllSessions(1000, 0);
13
+ const activeSessions = allSessions.filter(s => s.status === 'active').length;
14
+ // Average session duration (handle div by zero)
15
+ const totalDuration = tl.db.getTotalSessionDuration();
16
+ const avgSessionDuration = totalSessions > 0 ? Math.round(totalDuration / totalSessions) : 0;
17
+ return c.json({
18
+ totalSessions,
19
+ activeSessions,
20
+ totalContributors,
21
+ totalActivities: 0,
22
+ totalInsights,
23
+ avgSessionDuration,
24
+ today: overview.today,
25
+ week: overview.week,
26
+ month: overview.month,
27
+ roi: {
28
+ ...roi,
29
+ timeSavedSeconds: Math.round(roi.estimatedHoursSaved * 3600),
30
+ totalReuses: roi.knowledgeReuseCount,
31
+ uniqueInsights: totalInsights,
32
+ },
33
+ });
34
+ });
35
+ return route;
36
+ }
37
+ //# sourceMappingURL=overview.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"overview.js","sourceRoot":"","sources":["../../src/routes/overview.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,aAAa,CAAC,EAAY;IACxC,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;QAC3B,MAAM,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC;QAC5C,MAAM,GAAG,GAAG,EAAE,CAAC,SAAS,CAAC,aAAa,EAAE,CAAC;QACzC,MAAM,aAAa,GAAG,EAAE,CAAC,EAAE,CAAC,oBAAoB,EAAE,CAAC;QACnD,MAAM,iBAAiB,GAAG,EAAE,CAAC,EAAE,CAAC,qBAAqB,EAAE,CAAC,MAAM,CAAC;QAC/D,MAAM,WAAW,GAAG,EAAE,CAAC,EAAE,CAAC,cAAc,EAAE,CAAC;QAC3C,MAAM,aAAa,GAAG,WAAW,CAAC,IAAI,CAAC;QAEvC,wBAAwB;QACxB,MAAM,WAAW,GAAG,EAAE,CAAC,EAAE,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC;QAClD,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,MAAM,CAAC;QAE7E,gDAAgD;QAChD,MAAM,aAAa,GAAG,EAAE,CAAC,EAAE,CAAC,uBAAuB,EAAE,CAAC;QACtD,MAAM,kBAAkB,GAAG,aAAa,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAE7F,OAAO,CAAC,CAAC,IAAI,CAAC;YACZ,aAAa;YACb,cAAc;YACd,iBAAiB;YACjB,eAAe,EAAE,CAAC;YAClB,aAAa;YACb,kBAAkB;YAClB,KAAK,EAAE,QAAQ,CAAC,KAAK;YACrB,IAAI,EAAE,QAAQ,CAAC,IAAI;YACnB,KAAK,EAAE,QAAQ,CAAC,KAAK;YACrB,GAAG,EAAE;gBACH,GAAG,GAAG;gBACN,gBAAgB,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,mBAAmB,GAAG,IAAI,CAAC;gBAC5D,WAAW,EAAE,GAAG,CAAC,mBAAmB;gBACpC,cAAc,EAAE,aAAa;aAC9B;SACF,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function sessionsRoute(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=sessions.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sessions.d.ts","sourceRoot":"","sources":["../../src/routes/sessions.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAE/C,wBAAgB,aAAa,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CA4BhD"}
@@ -0,0 +1,27 @@
1
+ import { Hono } from 'hono';
2
+ export function sessionsRoute(tl) {
3
+ const route = new Hono();
4
+ route.get('/sessions', (c) => {
5
+ const limit = Number(c.req.query('limit') ?? '50');
6
+ const offset = Number(c.req.query('offset') ?? '0');
7
+ const sessions = tl.db.getAllSessions(limit, offset);
8
+ return c.json({ sessions, total: tl.db.getTotalSessionCount() });
9
+ });
10
+ route.get('/sessions/:id', (c) => {
11
+ const id = c.req.param('id');
12
+ const session = tl.db.getSession(id);
13
+ if (!session)
14
+ return c.json({ error: 'Session not found' }, 404);
15
+ const activities = tl.db.getActivitiesBySession(id);
16
+ const insights = tl.db.getMemoriesBySessionId(id);
17
+ return c.json({ session, activities, insights: insights.map(m => ({
18
+ id: m.id,
19
+ content: m.content,
20
+ category: m.category,
21
+ author: m.author,
22
+ createdAt: m.createdAt,
23
+ })) });
24
+ });
25
+ return route;
26
+ }
27
+ //# sourceMappingURL=sessions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sessions.js","sourceRoot":"","sources":["../../src/routes/sessions.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,MAAM,UAAU,aAAa,CAAC,EAAY;IACxC,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IAEzB,KAAK,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,EAAE;QAC3B,MAAM,KAAK,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,CAAC;QACnD,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,CAAC;QACpD,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,CAAC,cAAc,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;QACrD,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,EAAE,CAAC,EAAE,CAAC,oBAAoB,EAAE,EAAE,CAAC,CAAC;IACnE,CAAC,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,CAAC,eAAe,EAAE,CAAC,CAAC,EAAE,EAAE;QAC/B,MAAM,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAC7B,MAAM,OAAO,GAAG,EAAE,CAAC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QACrC,IAAI,CAAC,OAAO;YAAE,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,EAAE,GAAG,CAAC,CAAC;QAEjE,MAAM,UAAU,GAAG,EAAE,CAAC,EAAE,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC;QACpD,MAAM,QAAQ,GAAG,EAAE,CAAC,EAAE,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC;QAElD,OAAO,CAAC,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAChE,EAAE,EAAE,CAAC,CAAC,EAAE;gBACR,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,QAAQ,EAAE,CAAC,CAAC,QAAQ;gBACpB,MAAM,EAAE,CAAC,CAAC,MAAM;gBAChB,SAAS,EAAE,CAAC,CAAC,SAAS;aACvB,CAAC,CAAC,EAAE,CAAC,CAAC;IACT,CAAC,CAAC,CAAC;IAEH,OAAO,KAAK,CAAC;AACf,CAAC"}
@@ -0,0 +1,4 @@
1
+ import { Hono } from 'hono';
2
+ import type { TeamLens } from '@teamlens/core';
3
+ export declare function createApp(tl: TeamLens): Hono;
4
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAI5B,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,gBAAgB,CAAC;AAQ/C,wBAAgB,SAAS,CAAC,EAAE,EAAE,QAAQ,GAAG,IAAI,CA+D5C"}
package/dist/server.js ADDED
@@ -0,0 +1,65 @@
1
+ import { Hono } from 'hono';
2
+ import fs from 'node:fs';
3
+ import path from 'node:path';
4
+ import { fileURLToPath } from 'node:url';
5
+ import { overviewRoute } from './routes/overview.js';
6
+ import { sessionsRoute } from './routes/sessions.js';
7
+ import { insightsRoute } from './routes/insights.js';
8
+ import { contributorsRoute } from './routes/contributors.js';
9
+ import { analyticsRoute } from './routes/analytics.js';
10
+ import { hotFilesRoute } from './routes/hot-files.js';
11
+ export function createApp(tl) {
12
+ const app = new Hono();
13
+ // Reload DB from disk before each API request so we see data
14
+ // written by the MCP server (which runs in a separate process)
15
+ app.use('/api/*', async (c, next) => {
16
+ tl.db.reload();
17
+ await next();
18
+ });
19
+ // API routes
20
+ app.route('/api', overviewRoute(tl));
21
+ app.route('/api', sessionsRoute(tl));
22
+ app.route('/api', insightsRoute(tl));
23
+ app.route('/api', contributorsRoute(tl));
24
+ app.route('/api', analyticsRoute(tl));
25
+ app.route('/api', hotFilesRoute(tl));
26
+ // Static files - serve from public/ directory
27
+ // Use manual static file serving since we can't use hono/node-server in ESM easily
28
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
29
+ const publicDir = path.resolve(__dirname, '..', 'public');
30
+ const MIME_TYPES = {
31
+ '.html': 'text/html',
32
+ '.css': 'text/css',
33
+ '.js': 'application/javascript',
34
+ '.json': 'application/json',
35
+ '.png': 'image/png',
36
+ '.svg': 'image/svg+xml',
37
+ };
38
+ app.get('*', async (c) => {
39
+ const urlPath = c.req.path === '/' ? '/index.html' : c.req.path;
40
+ const filePath = path.join(publicDir, urlPath);
41
+ // Security: prevent directory traversal
42
+ if (!filePath.startsWith(publicDir)) {
43
+ return c.text('Forbidden', 403);
44
+ }
45
+ if (fs.existsSync(filePath) && fs.statSync(filePath).isFile()) {
46
+ const ext = path.extname(filePath);
47
+ const contentType = MIME_TYPES[ext] || 'application/octet-stream';
48
+ const content = fs.readFileSync(filePath);
49
+ return new Response(content, {
50
+ headers: { 'Content-Type': contentType },
51
+ });
52
+ }
53
+ // SPA fallback — serve index.html for non-API routes
54
+ const indexPath = path.join(publicDir, 'index.html');
55
+ if (fs.existsSync(indexPath)) {
56
+ const content = fs.readFileSync(indexPath);
57
+ return new Response(content, {
58
+ headers: { 'Content-Type': 'text/html' },
59
+ });
60
+ }
61
+ return c.text('Not Found', 404);
62
+ });
63
+ return app;
64
+ }
65
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,34 @@
1
+ {
2
+ "name": "@teamlens/web",
3
+ "version": "0.1.0",
4
+ "description": "TeamLens local web dashboard — session analytics, insights feed, contributor leaderboard",
5
+ "type": "module",
6
+ "license": "MIT",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "https://github.com/sanketbabar/teamlens",
10
+ "directory": "packages/web"
11
+ },
12
+ "publishConfig": {
13
+ "access": "public"
14
+ },
15
+ "main": "dist/index.js",
16
+ "types": "dist/index.d.ts",
17
+ "files": [
18
+ "dist/",
19
+ "public/"
20
+ ],
21
+ "dependencies": {
22
+ "hono": "^4.0.0",
23
+ "@teamlens/core": "0.1.0"
24
+ },
25
+ "devDependencies": {
26
+ "typescript": "^5.7.0",
27
+ "@types/node": "^22.0.0"
28
+ },
29
+ "scripts": {
30
+ "build": "tsc",
31
+ "dev": "tsc --watch",
32
+ "clean": "rm -rf dist .tsbuildinfo"
33
+ }
34
+ }