@mp3wizard/figma-console-mcp 1.14.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 (201) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +816 -0
  3. package/dist/apps/design-system-dashboard/scoring/accessibility.d.ts +14 -0
  4. package/dist/apps/design-system-dashboard/scoring/accessibility.d.ts.map +1 -0
  5. package/dist/apps/design-system-dashboard/scoring/accessibility.js +278 -0
  6. package/dist/apps/design-system-dashboard/scoring/accessibility.js.map +1 -0
  7. package/dist/apps/design-system-dashboard/scoring/component-metadata.d.ts +29 -0
  8. package/dist/apps/design-system-dashboard/scoring/component-metadata.d.ts.map +1 -0
  9. package/dist/apps/design-system-dashboard/scoring/component-metadata.js +358 -0
  10. package/dist/apps/design-system-dashboard/scoring/component-metadata.js.map +1 -0
  11. package/dist/apps/design-system-dashboard/scoring/consistency.d.ts +14 -0
  12. package/dist/apps/design-system-dashboard/scoring/consistency.d.ts.map +1 -0
  13. package/dist/apps/design-system-dashboard/scoring/consistency.js +342 -0
  14. package/dist/apps/design-system-dashboard/scoring/consistency.js.map +1 -0
  15. package/dist/apps/design-system-dashboard/scoring/coverage.d.ts +14 -0
  16. package/dist/apps/design-system-dashboard/scoring/coverage.d.ts.map +1 -0
  17. package/dist/apps/design-system-dashboard/scoring/coverage.js +231 -0
  18. package/dist/apps/design-system-dashboard/scoring/coverage.js.map +1 -0
  19. package/dist/apps/design-system-dashboard/scoring/engine.d.ts +27 -0
  20. package/dist/apps/design-system-dashboard/scoring/engine.d.ts.map +1 -0
  21. package/dist/apps/design-system-dashboard/scoring/engine.js +93 -0
  22. package/dist/apps/design-system-dashboard/scoring/engine.js.map +1 -0
  23. package/dist/apps/design-system-dashboard/scoring/naming-semantics.d.ts +14 -0
  24. package/dist/apps/design-system-dashboard/scoring/naming-semantics.d.ts.map +1 -0
  25. package/dist/apps/design-system-dashboard/scoring/naming-semantics.js +309 -0
  26. package/dist/apps/design-system-dashboard/scoring/naming-semantics.js.map +1 -0
  27. package/dist/apps/design-system-dashboard/scoring/token-architecture.d.ts +14 -0
  28. package/dist/apps/design-system-dashboard/scoring/token-architecture.d.ts.map +1 -0
  29. package/dist/apps/design-system-dashboard/scoring/token-architecture.js +350 -0
  30. package/dist/apps/design-system-dashboard/scoring/token-architecture.js.map +1 -0
  31. package/dist/apps/design-system-dashboard/scoring/types.d.ts +89 -0
  32. package/dist/apps/design-system-dashboard/scoring/types.d.ts.map +1 -0
  33. package/dist/apps/design-system-dashboard/scoring/types.js +41 -0
  34. package/dist/apps/design-system-dashboard/scoring/types.js.map +1 -0
  35. package/dist/apps/design-system-dashboard/server.d.ts +24 -0
  36. package/dist/apps/design-system-dashboard/server.d.ts.map +1 -0
  37. package/dist/apps/design-system-dashboard/server.js +160 -0
  38. package/dist/apps/design-system-dashboard/server.js.map +1 -0
  39. package/dist/apps/token-browser/server.d.ts +26 -0
  40. package/dist/apps/token-browser/server.d.ts.map +1 -0
  41. package/dist/apps/token-browser/server.js +137 -0
  42. package/dist/apps/token-browser/server.js.map +1 -0
  43. package/dist/browser/base.d.ts +58 -0
  44. package/dist/browser/base.d.ts.map +1 -0
  45. package/dist/browser/base.js +6 -0
  46. package/dist/browser/base.js.map +1 -0
  47. package/dist/browser/local.d.ts +87 -0
  48. package/dist/browser/local.d.ts.map +1 -0
  49. package/dist/browser/local.js +318 -0
  50. package/dist/browser/local.js.map +1 -0
  51. package/dist/cloudflare/apps/design-system-dashboard/scoring/accessibility.js +277 -0
  52. package/dist/cloudflare/apps/design-system-dashboard/scoring/component-metadata.js +357 -0
  53. package/dist/cloudflare/apps/design-system-dashboard/scoring/consistency.js +341 -0
  54. package/dist/cloudflare/apps/design-system-dashboard/scoring/coverage.js +230 -0
  55. package/dist/cloudflare/apps/design-system-dashboard/scoring/engine.js +92 -0
  56. package/dist/cloudflare/apps/design-system-dashboard/scoring/naming-semantics.js +308 -0
  57. package/dist/cloudflare/apps/design-system-dashboard/scoring/token-architecture.js +349 -0
  58. package/dist/cloudflare/apps/design-system-dashboard/scoring/types.js +40 -0
  59. package/dist/cloudflare/apps/design-system-dashboard/server.js +159 -0
  60. package/dist/cloudflare/apps/token-browser/server.js +136 -0
  61. package/dist/cloudflare/browser/base.js +5 -0
  62. package/dist/cloudflare/browser/cloudflare.js +156 -0
  63. package/dist/cloudflare/browser-manager.js +157 -0
  64. package/dist/cloudflare/core/cloud-websocket-connector.js +267 -0
  65. package/dist/cloudflare/core/cloud-websocket-relay.js +199 -0
  66. package/dist/cloudflare/core/comment-tools.js +292 -0
  67. package/dist/cloudflare/core/config.js +161 -0
  68. package/dist/cloudflare/core/console-monitor.js +427 -0
  69. package/dist/cloudflare/core/design-code-tools.js +2504 -0
  70. package/dist/cloudflare/core/design-system-manifest.js +260 -0
  71. package/dist/cloudflare/core/design-system-tools.js +863 -0
  72. package/dist/cloudflare/core/enrichment/enrichment-service.js +272 -0
  73. package/dist/cloudflare/core/enrichment/index.js +7 -0
  74. package/dist/cloudflare/core/enrichment/relationship-mapper.js +351 -0
  75. package/dist/cloudflare/core/enrichment/style-resolver.js +326 -0
  76. package/dist/cloudflare/core/figma-api.js +409 -0
  77. package/dist/cloudflare/core/figma-connector.js +7 -0
  78. package/dist/cloudflare/core/figma-desktop-connector.js +1184 -0
  79. package/dist/cloudflare/core/figma-reconstruction-spec.js +402 -0
  80. package/dist/cloudflare/core/figma-style-extractor.js +311 -0
  81. package/dist/cloudflare/core/figma-tools.js +2947 -0
  82. package/dist/cloudflare/core/logger.js +53 -0
  83. package/dist/cloudflare/core/port-discovery.js +282 -0
  84. package/dist/cloudflare/core/snippet-injector.js +96 -0
  85. package/dist/cloudflare/core/types/design-code.js +4 -0
  86. package/dist/cloudflare/core/types/enriched.js +5 -0
  87. package/dist/cloudflare/core/types/index.js +4 -0
  88. package/dist/cloudflare/core/websocket-connector.js +256 -0
  89. package/dist/cloudflare/core/websocket-server.js +646 -0
  90. package/dist/cloudflare/core/write-tools.js +2091 -0
  91. package/dist/cloudflare/index.js +2899 -0
  92. package/dist/cloudflare/test-browser.js +88 -0
  93. package/dist/core/comment-tools.d.ts +11 -0
  94. package/dist/core/comment-tools.d.ts.map +1 -0
  95. package/dist/core/comment-tools.js +293 -0
  96. package/dist/core/comment-tools.js.map +1 -0
  97. package/dist/core/config.d.ts +17 -0
  98. package/dist/core/config.d.ts.map +1 -0
  99. package/dist/core/config.js +162 -0
  100. package/dist/core/config.js.map +1 -0
  101. package/dist/core/console-monitor.d.ts +82 -0
  102. package/dist/core/console-monitor.d.ts.map +1 -0
  103. package/dist/core/console-monitor.js +428 -0
  104. package/dist/core/console-monitor.js.map +1 -0
  105. package/dist/core/design-code-tools.d.ts +127 -0
  106. package/dist/core/design-code-tools.d.ts.map +1 -0
  107. package/dist/core/design-code-tools.js +2505 -0
  108. package/dist/core/design-code-tools.js.map +1 -0
  109. package/dist/core/design-system-manifest.d.ts +272 -0
  110. package/dist/core/design-system-manifest.d.ts.map +1 -0
  111. package/dist/core/design-system-manifest.js +261 -0
  112. package/dist/core/design-system-manifest.js.map +1 -0
  113. package/dist/core/design-system-tools.d.ts +17 -0
  114. package/dist/core/design-system-tools.d.ts.map +1 -0
  115. package/dist/core/design-system-tools.js +864 -0
  116. package/dist/core/design-system-tools.js.map +1 -0
  117. package/dist/core/enrichment/enrichment-service.d.ts +52 -0
  118. package/dist/core/enrichment/enrichment-service.d.ts.map +1 -0
  119. package/dist/core/enrichment/enrichment-service.js +273 -0
  120. package/dist/core/enrichment/enrichment-service.js.map +1 -0
  121. package/dist/core/enrichment/index.d.ts +8 -0
  122. package/dist/core/enrichment/index.d.ts.map +1 -0
  123. package/dist/core/enrichment/index.js +8 -0
  124. package/dist/core/enrichment/index.js.map +1 -0
  125. package/dist/core/enrichment/relationship-mapper.d.ts +106 -0
  126. package/dist/core/enrichment/relationship-mapper.d.ts.map +1 -0
  127. package/dist/core/enrichment/relationship-mapper.js +352 -0
  128. package/dist/core/enrichment/relationship-mapper.js.map +1 -0
  129. package/dist/core/enrichment/style-resolver.d.ts +80 -0
  130. package/dist/core/enrichment/style-resolver.d.ts.map +1 -0
  131. package/dist/core/enrichment/style-resolver.js +327 -0
  132. package/dist/core/enrichment/style-resolver.js.map +1 -0
  133. package/dist/core/figma-api.d.ts +201 -0
  134. package/dist/core/figma-api.d.ts.map +1 -0
  135. package/dist/core/figma-api.js +410 -0
  136. package/dist/core/figma-api.js.map +1 -0
  137. package/dist/core/figma-connector.d.ts +48 -0
  138. package/dist/core/figma-connector.d.ts.map +1 -0
  139. package/dist/core/figma-connector.js +8 -0
  140. package/dist/core/figma-connector.js.map +1 -0
  141. package/dist/core/figma-desktop-connector.d.ts +265 -0
  142. package/dist/core/figma-desktop-connector.d.ts.map +1 -0
  143. package/dist/core/figma-desktop-connector.js +1184 -0
  144. package/dist/core/figma-desktop-connector.js.map +1 -0
  145. package/dist/core/figma-reconstruction-spec.d.ts +166 -0
  146. package/dist/core/figma-reconstruction-spec.d.ts.map +1 -0
  147. package/dist/core/figma-reconstruction-spec.js +403 -0
  148. package/dist/core/figma-reconstruction-spec.js.map +1 -0
  149. package/dist/core/figma-style-extractor.d.ts +76 -0
  150. package/dist/core/figma-style-extractor.d.ts.map +1 -0
  151. package/dist/core/figma-style-extractor.js +312 -0
  152. package/dist/core/figma-style-extractor.js.map +1 -0
  153. package/dist/core/figma-tools.d.ts +23 -0
  154. package/dist/core/figma-tools.d.ts.map +1 -0
  155. package/dist/core/figma-tools.js +2948 -0
  156. package/dist/core/figma-tools.js.map +1 -0
  157. package/dist/core/logger.d.ts +22 -0
  158. package/dist/core/logger.d.ts.map +1 -0
  159. package/dist/core/logger.js +54 -0
  160. package/dist/core/logger.js.map +1 -0
  161. package/dist/core/port-discovery.d.ts +110 -0
  162. package/dist/core/port-discovery.d.ts.map +1 -0
  163. package/dist/core/port-discovery.js +283 -0
  164. package/dist/core/port-discovery.js.map +1 -0
  165. package/dist/core/snippet-injector.d.ts +24 -0
  166. package/dist/core/snippet-injector.d.ts.map +1 -0
  167. package/dist/core/snippet-injector.js +97 -0
  168. package/dist/core/snippet-injector.js.map +1 -0
  169. package/dist/core/types/design-code.d.ts +262 -0
  170. package/dist/core/types/design-code.d.ts.map +1 -0
  171. package/dist/core/types/design-code.js +5 -0
  172. package/dist/core/types/design-code.js.map +1 -0
  173. package/dist/core/types/enriched.d.ts +213 -0
  174. package/dist/core/types/enriched.d.ts.map +1 -0
  175. package/dist/core/types/enriched.js +6 -0
  176. package/dist/core/types/enriched.js.map +1 -0
  177. package/dist/core/types/index.d.ts +112 -0
  178. package/dist/core/types/index.d.ts.map +1 -0
  179. package/dist/core/types/index.js +5 -0
  180. package/dist/core/types/index.js.map +1 -0
  181. package/dist/core/websocket-connector.d.ts +55 -0
  182. package/dist/core/websocket-connector.d.ts.map +1 -0
  183. package/dist/core/websocket-connector.js +257 -0
  184. package/dist/core/websocket-connector.js.map +1 -0
  185. package/dist/core/websocket-server.d.ts +191 -0
  186. package/dist/core/websocket-server.d.ts.map +1 -0
  187. package/dist/core/websocket-server.js +647 -0
  188. package/dist/core/websocket-server.js.map +1 -0
  189. package/dist/core/write-tools.d.ts +7 -0
  190. package/dist/core/write-tools.d.ts.map +1 -0
  191. package/dist/core/write-tools.js +2092 -0
  192. package/dist/core/write-tools.js.map +1 -0
  193. package/dist/local.d.ts +84 -0
  194. package/dist/local.d.ts.map +1 -0
  195. package/dist/local.js +5039 -0
  196. package/dist/local.js.map +1 -0
  197. package/figma-desktop-bridge/README.md +313 -0
  198. package/figma-desktop-bridge/code.js +2818 -0
  199. package/figma-desktop-bridge/manifest.json +67 -0
  200. package/figma-desktop-bridge/ui.html +1236 -0
  201. package/package.json +87 -0
@@ -0,0 +1,427 @@
1
+ /**
2
+ * Console Monitor
3
+ * Captures and manages console logs from Figma plugins
4
+ * Monitors both main page console AND Web Worker consoles (where Figma plugins run)
5
+ */
6
+ import { createChildLogger } from './logger.js';
7
+ const logger = createChildLogger({ component: 'console-monitor' });
8
+ /**
9
+ * Console Monitor
10
+ * Listens to page console events and maintains a circular buffer of logs
11
+ * Also monitors Web Workers to capture Figma plugin console logs
12
+ */
13
+ export class ConsoleMonitor {
14
+ constructor(config) {
15
+ this.logs = [];
16
+ this.isMonitoring = false;
17
+ this.page = null; // Supports both puppeteer-core and @cloudflare/puppeteer
18
+ this.workers = new Set();
19
+ this.lastUrl = ''; // Track the last URL to detect real navigations vs hash changes
20
+ this.config = config;
21
+ }
22
+ /**
23
+ * Start monitoring console logs on a page
24
+ * Accepts any puppeteer Page type (puppeteer-core or @cloudflare/puppeteer)
25
+ */
26
+ async startMonitoring(page) {
27
+ if (this.isMonitoring && this.page === page) {
28
+ logger.info('Already monitoring this page');
29
+ return;
30
+ }
31
+ this.page = page;
32
+ this.isMonitoring = true;
33
+ logger.info('Starting console monitoring (page + workers + frames)');
34
+ // DIAGNOSTIC: Log all frames on the page and add to console
35
+ const frames = page.frames();
36
+ logger.info({ frameCount: frames.length }, 'Frames detected on page');
37
+ // Add diagnostic marker to console logs
38
+ this.addLog({
39
+ timestamp: Date.now(),
40
+ level: 'info',
41
+ message: `[MCP DIAGNOSTIC] Monitoring started. Detected ${frames.length} frames and ${page.workers().length} workers.`,
42
+ args: [],
43
+ source: 'page',
44
+ });
45
+ for (const frame of frames) {
46
+ const frameUrl = frame.url();
47
+ const frameName = frame.name() || 'unnamed';
48
+ logger.info({
49
+ frameUrl,
50
+ isDetached: frame.isDetached(),
51
+ name: frameName
52
+ }, 'Frame details');
53
+ // Add frame detection to console logs
54
+ this.addLog({
55
+ timestamp: Date.now(),
56
+ level: 'info',
57
+ message: `[MCP DIAGNOSTIC] Frame detected: ${frameName} - ${frameUrl}`,
58
+ args: [],
59
+ source: 'page',
60
+ });
61
+ }
62
+ // Listen to ALL console events (includes main page, iframes, and workers)
63
+ page.on('console', async (msg) => {
64
+ try {
65
+ const location = msg.location();
66
+ const url = location?.url || 'unknown';
67
+ const text = msg.text();
68
+ const type = msg.type();
69
+ // DIAGNOSTIC: Log every console event with its source
70
+ logger.info({
71
+ type,
72
+ url,
73
+ textPreview: text.substring(0, 100),
74
+ location
75
+ }, 'Console event captured');
76
+ const entry = await this.processConsoleMessage(msg, 'page', url);
77
+ if (entry) {
78
+ this.addLog(entry);
79
+ }
80
+ }
81
+ catch (error) {
82
+ logger.error({ error }, 'Failed to process console message');
83
+ }
84
+ });
85
+ // Listen to page errors
86
+ page.on('pageerror', (error) => {
87
+ this.addLog({
88
+ timestamp: Date.now(),
89
+ level: 'error',
90
+ message: error.message,
91
+ args: [],
92
+ stackTrace: {
93
+ callFrames: error.stack
94
+ ? error.stack.split('\n').map((line) => ({
95
+ functionName: line.trim(),
96
+ url: '',
97
+ lineNumber: 0,
98
+ columnNumber: 0,
99
+ }))
100
+ : [],
101
+ },
102
+ source: 'page',
103
+ });
104
+ });
105
+ // Monitor existing workers (for Figma plugin console logs)
106
+ const existingWorkers = page.workers();
107
+ logger.info({ workerCount: existingWorkers.length }, 'Found existing workers');
108
+ for (const worker of existingWorkers) {
109
+ this.attachWorkerListeners(worker);
110
+ }
111
+ // Listen for new workers being created (e.g., when plugin starts)
112
+ page.on('workercreated', (worker) => {
113
+ logger.info({ workerUrl: worker.url() }, 'New worker created');
114
+ this.attachWorkerListeners(worker);
115
+ });
116
+ // Listen for workers being destroyed
117
+ page.on('workerdestroyed', (worker) => {
118
+ logger.info({ workerUrl: worker.url() }, 'Worker destroyed');
119
+ this.workers.delete(worker);
120
+ });
121
+ // Listen for new frames being attached (e.g., when plugin UI loads)
122
+ page.on('frameattached', (frame) => {
123
+ const frameUrl = frame.url();
124
+ const frameName = frame.name() || 'unnamed';
125
+ logger.info({ frameUrl, frameName }, 'New frame attached');
126
+ // Add diagnostic marker for new frame
127
+ this.addLog({
128
+ timestamp: Date.now(),
129
+ level: 'info',
130
+ message: `[MCP DIAGNOSTIC] New frame attached: ${frameName} - ${frameUrl}`,
131
+ args: [],
132
+ source: 'page',
133
+ });
134
+ });
135
+ // Listen for frames being detached
136
+ page.on('framedetached', (frame) => {
137
+ logger.info({ frameUrl: frame.url() }, 'Frame detached');
138
+ });
139
+ // Listen for main frame navigation - clear logs when navigating to a DIFFERENT page
140
+ // (not just hash changes or SPA navigations within the same Figma file)
141
+ page.on('framenavigated', (frame) => {
142
+ // Only handle main frame navigation (not iframe navigations)
143
+ if (frame === page.mainFrame()) {
144
+ const frameUrl = frame.url();
145
+ // Extract base URL without hash/query params to detect real navigation
146
+ const getBaseUrl = (url) => {
147
+ try {
148
+ const urlObj = new URL(url);
149
+ // For Figma, consider the file ID part of the path
150
+ // Example: https://figma.com/design/FILE_ID/...
151
+ // Only clear if the file ID changes
152
+ return urlObj.origin + urlObj.pathname.split('?')[0].split('#')[0];
153
+ }
154
+ catch {
155
+ return url;
156
+ }
157
+ };
158
+ const currentBaseUrl = getBaseUrl(frameUrl);
159
+ const previousBaseUrl = this.lastUrl ? getBaseUrl(this.lastUrl) : '';
160
+ // Only clear logs if we've actually navigated to a different file
161
+ // (not just hash changes like going from one component to another)
162
+ if (previousBaseUrl && currentBaseUrl !== previousBaseUrl) {
163
+ logger.info({
164
+ from: previousBaseUrl,
165
+ to: currentBaseUrl
166
+ }, 'Navigated to different file - clearing console logs');
167
+ // Clear old logs to prevent stale data from previous file
168
+ this.logs = [];
169
+ // Add diagnostic marker for navigation
170
+ this.addLog({
171
+ timestamp: Date.now(),
172
+ level: 'info',
173
+ message: `[MCP DIAGNOSTIC] Navigated to new file: ${frameUrl}. Console logs cleared.`,
174
+ args: [],
175
+ source: 'page',
176
+ });
177
+ }
178
+ // Update last URL for next comparison
179
+ this.lastUrl = frameUrl;
180
+ }
181
+ });
182
+ logger.info({
183
+ pageMonitoring: true,
184
+ workerMonitoring: true,
185
+ frameMonitoring: true,
186
+ initialWorkerCount: existingWorkers.length,
187
+ initialFrameCount: frames.length
188
+ }, 'Console monitoring started');
189
+ }
190
+ /**
191
+ * Attach console listeners to a Web Worker
192
+ * This captures Figma plugin console logs
193
+ */
194
+ attachWorkerListeners(worker) {
195
+ this.workers.add(worker);
196
+ const workerUrl = worker.url();
197
+ logger.info({ workerUrl }, 'Attaching console listener to worker');
198
+ // DIAGNOSTIC: Log a marker when worker listener is attached
199
+ this.addLog({
200
+ timestamp: Date.now(),
201
+ level: 'info',
202
+ message: `[MCP] Worker detected and monitored: ${workerUrl}`,
203
+ args: [],
204
+ source: 'plugin',
205
+ workerUrl,
206
+ });
207
+ // Listen to worker console events
208
+ worker.on('console', async (msg) => {
209
+ try {
210
+ logger.info({ workerUrl, type: msg.type(), text: msg.text() }, 'Worker console event received');
211
+ const entry = await this.processConsoleMessage(msg, 'worker', workerUrl);
212
+ if (entry) {
213
+ this.addLog(entry);
214
+ }
215
+ }
216
+ catch (error) {
217
+ logger.error({ error, workerUrl }, 'Failed to process worker console message');
218
+ }
219
+ });
220
+ // Worker doesn't have pageerror, but console.error will be captured above
221
+ }
222
+ /**
223
+ * Process console message from Puppeteer
224
+ * @param msg - Console message from page or worker
225
+ * @param context - Where the message came from ('page' or 'worker')
226
+ * @param workerUrl - URL of the worker (if context is 'worker')
227
+ */
228
+ async processConsoleMessage(msg, context, workerUrl) {
229
+ const level = msg.type();
230
+ // Filter by configured levels
231
+ if (this.config.filterLevels.length > 0 &&
232
+ !this.config.filterLevels.includes(level)) {
233
+ return null;
234
+ }
235
+ try {
236
+ // Extract message text
237
+ const message = msg.text();
238
+ // Extract arguments (with truncation)
239
+ const args = await Promise.all(msg.args().map(async (arg) => {
240
+ try {
241
+ const jsonValue = await arg.jsonValue();
242
+ return this.truncateValue(jsonValue);
243
+ }
244
+ catch {
245
+ return String(arg);
246
+ }
247
+ }));
248
+ // Determine source based on context
249
+ let source;
250
+ if (context === 'worker') {
251
+ // Workers are where Figma plugins run
252
+ source = 'plugin';
253
+ }
254
+ else {
255
+ // Page console - determine if plugin or figma
256
+ const location = msg.location();
257
+ source = this.determineSource(location?.url);
258
+ }
259
+ const entry = {
260
+ timestamp: Date.now(),
261
+ level,
262
+ message: this.truncateString(message),
263
+ args,
264
+ source,
265
+ };
266
+ // Add worker URL metadata for debugging
267
+ if (workerUrl && context === 'worker') {
268
+ entry.workerUrl = workerUrl;
269
+ }
270
+ // Add stack trace for errors
271
+ if (level === 'error' && msg.stackTrace) {
272
+ entry.stackTrace = {
273
+ callFrames: msg.stackTrace().callFrames || [],
274
+ };
275
+ }
276
+ return entry;
277
+ }
278
+ catch (error) {
279
+ logger.error({ error, context, workerUrl }, 'Failed to extract console message details');
280
+ return null;
281
+ }
282
+ }
283
+ /**
284
+ * Determine if log is from plugin or Figma based on URL
285
+ */
286
+ determineSource(url) {
287
+ if (!url)
288
+ return 'unknown';
289
+ // Check for plugin-related URLs
290
+ // Plugins might run in:
291
+ // - iframes with plugin-specific URLs
292
+ // - blob: URLs created by the plugin
293
+ // - chrome-extension: URLs (for dev mode)
294
+ // - any URL containing "plugin"
295
+ if (url.includes('plugin') ||
296
+ url.includes('iframe') ||
297
+ url.startsWith('blob:') ||
298
+ url.startsWith('chrome-extension:') ||
299
+ url.includes('figma.com/plugin') ||
300
+ url.includes('/plugin-')) {
301
+ return 'plugin';
302
+ }
303
+ // Main Figma application
304
+ if (url.includes('figma.com')) {
305
+ return 'figma';
306
+ }
307
+ return 'unknown';
308
+ }
309
+ /**
310
+ * Add log to circular buffer
311
+ */
312
+ addLog(entry) {
313
+ this.logs.push(entry);
314
+ // Maintain buffer size
315
+ if (this.logs.length > this.config.bufferSize) {
316
+ this.logs.shift();
317
+ }
318
+ logger.debug({ level: entry.level, source: entry.source }, 'Log captured');
319
+ }
320
+ /**
321
+ * Truncate string to max length
322
+ */
323
+ truncateString(str) {
324
+ const maxLength = this.config.truncation.maxStringLength;
325
+ if (str.length <= maxLength) {
326
+ return str;
327
+ }
328
+ return str.substring(0, maxLength) + '... (truncated)';
329
+ }
330
+ /**
331
+ * Truncate value (string, array, object) intelligently
332
+ * Based on AgentDesk pattern to prevent context overflow
333
+ */
334
+ truncateValue(value, depth = 0) {
335
+ const { maxStringLength, maxArrayLength, maxObjectDepth } = this.config.truncation;
336
+ // Max depth reached
337
+ if (depth >= maxObjectDepth) {
338
+ return '[Max depth reached]';
339
+ }
340
+ // Handle null/undefined
341
+ if (value === null || value === undefined) {
342
+ return value;
343
+ }
344
+ // Handle strings
345
+ if (typeof value === 'string') {
346
+ return this.truncateString(value);
347
+ }
348
+ // Handle arrays
349
+ if (Array.isArray(value)) {
350
+ const truncated = value.slice(0, maxArrayLength).map((item) => this.truncateValue(item, depth + 1));
351
+ if (value.length > maxArrayLength) {
352
+ truncated.push(`... (${value.length - maxArrayLength} more items)`);
353
+ }
354
+ return truncated;
355
+ }
356
+ // Handle objects
357
+ if (typeof value === 'object') {
358
+ const result = {};
359
+ let count = 0;
360
+ for (const [key, val] of Object.entries(value)) {
361
+ if (count >= 10) {
362
+ // Limit object properties
363
+ result['...'] = '(more properties)';
364
+ break;
365
+ }
366
+ result[key] = this.truncateValue(val, depth + 1);
367
+ count++;
368
+ }
369
+ return result;
370
+ }
371
+ // Primitives (number, boolean, etc.)
372
+ return value;
373
+ }
374
+ /**
375
+ * Get logs with optional filtering
376
+ */
377
+ getLogs(options) {
378
+ let filtered = [...this.logs];
379
+ // Filter by timestamp
380
+ if (options?.since) {
381
+ filtered = filtered.filter((log) => log.timestamp >= options.since);
382
+ }
383
+ // Filter by level
384
+ if (options?.level && options.level !== 'all') {
385
+ filtered = filtered.filter((log) => log.level === options.level);
386
+ }
387
+ // Limit count (get most recent)
388
+ if (options?.count) {
389
+ filtered = filtered.slice(-options.count);
390
+ }
391
+ return filtered;
392
+ }
393
+ /**
394
+ * Clear log buffer
395
+ */
396
+ clear() {
397
+ const count = this.logs.length;
398
+ this.logs = [];
399
+ logger.info({ clearedCount: count }, 'Console buffer cleared');
400
+ return count;
401
+ }
402
+ /**
403
+ * Stop monitoring
404
+ */
405
+ stopMonitoring() {
406
+ if (!this.isMonitoring) {
407
+ return;
408
+ }
409
+ this.isMonitoring = false;
410
+ this.page = null;
411
+ this.lastUrl = ''; // Clear last URL to prevent stale comparisons after restart
412
+ logger.info('Console monitoring stopped');
413
+ }
414
+ /**
415
+ * Get monitoring status
416
+ */
417
+ getStatus() {
418
+ return {
419
+ isMonitoring: this.isMonitoring,
420
+ logCount: this.logs.length,
421
+ bufferSize: this.config.bufferSize,
422
+ workerCount: this.workers.size,
423
+ oldestTimestamp: this.logs[0]?.timestamp,
424
+ newestTimestamp: this.logs[this.logs.length - 1]?.timestamp,
425
+ };
426
+ }
427
+ }