baseguard 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (244) hide show
  1. package/.eslintrc.json +25 -0
  2. package/.prettierrc +8 -0
  3. package/README.md +94 -0
  4. package/bin/base.js +494 -0
  5. package/dist/ai/fix-manager.d.ts +67 -0
  6. package/dist/ai/fix-manager.d.ts.map +1 -0
  7. package/dist/ai/fix-manager.js +326 -0
  8. package/dist/ai/fix-manager.js.map +1 -0
  9. package/dist/ai/gemini-analyzer.d.ts +116 -0
  10. package/dist/ai/gemini-analyzer.d.ts.map +1 -0
  11. package/dist/ai/gemini-analyzer.js +572 -0
  12. package/dist/ai/gemini-analyzer.js.map +1 -0
  13. package/dist/ai/index.d.ts +4 -0
  14. package/dist/ai/index.d.ts.map +1 -0
  15. package/dist/ai/index.js +5 -0
  16. package/dist/ai/index.js.map +1 -0
  17. package/dist/ai/jules-implementer.d.ts +115 -0
  18. package/dist/ai/jules-implementer.d.ts.map +1 -0
  19. package/dist/ai/jules-implementer.js +387 -0
  20. package/dist/ai/jules-implementer.js.map +1 -0
  21. package/dist/commands/automation.d.ts +5 -0
  22. package/dist/commands/automation.d.ts.map +1 -0
  23. package/dist/commands/automation.js +305 -0
  24. package/dist/commands/automation.js.map +1 -0
  25. package/dist/commands/check.d.ts +9 -0
  26. package/dist/commands/check.d.ts.map +1 -0
  27. package/dist/commands/check.js +113 -0
  28. package/dist/commands/check.js.map +1 -0
  29. package/dist/commands/config.d.ts +11 -0
  30. package/dist/commands/config.d.ts.map +1 -0
  31. package/dist/commands/config.js +324 -0
  32. package/dist/commands/config.js.map +1 -0
  33. package/dist/commands/fix.d.ts +9 -0
  34. package/dist/commands/fix.d.ts.map +1 -0
  35. package/dist/commands/fix.js +207 -0
  36. package/dist/commands/fix.js.map +1 -0
  37. package/dist/commands/index.d.ts +6 -0
  38. package/dist/commands/index.d.ts.map +1 -0
  39. package/dist/commands/index.js +7 -0
  40. package/dist/commands/index.js.map +1 -0
  41. package/dist/commands/init.d.ts +9 -0
  42. package/dist/commands/init.d.ts.map +1 -0
  43. package/dist/commands/init.js +125 -0
  44. package/dist/commands/init.js.map +1 -0
  45. package/dist/core/api-key-manager.d.ts +83 -0
  46. package/dist/core/api-key-manager.d.ts.map +1 -0
  47. package/dist/core/api-key-manager.js +244 -0
  48. package/dist/core/api-key-manager.js.map +1 -0
  49. package/dist/core/baseguard.d.ts +46 -0
  50. package/dist/core/baseguard.d.ts.map +1 -0
  51. package/dist/core/baseguard.js +132 -0
  52. package/dist/core/baseguard.js.map +1 -0
  53. package/dist/core/baseline-checker.d.ts +63 -0
  54. package/dist/core/baseline-checker.d.ts.map +1 -0
  55. package/dist/core/baseline-checker.js +502 -0
  56. package/dist/core/baseline-checker.js.map +1 -0
  57. package/dist/core/cache-manager.d.ts +88 -0
  58. package/dist/core/cache-manager.d.ts.map +1 -0
  59. package/dist/core/cache-manager.js +213 -0
  60. package/dist/core/cache-manager.js.map +1 -0
  61. package/dist/core/configuration.d.ts +140 -0
  62. package/dist/core/configuration.d.ts.map +1 -0
  63. package/dist/core/configuration.js +474 -0
  64. package/dist/core/configuration.js.map +1 -0
  65. package/dist/core/directory-filter.d.ts +90 -0
  66. package/dist/core/directory-filter.d.ts.map +1 -0
  67. package/dist/core/directory-filter.js +319 -0
  68. package/dist/core/directory-filter.js.map +1 -0
  69. package/dist/core/error-handler.d.ts +110 -0
  70. package/dist/core/error-handler.d.ts.map +1 -0
  71. package/dist/core/error-handler.js +392 -0
  72. package/dist/core/error-handler.js.map +1 -0
  73. package/dist/core/file-processor.d.ts +80 -0
  74. package/dist/core/file-processor.d.ts.map +1 -0
  75. package/dist/core/file-processor.js +259 -0
  76. package/dist/core/file-processor.js.map +1 -0
  77. package/dist/core/gitignore-manager.d.ts +44 -0
  78. package/dist/core/gitignore-manager.d.ts.map +1 -0
  79. package/dist/core/gitignore-manager.js +147 -0
  80. package/dist/core/gitignore-manager.js.map +1 -0
  81. package/dist/core/index.d.ts +13 -0
  82. package/dist/core/index.d.ts.map +1 -0
  83. package/dist/core/index.js +13 -0
  84. package/dist/core/index.js.map +1 -0
  85. package/dist/core/lazy-loader.d.ts +68 -0
  86. package/dist/core/lazy-loader.d.ts.map +1 -0
  87. package/dist/core/lazy-loader.js +260 -0
  88. package/dist/core/lazy-loader.js.map +1 -0
  89. package/dist/core/memory-manager.d.ts +1 -0
  90. package/dist/core/memory-manager.d.ts.map +1 -0
  91. package/dist/core/memory-manager.js +2 -0
  92. package/dist/core/memory-manager.js.map +1 -0
  93. package/dist/core/startup-optimizer.d.ts +45 -0
  94. package/dist/core/startup-optimizer.d.ts.map +1 -0
  95. package/dist/core/startup-optimizer.js +140 -0
  96. package/dist/core/startup-optimizer.js.map +1 -0
  97. package/dist/git/automation-engine.d.ts +58 -0
  98. package/dist/git/automation-engine.d.ts.map +1 -0
  99. package/dist/git/automation-engine.js +318 -0
  100. package/dist/git/automation-engine.js.map +1 -0
  101. package/dist/git/github-manager.d.ts +71 -0
  102. package/dist/git/github-manager.d.ts.map +1 -0
  103. package/dist/git/github-manager.js +226 -0
  104. package/dist/git/github-manager.js.map +1 -0
  105. package/dist/git/hook-manager.d.ts +43 -0
  106. package/dist/git/hook-manager.d.ts.map +1 -0
  107. package/dist/git/hook-manager.js +191 -0
  108. package/dist/git/hook-manager.js.map +1 -0
  109. package/dist/git/index.d.ts +4 -0
  110. package/dist/git/index.d.ts.map +1 -0
  111. package/dist/git/index.js +5 -0
  112. package/dist/git/index.js.map +1 -0
  113. package/dist/index.d.ts +8 -0
  114. package/dist/index.d.ts.map +1 -0
  115. package/dist/index.js +9 -0
  116. package/dist/index.js.map +1 -0
  117. package/dist/parsers/feature-validator.d.ts +60 -0
  118. package/dist/parsers/feature-validator.d.ts.map +1 -0
  119. package/dist/parsers/feature-validator.js +483 -0
  120. package/dist/parsers/feature-validator.js.map +1 -0
  121. package/dist/parsers/index.d.ts +8 -0
  122. package/dist/parsers/index.d.ts.map +1 -0
  123. package/dist/parsers/index.js +9 -0
  124. package/dist/parsers/index.js.map +1 -0
  125. package/dist/parsers/parser-manager.d.ts +103 -0
  126. package/dist/parsers/parser-manager.d.ts.map +1 -0
  127. package/dist/parsers/parser-manager.js +321 -0
  128. package/dist/parsers/parser-manager.js.map +1 -0
  129. package/dist/parsers/parser.d.ts +23 -0
  130. package/dist/parsers/parser.d.ts.map +1 -0
  131. package/dist/parsers/parser.js +6 -0
  132. package/dist/parsers/parser.js.map +1 -0
  133. package/dist/parsers/react-parser.d.ts +22 -0
  134. package/dist/parsers/react-parser.d.ts.map +1 -0
  135. package/dist/parsers/react-parser.js +307 -0
  136. package/dist/parsers/react-parser.js.map +1 -0
  137. package/dist/parsers/svelte-parser.d.ts +33 -0
  138. package/dist/parsers/svelte-parser.d.ts.map +1 -0
  139. package/dist/parsers/svelte-parser.js +408 -0
  140. package/dist/parsers/svelte-parser.js.map +1 -0
  141. package/dist/parsers/vanilla-parser.d.ts +31 -0
  142. package/dist/parsers/vanilla-parser.d.ts.map +1 -0
  143. package/dist/parsers/vanilla-parser.js +590 -0
  144. package/dist/parsers/vanilla-parser.js.map +1 -0
  145. package/dist/parsers/vue-parser.d.ts +9 -0
  146. package/dist/parsers/vue-parser.d.ts.map +1 -0
  147. package/dist/parsers/vue-parser.js +16 -0
  148. package/dist/parsers/vue-parser.js.map +1 -0
  149. package/dist/terminal-header.d.ts +12 -0
  150. package/dist/terminal-header.js +45 -0
  151. package/dist/types/index.d.ts +83 -0
  152. package/dist/types/index.d.ts.map +1 -0
  153. package/dist/types/index.js +5 -0
  154. package/dist/types/index.js.map +1 -0
  155. package/dist/ui/components.d.ts +133 -0
  156. package/dist/ui/components.d.ts.map +1 -0
  157. package/dist/ui/components.js +482 -0
  158. package/dist/ui/components.js.map +1 -0
  159. package/dist/ui/help.d.ts +11 -0
  160. package/dist/ui/help.d.ts.map +1 -0
  161. package/dist/ui/help.js +161 -0
  162. package/dist/ui/help.js.map +1 -0
  163. package/dist/ui/index.d.ts +5 -0
  164. package/dist/ui/index.d.ts.map +1 -0
  165. package/dist/ui/index.js +5 -0
  166. package/dist/ui/index.js.map +1 -0
  167. package/dist/ui/prompts.d.ts +63 -0
  168. package/dist/ui/prompts.d.ts.map +1 -0
  169. package/dist/ui/prompts.js +611 -0
  170. package/dist/ui/prompts.js.map +1 -0
  171. package/dist/ui/terminal-header.d.ts +13 -0
  172. package/dist/ui/terminal-header.d.ts.map +1 -0
  173. package/dist/ui/terminal-header.js +46 -0
  174. package/dist/ui/terminal-header.js.map +1 -0
  175. package/package.json +80 -0
  176. package/src/ai/__tests__/gemini-analyzer.test.ts +181 -0
  177. package/src/ai/fix-manager.ts +362 -0
  178. package/src/ai/gemini-analyzer.ts +671 -0
  179. package/src/ai/index.ts +4 -0
  180. package/src/ai/jules-implementer.ts +459 -0
  181. package/src/commands/automation.ts +344 -0
  182. package/src/commands/check.ts +299 -0
  183. package/src/commands/config.ts +365 -0
  184. package/src/commands/fix.ts +234 -0
  185. package/src/commands/index.ts +6 -0
  186. package/src/commands/init.ts +142 -0
  187. package/src/commands/status.ts +0 -0
  188. package/src/core/api-key-manager.ts +298 -0
  189. package/src/core/baseguard.ts +742 -0
  190. package/src/core/baseline-checker.ts +563 -0
  191. package/src/core/cache-manager.ts +270 -0
  192. package/src/core/configuration-recovery.ts +676 -0
  193. package/src/core/configuration.ts +559 -0
  194. package/src/core/debug-logger.ts +590 -0
  195. package/src/core/directory-filter.ts +421 -0
  196. package/src/core/error-handler.ts +517 -0
  197. package/src/core/file-processor.ts +331 -0
  198. package/src/core/gitignore-manager.ts +169 -0
  199. package/src/core/graceful-degradation-manager.ts +596 -0
  200. package/src/core/index.ts +13 -0
  201. package/src/core/lazy-loader.ts +307 -0
  202. package/src/core/logger.ts +0 -0
  203. package/src/core/memory-manager.ts +294 -0
  204. package/src/core/startup-optimizer.ts +173 -0
  205. package/src/core/system-error-handler.ts +746 -0
  206. package/src/git/automation-engine.ts +361 -0
  207. package/src/git/github-manager.ts +260 -0
  208. package/src/git/hook-manager.ts +210 -0
  209. package/src/git/index.ts +4 -0
  210. package/src/index.ts +8 -0
  211. package/src/parsers/feature-validator.ts +559 -0
  212. package/src/parsers/index.ts +8 -0
  213. package/src/parsers/parser-manager.ts +419 -0
  214. package/src/parsers/parser.ts +26 -0
  215. package/src/parsers/react-parser-optimized.ts +161 -0
  216. package/src/parsers/react-parser.ts +359 -0
  217. package/src/parsers/svelte-parser.ts +506 -0
  218. package/src/parsers/vanilla-parser.ts +682 -0
  219. package/src/parsers/vue-parser.ts +472 -0
  220. package/src/types/index.ts +92 -0
  221. package/src/ui/components.ts +567 -0
  222. package/src/ui/help.ts +193 -0
  223. package/src/ui/index.ts +4 -0
  224. package/src/ui/prompts.ts +688 -0
  225. package/src/ui/terminal-header.ts +59 -0
  226. package/test-config-commands.js +56 -0
  227. package/test-header-simple.js +33 -0
  228. package/test-terminal-header.js +12 -0
  229. package/test-ui.js +29 -0
  230. package/tests/e2e/baseguard.e2e.test.ts +516 -0
  231. package/tests/e2e/cross-platform.e2e.test.ts +420 -0
  232. package/tests/e2e/git-integration.e2e.test.ts +487 -0
  233. package/tests/fixtures/react-project/package.json +14 -0
  234. package/tests/fixtures/react-project/src/App.css +76 -0
  235. package/tests/fixtures/react-project/src/App.tsx +77 -0
  236. package/tests/fixtures/svelte-project/package.json +11 -0
  237. package/tests/fixtures/svelte-project/src/App.svelte +369 -0
  238. package/tests/fixtures/vanilla-project/index.html +76 -0
  239. package/tests/fixtures/vanilla-project/script.js +331 -0
  240. package/tests/fixtures/vanilla-project/styles.css +359 -0
  241. package/tests/fixtures/vue-project/package.json +12 -0
  242. package/tests/fixtures/vue-project/src/App.vue +216 -0
  243. package/tsconfig.json +36 -0
  244. package/vitest.config.ts +10 -0
@@ -0,0 +1,359 @@
1
+ import { Parser } from './parser.js';
2
+ import type { DetectedFeature } from '../types/index.js';
3
+ import { LazyLoader } from '../core/lazy-loader.js';
4
+
5
+ /**
6
+ * React/JSX parser using Babel - extracts ALL web platform features
7
+ * while ignoring React-specific APIs
8
+ */
9
+ export class ReactParser extends Parser {
10
+ private readonly REACT_SPECIFIC_APIS = new Set([
11
+ // React hooks
12
+ 'useState', 'useEffect', 'useContext', 'useReducer', 'useCallback', 'useMemo',
13
+ 'useRef', 'useImperativeHandle', 'useLayoutEffect', 'useDebugValue',
14
+ 'useId', 'useDeferredValue', 'useTransition', 'useSyncExternalStore',
15
+
16
+ // React components and APIs
17
+ 'React', 'Component', 'PureComponent', 'Fragment', 'StrictMode',
18
+ 'Suspense', 'lazy', 'memo', 'forwardRef', 'createContext', 'createElement',
19
+ 'cloneElement', 'isValidElement', 'Children', 'createRef',
20
+
21
+ // React DOM
22
+ 'ReactDOM', 'render', 'hydrate', 'unmountComponentAtNode', 'findDOMNode',
23
+ 'createPortal', 'flushSync',
24
+
25
+ // JSX elements (these are handled separately)
26
+ 'jsx', 'jsxs', '_jsx', '_jsxs'
27
+ ]);
28
+
29
+ private readonly WEB_PLATFORM_APIS = new Set([
30
+ // Canvas APIs
31
+ 'getContext', 'CanvasRenderingContext2D', 'WebGLRenderingContext', 'WebGL2RenderingContext',
32
+ 'OffscreenCanvas', 'ImageBitmap', 'createImageBitmap', 'Path2D',
33
+
34
+ // WebRTC APIs
35
+ 'RTCPeerConnection', 'RTCDataChannel', 'RTCSessionDescription', 'RTCIceCandidate',
36
+ 'getUserMedia', 'getDisplayMedia', 'MediaStream', 'MediaStreamTrack',
37
+
38
+ // WebAssembly
39
+ 'WebAssembly', 'instantiate', 'compile', 'validate',
40
+
41
+ // Service Workers & PWA
42
+ 'ServiceWorker', 'serviceWorker', 'register', 'Cache', 'caches',
43
+ 'PushManager', 'Notification', 'showNotification',
44
+
45
+ // DOM APIs
46
+ 'querySelector', 'querySelectorAll', 'getElementById', 'getElementsByClassName',
47
+ 'addEventListener', 'removeEventListener', 'dispatchEvent', 'CustomEvent',
48
+ 'MutationObserver', 'ResizeObserver', 'IntersectionObserver', 'PerformanceObserver',
49
+ 'AbortController', 'AbortSignal', 'FormData', 'URLSearchParams', 'URL',
50
+ 'fetch', 'Request', 'Response', 'Headers', 'Blob', 'File', 'FileReader',
51
+
52
+ // Web APIs
53
+ 'navigator', 'geolocation', 'permissions', 'clipboard', 'share',
54
+ 'requestAnimationFrame', 'cancelAnimationFrame', 'requestIdleCallback',
55
+ 'setTimeout', 'setInterval', 'clearTimeout', 'clearInterval',
56
+ 'localStorage', 'sessionStorage', 'indexedDB', 'crypto', 'performance',
57
+
58
+ // Audio/Video APIs
59
+ 'AudioContext', 'MediaRecorder', 'MediaSource', 'SourceBuffer',
60
+ 'HTMLMediaElement', 'HTMLAudioElement', 'HTMLVideoElement',
61
+
62
+ // Modern JavaScript APIs
63
+ 'structuredClone', 'queueMicrotask', 'reportError',
64
+ 'WeakRef', 'FinalizationRegistry', 'AggregateError',
65
+
66
+ // Intl APIs
67
+ 'Intl', 'DateTimeFormat', 'NumberFormat', 'Collator', 'PluralRules',
68
+ 'RelativeTimeFormat', 'ListFormat', 'Locale'
69
+ ]);
70
+
71
+ private readonly CSS_PROPERTIES = new Set([
72
+ // Container Queries
73
+ 'container-type', 'container-name', 'container',
74
+
75
+ // Grid & Flexbox
76
+ 'display', 'grid-template-columns', 'grid-template-rows', 'gap', 'grid-gap',
77
+ 'flex', 'flex-direction', 'flex-wrap', 'justify-content', 'align-items',
78
+
79
+ // Modern CSS
80
+ 'aspect-ratio', 'object-fit', 'object-position', 'backdrop-filter',
81
+ 'color-scheme', 'accent-color', 'scroll-behavior', 'scroll-snap-type',
82
+ 'overscroll-behavior', 'touch-action', 'user-select',
83
+
84
+ // CSS Custom Properties
85
+ '--', 'var(', 'calc(', 'clamp(', 'min(', 'max(',
86
+
87
+ // Transforms & Animations
88
+ 'transform', 'transform-origin', 'perspective', 'backface-visibility',
89
+ 'animation', 'transition', 'will-change',
90
+
91
+ // Layout
92
+ 'position', 'top', 'right', 'bottom', 'left', 'z-index',
93
+ 'width', 'height', 'margin', 'padding', 'border', 'outline'
94
+ ]);
95
+
96
+ canParse(filePath: string): boolean {
97
+ return /\.(jsx|tsx)$/.test(filePath);
98
+ }
99
+
100
+ async parseFeatures(content: string, filePath: string): Promise<DetectedFeature[]> {
101
+ const features: DetectedFeature[] = [];
102
+
103
+ try {
104
+ // Lazy load Babel dependencies
105
+ const [parser, traverse, types] = await Promise.all([
106
+ LazyLoader.getBabelParser(),
107
+ LazyLoader.getBabelTraverse(),
108
+ import('@babel/types')
109
+ ]);
110
+
111
+ const t = types.default || types;
112
+ const ast = parser.parse(content, {
113
+ sourceType: 'module',
114
+ plugins: [
115
+ 'jsx',
116
+ 'typescript',
117
+ 'decorators-legacy',
118
+ 'classProperties',
119
+ 'objectRestSpread',
120
+ 'asyncGenerators',
121
+ 'functionBind',
122
+ 'exportDefaultFrom',
123
+ 'exportNamespaceFrom',
124
+ 'dynamicImport',
125
+ 'nullishCoalescingOperator',
126
+ 'optionalChaining',
127
+ 'topLevelAwait'
128
+ ]
129
+ });
130
+
131
+ traverse(ast, {
132
+ // Extract JavaScript Web APIs
133
+ MemberExpression: (path: any) => {
134
+ const feature = this.extractWebAPIFeature(path.node, content, t);
135
+ if (feature) {
136
+ features.push({
137
+ ...feature,
138
+ file: filePath,
139
+ line: path.node.loc?.start.line || 0,
140
+ column: path.node.loc?.start.column || 0
141
+ });
142
+ }
143
+ },
144
+
145
+ // Extract function calls to Web APIs
146
+ CallExpression: (path: any) => {
147
+ const feature = this.extractWebAPICall(path.node, content, t);
148
+ if (feature) {
149
+ features.push({
150
+ ...feature,
151
+ file: filePath,
152
+ line: path.node.loc?.start.line || 0,
153
+ column: path.node.loc?.start.column || 0
154
+ });
155
+ }
156
+ },
157
+
158
+ // Extract CSS from inline styles
159
+ ObjectExpression: (path: any) => {
160
+ const cssFeatures = this.extractInlineCSS(path.node, content, t);
161
+ cssFeatures.forEach(feature => {
162
+ features.push({
163
+ ...feature,
164
+ file: filePath,
165
+ line: path.node.loc?.start.line || 0,
166
+ column: path.node.loc?.start.column || 0
167
+ });
168
+ });
169
+ },
170
+
171
+ // Extract modern JavaScript syntax features
172
+ OptionalMemberExpression: (path: any) => {
173
+ features.push({
174
+ feature: 'optional-chaining',
175
+ type: 'js',
176
+ context: this.getContext(content, path.node.loc?.start.line || 0),
177
+ line: path.node.loc?.start.line || 0,
178
+ column: path.node.loc?.start.column || 0,
179
+ file: filePath
180
+ });
181
+ },
182
+
183
+ // Nullish coalescing
184
+ LogicalExpression: (path: any) => {
185
+ if (path.node.operator === '??') {
186
+ features.push({
187
+ feature: 'nullish-coalescing',
188
+ type: 'js',
189
+ context: this.getContext(content, path.node.loc?.start.line || 0),
190
+ line: path.node.loc?.start.line || 0,
191
+ column: path.node.loc?.start.column || 0,
192
+ file: filePath
193
+ });
194
+ }
195
+ },
196
+
197
+ // Private class fields
198
+ ClassPrivateProperty: (path: any) => {
199
+ features.push({
200
+ feature: 'private-fields',
201
+ type: 'js',
202
+ context: this.getContext(content, path.node.loc?.start.line || 0),
203
+ line: path.node.loc?.start.line || 0,
204
+ column: path.node.loc?.start.column || 0,
205
+ file: filePath
206
+ });
207
+ },
208
+
209
+ // Top-level await
210
+ AwaitExpression: (path: any) => {
211
+ if (this.isTopLevelAwait(path, t)) {
212
+ features.push({
213
+ feature: 'top-level-await',
214
+ type: 'js',
215
+ context: this.getContext(content, path.node.loc?.start.line || 0),
216
+ line: path.node.loc?.start.line || 0,
217
+ column: path.node.loc?.start.column || 0,
218
+ file: filePath
219
+ });
220
+ }
221
+ }
222
+ });
223
+
224
+ } catch (error) {
225
+ console.warn(`Warning: Could not parse ${filePath}: ${error instanceof Error ? error.message : 'Unknown error'}`);
226
+ }
227
+
228
+ return features;
229
+ }
230
+
231
+ private extractWebAPIFeature(node: any, content: string, t: any): DetectedFeature | null {
232
+ const apiName = this.getMemberExpressionName(node, t);
233
+
234
+ if (!apiName || this.REACT_SPECIFIC_APIS.has(apiName)) {
235
+ return null;
236
+ }
237
+
238
+ if (this.WEB_PLATFORM_APIS.has(apiName)) {
239
+ return {
240
+ feature: apiName,
241
+ type: 'js',
242
+ context: this.getContext(content, node.loc?.start.line || 0),
243
+ line: node.loc?.start.line || 0,
244
+ column: node.loc?.start.column || 0
245
+ };
246
+ }
247
+
248
+ return null;
249
+ }
250
+
251
+ private extractWebAPICall(node: any, content: string, t: any): DetectedFeature | null {
252
+ let apiName = '';
253
+
254
+ if (t.isIdentifier(node.callee)) {
255
+ apiName = node.callee.name;
256
+ } else if (t.isMemberExpression(node.callee)) {
257
+ apiName = this.getMemberExpressionName(node.callee, t);
258
+ }
259
+
260
+ if (!apiName || this.REACT_SPECIFIC_APIS.has(apiName)) {
261
+ return null;
262
+ }
263
+
264
+ if (this.WEB_PLATFORM_APIS.has(apiName)) {
265
+ return {
266
+ feature: apiName,
267
+ type: 'js',
268
+ context: this.getContext(content, node.loc?.start.line || 0),
269
+ line: node.loc?.start.line || 0,
270
+ column: node.loc?.start.column || 0
271
+ };
272
+ }
273
+
274
+ return null;
275
+ }
276
+
277
+ private extractInlineCSS(node: any, content: string, t: any): DetectedFeature[] {
278
+ const features: DetectedFeature[] = [];
279
+
280
+ // Check if this is likely a style object (has CSS-like properties)
281
+ const hasStyleProps = node.properties.some(prop => {
282
+ if (t.isObjectProperty(prop) && (t.isIdentifier(prop.key) || t.isStringLiteral(prop.key))) {
283
+ const key = t.isIdentifier(prop.key) ? prop.key.name : prop.key.value;
284
+ return this.CSS_PROPERTIES.has(key) || key.includes('-') || key.startsWith('--');
285
+ }
286
+ return false;
287
+ });
288
+
289
+ if (!hasStyleProps) {
290
+ return features;
291
+ }
292
+
293
+ node.properties.forEach(prop => {
294
+ if (t.isObjectProperty(prop)) {
295
+ let key = '';
296
+ if (t.isIdentifier(prop.key)) {
297
+ key = prop.key.name;
298
+ } else if (t.isStringLiteral(prop.key)) {
299
+ key = prop.key.value;
300
+ }
301
+
302
+ if (key && (this.CSS_PROPERTIES.has(key) || key.startsWith('--') || key.includes('-'))) {
303
+ features.push({
304
+ feature: key,
305
+ type: 'css',
306
+ context: this.getContext(content, prop.loc?.start.line || 0),
307
+ line: prop.loc?.start.line || 0,
308
+ column: prop.loc?.start.column || 0
309
+ });
310
+ }
311
+ }
312
+ });
313
+
314
+ return features;
315
+ }
316
+
317
+ private getMemberExpressionName(node: any, t: any): string {
318
+ const parts: string[] = [];
319
+
320
+ const traverse = (n: any): void => {
321
+ if (t.isIdentifier(n)) {
322
+ parts.unshift(n.name);
323
+ } else if (t.isMemberExpression(n)) {
324
+ if (t.isIdentifier(n.property)) {
325
+ parts.unshift(n.property.name);
326
+ }
327
+ traverse(n.object);
328
+ }
329
+ };
330
+
331
+ traverse(node);
332
+ return parts.join('.');
333
+ }
334
+
335
+ private isTopLevelAwait(path: any, t: any): boolean {
336
+ let currentPath = path.parentPath;
337
+ while (currentPath) {
338
+ if (t.isFunction(currentPath.node) || t.isArrowFunctionExpression(currentPath.node)) {
339
+ return false;
340
+ }
341
+ currentPath = currentPath.parentPath!;
342
+ }
343
+ return true;
344
+ }
345
+
346
+ private getContext(content: string, line: number): string {
347
+ const lines = content.split('\n');
348
+ const targetLine = lines[line - 1] || '';
349
+ return targetLine.trim();
350
+ }
351
+
352
+ getSupportedExtensions(): string[] {
353
+ return ['.jsx', '.tsx'];
354
+ }
355
+
356
+ getName(): string {
357
+ return 'ReactParser';
358
+ }
359
+ }