@amaster.ai/vite-plugins 1.0.0-beta.0 → 1.0.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,183 +1 @@
1
1
  # @amaster.ai/vite-plugins
2
-
3
- Collection of Vite plugins for React application development.
4
-
5
- ## Features
6
-
7
- - 📝 **Browser Logs Plugin** - Collect console logs and network requests to file
8
- - 🔧 **Component ID Plugin** - Add unique IDs to HTML elements for debugging
9
- - 🌉 **Editor Bridge Plugin** - Connect development UI with editor tools
10
- - 🛣️ **Routes Expose Plugin** - Expose React Router routes in development
11
-
12
- ## Installation
13
-
14
- ```bash
15
- pnpm add -D @amaster.ai/vite-plugins
16
- ```
17
-
18
- ## Usage
19
-
20
- ```typescript
21
- // vite.config.ts
22
- import { defineConfig } from "vite";
23
- import react from "@vitejs/plugin-react";
24
- import {
25
- browserLogsPlugin,
26
- componentIdPlugin,
27
- editorBridgePlugin,
28
- routesExposePlugin,
29
- } from "@amaster.ai/vite-plugins";
30
-
31
- export default defineConfig({
32
- plugins: [
33
- react(),
34
- browserLogsPlugin(), // Collect browser logs
35
- componentIdPlugin(),
36
- editorBridgePlugin(),
37
- routesExposePlugin(),
38
- ],
39
- });
40
- ```
41
-
42
- ## Plugins
43
-
44
- ### browserLogsPlugin()
45
-
46
- Collects console logs, network requests, and errors from the browser and writes them to a file.
47
-
48
- **Features:**
49
- - ✅ Intercepts console.log/info/warn/error/debug
50
- - ✅ Captures fetch and XMLHttpRequest
51
- - ✅ Records global errors and unhandled rejections
52
- - ✅ Filters out Vite internal requests and SSE
53
- - ✅ Writes to `browser.log` in project root
54
- - ✅ Only runs in development mode
55
-
56
- **Output:**
57
-
58
- All logs are written to `browser.log` in your project root:
59
-
60
- ```json
61
- {"type":"console","timestamp":"2026-01-17T10:30:45.123Z","level":"log","message":"User logged in","stack":[...]}
62
- {"type":"request","timestamp":"2026-01-17T10:30:46.456Z","method":"GET","url":"/api/users","status":200,...}
63
- {"type":"error","timestamp":"2026-01-17T10:30:47.789Z","level":"error","message":"Uncaught Error: ..."}
64
- ```
65
-
66
- **Example:**
67
-
68
- ```typescript
69
- browserLogsPlugin()
70
- ```
71
-
72
- **Log Format:**
73
-
74
- - **Console logs**: `{ type: 'console', level, message, stack }`
75
- - **Network requests**: `{ type: 'request', method, url, status, duration, requestBody, responseBody }`
76
- - **Errors**: `{ type: 'error', message, stack }`
77
-
78
- **Programmatic Access:**
79
-
80
- ```javascript
81
- // In browser console
82
-
83
- // Get queue status
84
- window.__getBrowserLogsStatus__();
85
- // => { queueLength: 0, isWriting: false }
86
-
87
- // Flush logs manually
88
- await window.__flushBrowserLogs__();
89
- ```
90
-
91
- ### componentIdPlugin()
92
-
93
- Adds unique `data-node-component-id` attributes to HTML elements in JSX.
94
-
95
- **Features:**
96
- - ✅ Only runs in development mode
97
- - ✅ Only affects native HTML tags (not custom components)
98
- - ✅ Generates stable IDs based on file path and position
99
- - ✅ IDs persist through hot reloads
100
-
101
- **Example output:**
102
-
103
- ```tsx
104
- // Input
105
- <div>
106
- <button>Click me</button>
107
- </div>
108
-
109
- // Output (development only)
110
- <div data-node-component-id="a1b2c3d4e5f6">
111
- <button data-node-component-id="f6e5d4c3b2a1">Click me</button>
112
- </div>
113
- ```
114
-
115
- ### editorBridgePlugin(options?)
116
-
117
- Injects bridge script for editor integration.
118
-
119
- **Options:**
120
- - `bridgeScriptPath` - Path to bridge script (default: `/scripts/bridge.js`)
121
-
122
- **Features:**
123
- - ✅ Injects click handler for custom navigation
124
- - ✅ Loads bridge.js in development mode
125
- - ✅ Prevents unwanted navigation in editor mode
126
-
127
- **Example:**
128
-
129
- ```typescript
130
- editorBridgePlugin({
131
- bridgeScriptPath: "/custom/bridge.js",
132
- })
133
- ```
134
-
135
- ### routesExposePlugin(options?)
136
-
137
- Exposes React Router routes to `window.__APP_ROUTES__` in development.
138
-
139
- **Options:**
140
- - `routesFilePath` - Path to routes file (default: `src/routes.tsx`)
141
-
142
- **Features:**
143
- - ✅ Only runs in development mode
144
- - ✅ Automatically detects routes export
145
- - ✅ Accessible via browser DevTools
146
-
147
- **Example:**
148
-
149
- ```typescript
150
- routesExposePlugin({
151
- routesFilePath: "src/app/routes.tsx",
152
- })
153
- ```
154
-
155
- **Access routes in browser:**
156
-
157
- ```javascript
158
- console.log(window.__APP_ROUTES__);
159
- ```
160
-
161
- ## Development Mode Only
162
-
163
- All plugins are automatically disabled in production builds. They only run when:
164
-
165
- ```bash
166
- vite dev # Development server
167
- vite serve # Development server
168
- ```
169
-
170
- ## TypeScript Support
171
-
172
- Full TypeScript support with proper types:
173
-
174
- ```typescript
175
- import type { Plugin } from "vite";
176
- import { componentIdPlugin } from "@amaster.ai/vite-plugins";
177
-
178
- const plugin: Plugin = componentIdPlugin();
179
- ```
180
-
181
- ## License
182
-
183
- MIT
package/dist/index.cjs CHANGED
@@ -1,18 +1,4 @@
1
- 'use strict';
2
-
3
- var fs = require('fs');
4
- var path = require('path');
5
- var crypto = require('crypto');
6
-
7
- function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
8
-
9
- var fs__default = /*#__PURE__*/_interopDefault(fs);
10
- var path__default = /*#__PURE__*/_interopDefault(path);
11
-
12
- // src/browser-logs.ts
13
- function browserLogsPlugin() {
14
- let logFilePath = "";
15
- const injectedScript = `
1
+ 'use strict';var f=require('fs'),h=require('path'),crypto=require('crypto');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var f__default=/*#__PURE__*/_interopDefault(f);var h__default=/*#__PURE__*/_interopDefault(h);function m(){let o="",t=`
16
2
  <script>
17
3
  (function() {
18
4
  'use strict';
@@ -529,272 +515,8 @@ function browserLogsPlugin() {
529
515
 
530
516
  originalConsole.log('[BrowserLogs] Log collection started');
531
517
  })();
532
- </script>`;
533
- return {
534
- name: "vite-plugin-browser-logs",
535
- configResolved(config) {
536
- const root = config.root || process.cwd();
537
- logFilePath = path__default.default.join(root, "browser.log");
538
- },
539
- configureServer(devServer) {
540
- devServer.middlewares.use((req, res, next) => {
541
- if (req.url === "/__browser__" && req.method === "POST") {
542
- const origin = req.headers.origin || "*";
543
- let body = "";
544
- req.on("data", (chunk) => {
545
- body += chunk.toString();
546
- });
547
- req.on("end", () => {
548
- try {
549
- const logDir = path__default.default.dirname(logFilePath);
550
- if (!fs__default.default.existsSync(logDir)) {
551
- fs__default.default.mkdirSync(logDir, { recursive: true });
552
- }
553
- fs__default.default.appendFileSync(logFilePath, `${body}
554
- `, "utf-8");
555
- res.writeHead(200, {
556
- "Content-Type": "application/json",
557
- "Access-Control-Allow-Origin": origin,
558
- "Access-Control-Allow-Methods": "POST, OPTIONS",
559
- "Access-Control-Allow-Headers": "Content-Type"
560
- });
561
- res.end(JSON.stringify({ success: true }));
562
- } catch (error) {
563
- console.error("[BrowserLogs] Write error:", error);
564
- res.writeHead(500, {
565
- "Content-Type": "application/json",
566
- "Access-Control-Allow-Origin": origin,
567
- "Access-Control-Allow-Methods": "POST, OPTIONS",
568
- "Access-Control-Allow-Headers": "Content-Type"
569
- });
570
- res.end(JSON.stringify({ success: false, error: String(error) }));
571
- }
572
- });
573
- } else if (req.url === "/__browser__" && req.method === "OPTIONS") {
574
- const origin = req.headers.origin || "*";
575
- res.writeHead(204, {
576
- "Access-Control-Allow-Origin": origin,
577
- "Access-Control-Allow-Methods": "POST, OPTIONS",
578
- "Access-Control-Allow-Headers": "Content-Type",
579
- "Access-Control-Max-Age": "86400"
580
- });
581
- res.end();
582
- } else if (req.url === "/__browser__") {
583
- const origin = req.headers.origin || "*";
584
- res.writeHead(405, {
585
- "Content-Type": "application/json",
586
- "Access-Control-Allow-Origin": origin
587
- });
588
- res.end(JSON.stringify({ error: "Method not allowed" }));
589
- } else {
590
- next();
591
- }
592
- });
593
- console.log("[BrowserLogs] Logs will be written to:", logFilePath);
594
- },
595
- transformIndexHtml(html) {
596
- return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);
597
- }
598
- };
599
- }
600
- function generateUniqueId(filePath, position) {
601
- const key = `${filePath}:${position}`;
602
- return crypto.createHash("md5").update(key).digest("hex").substring(0, 12);
603
- }
604
- var HTML_TAGS = /* @__PURE__ */ new Set([
605
- "a",
606
- "abbr",
607
- "address",
608
- "area",
609
- "article",
610
- "aside",
611
- "audio",
612
- "b",
613
- "base",
614
- "bdi",
615
- "bdo",
616
- "blockquote",
617
- "body",
618
- "br",
619
- "button",
620
- "canvas",
621
- "caption",
622
- "cite",
623
- "code",
624
- "col",
625
- "colgroup",
626
- "data",
627
- "datalist",
628
- "dd",
629
- "del",
630
- "details",
631
- "dfn",
632
- "dialog",
633
- "div",
634
- "dl",
635
- "dt",
636
- "em",
637
- "embed",
638
- "fieldset",
639
- "figcaption",
640
- "figure",
641
- "footer",
642
- "form",
643
- "h1",
644
- "h2",
645
- "h3",
646
- "h4",
647
- "h5",
648
- "h6",
649
- "head",
650
- "header",
651
- "hgroup",
652
- "hr",
653
- "html",
654
- "i",
655
- "iframe",
656
- "img",
657
- "input",
658
- "ins",
659
- "kbd",
660
- "label",
661
- "legend",
662
- "li",
663
- "link",
664
- "main",
665
- "map",
666
- "mark",
667
- "meta",
668
- "meter",
669
- "nav",
670
- "noscript",
671
- "object",
672
- "ol",
673
- "optgroup",
674
- "option",
675
- "output",
676
- "p",
677
- "param",
678
- "picture",
679
- "pre",
680
- "progress",
681
- "q",
682
- "rp",
683
- "rt",
684
- "ruby",
685
- "s",
686
- "samp",
687
- "script",
688
- "section",
689
- "select",
690
- "small",
691
- "source",
692
- "span",
693
- "strong",
694
- "style",
695
- "sub",
696
- "summary",
697
- "sup",
698
- "svg",
699
- "table",
700
- "tbody",
701
- "td",
702
- "template",
703
- "textarea",
704
- "tfoot",
705
- "th",
706
- "thead",
707
- "time",
708
- "title",
709
- "tr",
710
- "track",
711
- "u",
712
- "ul",
713
- "var",
714
- "video",
715
- "wbr"
716
- ]);
717
- function componentIdPlugin() {
718
- let isDev = false;
719
- return {
720
- name: "vite-plugin-component-id",
721
- enforce: "pre",
722
- configResolved(config) {
723
- isDev = config.command === "serve";
724
- },
725
- transform(code, id) {
726
- if (!isDev) {
727
- return null;
728
- }
729
- if (!/\.(tsx|jsx)$/.test(id)) {
730
- return null;
731
- }
732
- if (id.includes("node_modules")) {
733
- return null;
734
- }
735
- if (!/<[a-z]/.test(code)) {
736
- return null;
737
- }
738
- try {
739
- let transformedCode = code;
740
- const jsxOpenTagRegex = /<([a-z][\da-z]*)(?=[\s/>])/g;
741
- let match;
742
- const matches = [];
743
- while ((match = jsxOpenTagRegex.exec(code)) !== null) {
744
- const tag = match[1];
745
- if (!tag) continue;
746
- const tagStartIndex = match.index;
747
- const tagEndIndex = match.index + match[0].length;
748
- if (!HTML_TAGS.has(tag)) {
749
- continue;
750
- }
751
- let checkIndex = tagEndIndex;
752
- let foundClosing = false;
753
- let hasComponentId = false;
754
- while (checkIndex < code.length && !foundClosing) {
755
- const char = code[checkIndex];
756
- if (char === ">") {
757
- foundClosing = true;
758
- const tagContent = code.substring(tagStartIndex, checkIndex);
759
- if (tagContent.includes("data-node-component-id")) {
760
- hasComponentId = true;
761
- }
762
- }
763
- checkIndex++;
764
- }
765
- if (hasComponentId) {
766
- continue;
767
- }
768
- matches.push({
769
- index: tagStartIndex,
770
- tag,
771
- tagEndIndex
772
- });
773
- }
774
- let currentCode = code;
775
- for (let i = matches.length - 1; i >= 0; i--) {
776
- const item = matches[i];
777
- if (!item) continue;
778
- const { index, tagEndIndex } = item;
779
- const uniqueId = generateUniqueId(id, index);
780
- const before = currentCode.substring(0, tagEndIndex);
781
- const after = currentCode.substring(tagEndIndex);
782
- transformedCode = `${before} data-node-component-id="${uniqueId}"${after}`;
783
- currentCode = transformedCode;
784
- }
785
- return {
786
- code: transformedCode,
787
- map: null
788
- };
789
- } catch {
790
- return null;
791
- }
792
- }
793
- };
794
- }
795
-
796
- // src/editor-bridge.ts
797
- var clickHandlerScript = `<script>
518
+ </script>`;return {name:"vite-plugin-browser-logs",configResolved(r){let e=r.root||process.cwd();o=h__default.default.join(e,"browser.log");},configureServer(r){r.middlewares.use((e,s,i)=>{if(e.url==="/__browser__"&&e.method==="POST"){let a=e.headers.origin||"*",l="";e.on("data",n=>{l+=n.toString();}),e.on("end",()=>{try{let n=h__default.default.dirname(o);f__default.default.existsSync(n)||f__default.default.mkdirSync(n,{recursive:!0}),f__default.default.appendFileSync(o,`${l}
519
+ `,"utf-8"),s.writeHead(200,{"Content-Type":"application/json","Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),s.end(JSON.stringify({success:!0}));}catch(n){console.error("[BrowserLogs] Write error:",n),s.writeHead(500,{"Content-Type":"application/json","Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),s.end(JSON.stringify({success:false,error:String(n)}));}});}else if(e.url==="/__browser__"&&e.method==="OPTIONS"){let a=e.headers.origin||"*";s.writeHead(204,{"Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type","Access-Control-Max-Age":"86400"}),s.end();}else if(e.url==="/__browser__"){let a=e.headers.origin||"*";s.writeHead(405,{"Content-Type":"application/json","Access-Control-Allow-Origin":a}),s.end(JSON.stringify({error:"Method not allowed"}));}else i();}),console.log("[BrowserLogs] Logs will be written to:",o);},transformIndexHtml(r){return r.replace(/<head([^>]*)>/i,`<head$1>${t}`)}}}function b(o,t){let r=`${o}:${t}`;return crypto.createHash("md5").update(r).digest("hex").substring(0,12)}var w=new Set(["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rp","rt","ruby","s","samp","script","section","select","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"]);function S(){let o=false;return {name:"vite-plugin-component-id",enforce:"pre",configResolved(t){o=t.command==="serve";},transform(t,r){if(!o||!/\.(tsx|jsx)$/.test(r)||r.includes("node_modules")||!/<[a-z]/.test(t))return null;try{let e=t,s=/<([a-z][\da-z]*)(?=[\s/>])/g,i,a=[];for(;(i=s.exec(t))!==null;){let n=i[1];if(!n)continue;let d=i.index,c=i.index+i[0].length;if(!w.has(n))continue;let u=c,g=!1,p=!1;for(;u<t.length&&!g;)t[u]===">"&&(g=!0,t.substring(d,u).includes("data-node-component-id")&&(p=!0)),u++;p||a.push({index:d,tag:n,tagEndIndex:c});}let l=t;for(let n=a.length-1;n>=0;n--){let d=a[n];if(!d)continue;let{index:c,tagEndIndex:u}=d,g=b(r,c),p=l.substring(0,u),y=l.substring(u);e=`${p} data-node-component-id="${g}"${y}`,l=e;}return {code:e,map:null}}catch{return null}}}}var _=`<script>
798
520
  document.addEventListener("click", (e) => {
799
521
  const element = e.target;
800
522
  const noJumpOut = document.body.classList.contains("forbid-jump-out")
@@ -812,65 +534,11 @@ document.addEventListener("click", (e) => {
812
534
  e.preventDefault();
813
535
  }
814
536
  });
815
- </script>`;
816
- function editorBridgePlugin(options) {
817
- let isDev = false;
818
- const bridgePath = options?.bridgeScriptPath || "/scripts/bridge.js";
819
- return {
820
- name: "vite-plugin-editor-bridge",
821
- configResolved(config) {
822
- isDev = config.command === "serve";
823
- },
824
- transformIndexHtml(html) {
825
- const devScript = isDev ? `<script type="module" src="${bridgePath}"></script>` : "";
826
- const scriptsToInject = `${clickHandlerScript + devScript}</body>`;
827
- return html.replace("</body>", scriptsToInject);
828
- }
829
- };
830
- }
831
-
832
- // src/routes-expose.ts
833
- function routesExposePlugin(options) {
834
- let isDev = false;
835
- const routesPath = options?.routesFilePath || "src/routes.tsx";
836
- return {
837
- name: "vite-plugin-routes-expose",
838
- enforce: "post",
839
- configResolved(config) {
840
- isDev = config.command === "serve";
841
- },
842
- transform(code, id) {
843
- if (!isDev) {
844
- return null;
845
- }
846
- if (!id.endsWith(routesPath)) {
847
- return null;
848
- }
849
- try {
850
- if (code.includes("window.__APP_ROUTES__")) {
851
- return null;
852
- }
853
- const transformedCode = `${code}
537
+ </script>`;function x(o){let t=false,r=o?.bridgeScriptPath||"/scripts/bridge.js";return {name:"vite-plugin-editor-bridge",configResolved(e){t=e.command==="serve";},transformIndexHtml(e){let s=t?`<script type="module" src="${r}"></script>`:"",i=`${_+s}</body>`;return e.replace("</body>",i)}}}function T(o){let t=false,r=o?.routesFilePath||"src/routes.tsx";return {name:"vite-plugin-routes-expose",enforce:"post",configResolved(e){t=e.command==="serve";},transform(e,s){if(!t||!s.endsWith(r))return null;try{return e.includes("window.__APP_ROUTES__")?null:{code:`${e}
854
538
 
855
539
  // Development mode: Expose routes to window.__APP_ROUTES__
856
540
  if (typeof window !== 'undefined') {
857
541
  window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];
858
542
  }
859
- `;
860
- return {
861
- code: transformedCode,
862
- map: null
863
- };
864
- } catch {
865
- return null;
866
- }
867
- }
868
- };
869
- }
870
-
871
- exports.browserLogsPlugin = browserLogsPlugin;
872
- exports.componentIdPlugin = componentIdPlugin;
873
- exports.editorBridgePlugin = editorBridgePlugin;
874
- exports.routesExposePlugin = routesExposePlugin;
875
- //# sourceMappingURL=index.cjs.map
543
+ `,map:null}}catch{return null}}}}exports.browserLogsPlugin=m;exports.componentIdPlugin=S;exports.editorBridgePlugin=x;exports.routesExposePlugin=T;//# sourceMappingURL=index.cjs.map
876
544
  //# sourceMappingURL=index.cjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/browser-logs.ts","../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts"],"names":["path","fs","createHash"],"mappings":";;;;;;;;;;;;AASO,SAAS,iBAAA,GAA4B;AAC1C,EAAA,IAAI,WAAA,GAAc,EAAA;AAGlB,EAAA,MAAM,cAAA,GAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAA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","file":"index.cjs","sourcesContent":["import type { Plugin } from \"vite\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode\n */\nexport function editorBridgePlugin(options?: { bridgeScriptPath?: string }): Plugin {\n let isDev = false;\n const bridgePath = options?.bridgeScriptPath || \"/scripts/bridge.js\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transformIndexHtml(html) {\n const devScript = isDev ? `<script type=\"module\" src=\"${bridgePath}\"></script>` : \"\";\n\n const scriptsToInject = `${clickHandlerScript + devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n"]}
1
+ {"version":3,"sources":["../src/browser-logs.ts","../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts"],"names":["browserLogsPlugin","logFilePath","injectedScript","config","root","path","devServer","req","res","next","origin","body","chunk","logDir","fs","error","html","generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","editorBridgePlugin","options","bridgePath","devScript","scriptsToInject","routesExposePlugin","routesPath"],"mappings":"2OASO,SAASA,CAAAA,EAA4B,CAC1C,IAAIC,CAAAA,CAAc,EAAA,CAGZC,CAAAA,CAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","file":"index.cjs","sourcesContent":["import type { Plugin } from \"vite\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode\n */\nexport function editorBridgePlugin(options?: { bridgeScriptPath?: string }): Plugin {\n let isDev = false;\n const bridgePath = options?.bridgeScriptPath || \"/scripts/bridge.js\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transformIndexHtml(html) {\n const devScript = isDev ? `<script type=\"module\" src=\"${bridgePath}\"></script>` : \"\";\n\n const scriptsToInject = `${clickHandlerScript + devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n"]}
package/dist/index.js CHANGED
@@ -1,11 +1,4 @@
1
- import fs from 'fs';
2
- import path from 'path';
3
- import { createHash } from 'crypto';
4
-
5
- // src/browser-logs.ts
6
- function browserLogsPlugin() {
7
- let logFilePath = "";
8
- const injectedScript = `
1
+ import f from'fs';import h from'path';import {createHash}from'crypto';function m(){let o="",t=`
9
2
  <script>
10
3
  (function() {
11
4
  'use strict';
@@ -522,272 +515,8 @@ function browserLogsPlugin() {
522
515
 
523
516
  originalConsole.log('[BrowserLogs] Log collection started');
524
517
  })();
525
- </script>`;
526
- return {
527
- name: "vite-plugin-browser-logs",
528
- configResolved(config) {
529
- const root = config.root || process.cwd();
530
- logFilePath = path.join(root, "browser.log");
531
- },
532
- configureServer(devServer) {
533
- devServer.middlewares.use((req, res, next) => {
534
- if (req.url === "/__browser__" && req.method === "POST") {
535
- const origin = req.headers.origin || "*";
536
- let body = "";
537
- req.on("data", (chunk) => {
538
- body += chunk.toString();
539
- });
540
- req.on("end", () => {
541
- try {
542
- const logDir = path.dirname(logFilePath);
543
- if (!fs.existsSync(logDir)) {
544
- fs.mkdirSync(logDir, { recursive: true });
545
- }
546
- fs.appendFileSync(logFilePath, `${body}
547
- `, "utf-8");
548
- res.writeHead(200, {
549
- "Content-Type": "application/json",
550
- "Access-Control-Allow-Origin": origin,
551
- "Access-Control-Allow-Methods": "POST, OPTIONS",
552
- "Access-Control-Allow-Headers": "Content-Type"
553
- });
554
- res.end(JSON.stringify({ success: true }));
555
- } catch (error) {
556
- console.error("[BrowserLogs] Write error:", error);
557
- res.writeHead(500, {
558
- "Content-Type": "application/json",
559
- "Access-Control-Allow-Origin": origin,
560
- "Access-Control-Allow-Methods": "POST, OPTIONS",
561
- "Access-Control-Allow-Headers": "Content-Type"
562
- });
563
- res.end(JSON.stringify({ success: false, error: String(error) }));
564
- }
565
- });
566
- } else if (req.url === "/__browser__" && req.method === "OPTIONS") {
567
- const origin = req.headers.origin || "*";
568
- res.writeHead(204, {
569
- "Access-Control-Allow-Origin": origin,
570
- "Access-Control-Allow-Methods": "POST, OPTIONS",
571
- "Access-Control-Allow-Headers": "Content-Type",
572
- "Access-Control-Max-Age": "86400"
573
- });
574
- res.end();
575
- } else if (req.url === "/__browser__") {
576
- const origin = req.headers.origin || "*";
577
- res.writeHead(405, {
578
- "Content-Type": "application/json",
579
- "Access-Control-Allow-Origin": origin
580
- });
581
- res.end(JSON.stringify({ error: "Method not allowed" }));
582
- } else {
583
- next();
584
- }
585
- });
586
- console.log("[BrowserLogs] Logs will be written to:", logFilePath);
587
- },
588
- transformIndexHtml(html) {
589
- return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);
590
- }
591
- };
592
- }
593
- function generateUniqueId(filePath, position) {
594
- const key = `${filePath}:${position}`;
595
- return createHash("md5").update(key).digest("hex").substring(0, 12);
596
- }
597
- var HTML_TAGS = /* @__PURE__ */ new Set([
598
- "a",
599
- "abbr",
600
- "address",
601
- "area",
602
- "article",
603
- "aside",
604
- "audio",
605
- "b",
606
- "base",
607
- "bdi",
608
- "bdo",
609
- "blockquote",
610
- "body",
611
- "br",
612
- "button",
613
- "canvas",
614
- "caption",
615
- "cite",
616
- "code",
617
- "col",
618
- "colgroup",
619
- "data",
620
- "datalist",
621
- "dd",
622
- "del",
623
- "details",
624
- "dfn",
625
- "dialog",
626
- "div",
627
- "dl",
628
- "dt",
629
- "em",
630
- "embed",
631
- "fieldset",
632
- "figcaption",
633
- "figure",
634
- "footer",
635
- "form",
636
- "h1",
637
- "h2",
638
- "h3",
639
- "h4",
640
- "h5",
641
- "h6",
642
- "head",
643
- "header",
644
- "hgroup",
645
- "hr",
646
- "html",
647
- "i",
648
- "iframe",
649
- "img",
650
- "input",
651
- "ins",
652
- "kbd",
653
- "label",
654
- "legend",
655
- "li",
656
- "link",
657
- "main",
658
- "map",
659
- "mark",
660
- "meta",
661
- "meter",
662
- "nav",
663
- "noscript",
664
- "object",
665
- "ol",
666
- "optgroup",
667
- "option",
668
- "output",
669
- "p",
670
- "param",
671
- "picture",
672
- "pre",
673
- "progress",
674
- "q",
675
- "rp",
676
- "rt",
677
- "ruby",
678
- "s",
679
- "samp",
680
- "script",
681
- "section",
682
- "select",
683
- "small",
684
- "source",
685
- "span",
686
- "strong",
687
- "style",
688
- "sub",
689
- "summary",
690
- "sup",
691
- "svg",
692
- "table",
693
- "tbody",
694
- "td",
695
- "template",
696
- "textarea",
697
- "tfoot",
698
- "th",
699
- "thead",
700
- "time",
701
- "title",
702
- "tr",
703
- "track",
704
- "u",
705
- "ul",
706
- "var",
707
- "video",
708
- "wbr"
709
- ]);
710
- function componentIdPlugin() {
711
- let isDev = false;
712
- return {
713
- name: "vite-plugin-component-id",
714
- enforce: "pre",
715
- configResolved(config) {
716
- isDev = config.command === "serve";
717
- },
718
- transform(code, id) {
719
- if (!isDev) {
720
- return null;
721
- }
722
- if (!/\.(tsx|jsx)$/.test(id)) {
723
- return null;
724
- }
725
- if (id.includes("node_modules")) {
726
- return null;
727
- }
728
- if (!/<[a-z]/.test(code)) {
729
- return null;
730
- }
731
- try {
732
- let transformedCode = code;
733
- const jsxOpenTagRegex = /<([a-z][\da-z]*)(?=[\s/>])/g;
734
- let match;
735
- const matches = [];
736
- while ((match = jsxOpenTagRegex.exec(code)) !== null) {
737
- const tag = match[1];
738
- if (!tag) continue;
739
- const tagStartIndex = match.index;
740
- const tagEndIndex = match.index + match[0].length;
741
- if (!HTML_TAGS.has(tag)) {
742
- continue;
743
- }
744
- let checkIndex = tagEndIndex;
745
- let foundClosing = false;
746
- let hasComponentId = false;
747
- while (checkIndex < code.length && !foundClosing) {
748
- const char = code[checkIndex];
749
- if (char === ">") {
750
- foundClosing = true;
751
- const tagContent = code.substring(tagStartIndex, checkIndex);
752
- if (tagContent.includes("data-node-component-id")) {
753
- hasComponentId = true;
754
- }
755
- }
756
- checkIndex++;
757
- }
758
- if (hasComponentId) {
759
- continue;
760
- }
761
- matches.push({
762
- index: tagStartIndex,
763
- tag,
764
- tagEndIndex
765
- });
766
- }
767
- let currentCode = code;
768
- for (let i = matches.length - 1; i >= 0; i--) {
769
- const item = matches[i];
770
- if (!item) continue;
771
- const { index, tagEndIndex } = item;
772
- const uniqueId = generateUniqueId(id, index);
773
- const before = currentCode.substring(0, tagEndIndex);
774
- const after = currentCode.substring(tagEndIndex);
775
- transformedCode = `${before} data-node-component-id="${uniqueId}"${after}`;
776
- currentCode = transformedCode;
777
- }
778
- return {
779
- code: transformedCode,
780
- map: null
781
- };
782
- } catch {
783
- return null;
784
- }
785
- }
786
- };
787
- }
788
-
789
- // src/editor-bridge.ts
790
- var clickHandlerScript = `<script>
518
+ </script>`;return {name:"vite-plugin-browser-logs",configResolved(r){let e=r.root||process.cwd();o=h.join(e,"browser.log");},configureServer(r){r.middlewares.use((e,s,i)=>{if(e.url==="/__browser__"&&e.method==="POST"){let a=e.headers.origin||"*",l="";e.on("data",n=>{l+=n.toString();}),e.on("end",()=>{try{let n=h.dirname(o);f.existsSync(n)||f.mkdirSync(n,{recursive:!0}),f.appendFileSync(o,`${l}
519
+ `,"utf-8"),s.writeHead(200,{"Content-Type":"application/json","Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),s.end(JSON.stringify({success:!0}));}catch(n){console.error("[BrowserLogs] Write error:",n),s.writeHead(500,{"Content-Type":"application/json","Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type"}),s.end(JSON.stringify({success:false,error:String(n)}));}});}else if(e.url==="/__browser__"&&e.method==="OPTIONS"){let a=e.headers.origin||"*";s.writeHead(204,{"Access-Control-Allow-Origin":a,"Access-Control-Allow-Methods":"POST, OPTIONS","Access-Control-Allow-Headers":"Content-Type","Access-Control-Max-Age":"86400"}),s.end();}else if(e.url==="/__browser__"){let a=e.headers.origin||"*";s.writeHead(405,{"Content-Type":"application/json","Access-Control-Allow-Origin":a}),s.end(JSON.stringify({error:"Method not allowed"}));}else i();}),console.log("[BrowserLogs] Logs will be written to:",o);},transformIndexHtml(r){return r.replace(/<head([^>]*)>/i,`<head$1>${t}`)}}}function b(o,t){let r=`${o}:${t}`;return createHash("md5").update(r).digest("hex").substring(0,12)}var w=new Set(["a","abbr","address","area","article","aside","audio","b","base","bdi","bdo","blockquote","body","br","button","canvas","caption","cite","code","col","colgroup","data","datalist","dd","del","details","dfn","dialog","div","dl","dt","em","embed","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","iframe","img","input","ins","kbd","label","legend","li","link","main","map","mark","meta","meter","nav","noscript","object","ol","optgroup","option","output","p","param","picture","pre","progress","q","rp","rt","ruby","s","samp","script","section","select","small","source","span","strong","style","sub","summary","sup","svg","table","tbody","td","template","textarea","tfoot","th","thead","time","title","tr","track","u","ul","var","video","wbr"]);function S(){let o=false;return {name:"vite-plugin-component-id",enforce:"pre",configResolved(t){o=t.command==="serve";},transform(t,r){if(!o||!/\.(tsx|jsx)$/.test(r)||r.includes("node_modules")||!/<[a-z]/.test(t))return null;try{let e=t,s=/<([a-z][\da-z]*)(?=[\s/>])/g,i,a=[];for(;(i=s.exec(t))!==null;){let n=i[1];if(!n)continue;let d=i.index,c=i.index+i[0].length;if(!w.has(n))continue;let u=c,g=!1,p=!1;for(;u<t.length&&!g;)t[u]===">"&&(g=!0,t.substring(d,u).includes("data-node-component-id")&&(p=!0)),u++;p||a.push({index:d,tag:n,tagEndIndex:c});}let l=t;for(let n=a.length-1;n>=0;n--){let d=a[n];if(!d)continue;let{index:c,tagEndIndex:u}=d,g=b(r,c),p=l.substring(0,u),y=l.substring(u);e=`${p} data-node-component-id="${g}"${y}`,l=e;}return {code:e,map:null}}catch{return null}}}}var _=`<script>
791
520
  document.addEventListener("click", (e) => {
792
521
  const element = e.target;
793
522
  const noJumpOut = document.body.classList.contains("forbid-jump-out")
@@ -805,62 +534,11 @@ document.addEventListener("click", (e) => {
805
534
  e.preventDefault();
806
535
  }
807
536
  });
808
- </script>`;
809
- function editorBridgePlugin(options) {
810
- let isDev = false;
811
- const bridgePath = options?.bridgeScriptPath || "/scripts/bridge.js";
812
- return {
813
- name: "vite-plugin-editor-bridge",
814
- configResolved(config) {
815
- isDev = config.command === "serve";
816
- },
817
- transformIndexHtml(html) {
818
- const devScript = isDev ? `<script type="module" src="${bridgePath}"></script>` : "";
819
- const scriptsToInject = `${clickHandlerScript + devScript}</body>`;
820
- return html.replace("</body>", scriptsToInject);
821
- }
822
- };
823
- }
824
-
825
- // src/routes-expose.ts
826
- function routesExposePlugin(options) {
827
- let isDev = false;
828
- const routesPath = options?.routesFilePath || "src/routes.tsx";
829
- return {
830
- name: "vite-plugin-routes-expose",
831
- enforce: "post",
832
- configResolved(config) {
833
- isDev = config.command === "serve";
834
- },
835
- transform(code, id) {
836
- if (!isDev) {
837
- return null;
838
- }
839
- if (!id.endsWith(routesPath)) {
840
- return null;
841
- }
842
- try {
843
- if (code.includes("window.__APP_ROUTES__")) {
844
- return null;
845
- }
846
- const transformedCode = `${code}
537
+ </script>`;function x(o){let t=false,r=o?.bridgeScriptPath||"/scripts/bridge.js";return {name:"vite-plugin-editor-bridge",configResolved(e){t=e.command==="serve";},transformIndexHtml(e){let s=t?`<script type="module" src="${r}"></script>`:"",i=`${_+s}</body>`;return e.replace("</body>",i)}}}function T(o){let t=false,r=o?.routesFilePath||"src/routes.tsx";return {name:"vite-plugin-routes-expose",enforce:"post",configResolved(e){t=e.command==="serve";},transform(e,s){if(!t||!s.endsWith(r))return null;try{return e.includes("window.__APP_ROUTES__")?null:{code:`${e}
847
538
 
848
539
  // Development mode: Expose routes to window.__APP_ROUTES__
849
540
  if (typeof window !== 'undefined') {
850
541
  window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];
851
542
  }
852
- `;
853
- return {
854
- code: transformedCode,
855
- map: null
856
- };
857
- } catch {
858
- return null;
859
- }
860
- }
861
- };
862
- }
863
-
864
- export { browserLogsPlugin, componentIdPlugin, editorBridgePlugin, routesExposePlugin };
865
- //# sourceMappingURL=index.js.map
543
+ `,map:null}}catch{return null}}}}export{m as browserLogsPlugin,S as componentIdPlugin,x as editorBridgePlugin,T as routesExposePlugin};//# sourceMappingURL=index.js.map
866
544
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/browser-logs.ts","../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts"],"names":[],"mappings":";;;;;AASO,SAAS,iBAAA,GAA4B;AAC1C,EAAA,IAAI,WAAA,GAAc,EAAA;AAGlB,EAAA,MAAM,cAAA,GAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","file":"index.js","sourcesContent":["import type { Plugin } from \"vite\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode\n */\nexport function editorBridgePlugin(options?: { bridgeScriptPath?: string }): Plugin {\n let isDev = false;\n const bridgePath = options?.bridgeScriptPath || \"/scripts/bridge.js\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transformIndexHtml(html) {\n const devScript = isDev ? `<script type=\"module\" src=\"${bridgePath}\"></script>` : \"\";\n\n const scriptsToInject = `${clickHandlerScript + devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n"]}
1
+ {"version":3,"sources":["../src/browser-logs.ts","../src/component-id.ts","../src/editor-bridge.ts","../src/routes-expose.ts"],"names":["browserLogsPlugin","logFilePath","injectedScript","config","root","path","devServer","req","res","next","origin","body","chunk","logDir","fs","error","html","generateUniqueId","filePath","position","key","createHash","HTML_TAGS","componentIdPlugin","isDev","code","id","transformedCode","jsxOpenTagRegex","match","matches","tag","tagStartIndex","tagEndIndex","checkIndex","foundClosing","hasComponentId","currentCode","i","item","index","uniqueId","before","after","clickHandlerScript","editorBridgePlugin","options","bridgePath","devScript","scriptsToInject","routesExposePlugin","routesPath"],"mappings":"sEASO,SAASA,CAAAA,EAA4B,CAC1C,IAAIC,CAAAA,CAAc,EAAA,CAGZC,CAAAA,CAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;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","file":"index.js","sourcesContent":["import type { Plugin } from \"vite\";\nimport fs from \"node:fs\";\nimport path from \"node:path\";\n\n/**\n * Browser logs collection plugin\n * Injects script into HTML head to collect console logs and network requests\n * Logs are written directly to browser.log file (same level as index.html)\n */\nexport function browserLogsPlugin(): Plugin {\n let logFilePath = \"\";\n\n // Script to inject into browser\n const injectedScript = `\n<script>\n(function() {\n 'use strict';\n \n // Log API path (provided by Vite dev server)\n var LOG_API_PATH = '/__browser__';\n \n // Write queue to ensure sequential writes\n var writeQueue = [];\n var isWriting = false;\n \n // Process write queue to ensure sequential writes\n function processWriteQueue() {\n if (isWriting || writeQueue.length === 0) return;\n \n isWriting = true;\n var entry = writeQueue.shift();\n var logText = JSON.stringify(entry);\n \n fetch(LOG_API_PATH, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: logText\n })\n .then(function(response) {\n isWriting = false;\n if (!response.ok) {\n console.warn('[BrowserLogs] Failed to write log:', response.status);\n }\n // Continue processing next item in queue\n processWriteQueue();\n })\n .catch(function(error) {\n console.warn('[BrowserLogs] Failed to write log:', error.message);\n // On failure, put log back to queue head for retry\n writeQueue.unshift(entry);\n isWriting = false;\n // Retry after delay\n setTimeout(processWriteQueue, 1000);\n });\n }\n \n // Limit headers object size\n function truncateHeaders(headers) {\n if (!headers) return headers;\n var jsonStr = JSON.stringify(headers);\n if (jsonStr.length <= MAX_HEADER_SIZE) return headers;\n return '[Headers truncated, size: ' + jsonStr.length + ']';\n }\n \n // Truncate oversized log entries\n function truncateLogEntry(entry) {\n var jsonStr = JSON.stringify(entry);\n if (jsonStr.length <= MAX_LOG_ENTRY_SIZE) {\n return entry;\n }\n \n // If oversized, progressively truncate non-critical fields\n var truncated = Object.assign({}, entry);\n \n // 1. Truncate response body first\n if (truncated.responseBody && typeof truncated.responseBody === 'string') {\n truncated.responseBody = truncated.responseBody.substring(0, 500) + '... [truncated]';\n }\n \n // 2. Truncate request body\n if (truncated.requestBody && typeof truncated.requestBody === 'string') {\n truncated.requestBody = truncated.requestBody.substring(0, 500) + '... [truncated]';\n }\n \n // 3. Truncate message\n if (truncated.message && typeof truncated.message === 'string' && truncated.message.length > 1000) {\n truncated.message = truncated.message.substring(0, 1000) + '... [truncated]';\n }\n \n // 4. Truncate stack\n if (truncated.stack && Array.isArray(truncated.stack) && truncated.stack.length > 3) {\n truncated.stack = truncated.stack.slice(0, 3);\n }\n \n // Add truncation marker\n truncated._truncated = true;\n truncated._originalSize = jsonStr.length;\n \n return truncated;\n }\n \n // Add log and send immediately (ensure order)\n function addLog(entry) {\n var truncatedEntry = truncateLogEntry(entry);\n writeQueue.push(truncatedEntry);\n processWriteQueue();\n }\n \n // ============================================\n // Console log collection\n // ============================================\n var originalConsole = {\n log: console.log,\n info: console.info,\n warn: console.warn,\n error: console.error,\n debug: console.debug\n };\n \n function getStackTrace() {\n var stack = new Error().stack || '';\n var lines = stack.split('\\\\n').slice(3);\n // Limit stack lines\n return lines.slice(0, MAX_STACK_LINES).map(function(line) { return line.trim(); });\n }\n \n function formatMessage(args) {\n return Array.from(args).map(function(arg) {\n if (arg === null) return 'null';\n if (arg === undefined) return 'undefined';\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg);\n } catch (e) {\n return String(arg);\n }\n }\n return String(arg);\n }).join(' ');\n }\n \n function createLogEntry(level, args) {\n return {\n type: 'console',\n timestamp: new Date().toISOString(),\n level: level,\n message: formatMessage(args),\n stack: getStackTrace()\n };\n }\n \n // Check if message should be filtered (contains [vite] text)\n function shouldFilterConsoleLog(args) {\n for (var i = 0; i < args.length; i++) {\n var arg = args[i];\n if (typeof arg === 'string' && arg.indexOf('[vite]') !== -1) {\n return true;\n }\n }\n return false;\n }\n \n function wrapConsoleMethod(method, level) {\n return function() {\n var args = arguments;\n // Filter out logs containing [vite]\n if (shouldFilterConsoleLog(args)) {\n return originalConsole[method].apply(console, args);\n }\n var entry = createLogEntry(level, args);\n addLog(entry);\n return originalConsole[method].apply(console, args);\n };\n }\n \n console.log = wrapConsoleMethod('log', 'log');\n console.info = wrapConsoleMethod('info', 'info');\n console.warn = wrapConsoleMethod('warn', 'warn');\n console.error = wrapConsoleMethod('error', 'error');\n console.debug = wrapConsoleMethod('debug', 'debug');\n \n // ============================================\n // Network request collection (exclude SSE and log API requests)\n // ============================================\n \n var MAX_BODY_SIZE = 2000;\n var MAX_LOG_ENTRY_SIZE = 3000; // Max single log entry length\n var MAX_HEADER_SIZE = 500; // Max single header object length\n var MAX_STACK_LINES = 5; // Max stack lines to keep\n const FILTERED_URLS = ['/__browser__', '/builtin']; // Filter out log API and Vite built-in requests\n \n function isSSERequest(url, headers) {\n var sseUrlPatterns = ['/events', '/sse', '/stream', 'text/event-stream'];\n var urlStr = String(url).toLowerCase();\n for (var i = 0; i < sseUrlPatterns.length; i++) {\n if (urlStr.indexOf(sseUrlPatterns[i]) !== -1) {\n return true;\n }\n }\n \n if (headers) {\n if (typeof headers.get === 'function') {\n var accept = headers.get('Accept') || headers.get('accept');\n if (accept && accept.indexOf('text/event-stream') !== -1) {\n return true;\n }\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n if (key.toLowerCase() === 'accept' && headers[key].indexOf('text/event-stream') !== -1) {\n return true;\n }\n }\n }\n }\n \n return false;\n }\n \n function shouldFilterRequest(url) {\n return FILTERED_URLS.some(function(filteredUrl) {\n return String(url).indexOf(filteredUrl) !== -1;\n });\n }\n \n function formatRequestBody(body) {\n if (!body) return null;\n \n try {\n if (typeof body === 'string') {\n return body.substring(0, MAX_BODY_SIZE);\n }\n if (body instanceof FormData) {\n var formDataObj = {};\n body.forEach(function(value, key) {\n if (value instanceof File) {\n formDataObj[key] = '[File: ' + value.name + ', size: ' + value.size + ']';\n } else {\n formDataObj[key] = String(value).substring(0, 1000);\n }\n });\n return JSON.stringify(formDataObj);\n }\n if (body instanceof Blob) {\n return '[Blob: size=' + body.size + ', type=' + body.type + ']';\n }\n if (body instanceof ArrayBuffer || ArrayBuffer.isView(body)) {\n return '[Binary: size=' + body.byteLength + ']';\n }\n if (body instanceof URLSearchParams) {\n return body.toString().substring(0, MAX_BODY_SIZE);\n }\n return String(body).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading body: ' + e.message + ']';\n }\n }\n \n function formatResponseBody(response, responseType) {\n if (!response) return null;\n \n try {\n if (responseType === '' || responseType === 'text') {\n return String(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'json') {\n return JSON.stringify(response).substring(0, MAX_BODY_SIZE);\n }\n if (responseType === 'document') {\n return '[Document]';\n }\n if (responseType === 'blob') {\n return '[Blob: size=' + response.size + ']';\n }\n if (responseType === 'arraybuffer') {\n return '[ArrayBuffer: size=' + response.byteLength + ']';\n }\n return String(response).substring(0, MAX_BODY_SIZE);\n } catch (e) {\n return '[Error reading response: ' + e.message + ']';\n }\n }\n \n // Intercept XMLHttpRequest\n var originalXHROpen = XMLHttpRequest.prototype.open;\n var originalXHRSend = XMLHttpRequest.prototype.send;\n var originalXHRSetRequestHeader = XMLHttpRequest.prototype.setRequestHeader;\n \n XMLHttpRequest.prototype.open = function(method, url, async, user, password) {\n this.__requestInfo__ = {\n method: method,\n url: url,\n startTime: null,\n headers: {}\n };\n return originalXHROpen.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.setRequestHeader = function(name, value) {\n if (this.__requestInfo__ && this.__requestInfo__.headers) {\n this.__requestInfo__.headers[name] = value;\n }\n return originalXHRSetRequestHeader.apply(this, arguments);\n };\n \n XMLHttpRequest.prototype.send = function(body) {\n var xhr = this;\n var requestInfo = xhr.__requestInfo__;\n \n if (requestInfo) {\n // Skip SSE requests and filtered requests\n if (isSSERequest(requestInfo.url, requestInfo.headers) || shouldFilterRequest(requestInfo.url)) {\n return originalXHRSend.apply(this, arguments);\n }\n \n requestInfo.startTime = Date.now();\n requestInfo.requestBody = formatRequestBody(body);\n \n xhr.addEventListener('loadend', function() {\n var contentType = xhr.getResponseHeader('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return;\n }\n \n var responseHeaders = {};\n var allHeaders = xhr.getAllResponseHeaders();\n if (allHeaders) {\n allHeaders.split('\\\\r\\\\n').forEach(function(line) {\n var parts = line.split(': ');\n if (parts.length === 2) {\n responseHeaders[parts[0]] = parts[1];\n }\n });\n }\n \n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'xhr',\n method: requestInfo.method,\n url: requestInfo.url,\n status: xhr.status,\n statusText: xhr.statusText,\n duration: Date.now() - requestInfo.startTime,\n requestHeaders: truncateHeaders(requestInfo.headers),\n requestBody: requestInfo.requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseType: xhr.responseType,\n responseBody: formatResponseBody(xhr.response, xhr.responseType),\n responseSize: xhr.response ? (typeof xhr.response === 'string' ? xhr.response.length : (xhr.response.byteLength || xhr.response.size || null)) : null\n };\n \n addLog(entry);\n });\n }\n \n return originalXHRSend.apply(this, arguments);\n };\n \n // Intercept Fetch API\n var originalFetch = window.fetch;\n \n function headersToObject(headers) {\n var obj = {};\n if (!headers) return obj;\n \n if (typeof headers.forEach === 'function') {\n headers.forEach(function(value, key) {\n obj[key] = value;\n });\n } else if (typeof headers === 'object') {\n for (var key in headers) {\n obj[key] = headers[key];\n }\n }\n return obj;\n }\n \n window.fetch = function(input, init) {\n var startTime = Date.now();\n var method = (init && init.method) || 'GET';\n var url = typeof input === 'string' ? input : input.url;\n var headers = init && init.headers;\n \n // Skip SSE requests and filtered requests\n if (isSSERequest(url, headers) || shouldFilterRequest(url)) {\n return originalFetch.apply(this, arguments);\n }\n \n var requestHeaders = headersToObject(headers);\n var requestBody = formatRequestBody(init && init.body);\n \n return originalFetch.apply(this, arguments)\n .then(function(response) {\n var contentType = response.headers.get('Content-Type') || '';\n if (contentType.indexOf('text/event-stream') !== -1) {\n return response;\n }\n \n var clonedResponse = response.clone();\n var responseHeaders = headersToObject(response.headers);\n \n clonedResponse.text().then(function(bodyText) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: bodyText ? bodyText.substring(0, MAX_BODY_SIZE) : null,\n responseSize: bodyText ? bodyText.length : null,\n ok: response.ok\n };\n \n addLog(entry);\n }).catch(function(e) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: response.status,\n statusText: response.statusText,\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: truncateHeaders(responseHeaders),\n responseBody: '[Unable to read: ' + e.message + ']',\n responseSize: null,\n ok: response.ok\n };\n \n addLog(entry);\n });\n \n return response;\n })\n .catch(function(error) {\n var entry = {\n type: 'request',\n timestamp: new Date().toISOString(),\n requestType: 'fetch',\n method: method,\n url: url,\n status: 0,\n statusText: 'Network Error',\n duration: Date.now() - startTime,\n requestHeaders: truncateHeaders(requestHeaders),\n requestBody: requestBody,\n responseHeaders: null,\n responseBody: null,\n error: error.message\n };\n \n addLog(entry);\n throw error;\n });\n };\n \n // ============================================\n // Global error capture\n // ============================================\n window.addEventListener('error', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: event.message,\n stack: event.error ? event.error.stack : 'at ' + event.filename + ':' + event.lineno + ':' + event.colno\n };\n addLog(entry);\n });\n \n window.addEventListener('unhandledrejection', function(event) {\n var entry = {\n type: 'error',\n timestamp: new Date().toISOString(),\n level: 'error',\n message: 'Unhandled Promise Rejection: ' + (event.reason ? (event.reason.message || String(event.reason)) : 'Unknown'),\n stack: event.reason && event.reason.stack ? event.reason.stack : ''\n };\n addLog(entry);\n });\n \n // ============================================\n // Send remaining logs on page unload\n // ============================================\n window.addEventListener('beforeunload', function() {\n if (writeQueue.length > 0) {\n // Use sendBeacon to ensure logs are sent\n writeQueue.forEach(function(entry) {\n navigator.sendBeacon(LOG_API_PATH, JSON.stringify(entry));\n });\n writeQueue = [];\n }\n });\n \n // ============================================\n // Provide manual flush method\n // ============================================\n window.__flushBrowserLogs__ = function() {\n return new Promise(function(resolve) {\n // Wait for queue to finish processing\n function checkQueue() {\n if (writeQueue.length === 0 && !isWriting) {\n resolve();\n } else {\n setTimeout(checkQueue, 100);\n }\n }\n checkQueue();\n });\n };\n \n // Provide method to get queue status\n window.__getBrowserLogsStatus__ = function() {\n return {\n queueLength: writeQueue.length,\n isWriting: isWriting\n };\n };\n \n originalConsole.log('[BrowserLogs] Log collection started');\n})();\n</script>`;\n\n return {\n name: \"vite-plugin-browser-logs\",\n\n configResolved(config) {\n // Determine log file path: same level as index.html\n const root = config.root || process.cwd();\n logFilePath = path.join(root, \"browser.log\");\n },\n\n configureServer(devServer) {\n // Add log write API\n devServer.middlewares.use((req, res, next) => {\n if (req.url === \"/__browser__\" && req.method === \"POST\") {\n // Get request origin, dynamically set CORS headers to avoid protocol mismatch\n const origin = req.headers.origin || \"*\";\n\n let body = \"\";\n req.on(\"data\", (chunk: Buffer) => {\n body += chunk.toString();\n });\n req.on(\"end\", () => {\n try {\n // Ensure log directory exists\n const logDir = path.dirname(logFilePath);\n if (!fs.existsSync(logDir)) {\n fs.mkdirSync(logDir, { recursive: true });\n }\n // Append to log file\n fs.appendFileSync(logFilePath, `${body}\\n`, \"utf-8\");\n res.writeHead(200, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: true }));\n } catch (error) {\n console.error(\"[BrowserLogs] Write error:\", error);\n res.writeHead(500, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n });\n res.end(JSON.stringify({ success: false, error: String(error) }));\n }\n });\n } else if (req.url === \"/__browser__\" && req.method === \"OPTIONS\") {\n // Handle CORS preflight request\n const origin = req.headers.origin || \"*\";\n res.writeHead(204, {\n \"Access-Control-Allow-Origin\": origin,\n \"Access-Control-Allow-Methods\": \"POST, OPTIONS\",\n \"Access-Control-Allow-Headers\": \"Content-Type\",\n \"Access-Control-Max-Age\": \"86400\",\n });\n res.end();\n } else if (req.url === \"/__browser__\") {\n const origin = req.headers.origin || \"*\";\n res.writeHead(405, {\n \"Content-Type\": \"application/json\",\n \"Access-Control-Allow-Origin\": origin,\n });\n res.end(JSON.stringify({ error: \"Method not allowed\" }));\n } else {\n next();\n }\n });\n\n console.log(\"[BrowserLogs] Logs will be written to:\", logFilePath);\n },\n\n transformIndexHtml(html) {\n // Insert script after <head> tag to ensure earliest execution\n return html.replace(/<head([^>]*)>/i, `<head$1>${injectedScript}`);\n },\n };\n}\n","import { createHash } from \"node:crypto\";\nimport type { Plugin } from \"vite\";\n\nfunction generateUniqueId(filePath: string, position: number): string {\n const key = `${filePath}:${position}`;\n return createHash(\"md5\").update(key).digest(\"hex\").substring(0, 12);\n}\n\nconst HTML_TAGS = new Set([\n \"a\", \"abbr\", \"address\", \"area\", \"article\", \"aside\", \"audio\", \"b\", \"base\", \"bdi\", \"bdo\",\n \"blockquote\", \"body\", \"br\", \"button\", \"canvas\", \"caption\", \"cite\", \"code\", \"col\",\n \"colgroup\", \"data\", \"datalist\", \"dd\", \"del\", \"details\", \"dfn\", \"dialog\", \"div\", \"dl\",\n \"dt\", \"em\", \"embed\", \"fieldset\", \"figcaption\", \"figure\", \"footer\", \"form\", \"h1\", \"h2\",\n \"h3\", \"h4\", \"h5\", \"h6\", \"head\", \"header\", \"hgroup\", \"hr\", \"html\", \"i\", \"iframe\", \"img\",\n \"input\", \"ins\", \"kbd\", \"label\", \"legend\", \"li\", \"link\", \"main\", \"map\", \"mark\", \"meta\",\n \"meter\", \"nav\", \"noscript\", \"object\", \"ol\", \"optgroup\", \"option\", \"output\", \"p\", \"param\",\n \"picture\", \"pre\", \"progress\", \"q\", \"rp\", \"rt\", \"ruby\", \"s\", \"samp\", \"script\", \"section\",\n \"select\", \"small\", \"source\", \"span\", \"strong\", \"style\", \"sub\", \"summary\", \"sup\", \"svg\",\n \"table\", \"tbody\", \"td\", \"template\", \"textarea\", \"tfoot\", \"th\", \"thead\", \"time\", \"title\",\n \"tr\", \"track\", \"u\", \"ul\", \"var\", \"video\", \"wbr\",\n]);\n\n/**\n * Vite plugin: Add data-node-component-id to React components\n * Only enabled in development mode\n */\nexport function componentIdPlugin(): Plugin {\n let isDev = false;\n\n return {\n name: \"vite-plugin-component-id\",\n enforce: \"pre\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!/\\.(tsx|jsx)$/.test(id)) {\n return null;\n }\n\n if (id.includes(\"node_modules\")) {\n return null;\n }\n\n if (!/<[a-z]/.test(code)) {\n return null;\n }\n\n try {\n let transformedCode = code;\n const jsxOpenTagRegex = /<([a-z][\\da-z]*)(?=[\\s/>])/g;\n\n let match: RegExpExecArray | null;\n const matches: Array<{ index: number; tag: string; tagEndIndex: number }> = [];\n\n while ((match = jsxOpenTagRegex.exec(code)) !== null) {\n const tag = match[1];\n if (!tag) continue;\n \n const tagStartIndex = match.index;\n const tagEndIndex = match.index + match[0].length;\n\n if (!HTML_TAGS.has(tag)) {\n continue;\n }\n\n let checkIndex = tagEndIndex;\n let foundClosing = false;\n let hasComponentId = false;\n\n while (checkIndex < code.length && !foundClosing) {\n const char = code[checkIndex];\n if (char === \">\") {\n foundClosing = true;\n const tagContent = code.substring(tagStartIndex, checkIndex);\n if (tagContent.includes(\"data-node-component-id\")) {\n hasComponentId = true;\n }\n }\n checkIndex++;\n }\n\n if (hasComponentId) {\n continue;\n }\n\n matches.push({\n index: tagStartIndex,\n tag,\n tagEndIndex,\n });\n }\n\n let currentCode = code;\n for (let i = matches.length - 1; i >= 0; i--) {\n const item = matches[i];\n if (!item) continue;\n \n const { index, tagEndIndex } = item;\n const uniqueId = generateUniqueId(id, index);\n\n const before = currentCode.substring(0, tagEndIndex);\n const after = currentCode.substring(tagEndIndex);\n\n transformedCode = `${before} data-node-component-id=\"${uniqueId}\"${after}`;\n currentCode = transformedCode;\n }\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\nconst clickHandlerScript = `<script>\ndocument.addEventListener(\"click\", (e) => {\n const element = e.target;\n const noJumpOut = document.body.classList.contains(\"forbid-jump-out\")\n if (element.hasAttribute(\"data-link-href\") && !noJumpOut) {\n const href = element.getAttribute(\"data-link-href\");\n const target = element.getAttribute(\"data-link-target\");\n if (href) {\n if (target === \"_blank\") {\n window.open(href, \"_blank\");\n } else {\n window.location.href = href;\n }\n }\n } else if(noJumpOut && element.tagName === \"A\") {\n e.preventDefault();\n }\n});\n</script>`;\n\n/**\n * Vite plugin: Inject editor bridge script\n * Injects bridge.js in development mode\n */\nexport function editorBridgePlugin(options?: { bridgeScriptPath?: string }): Plugin {\n let isDev = false;\n const bridgePath = options?.bridgeScriptPath || \"/scripts/bridge.js\";\n\n return {\n name: \"vite-plugin-editor-bridge\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transformIndexHtml(html) {\n const devScript = isDev ? `<script type=\"module\" src=\"${bridgePath}\"></script>` : \"\";\n\n const scriptsToInject = `${clickHandlerScript + devScript}</body>`;\n\n return html.replace(\"</body>\", scriptsToInject);\n },\n };\n}\n","import type { Plugin } from \"vite\";\n\n/**\n * Vite plugin: Expose routes to window.__APP_ROUTES__\n * Only enabled in development mode\n */\nexport function routesExposePlugin(options?: { routesFilePath?: string }): Plugin {\n let isDev = false;\n const routesPath = options?.routesFilePath || \"src/routes.tsx\";\n\n return {\n name: \"vite-plugin-routes-expose\",\n enforce: \"post\",\n\n configResolved(config) {\n isDev = config.command === \"serve\";\n },\n\n transform(code: string, id: string) {\n if (!isDev) {\n return null;\n }\n\n if (!id.endsWith(routesPath)) {\n return null;\n }\n\n try {\n if (code.includes(\"window.__APP_ROUTES__\")) {\n return null;\n }\n\n const transformedCode = `${code}\n\n// Development mode: Expose routes to window.__APP_ROUTES__\nif (typeof window !== 'undefined') {\n window.__APP_ROUTES__ = typeof routes !== 'undefined' && Array.isArray(routes) ? routes : [];\n}\n`;\n\n return {\n code: transformedCode,\n map: null,\n };\n } catch {\n return null;\n }\n },\n };\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@amaster.ai/vite-plugins",
3
- "version": "1.0.0-beta.0",
3
+ "version": "1.0.0-beta.1",
4
4
  "description": "Collection of Vite plugins for React applications",
5
5
  "type": "module",
6
6
  "main": "./dist/index.cjs",