yo-bug 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.
Files changed (140) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +160 -0
  3. package/bin/cli.ts +17 -0
  4. package/bin/install.ts +34 -0
  5. package/bin/mcp.ts +2 -0
  6. package/dist/bin/cli.d.ts +3 -0
  7. package/dist/bin/cli.d.ts.map +1 -0
  8. package/dist/bin/cli.js +19 -0
  9. package/dist/bin/cli.js.map +1 -0
  10. package/dist/bin/install.d.ts +3 -0
  11. package/dist/bin/install.d.ts.map +1 -0
  12. package/dist/bin/install.js +28 -0
  13. package/dist/bin/install.js.map +1 -0
  14. package/dist/bin/mcp.d.ts +3 -0
  15. package/dist/bin/mcp.d.ts.map +1 -0
  16. package/dist/bin/mcp.js +3 -0
  17. package/dist/bin/mcp.js.map +1 -0
  18. package/dist/src/detect/dev-server.d.ts +11 -0
  19. package/dist/src/detect/dev-server.d.ts.map +1 -0
  20. package/dist/src/detect/dev-server.js +92 -0
  21. package/dist/src/detect/dev-server.js.map +1 -0
  22. package/dist/src/detect/spawn-dev.d.ts +4 -0
  23. package/dist/src/detect/spawn-dev.d.ts.map +1 -0
  24. package/dist/src/detect/spawn-dev.js +37 -0
  25. package/dist/src/detect/spawn-dev.js.map +1 -0
  26. package/dist/src/install/detect-ai-tool.d.ts +7 -0
  27. package/dist/src/install/detect-ai-tool.d.ts.map +1 -0
  28. package/dist/src/install/detect-ai-tool.js +38 -0
  29. package/dist/src/install/detect-ai-tool.js.map +1 -0
  30. package/dist/src/install/write-mcp-config.d.ts +6 -0
  31. package/dist/src/install/write-mcp-config.d.ts.map +1 -0
  32. package/dist/src/install/write-mcp-config.js +47 -0
  33. package/dist/src/install/write-mcp-config.js.map +1 -0
  34. package/dist/src/mcp/index.d.ts +2 -0
  35. package/dist/src/mcp/index.d.ts.map +1 -0
  36. package/dist/src/mcp/index.js +263 -0
  37. package/dist/src/mcp/index.js.map +1 -0
  38. package/dist/src/mcp/tools/checklist.d.ts +22 -0
  39. package/dist/src/mcp/tools/checklist.d.ts.map +1 -0
  40. package/dist/src/mcp/tools/checklist.js +58 -0
  41. package/dist/src/mcp/tools/checklist.js.map +1 -0
  42. package/dist/src/mcp/tools/get-feedback.d.ts +13 -0
  43. package/dist/src/mcp/tools/get-feedback.d.ts.map +1 -0
  44. package/dist/src/mcp/tools/get-feedback.js +27 -0
  45. package/dist/src/mcp/tools/get-feedback.js.map +1 -0
  46. package/dist/src/mcp/tools/list-feedbacks.d.ts +20 -0
  47. package/dist/src/mcp/tools/list-feedbacks.d.ts.map +1 -0
  48. package/dist/src/mcp/tools/list-feedbacks.js +29 -0
  49. package/dist/src/mcp/tools/list-feedbacks.js.map +1 -0
  50. package/dist/src/mcp/tools/resolve-feedback.d.ts +8 -0
  51. package/dist/src/mcp/tools/resolve-feedback.d.ts.map +1 -0
  52. package/dist/src/mcp/tools/resolve-feedback.js +28 -0
  53. package/dist/src/mcp/tools/resolve-feedback.js.map +1 -0
  54. package/dist/src/mcp/tools/start-session.d.ts +11 -0
  55. package/dist/src/mcp/tools/start-session.d.ts.map +1 -0
  56. package/dist/src/mcp/tools/start-session.js +56 -0
  57. package/dist/src/mcp/tools/start-session.js.map +1 -0
  58. package/dist/src/mcp/tools/stop-session.d.ts +6 -0
  59. package/dist/src/mcp/tools/stop-session.d.ts.map +1 -0
  60. package/dist/src/mcp/tools/stop-session.js +80 -0
  61. package/dist/src/mcp/tools/stop-session.js.map +1 -0
  62. package/dist/src/mcp/tools/test-history.d.ts +33 -0
  63. package/dist/src/mcp/tools/test-history.d.ts.map +1 -0
  64. package/dist/src/mcp/tools/test-history.js +66 -0
  65. package/dist/src/mcp/tools/test-history.js.map +1 -0
  66. package/dist/src/server/feedback-api.d.ts +4 -0
  67. package/dist/src/server/feedback-api.d.ts.map +1 -0
  68. package/dist/src/server/feedback-api.js +152 -0
  69. package/dist/src/server/feedback-api.js.map +1 -0
  70. package/dist/src/server/html-injector.d.ts +10 -0
  71. package/dist/src/server/html-injector.d.ts.map +1 -0
  72. package/dist/src/server/html-injector.js +34 -0
  73. package/dist/src/server/html-injector.js.map +1 -0
  74. package/dist/src/server/proxy-server.d.ts +8 -0
  75. package/dist/src/server/proxy-server.d.ts.map +1 -0
  76. package/dist/src/server/proxy-server.js +87 -0
  77. package/dist/src/server/proxy-server.js.map +1 -0
  78. package/dist/src/server/sdk-serve.d.ts +3 -0
  79. package/dist/src/server/sdk-serve.d.ts.map +1 -0
  80. package/dist/src/server/sdk-serve.js +20 -0
  81. package/dist/src/server/sdk-serve.js.map +1 -0
  82. package/dist/src/storage/store.d.ts +21 -0
  83. package/dist/src/storage/store.d.ts.map +1 -0
  84. package/dist/src/storage/store.js +138 -0
  85. package/dist/src/storage/store.js.map +1 -0
  86. package/dist/src/storage/types.d.ts +74 -0
  87. package/dist/src/storage/types.d.ts.map +1 -0
  88. package/dist/src/storage/types.js +2 -0
  89. package/dist/src/storage/types.js.map +1 -0
  90. package/dist/vibe-feedback.js +399 -0
  91. package/package.json +67 -0
  92. package/src/client/annotation-mode/canvas-editor.ts +178 -0
  93. package/src/client/annotation-mode/history.ts +32 -0
  94. package/src/client/annotation-mode/region-selector.ts +123 -0
  95. package/src/client/annotation-mode/screenshot.ts +17 -0
  96. package/src/client/annotation-mode/toolbar.ts +139 -0
  97. package/src/client/annotation-mode/tools/arrow.ts +57 -0
  98. package/src/client/annotation-mode/tools/base-tool.ts +25 -0
  99. package/src/client/annotation-mode/tools/circle.ts +37 -0
  100. package/src/client/annotation-mode/tools/freehand.ts +23 -0
  101. package/src/client/annotation-mode/tools/rect.ts +32 -0
  102. package/src/client/annotation-mode/tools/text.ts +93 -0
  103. package/src/client/api/client.ts +48 -0
  104. package/src/client/capture/action-recorder.ts +157 -0
  105. package/src/client/capture/console-interceptor.ts +65 -0
  106. package/src/client/capture/context-buffer.ts +23 -0
  107. package/src/client/capture/error-interceptor.ts +52 -0
  108. package/src/client/capture/network-interceptor.ts +143 -0
  109. package/src/client/core/event-bus.ts +20 -0
  110. package/src/client/core/i18n.ts +83 -0
  111. package/src/client/core/sdk.ts +373 -0
  112. package/src/client/core/shadow-host.ts +27 -0
  113. package/src/client/element-mode/highlighter.ts +79 -0
  114. package/src/client/element-mode/inspector.ts +73 -0
  115. package/src/client/element-mode/selector.ts +22 -0
  116. package/src/client/index.ts +10 -0
  117. package/src/client/styles/sdk.css.ts +222 -0
  118. package/src/client/ui/checklist-panel.ts +279 -0
  119. package/src/client/ui/feedback-panel.ts +149 -0
  120. package/src/client/ui/floating-button.ts +103 -0
  121. package/src/client/ui/toast.ts +17 -0
  122. package/src/client/ui/verify-panel.ts +111 -0
  123. package/src/detect/dev-server.ts +110 -0
  124. package/src/detect/spawn-dev.ts +50 -0
  125. package/src/install/detect-ai-tool.ts +49 -0
  126. package/src/install/write-mcp-config.ts +49 -0
  127. package/src/mcp/index.ts +327 -0
  128. package/src/mcp/tools/checklist.ts +61 -0
  129. package/src/mcp/tools/get-feedback.ts +34 -0
  130. package/src/mcp/tools/list-feedbacks.ts +37 -0
  131. package/src/mcp/tools/resolve-feedback.ts +34 -0
  132. package/src/mcp/tools/start-session.ts +65 -0
  133. package/src/mcp/tools/stop-session.ts +93 -0
  134. package/src/mcp/tools/test-history.ts +97 -0
  135. package/src/server/feedback-api.ts +164 -0
  136. package/src/server/html-injector.ts +41 -0
  137. package/src/server/proxy-server.ts +107 -0
  138. package/src/server/sdk-serve.ts +24 -0
  139. package/src/storage/store.ts +172 -0
  140. package/src/storage/types.ts +68 -0
@@ -0,0 +1,80 @@
1
+ import { stopProxyServer, isProxyRunning } from '../../server/proxy-server.js';
2
+ import { killDevServer, isDevServerSpawned } from '../../detect/spawn-dev.js';
3
+ export async function stopTestSession(store) {
4
+ if (!isProxyRunning()) {
5
+ return {
6
+ status: 'not_running',
7
+ message: '测试模式未在运行。',
8
+ };
9
+ }
10
+ // Gather session summary before stopping
11
+ const { items: allItems } = await store.list({ status: 'all', limit: 100 });
12
+ const openItems = allItems.filter(i => i.status === 'open');
13
+ const resolvedItems = allItems.filter(i => i.status === 'resolved');
14
+ // Count by type
15
+ const typeCounts = {};
16
+ for (const item of openItems) {
17
+ typeCounts[item.problemType] = (typeCounts[item.problemType] || 0) + 1;
18
+ }
19
+ // Check checklist status
20
+ let checklistSummary = '';
21
+ try {
22
+ const res = await fetch('http://localhost:3695/api/checklist');
23
+ const cl = await res.json();
24
+ if (cl.items && cl.items.length > 0) {
25
+ const passed = cl.items.filter((i) => i.status === 'passed').length;
26
+ const failed = cl.items.filter((i) => i.status === 'failed').length;
27
+ const pending = cl.items.filter((i) => i.status === 'pending').length;
28
+ checklistSummary = `\n测试清单「${cl.title}」: ${passed} 通过 / ${failed} 失败 / ${pending} 未测`;
29
+ // Identify weak dimensions
30
+ const failedDimensions = cl.items
31
+ .filter((i) => i.status === 'failed' && i.dimension)
32
+ .map((i) => i.dimension);
33
+ const pendingDimensions = cl.items
34
+ .filter((i) => i.status === 'pending' && i.dimension)
35
+ .map((i) => i.dimension);
36
+ if (failedDimensions.length > 0) {
37
+ const unique = [...new Set(failedDimensions)];
38
+ checklistSummary += `\n失败集中维度: ${unique.join('、')}`;
39
+ }
40
+ if (pendingDimensions.length > 0) {
41
+ const unique = [...new Set(pendingDimensions)];
42
+ checklistSummary += `\n未测维度: ${unique.join('、')}`;
43
+ }
44
+ }
45
+ }
46
+ catch { }
47
+ // Stop proxy
48
+ stopProxyServer();
49
+ // Kill spawned dev server (if we started it)
50
+ if (isDevServerSpawned()) {
51
+ killDevServer();
52
+ }
53
+ // Build summary
54
+ const lines = ['测试模式已关闭。\n', '── 测试会话总结 ──'];
55
+ lines.push(`反馈: ${allItems.length} 条(${openItems.length} 待修复, ${resolvedItems.length} 已修复)`);
56
+ if (Object.keys(typeCounts).length > 0) {
57
+ const typeStr = Object.entries(typeCounts)
58
+ .map(([type, count]) => `${type}: ${count}`)
59
+ .join(', ');
60
+ lines.push(`问题分布: ${typeStr}`);
61
+ }
62
+ if (checklistSummary) {
63
+ lines.push(checklistSummary);
64
+ }
65
+ // Actionable conclusion
66
+ if (openItems.length > 0) {
67
+ lines.push(`\n建议: 还有 ${openItems.length} 条未修复反馈,请查看 list_feedbacks() 并逐一处理。`);
68
+ }
69
+ else if (allItems.length > 0) {
70
+ lines.push('\n结论: 所有反馈已修复。');
71
+ }
72
+ else {
73
+ lines.push('\n结论: 本次未收到反馈。');
74
+ }
75
+ return {
76
+ status: 'stopped',
77
+ message: lines.join('\n'),
78
+ };
79
+ }
80
+ //# sourceMappingURL=stop-session.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stop-session.js","sourceRoot":"","sources":["../../../../src/mcp/tools/stop-session.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ interface TestRecord {
2
+ module: string;
3
+ title: string;
4
+ items: any[];
5
+ results: {
6
+ passed: number;
7
+ failed: number;
8
+ total: number;
9
+ };
10
+ failedItems: string[];
11
+ timestamp: string;
12
+ }
13
+ export declare function saveTestRecord(args: {
14
+ module: string;
15
+ title: string;
16
+ items: any[];
17
+ results: {
18
+ passed: number;
19
+ failed: number;
20
+ total: number;
21
+ };
22
+ failedItems: string[];
23
+ }): Promise<{
24
+ text: string;
25
+ }>;
26
+ export declare function getTestHistory(args: {
27
+ module: string;
28
+ }): Promise<{
29
+ text: string;
30
+ records: TestRecord[];
31
+ }>;
32
+ export {};
33
+ //# sourceMappingURL=test-history.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test-history.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/test-history.ts"],"names":[],"mappings":"AASA,UAAU,UAAU;IAClB,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,GAAG,EAAE,CAAC;IACb,OAAO,EAAE;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC;IAC3D,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,wBAAsB,cAAc,CAAC,IAAI,EAAE;IACzC,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,GAAG,EAAE,CAAC;IACb,OAAO,EAAE;QAAE,MAAM,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC;IAC3D,WAAW,EAAE,MAAM,EAAE,CAAC;CACvB,GAAG,OAAO,CAAC;IAAE,IAAI,EAAE,MAAM,CAAA;CAAE,CAAC,CAyB5B;AAED,wBAAsB,cAAc,CAAC,IAAI,EAAE;IACzC,MAAM,EAAE,MAAM,CAAC;CAChB,GAAG,OAAO,CAAC;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,OAAO,EAAE,UAAU,EAAE,CAAA;CAAE,CAAC,CAuCnD"}
@@ -0,0 +1,66 @@
1
+ import fs from 'fs/promises';
2
+ import path from 'path';
3
+ const HISTORY_DIR = path.join(process.env.HOME || process.env.USERPROFILE || '.', '.yo-bug', 'test-history');
4
+ export async function saveTestRecord(args) {
5
+ await fs.mkdir(HISTORY_DIR, { recursive: true });
6
+ const record = {
7
+ ...args,
8
+ timestamp: new Date().toISOString(),
9
+ };
10
+ // Load existing records for this module
11
+ const filePath = path.join(HISTORY_DIR, `${sanitize(args.module)}.json`);
12
+ let records = [];
13
+ try {
14
+ const data = await fs.readFile(filePath, 'utf-8');
15
+ records = JSON.parse(data);
16
+ }
17
+ catch { }
18
+ records.push(record);
19
+ // Keep last 20 records per module
20
+ if (records.length > 20)
21
+ records = records.slice(-20);
22
+ await fs.writeFile(filePath, JSON.stringify(records, null, 2), 'utf-8');
23
+ return {
24
+ text: `测试记录已保存。模块「${args.module}」累计 ${records.length} 次测试记录。`,
25
+ };
26
+ }
27
+ export async function getTestHistory(args) {
28
+ const filePath = path.join(HISTORY_DIR, `${sanitize(args.module)}.json`);
29
+ let records = [];
30
+ try {
31
+ const data = await fs.readFile(filePath, 'utf-8');
32
+ records = JSON.parse(data);
33
+ }
34
+ catch {
35
+ return { text: `模块「${args.module}」没有历史测试记录。`, records: [] };
36
+ }
37
+ const lines = [`模块「${args.module}」历史测试记录(${records.length} 次):\n`];
38
+ for (const r of records.slice(-5)) { // Show last 5
39
+ lines.push(`[${r.timestamp.slice(0, 10)}] ${r.title}`);
40
+ lines.push(` 结果: ${r.results.passed}/${r.results.total} 通过`);
41
+ if (r.failedItems.length > 0) {
42
+ lines.push(` 失败项: ${r.failedItems.join('; ')}`);
43
+ }
44
+ }
45
+ // Aggregate: which test scenarios fail most often?
46
+ const failCounts = {};
47
+ for (const r of records) {
48
+ for (const f of r.failedItems) {
49
+ failCounts[f] = (failCounts[f] || 0) + 1;
50
+ }
51
+ }
52
+ const frequentFails = Object.entries(failCounts)
53
+ .sort((a, b) => b[1] - a[1])
54
+ .slice(0, 5);
55
+ if (frequentFails.length > 0) {
56
+ lines.push('\n高频失败场景(生成清单时请重点覆盖):');
57
+ for (const [scenario, count] of frequentFails) {
58
+ lines.push(` ${count}x 失败: ${scenario}`);
59
+ }
60
+ }
61
+ return { text: lines.join('\n'), records };
62
+ }
63
+ function sanitize(name) {
64
+ return name.replace(/[^a-zA-Z0-9\u4e00-\u9fff_-]/g, '_').slice(0, 50);
65
+ }
66
+ //# sourceMappingURL=test-history.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"test-history.js","sourceRoot":"","sources":["../../../../src/mcp/tools/test-history.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import { Router } from 'express';
2
+ import { FeedbackStore } from '../storage/store.js';
3
+ export declare function createFeedbackRouter(store: FeedbackStore): Router;
4
+ //# sourceMappingURL=feedback-api.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"feedback-api.d.ts","sourceRoot":"","sources":["../../../src/server/feedback-api.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAIpD,wBAAgB,oBAAoB,CAAC,KAAK,EAAE,aAAa,GAAG,MAAM,CA6JjE"}
@@ -0,0 +1,152 @@
1
+ import { Router } from 'express';
2
+ import multer from 'multer';
3
+ const upload = multer({ storage: multer.memoryStorage() });
4
+ export function createFeedbackRouter(store) {
5
+ const router = Router();
6
+ // Submit feedback
7
+ router.post('/api/feedback', upload.single('screenshot'), async (req, res) => {
8
+ try {
9
+ let feedbackData;
10
+ if (req.body.feedback) {
11
+ // multipart form: feedback JSON string + screenshot file
12
+ feedbackData = JSON.parse(req.body.feedback);
13
+ }
14
+ else {
15
+ // plain JSON body
16
+ feedbackData = req.body;
17
+ }
18
+ // Sanitize: only allow known fields, strip anything dangerous
19
+ const sanitized = {
20
+ mode: feedbackData.mode,
21
+ problemType: feedbackData.problemType,
22
+ description: String(feedbackData.description || '').slice(0, 2000),
23
+ element: feedbackData.element,
24
+ pageUrl: String(feedbackData.pageUrl || '').slice(0, 500),
25
+ pageTitle: String(feedbackData.pageTitle || '').slice(0, 200),
26
+ viewport: feedbackData.viewport,
27
+ userAgent: String(feedbackData.userAgent || '').slice(0, 300),
28
+ timestamp: feedbackData.timestamp,
29
+ consoleErrors: Array.isArray(feedbackData.consoleErrors) ? feedbackData.consoleErrors.slice(0, 50) : [],
30
+ networkErrors: Array.isArray(feedbackData.networkErrors) ? feedbackData.networkErrors.slice(0, 50) : [],
31
+ unhandledErrors: Array.isArray(feedbackData.unhandledErrors) ? feedbackData.unhandledErrors.slice(0, 50) : [],
32
+ actionSteps: Array.isArray(feedbackData.actionSteps) ? feedbackData.actionSteps.slice(0, 100) : [],
33
+ hasScreenshot: false,
34
+ };
35
+ const screenshot = req.file?.buffer;
36
+ const item = await store.create(sanitized, screenshot);
37
+ res.json({ id: item.id, createdAt: item.createdAt });
38
+ }
39
+ catch (err) {
40
+ res.status(400).json({ error: err.message });
41
+ }
42
+ });
43
+ // List feedbacks
44
+ router.get('/api/feedback', async (req, res) => {
45
+ try {
46
+ const result = await store.list({
47
+ limit: Number(req.query.limit) || 20,
48
+ status: req.query.status || 'open',
49
+ type: req.query.type,
50
+ });
51
+ res.json(result);
52
+ }
53
+ catch (err) {
54
+ res.status(500).json({ error: err.message });
55
+ }
56
+ });
57
+ // Get single feedback
58
+ router.get('/api/feedback/:id', async (req, res) => {
59
+ try {
60
+ const item = await store.getById(req.params.id);
61
+ if (!item) {
62
+ return res.status(404).json({ error: 'Not found' });
63
+ }
64
+ res.json(item);
65
+ }
66
+ catch (err) {
67
+ res.status(500).json({ error: err.message });
68
+ }
69
+ });
70
+ // Get screenshot
71
+ router.get('/api/feedback/:id/screenshot', async (req, res) => {
72
+ try {
73
+ const screenshot = await store.getScreenshot(req.params.id);
74
+ if (!screenshot) {
75
+ return res.status(404).json({ error: 'No screenshot' });
76
+ }
77
+ res.setHeader('Content-Type', 'image/png');
78
+ res.end(screenshot);
79
+ }
80
+ catch (err) {
81
+ res.status(500).json({ error: err.message });
82
+ }
83
+ });
84
+ // --- Checklist API (AI → Browser → AI) ---
85
+ let checklist = {
86
+ items: [],
87
+ title: '',
88
+ createdAt: '',
89
+ };
90
+ // AI pushes checklist (supports both string[] and structured items)
91
+ router.post('/api/checklist', (req, res) => {
92
+ const { title, items } = req.body;
93
+ checklist = {
94
+ title: title || 'Test Checklist',
95
+ items: (items || []).map((item, i) => ({
96
+ id: i,
97
+ step: typeof item === 'string' ? item : (item.step || item.text || ''),
98
+ expect: typeof item === 'string' ? '' : (item.expect || ''),
99
+ priority: (typeof item === 'object' && item.priority) || 'normal',
100
+ dimension: (typeof item === 'object' && item.dimension) || '',
101
+ status: 'pending',
102
+ feedback: '',
103
+ })),
104
+ createdAt: new Date().toISOString(),
105
+ };
106
+ res.json({ ok: true, count: checklist.items.length });
107
+ });
108
+ // Browser polls checklist
109
+ router.get('/api/checklist', (_req, res) => {
110
+ res.json(checklist);
111
+ });
112
+ // Browser updates a checklist item
113
+ router.patch('/api/checklist/:itemId', (req, res) => {
114
+ const id = parseInt(req.params.itemId);
115
+ const item = checklist.items.find((i) => i.id === id);
116
+ if (!item)
117
+ return res.status(404).json({ error: 'Not found' });
118
+ if (req.body.status)
119
+ item.status = req.body.status;
120
+ if (req.body.feedback !== undefined)
121
+ item.feedback = req.body.feedback;
122
+ res.json(item);
123
+ });
124
+ // --- Verification API (AI → Browser → AI) ---
125
+ let pendingVerifications = [];
126
+ // AI pushes verification request
127
+ router.post('/api/verify', (req, res) => {
128
+ pendingVerifications.push(req.body);
129
+ res.json({ ok: true });
130
+ });
131
+ // Browser polls for pending verifications
132
+ router.get('/api/verify', (_req, res) => {
133
+ res.json({ items: pendingVerifications });
134
+ });
135
+ // Browser confirms/rejects verification
136
+ router.patch('/api/verify/:feedbackId', async (req, res) => {
137
+ const { feedbackId } = req.params;
138
+ const { confirmed } = req.body; // true = fix works, false = still broken
139
+ const newStatus = confirmed ? 'resolved' : 'open';
140
+ await store.updateStatus(feedbackId, newStatus);
141
+ pendingVerifications = pendingVerifications.filter(v => v.feedbackId !== feedbackId);
142
+ res.json({ ok: true, status: newStatus });
143
+ });
144
+ // Reset in-memory state
145
+ router.post('/api/reset', (_req, res) => {
146
+ checklist = { items: [], title: '', createdAt: '' };
147
+ pendingVerifications = [];
148
+ res.json({ ok: true });
149
+ });
150
+ return router;
151
+ }
152
+ //# sourceMappingURL=feedback-api.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"feedback-api.js","sourceRoot":"","sources":["../../../src/server/feedback-api.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import type { IncomingMessage, ServerResponse } from 'http';
2
+ /**
3
+ * Middleware that intercepts HTML responses from the proxy
4
+ * and injects the vibe-feedback SDK script before </body>.
5
+ */
6
+ export declare function createHtmlInjector(proxyRes: IncomingMessage, res: ServerResponse): {
7
+ onData(chunk: Buffer): void;
8
+ onEnd(): void;
9
+ } | null;
10
+ //# sourceMappingURL=html-injector.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"html-injector.d.ts","sourceRoot":"","sources":["../../../src/server/html-injector.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,cAAc,EAAE,MAAM,MAAM,CAAC;AAI5D;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,QAAQ,EAAE,eAAe,EAAE,GAAG,EAAE,cAAc;kBAc/D,MAAM;;SAkBvB"}
@@ -0,0 +1,34 @@
1
+ const INJECT_SCRIPT = `<script src="/vibe-feedback.js"></script>`;
2
+ /**
3
+ * Middleware that intercepts HTML responses from the proxy
4
+ * and injects the vibe-feedback SDK script before </body>.
5
+ */
6
+ export function createHtmlInjector(proxyRes, res) {
7
+ const contentType = proxyRes.headers['content-type'] || '';
8
+ if (!contentType.includes('text/html')) {
9
+ return null; // Not HTML, don't intercept
10
+ }
11
+ // Copy all response headers except content-length (we'll modify the body)
12
+ res.writeHead(proxyRes.statusCode || 200, Object.fromEntries(Object.entries(proxyRes.headers).filter(([key]) => key !== 'content-length')));
13
+ const chunks = [];
14
+ return {
15
+ onData(chunk) {
16
+ chunks.push(chunk);
17
+ },
18
+ onEnd() {
19
+ let html = Buffer.concat(chunks).toString('utf-8');
20
+ // Inject before </body> or at end if no </body>
21
+ if (html.includes('</body>')) {
22
+ html = html.replace('</body>', `${INJECT_SCRIPT}\n</body>`);
23
+ }
24
+ else if (html.includes('</html>')) {
25
+ html = html.replace('</html>', `${INJECT_SCRIPT}\n</html>`);
26
+ }
27
+ else {
28
+ html += `\n${INJECT_SCRIPT}`;
29
+ }
30
+ res.end(html);
31
+ },
32
+ };
33
+ }
34
+ //# sourceMappingURL=html-injector.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"html-injector.js","sourceRoot":"","sources":["../../../src/server/html-injector.ts"],"names":[],"mappings":"AAEA,MAAM,aAAa,GAAG,2CAA2C,CAAC;AAElE;;;GAGG;AACH,MAAM,UAAU,kBAAkB,CAAC,QAAyB,EAAE,GAAmB;IAC/E,MAAM,WAAW,GAAG,QAAQ,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;IAC3D,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;QACvC,OAAO,IAAI,CAAC,CAAC,4BAA4B;IAC3C,CAAC;IAED,0EAA0E;IAC1E,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,UAAU,IAAI,GAAG,EAAE,MAAM,CAAC,WAAW,CAC1D,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC,GAAG,KAAK,gBAAgB,CAAC,CAC7E,CAAC,CAAC;IAEH,MAAM,MAAM,GAAa,EAAE,CAAC;IAE5B,OAAO;QACL,MAAM,CAAC,KAAa;YAClB,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrB,CAAC;QACD,KAAK;YACH,IAAI,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;YAEnD,gDAAgD;YAChD,IAAI,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC7B,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,GAAG,aAAa,WAAW,CAAC,CAAC;YAC9D,CAAC;iBAAM,IAAI,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBACpC,IAAI,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,GAAG,aAAa,WAAW,CAAC,CAAC;YAC9D,CAAC;iBAAM,CAAC;gBACN,IAAI,IAAI,KAAK,aAAa,EAAE,CAAC;YAC/B,CAAC;YAED,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAChB,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,8 @@
1
+ import { FeedbackStore } from '../storage/store.js';
2
+ export declare function startProxyServer(targetPort: number, store: FeedbackStore): Promise<{
3
+ proxyUrl: string;
4
+ targetUrl: string;
5
+ }>;
6
+ export declare function stopProxyServer(): void;
7
+ export declare function isProxyRunning(): boolean;
8
+ //# sourceMappingURL=proxy-server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"proxy-server.d.ts","sourceRoot":"","sources":["../../../src/server/proxy-server.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAUpD,wBAAsB,gBAAgB,CACpC,UAAU,EAAE,MAAM,EAClB,KAAK,EAAE,aAAa,GACnB,OAAO,CAAC;IAAE,QAAQ,EAAE,MAAM,CAAC;IAAC,SAAS,EAAE,MAAM,CAAA;CAAE,CAAC,CA0ElD;AAED,wBAAgB,eAAe,IAAI,IAAI,CAStC;AAED,wBAAgB,cAAc,IAAI,OAAO,CAExC"}
@@ -0,0 +1,87 @@
1
+ import express from 'express';
2
+ import cors from 'cors';
3
+ import httpProxy from 'http-proxy';
4
+ import { createFeedbackRouter } from './feedback-api.js';
5
+ import { createSdkRouter } from './sdk-serve.js';
6
+ import { createHtmlInjector } from './html-injector.js';
7
+ const PROXY_PORT = 3695;
8
+ let server = null;
9
+ let proxy = null;
10
+ export async function startProxyServer(targetPort, store) {
11
+ if (server) {
12
+ throw new Error('Proxy server is already running');
13
+ }
14
+ const targetUrl = `http://localhost:${targetPort}`;
15
+ const proxyUrl = `http://localhost:${PROXY_PORT}`;
16
+ const app = express();
17
+ // CORS for SDK to submit feedback
18
+ app.use(cors());
19
+ // Parse JSON for feedback API
20
+ app.use(express.json({ limit: '10mb' }));
21
+ // Serve SDK JS file
22
+ app.use(createSdkRouter());
23
+ // Feedback API routes
24
+ app.use(createFeedbackRouter(store));
25
+ // Create proxy
26
+ proxy = httpProxy.createProxyServer({
27
+ target: targetUrl,
28
+ ws: true, // WebSocket support for HMR
29
+ changeOrigin: true,
30
+ });
31
+ proxy.on('error', (_err, _req, res) => {
32
+ if (res && 'writeHead' in res) {
33
+ res.writeHead?.(502, { 'Content-Type': 'text/plain' });
34
+ res.end?.('Dev server not responding');
35
+ }
36
+ });
37
+ // Intercept proxy responses to inject SDK into HTML
38
+ proxy.on('proxyRes', (proxyRes, _req, res) => {
39
+ const injector = createHtmlInjector(proxyRes, res);
40
+ if (injector) {
41
+ // HTML response — buffer and inject
42
+ proxyRes.on('data', (chunk) => injector.onData(chunk));
43
+ proxyRes.on('end', () => injector.onEnd());
44
+ }
45
+ else {
46
+ // Non-HTML — pipe through directly
47
+ proxyRes.pipe(res);
48
+ }
49
+ });
50
+ // All other requests → proxy to dev server
51
+ app.all('*', (req, res) => {
52
+ proxy.web(req, res, {
53
+ selfHandleResponse: true, // We handle response in proxyRes event
54
+ });
55
+ });
56
+ // Start server
57
+ await new Promise((resolve, reject) => {
58
+ server = app.listen(PROXY_PORT, () => resolve());
59
+ server.on('error', (err) => {
60
+ if (err.code === 'EADDRINUSE') {
61
+ reject(new Error(`Port ${PROXY_PORT} is already in use. Stop the other process or restart.`));
62
+ }
63
+ else {
64
+ reject(err);
65
+ }
66
+ });
67
+ });
68
+ // Handle WebSocket upgrade for HMR
69
+ server.on('upgrade', (req, socket, head) => {
70
+ proxy.ws(req, socket, head);
71
+ });
72
+ return { proxyUrl, targetUrl };
73
+ }
74
+ export function stopProxyServer() {
75
+ if (proxy) {
76
+ proxy.close();
77
+ proxy = null;
78
+ }
79
+ if (server) {
80
+ server.close();
81
+ server = null;
82
+ }
83
+ }
84
+ export function isProxyRunning() {
85
+ return server !== null;
86
+ }
87
+ //# sourceMappingURL=proxy-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"proxy-server.js","sourceRoot":"","sources":["../../../src/server/proxy-server.ts"],"names":[],"mappings":"AAAA,OAAO,OAAO,MAAM,SAAS,CAAC;AAC9B,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,SAAS,MAAM,YAAY,CAAC;AAGnC,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,kBAAkB,EAAE,MAAM,oBAAoB,CAAC;AAExD,MAAM,UAAU,GAAG,IAAI,CAAC;AAExB,IAAI,MAAM,GAAkB,IAAI,CAAC;AACjC,IAAI,KAAK,GAAqB,IAAI,CAAC;AAEnC,MAAM,CAAC,KAAK,UAAU,gBAAgB,CACpC,UAAkB,EAClB,KAAoB;IAEpB,IAAI,MAAM,EAAE,CAAC;QACX,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;IACrD,CAAC;IAED,MAAM,SAAS,GAAG,oBAAoB,UAAU,EAAE,CAAC;IACnD,MAAM,QAAQ,GAAG,oBAAoB,UAAU,EAAE,CAAC;IAElD,MAAM,GAAG,GAAG,OAAO,EAAE,CAAC;IAEtB,kCAAkC;IAClC,GAAG,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,CAAC;IAEhB,8BAA8B;IAC9B,GAAG,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,CAAC;IAEzC,oBAAoB;IACpB,GAAG,CAAC,GAAG,CAAC,eAAe,EAAE,CAAC,CAAC;IAE3B,sBAAsB;IACtB,GAAG,CAAC,GAAG,CAAC,oBAAoB,CAAC,KAAK,CAAC,CAAC,CAAC;IAErC,eAAe;IACf,KAAK,GAAG,SAAS,CAAC,iBAAiB,CAAC;QAClC,MAAM,EAAE,SAAS;QACjB,EAAE,EAAE,IAAI,EAAE,4BAA4B;QACtC,YAAY,EAAE,IAAI;KACnB,CAAC,CAAC;IAEH,KAAK,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,EAAE,IAAI,EAAE,GAAG,EAAE,EAAE;QACpC,IAAI,GAAG,IAAI,WAAW,IAAI,GAAG,EAAE,CAAC;YAC7B,GAAW,CAAC,SAAS,EAAE,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,YAAY,EAAE,CAAC,CAAC;YAC/D,GAAW,CAAC,GAAG,EAAE,CAAC,2BAA2B,CAAC,CAAC;QAClD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,oDAAoD;IACpD,KAAK,CAAC,EAAE,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,IAAI,EAAE,GAAG,EAAE,EAAE;QAC3C,MAAM,QAAQ,GAAG,kBAAkB,CAAC,QAAQ,EAAE,GAAU,CAAC,CAAC;QAC1D,IAAI,QAAQ,EAAE,CAAC;YACb,oCAAoC;YACpC,QAAQ,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;YACvD,QAAQ,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC,CAAC;QAC7C,CAAC;aAAM,CAAC;YACN,mCAAmC;YACnC,QAAQ,CAAC,IAAI,CAAC,GAAU,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,2CAA2C;IAC3C,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACxB,KAAM,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE;YACnB,kBAAkB,EAAE,IAAI,EAAE,uCAAuC;SAClE,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,eAAe;IACf,MAAM,IAAI,OAAO,CAAO,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QAC1C,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,UAAU,EAAE,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC;QACjD,MAAO,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAA0B,EAAE,EAAE;YACjD,IAAI,GAAG,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;gBAC9B,MAAM,CAAC,IAAI,KAAK,CAAC,QAAQ,UAAU,wDAAwD,CAAC,CAAC,CAAC;YAChG,CAAC;iBAAM,CAAC;gBACN,MAAM,CAAC,GAAG,CAAC,CAAC;YACd,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,mCAAmC;IACnC,MAAO,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,GAAQ,EAAE,MAAW,EAAE,IAAS,EAAE,EAAE;QACzD,KAAM,CAAC,EAAE,CAAC,GAAG,EAAE,MAAM,EAAE,IAAI,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,CAAC;AACjC,CAAC;AAED,MAAM,UAAU,eAAe;IAC7B,IAAI,KAAK,EAAE,CAAC;QACV,KAAK,CAAC,KAAK,EAAE,CAAC;QACd,KAAK,GAAG,IAAI,CAAC;IACf,CAAC;IACD,IAAI,MAAM,EAAE,CAAC;QACX,MAAM,CAAC,KAAK,EAAE,CAAC;QACf,MAAM,GAAG,IAAI,CAAC;IAChB,CAAC;AACH,CAAC;AAED,MAAM,UAAU,cAAc;IAC5B,OAAO,MAAM,KAAK,IAAI,CAAC;AACzB,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Router } from 'express';
2
+ export declare function createSdkRouter(): Router;
3
+ //# sourceMappingURL=sdk-serve.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sdk-serve.d.ts","sourceRoot":"","sources":["../../../src/server/sdk-serve.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAKjC,wBAAgB,eAAe,IAAI,MAAM,CAkBxC"}
@@ -0,0 +1,20 @@
1
+ import { Router } from 'express';
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import { fileURLToPath } from 'url';
5
+ export function createSdkRouter() {
6
+ const router = Router();
7
+ router.get('/vibe-feedback.js', (_req, res) => {
8
+ // Resolve SDK path relative to this file's location
9
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
10
+ const sdkPath = path.resolve(__dirname, '../../dist/vibe-feedback.js');
11
+ if (!fs.existsSync(sdkPath)) {
12
+ return res.status(404).send('// SDK not built yet');
13
+ }
14
+ res.setHeader('Content-Type', 'application/javascript');
15
+ res.setHeader('Cache-Control', 'no-cache');
16
+ fs.createReadStream(sdkPath).pipe(res);
17
+ });
18
+ return router;
19
+ }
20
+ //# sourceMappingURL=sdk-serve.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sdk-serve.js","sourceRoot":"","sources":["../../../src/server/sdk-serve.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,aAAa,EAAE,MAAM,KAAK,CAAC;AAEpC,MAAM,UAAU,eAAe;IAC7B,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC;IAExB,MAAM,CAAC,GAAG,CAAC,mBAAmB,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE;QAC5C,oDAAoD;QACpD,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;QAC/D,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,6BAA6B,CAAC,CAAC;QAEvE,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;YAC5B,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;QACtD,CAAC;QAED,GAAG,CAAC,SAAS,CAAC,cAAc,EAAE,wBAAwB,CAAC,CAAC;QACxD,GAAG,CAAC,SAAS,CAAC,eAAe,EAAE,UAAU,CAAC,CAAC;QAC3C,EAAE,CAAC,gBAAgB,CAAC,OAAO,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,OAAO,MAAM,CAAC;AAChB,CAAC"}
@@ -0,0 +1,21 @@
1
+ import type { FeedbackItem } from './types.js';
2
+ export declare class FeedbackStore {
3
+ private indexPath;
4
+ constructor();
5
+ init(): Promise<void>;
6
+ create(data: Omit<FeedbackItem, 'id' | 'createdAt' | 'status'>, screenshot?: Buffer): Promise<FeedbackItem>;
7
+ list(options?: {
8
+ limit?: number;
9
+ status?: string;
10
+ type?: string;
11
+ }): Promise<{
12
+ items: FeedbackItem[];
13
+ total: number;
14
+ }>;
15
+ getById(id: string): Promise<FeedbackItem | null>;
16
+ getScreenshot(id: string): Promise<Buffer | null>;
17
+ updateStatus(id: string, status: 'open' | 'verify' | 'resolved'): Promise<void>;
18
+ clear(): Promise<void>;
19
+ private readIndex;
20
+ }
21
+ //# sourceMappingURL=store.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"store.d.ts","sourceRoot":"","sources":["../../../src/storage/store.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAc/C,qBAAa,aAAa;IACxB,OAAO,CAAC,SAAS,CAAS;;IAMpB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IASrB,MAAM,CACV,IAAI,EAAE,IAAI,CAAC,YAAY,EAAE,IAAI,GAAG,WAAW,GAAG,QAAQ,CAAC,EACvD,UAAU,CAAC,EAAE,MAAM,GAClB,OAAO,CAAC,YAAY,CAAC;IA8ClB,IAAI,CAAC,OAAO,CAAC,EAAE;QACnB,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,IAAI,CAAC,EAAE,MAAM,CAAC;KACf,GAAG,OAAO,CAAC;QAAE,KAAK,EAAE,YAAY,EAAE,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC;IAyB/C,OAAO,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;IAajD,aAAa,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IASjD,YAAY,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,QAAQ,GAAG,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC;IAqB/E,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;YASd,SAAS;CAQxB"}