ATlast โ€” you'll never need to find your favorites on another platform again. Find your favs in the ATmosphere.
atproto

Compare changes

Choose any two refs to compare.

-165
EXTENSION_STATUS.md
··· 1 - # Extension Implementation Status 2 - 3 - ## Current State: โœ… COMPLETE - Ready for Production Testing 4 - 5 - ### What's Complete โœ… 6 - 7 - 1. **Environment Configuration** 8 - - Dev/prod builds with correct API URLs 9 - - Build: `npm run build` (dev) or `npm run build:prod` 10 - - Dev: `http://127.0.0.1:8888` 11 - - Prod: `https://atlast.byarielm.fyi` 12 - 13 - 2. **Server Health Check** 14 - - Extension checks if dev server is running (dev mode only) 15 - - Shows "Server offline" state with instructions 16 - - "Check Again" button to retry 17 - 18 - 3. **Authentication Flow** 19 - - Extension checks `/session` endpoint on init 20 - - Shows "Not logged in" state if no session 21 - - "Open ATlast" button to log in 22 - - "Check Again" to retry after login 23 - - **User must be logged in to ATlast BEFORE using extension** 24 - 25 - 4. **Upload Flow** (matches file upload) 26 - - Scan Twitter Following page 27 - - POST to `/extension-import` (requires auth) 28 - - Backend: 29 - - Gets DID from session 30 - - Creates `user_upload` entry 31 - - Saves to `source_accounts` table 32 - - Returns `uploadId` 33 - - Opens `/results?uploadId={id}` 34 - - Frontend searches and displays (same as file upload) 35 - 36 - 5. **CORS Permissions** 37 - - Extension has host_permissions for: 38 - - `http://127.0.0.1:8888/*` 39 - - `http://localhost:8888/*` 40 - - `https://atlast.byarielm.fyi/*` 41 - 42 - 6. **Cleanup Complete** 43 - - โŒ Removed `extension_imports` table 44 - - โŒ Removed `get-extension-import` function 45 - - โŒ Removed `ExtensionImport.tsx` page 46 - - โŒ Removed `/import/:id` route 47 - - โŒ Removed `utils/import-store.ts` 48 - 49 - ### What Needs Testing ๐Ÿงช 50 - 51 - 1. **Full Flow Test** 52 - ```bash 53 - # 1. Start dev server 54 - npx netlify-cli dev --filter @atlast/web 55 - 56 - # 2. Build extension 57 - cd packages/extension 58 - npm run build 59 - 60 - # 3. Load extension in Chrome 61 - chrome://extensions/ โ†’ Load unpacked โ†’ packages/extension/dist/chrome/ 62 - 63 - # 4. Log in to ATlast 64 - Open http://127.0.0.1:8888 โ†’ Log in 65 - 66 - # 5. Go to Twitter 67 - https://twitter.com/justadev_atlast/following 68 - 69 - # 6. Open extension popup 70 - - Should show "Ready to scan Twitter/X" 71 - - Click "Start Scan" 72 - - Wait for completion 73 - - Click "Open in ATlast" 74 - - Should open /results?uploadId={id} 75 - - Results should load and search automatically 76 - ``` 77 - 78 - 2. **Error Cases to Test** 79 - - Not logged in โ†’ should show login prompt 80 - - Server offline โ†’ should show offline state 81 - - Empty results โ†’ should show appropriate message 82 - - Network errors โ†’ should handle gracefully 83 - 84 - ### Recently Completed (Dec 2024 - Jan 2025) ๐ŸŽ‰ 85 - 86 - **Extension Flow Fixes:** 87 - - โœ… NaN database error - Fixed missing `matchedUsers` parameter in `extension-import.ts` (node #287) 88 - - โœ… Database initialized successfully (node #288) 89 - - โœ… API response unwrapping - Fixed api-client to access ApiResponse.data field (nodes #290-295) 90 - - โœ… Loading screen during extension upload search (node #325) 91 - - โœ… Timezone fixes - All timestamp columns use TIMESTAMPTZ (node #326) 92 - - โœ… Vite dev server optimization - Pre-bundling dependencies for faster startup (node #327) 93 - 94 - **Decision Graph Documentation:** 95 - - โœ… Fixed 18 orphan nodes and linked to parent goals (nodes #328-331) 96 - - โœ… Improved decision graph workflow with lifecycle management (node #332) 97 - - โœ… Updated CLAUDE.md with node status transitions and common mistakes 98 - 99 - ### Current Status ๐Ÿ“Š 100 - 101 - **All extension bugs resolved!** The extension is fully functional and ready for production testing and deployment. 102 - 103 - ### Next Steps ๐Ÿ“‹ 104 - 105 - 1. โœ… Build extension: `cd packages/extension && pnpm run build` 106 - 2. โœ… Reload extension in Chrome 107 - 3. โœ… Test login flow 108 - 4. โœ… Test scan and upload 109 - 5. โœ… Verify results page works 110 - 6. โœ… All bugs fixed 111 - 7. ๐Ÿ”œ Test production build: `pnpm run build:prod` 112 - 8. ๐Ÿ”œ Chrome Web Store submission 113 - 9. ๐Ÿ”œ Firefox Add-ons support and submission 114 - 115 - ### Architecture Notes ๐Ÿ“ 116 - 117 - **Removed temporary import storage approach:** 118 - - Previously tried in-memory storage (doesn't work in serverless) 119 - - Then tried database storage with temp table (overkill) 120 - 121 - **Current approach:** 122 - - User logs in to ATlast FIRST 123 - - Extension requires authentication 124 - - Upload creates permanent records immediately 125 - - No temporary storage needed 126 - - Matches file upload behavior exactly 127 - 128 - **Why this is better:** 129 - - Simpler architecture 130 - - No temporary storage to expire 131 - - Proper user association from the start 132 - - Reuses existing upload/search infrastructure 133 - - Same flow as file uploads (consistency) 134 - 135 - ### Files Modified in Latest Refactor 136 - 137 - **Deleted:** 138 - - `packages/functions/src/get-extension-import.ts` 139 - - `packages/functions/src/utils/import-store.ts` 140 - - `packages/web/src/pages/ExtensionImport.tsx` 141 - 142 - **Modified:** 143 - - `packages/functions/src/extension-import.ts` - Now requires auth, creates upload 144 - - `packages/functions/src/infrastructure/database/DatabaseService.ts` - Removed extension_imports table 145 - - `packages/functions/src/core/types/database.types.ts` - Removed ExtensionImportRow 146 - - `packages/web/src/Router.tsx` - Removed /import/:id route 147 - - `packages/extension/src/popup/popup.ts` - Added session check, login state 148 - - `packages/extension/src/popup/popup.html` - Added not-logged-in state 149 - - `packages/extension/src/lib/api-client.ts` - Added checkSession(), credentials: 'include' 150 - 151 - ### Decision Graph Summary 152 - 153 - **Total nodes:** 332 nodes, 333 edges 154 - **Key decisions tracked:** 155 - - Environment configuration approach (#261-269) 156 - - Port 8888 conflict resolution (#270-274) 157 - - CORS permissions fix (#275-277) 158 - - Storage approach: in-memory โ†’ database โ†’ proper auth flow (#278-284) 159 - - Refactor and build (#285-286) 160 - - Bug fixes: NaN parameter error (#287), database initialization (#288) 161 - - API response unwrapping fix (#290-295) 162 - - Extension upload flow fixes (#296-327) 163 - - Decision graph integrity fixes (#328-332) 164 - 165 - **Live graph:** https://notactuallytreyanastasio.github.io/deciduous/
+44 -12
docs/git-history.json
··· 1 1 [ 2 2 { 3 - "hash": "fcf682bb8969aca108262348e7e17531077713be", 4 - "short_hash": "fcf682b", 3 + "hash": "15b67054a684ebb2a21761a1774ba15f9b1c29e2", 4 + "short_hash": "15b6705", 5 5 "author": "Ariel M. Lighty", 6 - "date": "2025-12-27T15:48:44-05:00", 7 - "message": "docs: improve decision graph workflow with lifecycle management\n\nUpdated CLAUDE.md with comprehensive node lifecycle management:\n- Added node status transitions (pending โ†’ in_progress โ†’ completed)\n- Correct orphan detection commands (awk instead of cut)\n- Common mistakes section with examples\n- Enhanced audit checklist with status verification\n- Verification workflow after node creation\n\nAlso updated extension popup with ATmosphere branding.\n\nDecision graph now at 331 nodes, 332 edges - all orphans resolved.", 6 + "date": "2025-12-28T20:38:38-05:00", 7 + "message": "fix: add health check function for extension server detection\n\n- Created /health function endpoint with CORS support\n- Updated checkServerHealth to use function endpoint instead of root URL\n- Fixes Firefox extension server detection with proper CORS headers", 8 + "files_changed": 5 9 + }, 10 + { 11 + "hash": "603cf0a187850664336a12c9e5cbb49038906f53", 12 + "short_hash": "603cf0a", 13 + "author": "Ariel M. Lighty", 14 + "date": "2025-12-27T22:42:43-05:00", 15 + "message": "fix: CORS for extension credentialed requests\n\nUpdated CORS headers to support credentials from Chrome extensions:\n- Added getCorsHeaders() to detect chrome-extension:// origins\n- Changed from wildcard Access-Control-Allow-Origin to specific origin\n- Added Access-Control-Allow-Credentials: true for credentialed requests\n- Updated session endpoint to pass event for CORS header detection", 8 16 "files_changed": 4 9 17 }, 10 18 { 19 + "hash": "bd3aabb75abb1875aef125610fcdccb14967a8e3", 20 + "short_hash": "bd3aabb", 21 + "author": "Ariel M. Lighty", 22 + "date": "2025-12-27T22:10:11-05:00", 23 + "message": "fix: extension dark mode and build mode messaging\n\n- Changed darkMode from 'class' to 'media' for automatic system preference detection\n- Made server offline message conditional on build mode (dev vs prod)\n- Hide dev server instructions in production builds", 24 + "files_changed": 5 25 + }, 26 + { 27 + "hash": "bd3aabb75abb1875aef125610fcdccb14967a8e3", 28 + "short_hash": "bd3aabb", 29 + "author": "Ariel M. Lighty", 30 + "date": "2025-12-27T22:10:11-05:00", 31 + "message": "fix: extension dark mode and build mode messaging\n\n- Changed darkMode from 'class' to 'media' for automatic system preference detection\n- Made server offline message conditional on build mode (dev vs prod)\n- Hide dev server instructions in production builds", 32 + "files_changed": 5 33 + }, 34 + { 35 + "hash": "d07180cd3a19328b82b35118e525b59d4e2e060b", 36 + "short_hash": "d07180c", 37 + "author": "Ariel M. Lighty", 38 + "date": "2025-12-27T18:38:39-05:00", 39 + "message": "feat: add Tailwind CSS to extension\n\nReplaced 299 lines of vanilla CSS with Tailwind for design consistency with web app. Production build minified to 13KB.", 40 + "files_changed": 9 41 + }, 42 + { 43 + "hash": "fe29bb3e5faa0151f63c14724f7509af669860de", 44 + "short_hash": "fe29bb3", 45 + "author": "Ariel M. Lighty", 46 + "date": "2025-12-27T16:02:10-05:00", 47 + "message": "docs: update all .md files to reflect current project status\n\nUpdated 4 markdown files with current state:\n\nEXTENSION_STATUS.md:\n- Changed status from DEBUGGING to COMPLETE\n- Updated decision graph count (295 โ†’ 332 nodes)\n- Added recently completed section (nodes 296-332)\n- Marked all extension bugs as resolved\n\nCONTRIBUTING.md:\n- Replaced npm with pnpm throughout\n- Added monorepo structure documentation\n- Updated development commands (netlify-cli dev --filter)\n- Added extension development workflow\n\nPLAN.md:\n- Updated status to Phase 1 COMPLETE\n- Added all recent fixes to completion list\n- Updated decision graph count to 332 nodes\n- Added changelog entries for latest work\n\npackages/extension/README.md:\n- Added prerequisites section (dev server + login required)\n- Updated build commands with dev/prod distinction\n- Added Step 0: Start ATlast Dev Server\n- Added common issues for auth and server states\n\nAll files now accurately reflect completion status and use pnpm.", 48 + "files_changed": 6 49 + }, 50 + { 11 51 "hash": "fcf682bb8969aca108262348e7e17531077713be", 12 52 "short_hash": "fcf682b", 13 53 "author": "Ariel M. Lighty", ··· 21 61 "author": "Ariel M. Lighty", 22 62 "date": "2025-12-26T21:57:05-05:00", 23 63 "message": "perf: optimize Vite dev server startup\n\nAdded explicit optimizeDeps.include to pre-bundle common dependencies:\n- React ecosystem (react, react-dom, react-router-dom)\n- Icon libraries (@icons-pack/react-simple-icons, lucide-react)\n- Other deps (date-fns, jszip, zustand, @tanstack/react-virtual)\n\nAlso added server.fs.allow config for monorepo file serving.\n\nThis should speed up subsequent dev server starts by ensuring these\ndependencies are consistently pre-bundled.", 24 - "files_changed": 1 25 - }, 26 - { 27 - "hash": "aacbbaa27797781098dacdfd0194c93cd71d7bd2", 28 - "short_hash": "aacbbaa", 29 - "author": "Ariel M. Lighty", 30 - "date": "2025-12-26T21:46:06-05:00", 31 - "message": "fix: use TIMESTAMPTZ for all timestamp columns\n\nChanged all TIMESTAMP columns to TIMESTAMPTZ (timestamp with timezone) to\nproperly handle timezone-aware timestamps across all tables:\n- oauth_states (created_at, expires_at)\n- oauth_sessions (created_at, expires_at)\n- user_sessions (created_at, expires_at)\n- user_uploads (created_at, last_checked)\n- source_accounts (last_checked, match_found_at, created_at)\n- user_source_follows (created_at)\n- atproto_matches (found_at, last_verified, last_follow_check)\n- user_match_status (notified_at, viewed_at, followed_at, dismissed_at)\n- notification_queue (created_at, sent_at)\n\nThis fixes the 5-hour timezone offset issue where timestamps were stored\nwithout timezone info, causing display errors across different timezones.", 32 64 "files_changed": 1 33 65 }, 34 66 {
+1204 -71
docs/graph-data.json
··· 3185 3185 "node_type": "goal", 3186 3186 "title": "Fix extension upload errors - undefined response and invalid URL", 3187 3187 "description": null, 3188 - "status": "pending", 3188 + "status": "completed", 3189 3189 "created_at": "2025-12-26T13:31:45.695565800-05:00", 3190 - "updated_at": "2025-12-26T13:31:45.695565800-05:00", 3190 + "updated_at": "2025-12-27T17:49:55.246500-05:00", 3191 3191 "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 3192 3192 }, 3193 3193 { ··· 3196 3196 "node_type": "observation", 3197 3197 "title": "Backend returns correct structure but response might be wrapped by successResponse helper", 3198 3198 "description": null, 3199 - "status": "pending", 3199 + "status": "completed", 3200 3200 "created_at": "2025-12-26T13:32:20.697112800-05:00", 3201 - "updated_at": "2025-12-26T13:32:20.697112800-05:00", 3201 + "updated_at": "2025-12-27T17:49:55.310376600-05:00", 3202 3202 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3203 3203 }, 3204 3204 { ··· 3207 3207 "node_type": "observation", 3208 3208 "title": "successResponse wraps data in {success: true, data: {...}} structure - extension expects flat response", 3209 3209 "description": null, 3210 - "status": "pending", 3210 + "status": "completed", 3211 3211 "created_at": "2025-12-26T13:32:50.409160400-05:00", 3212 - "updated_at": "2025-12-26T13:32:50.409160400-05:00", 3212 + "updated_at": "2025-12-27T17:49:55.384830800-05:00", 3213 3213 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3214 3214 }, 3215 3215 { ··· 3218 3218 "node_type": "action", 3219 3219 "title": "Fix api-client.ts to unwrap ApiResponse.data field", 3220 3220 "description": null, 3221 - "status": "pending", 3221 + "status": "completed", 3222 3222 "created_at": "2025-12-26T13:32:54.625124500-05:00", 3223 - "updated_at": "2025-12-26T13:32:54.625124500-05:00", 3223 + "updated_at": "2025-12-27T17:49:55.449186500-05:00", 3224 3224 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3225 3225 }, 3226 3226 { ··· 3229 3229 "node_type": "outcome", 3230 3230 "title": "Fixed API client to unwrap ApiResponse.data - both uploadToATlast and checkSession now correctly access nested data field", 3231 3231 "description": null, 3232 - "status": "pending", 3232 + "status": "completed", 3233 3233 "created_at": "2025-12-26T13:34:09.012837500-05:00", 3234 - "updated_at": "2025-12-26T13:34:09.012837500-05:00", 3234 + "updated_at": "2025-12-27T17:49:55.512809400-05:00", 3235 3235 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3236 3236 }, 3237 3237 { ··· 3240 3240 "node_type": "outcome", 3241 3241 "title": "Committed API response fix to git", 3242 3242 "description": null, 3243 - "status": "pending", 3243 + "status": "completed", 3244 3244 "created_at": "2025-12-26T13:36:02.733197600-05:00", 3245 - "updated_at": "2025-12-26T13:36:02.733197600-05:00", 3245 + "updated_at": "2025-12-27T17:49:55.576426900-05:00", 3246 3246 "metadata_json": "{\"branch\":\"master\",\"commit\":\"9563633\",\"confidence\":95}" 3247 3247 }, 3248 3248 { ··· 3251 3251 "node_type": "observation", 3252 3252 "title": "Extension upload flow fixed and ready for testing - API response unwrapping resolves undefined errors", 3253 3253 "description": null, 3254 - "status": "pending", 3254 + "status": "completed", 3255 3255 "created_at": "2025-12-26T13:37:35.844832-05:00", 3256 - "updated_at": "2025-12-26T13:37:35.844832-05:00", 3256 + "updated_at": "2025-12-27T17:49:55.653339900-05:00", 3257 3257 "metadata_json": "{\"branch\":\"master\",\"commit\":\"9ca7347\",\"confidence\":95}" 3258 3258 }, 3259 3259 { ··· 3262 3262 "node_type": "goal", 3263 3263 "title": "Fix backend repository method error and missing frontend route", 3264 3264 "description": null, 3265 - "status": "pending", 3265 + "status": "completed", 3266 3266 "created_at": "2025-12-26T13:43:03.332690700-05:00", 3267 - "updated_at": "2025-12-26T13:43:03.332690700-05:00", 3267 + "updated_at": "2025-12-27T17:49:55.729232100-05:00", 3268 3268 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3269 3269 }, 3270 3270 { ··· 3273 3273 "node_type": "observation", 3274 3274 "title": "Two issues: 1) SourceAccountRepository has getOrCreate/bulkCreate not upsertSourceAccount, 2) Router only has / route, no /results route", 3275 3275 "description": null, 3276 - "status": "pending", 3276 + "status": "completed", 3277 3277 "created_at": "2025-12-26T13:43:28.902663600-05:00", 3278 - "updated_at": "2025-12-26T13:43:28.902663600-05:00", 3278 + "updated_at": "2025-12-27T17:49:55.791246300-05:00", 3279 3279 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3280 3280 }, 3281 3281 { ··· 3284 3284 "node_type": "action", 3285 3285 "title": "Fix backend to use bulkCreate and frontend to handle uploadId param", 3286 3286 "description": null, 3287 - "status": "pending", 3287 + "status": "completed", 3288 3288 "created_at": "2025-12-26T13:44:28.406069900-05:00", 3289 - "updated_at": "2025-12-26T13:44:28.406069900-05:00", 3289 + "updated_at": "2025-12-27T17:49:55.863335500-05:00", 3290 3290 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3291 3291 }, 3292 3292 { ··· 3295 3295 "node_type": "outcome", 3296 3296 "title": "Fixed both issues: backend uses bulkCreate, redirects to /?uploadId, frontend loads results from uploadId param", 3297 3297 "description": null, 3298 - "status": "pending", 3298 + "status": "completed", 3299 3299 "created_at": "2025-12-26T13:45:58.309042200-05:00", 3300 - "updated_at": "2025-12-26T13:45:58.309042200-05:00", 3300 + "updated_at": "2025-12-27T17:49:55.947393200-05:00", 3301 3301 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3302 3302 }, 3303 3303 { ··· 3306 3306 "node_type": "outcome", 3307 3307 "title": "Committed fixes for bulkCreate and uploadId handling", 3308 3308 "description": null, 3309 - "status": "pending", 3309 + "status": "completed", 3310 3310 "created_at": "2025-12-26T13:47:48.770693200-05:00", 3311 - "updated_at": "2025-12-26T13:47:48.770693200-05:00", 3311 + "updated_at": "2025-12-27T17:49:56.029469300-05:00", 3312 3312 "metadata_json": "{\"branch\":\"master\",\"commit\":\"581ed00\",\"confidence\":95}" 3313 3313 }, 3314 3314 { ··· 3317 3317 "node_type": "observation", 3318 3318 "title": "Frontend error: loadUploadResults not defined - need to check function scope", 3319 3319 "description": null, 3320 - "status": "pending", 3320 + "status": "completed", 3321 3321 "created_at": "2025-12-26T13:50:59.977950500-05:00", 3322 - "updated_at": "2025-12-26T13:50:59.977950500-05:00", 3322 + "updated_at": "2025-12-27T17:49:56.093781100-05:00", 3323 3323 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3324 3324 }, 3325 3325 { ··· 3328 3328 "node_type": "action", 3329 3329 "title": "Fix useEffect to call handleLoadUpload instead of non-existent loadUploadResults", 3330 3330 "description": null, 3331 - "status": "pending", 3331 + "status": "completed", 3332 3332 "created_at": "2025-12-26T13:51:36.007564400-05:00", 3333 - "updated_at": "2025-12-26T13:51:36.007564400-05:00", 3333 + "updated_at": "2025-12-27T17:49:56.169258900-05:00", 3334 3334 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3335 3335 }, 3336 3336 { ··· 3339 3339 "node_type": "outcome", 3340 3340 "title": "Fixed function name - now calls handleLoadUpload correctly", 3341 3341 "description": null, 3342 - "status": "pending", 3342 + "status": "completed", 3343 3343 "created_at": "2025-12-26T13:51:52.256909300-05:00", 3344 - "updated_at": "2025-12-26T13:51:52.256909300-05:00", 3344 + "updated_at": "2025-12-27T17:49:56.234188500-05:00", 3345 3345 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3346 3346 }, 3347 3347 { ··· 3350 3350 "node_type": "goal", 3351 3351 "title": "Fix extension flow: auto-search after load, history navigation, time formatting", 3352 3352 "description": null, 3353 - "status": "pending", 3353 + "status": "completed", 3354 3354 "created_at": "2025-12-26T14:05:53.798547500-05:00", 3355 - "updated_at": "2025-12-26T14:05:53.798547500-05:00", 3355 + "updated_at": "2025-12-27T17:49:56.309329800-05:00", 3356 3356 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3357 3357 }, 3358 3358 { ··· 3361 3361 "node_type": "observation", 3362 3362 "title": "handleLoadUpload expects existing results but extension creates empty upload - need to load source accounts and trigger search", 3363 3363 "description": null, 3364 - "status": "pending", 3364 + "status": "completed", 3365 3365 "created_at": "2025-12-26T14:06:18.067673100-05:00", 3366 - "updated_at": "2025-12-26T14:06:18.067673100-05:00", 3366 + "updated_at": "2025-12-27T17:49:56.384145700-05:00", 3367 3367 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3368 3368 }, 3369 3369 { ··· 3372 3372 "node_type": "observation", 3373 3373 "title": "Extension-import creates upload and source_accounts but NOT user_source_follows - get-upload-details returns empty because it queries FROM user_source_follows", 3374 3374 "description": null, 3375 - "status": "pending", 3375 + "status": "completed", 3376 3376 "created_at": "2025-12-26T14:08:57.918421600-05:00", 3377 - "updated_at": "2025-12-26T14:08:57.918421600-05:00", 3377 + "updated_at": "2025-12-27T17:49:56.459539400-05:00", 3378 3378 "metadata_json": "{\"branch\":\"master\",\"confidence\":100}" 3379 3379 }, 3380 3380 { ··· 3383 3383 "node_type": "action", 3384 3384 "title": "Add user_source_follows creation to extension-import endpoint", 3385 3385 "description": null, 3386 - "status": "pending", 3386 + "status": "completed", 3387 3387 "created_at": "2025-12-26T14:09:03.035871-05:00", 3388 - "updated_at": "2025-12-26T14:09:03.035871-05:00", 3388 + "updated_at": "2025-12-27T17:49:56.523841100-05:00", 3389 3389 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3390 3390 }, 3391 3391 { ··· 3394 3394 "node_type": "outcome", 3395 3395 "title": "Fixed all extension flow issues: added user_source_follows creation, auto-search after load, time formatting", 3396 3396 "description": null, 3397 - "status": "pending", 3397 + "status": "completed", 3398 3398 "created_at": "2025-12-26T14:11:09.055850200-05:00", 3399 - "updated_at": "2025-12-26T14:11:09.055850200-05:00", 3399 + "updated_at": "2025-12-27T17:49:56.588486100-05:00", 3400 3400 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3401 3401 }, 3402 3402 { ··· 3405 3405 "node_type": "outcome", 3406 3406 "title": "Committed all extension flow fixes", 3407 3407 "description": null, 3408 - "status": "pending", 3408 + "status": "completed", 3409 3409 "created_at": "2025-12-26T14:16:08.387214900-05:00", 3410 - "updated_at": "2025-12-26T14:16:08.387214900-05:00", 3410 + "updated_at": "2025-12-27T17:49:56.670180800-05:00", 3411 3411 "metadata_json": "{\"branch\":\"master\",\"commit\":\"6ced3f0\",\"confidence\":95}" 3412 3412 }, 3413 3413 { ··· 3416 3416 "node_type": "observation", 3417 3417 "title": "searchAllUsers called with wrong parameters - missing onProgressUpdate callback", 3418 3418 "description": null, 3419 - "status": "pending", 3419 + "status": "completed", 3420 3420 "created_at": "2025-12-26T16:07:21.838974100-05:00", 3421 - "updated_at": "2025-12-26T16:07:21.838974100-05:00", 3421 + "updated_at": "2025-12-27T17:49:56.746464900-05:00", 3422 3422 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3423 3423 }, 3424 3424 { ··· 3427 3427 "node_type": "action", 3428 3428 "title": "Fix searchAllUsers call with correct parameters and callbacks", 3429 3429 "description": null, 3430 - "status": "pending", 3430 + "status": "completed", 3431 3431 "created_at": "2025-12-26T16:08:18.523845400-05:00", 3432 - "updated_at": "2025-12-26T16:08:18.523845400-05:00", 3432 + "updated_at": "2025-12-27T17:49:56.809583600-05:00", 3433 3433 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3434 3434 }, 3435 3435 { ··· 3438 3438 "node_type": "outcome", 3439 3439 "title": "Fixed searchAllUsers call - now passes onProgressUpdate and onComplete callbacks", 3440 3440 "description": null, 3441 - "status": "pending", 3441 + "status": "completed", 3442 3442 "created_at": "2025-12-26T16:08:24.248208800-05:00", 3443 - "updated_at": "2025-12-26T16:08:24.248208800-05:00", 3443 + "updated_at": "2025-12-27T17:49:56.884711900-05:00", 3444 3444 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3445 3445 }, 3446 3446 { ··· 3449 3449 "node_type": "goal", 3450 3450 "title": "Fix validation error and undefined localeCompare in extension flow", 3451 3451 "description": null, 3452 - "status": "pending", 3452 + "status": "completed", 3453 3453 "created_at": "2025-12-26T20:17:59.516959100-05:00", 3454 - "updated_at": "2025-12-26T20:17:59.516959100-05:00", 3454 + "updated_at": "2025-12-27T17:49:56.971434500-05:00", 3455 3455 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3456 3456 }, 3457 3457 { ··· 3460 3460 "node_type": "observation", 3461 3461 "title": "Two errors: 1) batch-search-actors gets null in usernames array, 2) Frontend localeCompare on undefined - likely wrong SearchResult structure", 3462 3462 "description": null, 3463 - "status": "pending", 3463 + "status": "completed", 3464 3464 "created_at": "2025-12-26T20:18:03.693879700-05:00", 3465 - "updated_at": "2025-12-26T20:18:03.693879700-05:00", 3465 + "updated_at": "2025-12-27T17:49:57.049131800-05:00", 3466 3466 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3467 3467 }, 3468 3468 { ··· 3471 3471 "node_type": "action", 3472 3472 "title": "Fix SearchResult structure - sourceUser should be object not string", 3473 3473 "description": null, 3474 - "status": "pending", 3474 + "status": "completed", 3475 3475 "created_at": "2025-12-26T20:19:47.621459800-05:00", 3476 - "updated_at": "2025-12-26T20:19:47.621459800-05:00", 3476 + "updated_at": "2025-12-27T17:49:57.127563700-05:00", 3477 3477 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3478 3478 }, 3479 3479 { ··· 3482 3482 "node_type": "outcome", 3483 3483 "title": "Fixed SearchResult structure - sourceUser is now correct SourceUser object instead of string", 3484 3484 "description": null, 3485 - "status": "pending", 3485 + "status": "completed", 3486 3486 "created_at": "2025-12-26T20:20:22.507291300-05:00", 3487 - "updated_at": "2025-12-26T20:20:22.507291300-05:00", 3487 + "updated_at": "2025-12-27T17:49:57.190209200-05:00", 3488 3488 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3489 3489 }, 3490 3490 { ··· 3493 3493 "node_type": "goal", 3494 3494 "title": "Fix results not saving to database and timestamp timezone issue", 3495 3495 "description": null, 3496 - "status": "pending", 3496 + "status": "completed", 3497 3497 "created_at": "2025-12-26T20:37:03.493239600-05:00", 3498 - "updated_at": "2025-12-26T20:37:03.493239600-05:00", 3498 + "updated_at": "2025-12-27T17:49:57.263765-05:00", 3499 3499 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3500 3500 }, 3501 3501 { ··· 3603 3603 "node_type": "outcome", 3604 3604 "title": "Fixed decision graph integrity - linked 18 orphan nodes to parent goals, marked nodes 319-327 as completed", 3605 3605 "description": null, 3606 - "status": "pending", 3606 + "status": "completed", 3607 3607 "created_at": "2025-12-27T15:38:21.291457500-05:00", 3608 - "updated_at": "2025-12-27T15:38:21.291457500-05:00", 3608 + "updated_at": "2025-12-27T17:49:54.129059900-05:00", 3609 3609 "metadata_json": "{\"branch\":\"master\",\"confidence\":100}" 3610 3610 }, 3611 3611 { ··· 3614 3614 "node_type": "observation", 3615 3615 "title": "Decision graph audit revealed systematic issues: 18 orphan nodes, incorrect status (pending vs completed), wrong orphan detection commands in recovery workflow", 3616 3616 "description": null, 3617 - "status": "pending", 3617 + "status": "completed", 3618 3618 "created_at": "2025-12-27T15:40:23.238704300-05:00", 3619 - "updated_at": "2025-12-27T15:40:23.238704300-05:00", 3619 + "updated_at": "2025-12-27T17:49:57.327650700-05:00", 3620 3620 "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3621 3621 }, 3622 3622 { ··· 3627 3627 "description": null, 3628 3628 "status": "completed", 3629 3629 "created_at": "2025-12-27T15:41:04.067444-05:00", 3630 - "updated_at": "2025-12-27T15:47:53.643256600-05:00", 3630 + "updated_at": "2025-12-27T17:49:57.403361400-05:00", 3631 3631 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3632 3632 }, 3633 3633 { ··· 3638 3638 "description": null, 3639 3639 "status": "completed", 3640 3640 "created_at": "2025-12-27T15:47:49.308750700-05:00", 3641 - "updated_at": "2025-12-27T15:47:55.838439500-05:00", 3641 + "updated_at": "2025-12-27T17:49:57.478252800-05:00", 3642 3642 "metadata_json": "{\"branch\":\"master\",\"confidence\":100}" 3643 3643 }, 3644 3644 { ··· 3649 3649 "description": null, 3650 3650 "status": "completed", 3651 3651 "created_at": "2025-12-27T15:48:47.658343800-05:00", 3652 - "updated_at": "2025-12-27T15:48:51.950405900-05:00", 3652 + "updated_at": "2025-12-27T17:49:57.553143200-05:00", 3653 3653 "metadata_json": "{\"branch\":\"master\",\"commit\":\"fcf682b\",\"confidence\":100}" 3654 3654 }, 3655 3655 { ··· 3660 3660 "description": null, 3661 3661 "status": "completed", 3662 3662 "created_at": "2025-12-27T15:50:48.815758500-05:00", 3663 - "updated_at": "2025-12-27T15:59:50.747036600-05:00", 3663 + "updated_at": "2025-12-27T17:49:57.630386-05:00", 3664 3664 "metadata_json": "{\"branch\":\"master\",\"confidence\":90,\"prompt\":\"review and update the .md files thru the project based on current project status.\"}" 3665 3665 }, 3666 3666 { ··· 3671 3671 "description": null, 3672 3672 "status": "completed", 3673 3673 "created_at": "2025-12-27T15:51:22.583189100-05:00", 3674 - "updated_at": "2025-12-27T15:52:10.859032400-05:00", 3674 + "updated_at": "2025-12-27T17:49:57.707946400-05:00", 3675 3675 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3676 3676 }, 3677 3677 { ··· 3680 3680 "node_type": "observation", 3681 3681 "title": "Analyzed all project .md files - found outdated information in CONTRIBUTING.md (npmโ†’pnpm), EXTENSION_STATUS.md (debuggingโ†’completed), PLAN.md (optimization status), extension README (build commands)", 3682 3682 "description": null, 3683 - "status": "pending", 3683 + "status": "completed", 3684 3684 "created_at": "2025-12-27T15:52:06.741629200-05:00", 3685 - "updated_at": "2025-12-27T15:52:06.741629200-05:00", 3685 + "updated_at": "2025-12-27T17:49:57.786343300-05:00", 3686 3686 "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3687 3687 }, 3688 3688 { ··· 3691 3691 "node_type": "decision", 3692 3692 "title": "Choose which .md files to update based on priority and impact", 3693 3693 "description": null, 3694 - "status": "pending", 3694 + "status": "completed", 3695 3695 "created_at": "2025-12-27T15:52:30.322805700-05:00", 3696 - "updated_at": "2025-12-27T15:52:30.322805700-05:00", 3696 + "updated_at": "2025-12-27T17:49:57.849977800-05:00", 3697 3697 "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 3698 3698 }, 3699 3699 { ··· 3761 3761 "created_at": "2025-12-27T15:59:41.457774700-05:00", 3762 3762 "updated_at": "2025-12-27T15:59:45.883622500-05:00", 3763 3763 "metadata_json": "{\"branch\":\"master\",\"confidence\":100}" 3764 + }, 3765 + { 3766 + "id": 343, 3767 + "change_id": "9e0fcead-ea30-4b31-974b-4e07f7fc6787", 3768 + "node_type": "action", 3769 + "title": "Committed all markdown documentation updates", 3770 + "description": null, 3771 + "status": "completed", 3772 + "created_at": "2025-12-27T16:02:13.397776700-05:00", 3773 + "updated_at": "2025-12-27T16:02:56.131931100-05:00", 3774 + "metadata_json": "{\"branch\":\"master\",\"commit\":\"fe29bb3\",\"confidence\":100}" 3775 + }, 3776 + { 3777 + "id": 344, 3778 + "change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 3779 + "node_type": "goal", 3780 + "title": "Add Tailwind CSS to extension for design consistency", 3781 + "description": null, 3782 + "status": "completed", 3783 + "created_at": "2025-12-27T17:59:23.523767600-05:00", 3784 + "updated_at": "2025-12-27T18:07:53.271415-05:00", 3785 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90,\"prompt\":\"yes\"}" 3786 + }, 3787 + { 3788 + "id": 345, 3789 + "change_id": "0ef352ed-538b-4632-8b62-ebb17603f944", 3790 + "node_type": "action", 3791 + "title": "Installing Tailwind CSS and PostCSS dependencies", 3792 + "description": null, 3793 + "status": "completed", 3794 + "created_at": "2025-12-27T18:00:41.652670100-05:00", 3795 + "updated_at": "2025-12-27T18:00:43.901523100-05:00", 3796 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3797 + }, 3798 + { 3799 + "id": 346, 3800 + "change_id": "888e6ad0-5002-4cdb-b35e-f4214ca07dfa", 3801 + "node_type": "action", 3802 + "title": "Creating Tailwind and PostCSS config files", 3803 + "description": null, 3804 + "status": "completed", 3805 + "created_at": "2025-12-27T18:01:27.404433500-05:00", 3806 + "updated_at": "2025-12-27T18:01:29.980132200-05:00", 3807 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3808 + }, 3809 + { 3810 + "id": 347, 3811 + "change_id": "fae7a634-d921-4b6f-9620-0c58d88b863e", 3812 + "node_type": "action", 3813 + "title": "Updating build.js to process CSS with PostCSS + Tailwind", 3814 + "description": null, 3815 + "status": "completed", 3816 + "created_at": "2025-12-27T18:01:50.537140900-05:00", 3817 + "updated_at": "2025-12-27T18:01:53.031316700-05:00", 3818 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3819 + }, 3820 + { 3821 + "id": 348, 3822 + "change_id": "c25a8f4b-8bf1-4a33-bef9-3731dfd83627", 3823 + "node_type": "action", 3824 + "title": "Converting popup.css to use Tailwind directives", 3825 + "description": null, 3826 + "status": "completed", 3827 + "created_at": "2025-12-27T18:02:42.167814700-05:00", 3828 + "updated_at": "2025-12-27T18:02:44.488653900-05:00", 3829 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3830 + }, 3831 + { 3832 + "id": 349, 3833 + "change_id": "c65ee3d9-62a0-47aa-870a-f6422ff2536a", 3834 + "node_type": "action", 3835 + "title": "Converting popup.html to use Tailwind utility classes", 3836 + "description": null, 3837 + "status": "completed", 3838 + "created_at": "2025-12-27T18:03:00.465637900-05:00", 3839 + "updated_at": "2025-12-27T18:03:02.815261100-05:00", 3840 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3841 + }, 3842 + { 3843 + "id": 350, 3844 + "change_id": "8136e615-5baa-4fe5-9a7d-d672ff1a6f85", 3845 + "node_type": "outcome", 3846 + "title": "Successfully integrated Tailwind CSS into extension", 3847 + "description": null, 3848 + "status": "completed", 3849 + "created_at": "2025-12-27T18:07:49.869572400-05:00", 3850 + "updated_at": "2025-12-27T18:07:52.136827400-05:00", 3851 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3852 + }, 3853 + { 3854 + "id": 351, 3855 + "change_id": "9468bcb3-78ec-4dae-8d8f-968ba6f5b3fe", 3856 + "node_type": "outcome", 3857 + "title": "Committed Tailwind CSS integration to git", 3858 + "description": null, 3859 + "status": "completed", 3860 + "created_at": "2025-12-27T18:38:55.689869700-05:00", 3861 + "updated_at": "2025-12-27T18:39:01.013284600-05:00", 3862 + "metadata_json": "{\"branch\":\"master\",\"commit\":\"d07180c\",\"confidence\":95}" 3863 + }, 3864 + { 3865 + "id": 352, 3866 + "change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 3867 + "node_type": "goal", 3868 + "title": "Fix extension dark mode and dev/prod detection issues", 3869 + "description": null, 3870 + "status": "completed", 3871 + "created_at": "2025-12-27T22:05:50.675487800-05:00", 3872 + "updated_at": "2025-12-27T22:09:32.111749500-05:00", 3873 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90,\"prompt\":\"there now seems to be an issue with dark mode not activating, and either an issue with detecting dev vs prod or the copy is just wrong. analyze and fix.\"}" 3874 + }, 3875 + { 3876 + "id": 353, 3877 + "change_id": "eaed6e9b-9f16-4b45-8783-44ea2ea1f2a9", 3878 + "node_type": "observation", 3879 + "title": "Found two issues: 1) darkMode: 'class' requires manual .dark class addition, 2) Dev/prod detection may be incorrect", 3880 + "description": null, 3881 + "status": "completed", 3882 + "created_at": "2025-12-27T22:06:19.509001-05:00", 3883 + "updated_at": "2025-12-27T22:06:23.515277300-05:00", 3884 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3885 + }, 3886 + { 3887 + "id": 354, 3888 + "change_id": "d66fc83e-9737-4047-8ce2-e2ba857aeea9", 3889 + "node_type": "decision", 3890 + "title": "Choose dark mode strategy: media queries vs class-based with JS", 3891 + "description": null, 3892 + "status": "completed", 3893 + "created_at": "2025-12-27T22:07:01.587088200-05:00", 3894 + "updated_at": "2025-12-27T22:07:07.798171700-05:00", 3895 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 3896 + }, 3897 + { 3898 + "id": 355, 3899 + "change_id": "76e2a379-7803-4c82-8013-be6b62f2d360", 3900 + "node_type": "outcome", 3901 + "title": "Chose media queries - simpler and matches original behavior", 3902 + "description": null, 3903 + "status": "completed", 3904 + "created_at": "2025-12-27T22:07:04.660558100-05:00", 3905 + "updated_at": "2025-12-27T22:07:07.897193100-05:00", 3906 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3907 + }, 3908 + { 3909 + "id": 356, 3910 + "change_id": "df681aa8-e470-4ead-a0d2-a4095febfa3d", 3911 + "node_type": "action", 3912 + "title": "Fixing dark mode config to use media queries", 3913 + "description": null, 3914 + "status": "completed", 3915 + "created_at": "2025-12-27T22:07:24.774976300-05:00", 3916 + "updated_at": "2025-12-27T22:07:30.392290200-05:00", 3917 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3918 + }, 3919 + { 3920 + "id": 357, 3921 + "change_id": "57060303-5a30-4f11-a752-a02376df5ea7", 3922 + "node_type": "action", 3923 + "title": "Making server offline message conditional on build mode", 3924 + "description": null, 3925 + "status": "completed", 3926 + "created_at": "2025-12-27T22:07:49.952419800-05:00", 3927 + "updated_at": "2025-12-27T22:09:00.514201500-05:00", 3928 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3929 + }, 3930 + { 3931 + "id": 358, 3932 + "change_id": "fc211ac7-7a1a-4b69-835a-992c354e8237", 3933 + "node_type": "outcome", 3934 + "title": "Successfully fixed dark mode and dev/prod messaging", 3935 + "description": null, 3936 + "status": "completed", 3937 + "created_at": "2025-12-27T22:09:28.843864300-05:00", 3938 + "updated_at": "2025-12-27T22:09:32.017503200-05:00", 3939 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3940 + }, 3941 + { 3942 + "id": 359, 3943 + "change_id": "4a7d5885-1713-4ba7-ad13-bb12b58c9410", 3944 + "node_type": "outcome", 3945 + "title": "Committed fixes to git", 3946 + "description": null, 3947 + "status": "completed", 3948 + "created_at": "2025-12-27T22:10:25.576235500-05:00", 3949 + "updated_at": "2025-12-27T22:10:28.961887300-05:00", 3950 + "metadata_json": "{\"branch\":\"master\",\"commit\":\"bd3aabb\",\"confidence\":95}" 3951 + }, 3952 + { 3953 + "id": 360, 3954 + "change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 3955 + "node_type": "goal", 3956 + "title": "Fix extension not detecting login session despite dev server running", 3957 + "description": null, 3958 + "status": "completed", 3959 + "created_at": "2025-12-27T22:23:13.072419900-05:00", 3960 + "updated_at": "2025-12-27T22:41:49.160848100-05:00", 3961 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90,\"prompt\":\"dark mode is fixed, but the extension in /chrome/ uploaded still is saying login with atlast and dev server is running\"}" 3962 + }, 3963 + { 3964 + "id": 361, 3965 + "change_id": "aecf2327-d20d-4c6c-b6b0-06ccf26a2b27", 3966 + "node_type": "observation", 3967 + "title": "Extension dist/chrome contains production build, not dev build. User ran build:prod last.", 3968 + "description": null, 3969 + "status": "completed", 3970 + "created_at": "2025-12-27T22:23:45.918832500-05:00", 3971 + "updated_at": "2025-12-27T22:23:48.919570500-05:00", 3972 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3973 + }, 3974 + { 3975 + "id": 362, 3976 + "change_id": "e897db97-44d8-4993-b4c3-0d829265b2f8", 3977 + "node_type": "observation", 3978 + "title": "Dev build now deployed. Extension will check session at http://127.0.0.1:8888/.netlify/functions/session with credentials:include", 3979 + "description": null, 3980 + "status": "completed", 3981 + "created_at": "2025-12-27T22:24:17.767230200-05:00", 3982 + "updated_at": "2025-12-27T22:24:20.981953100-05:00", 3983 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 3984 + }, 3985 + { 3986 + "id": 363, 3987 + "change_id": "2c62bfa3-d148-4448-8c2b-d0cf1e94ceb0", 3988 + "node_type": "observation", 3989 + "title": "Found CORS issue: successResponse uses 'Access-Control-Allow-Origin: *' which blocks credentialed requests from extension", 3990 + "description": null, 3991 + "status": "completed", 3992 + "created_at": "2025-12-27T22:24:51.861265800-05:00", 3993 + "updated_at": "2025-12-27T22:24:55.482724500-05:00", 3994 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3995 + }, 3996 + { 3997 + "id": 364, 3998 + "change_id": "560d6bea-47ec-408d-919b-15ca7198aac9", 3999 + "node_type": "action", 4000 + "title": "Updating CORS headers to support credentialed requests from extension", 4001 + "description": null, 4002 + "status": "completed", 4003 + "created_at": "2025-12-27T22:25:23.035212700-05:00", 4004 + "updated_at": "2025-12-27T22:26:03.046221900-05:00", 4005 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4006 + }, 4007 + { 4008 + "id": 365, 4009 + "change_id": "3ef0c9e9-aa40-4914-a5f4-32bcfaf68d04", 4010 + "node_type": "outcome", 4011 + "title": "Fixed CORS to support credentialed requests from extensions", 4012 + "description": null, 4013 + "status": "completed", 4014 + "created_at": "2025-12-27T22:41:38.430661200-05:00", 4015 + "updated_at": "2025-12-27T22:41:48.981429600-05:00", 4016 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4017 + }, 4018 + { 4019 + "id": 366, 4020 + "change_id": "77b7ed7e-a113-41f6-a677-50d376f3f008", 4021 + "node_type": "outcome", 4022 + "title": "Committed CORS fixes to git", 4023 + "description": null, 4024 + "status": "completed", 4025 + "created_at": "2025-12-27T22:42:49.037783-05:00", 4026 + "updated_at": "2025-12-27T22:42:54.162857-05:00", 4027 + "metadata_json": "{\"branch\":\"master\",\"commit\":\"603cf0a\",\"confidence\":95}" 4028 + }, 4029 + { 4030 + "id": 367, 4031 + "change_id": "df6abf7a-e7a4-45f3-8485-b933319416d9", 4032 + "node_type": "goal", 4033 + "title": "Create Firefox-compatible version of Twitter scraper extension", 4034 + "description": null, 4035 + "status": "completed", 4036 + "created_at": "2025-12-28T18:09:33.241860800-05:00", 4037 + "updated_at": "2025-12-28T19:21:32.412499-05:00", 4038 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85,\"prompt\":\"let's make the extension have a firefox compatible version too.\"}" 4039 + }, 4040 + { 4041 + "id": 368, 4042 + "change_id": "79721edf-aa05-4580-8c28-7d20941ef155", 4043 + "node_type": "observation", 4044 + "title": "Current extension uses Manifest V3 with Chrome-specific APIs", 4045 + "description": null, 4046 + "status": "pending", 4047 + "created_at": "2025-12-28T18:10:08.441348100-05:00", 4048 + "updated_at": "2025-12-28T18:10:08.441348100-05:00", 4049 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4050 + }, 4051 + { 4052 + "id": 369, 4053 + "change_id": "783841d0-c096-48f6-be18-193a9dcc7d4b", 4054 + "node_type": "observation", 4055 + "title": "Firefox compatibility analysis: Extension uses chrome.* APIs (runtime.sendMessage, storage.local, tabs.query/sendMessage), MV3 service worker. Firefox supports MV3 but has differences. Options: 1) Use webextension-polyfill for cross-browser, 2) Dual manifests (MV3 Chrome + MV2 Firefox), 3) Keep MV3 for both with minimal changes. Current build outputs to dist/chrome only.", 4056 + "description": null, 4057 + "status": "pending", 4058 + "created_at": "2025-12-28T18:10:48.087066800-05:00", 4059 + "updated_at": "2025-12-28T18:10:48.087066800-05:00", 4060 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4061 + }, 4062 + { 4063 + "id": 370, 4064 + "change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 4065 + "node_type": "decision", 4066 + "title": "Choose Firefox compatibility approach: webextension-polyfill, dual manifests, or minimal MV3 changes", 4067 + "description": null, 4068 + "status": "pending", 4069 + "created_at": "2025-12-28T18:10:50.375270400-05:00", 4070 + "updated_at": "2025-12-28T18:10:50.375270400-05:00", 4071 + "metadata_json": "{\"branch\":\"master\",\"confidence\":80}" 4072 + }, 4073 + { 4074 + "id": 371, 4075 + "change_id": "159906da-984f-4a1d-a1a6-98e0fc0cf369", 4076 + "node_type": "option", 4077 + "title": "Use webextension-polyfill library for unified cross-browser API", 4078 + "description": null, 4079 + "status": "pending", 4080 + "created_at": "2025-12-28T18:11:05.947924200-05:00", 4081 + "updated_at": "2025-12-28T18:11:05.947924200-05:00", 4082 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 4083 + }, 4084 + { 4085 + "id": 372, 4086 + "change_id": "df5e42e6-53c1-4b30-8b6f-f2385cd9e247", 4087 + "node_type": "option", 4088 + "title": "Dual manifests: MV3 for Chrome, MV2 for Firefox with separate builds", 4089 + "description": null, 4090 + "status": "pending", 4091 + "created_at": "2025-12-28T18:11:08.179938100-05:00", 4092 + "updated_at": "2025-12-28T18:11:08.179938100-05:00", 4093 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 4094 + }, 4095 + { 4096 + "id": 373, 4097 + "change_id": "7bb58202-7a9b-4e8b-8b9e-927e5106bce7", 4098 + "node_type": "option", 4099 + "title": "Keep MV3 for both browsers with minimal manifest tweaks", 4100 + "description": null, 4101 + "status": "pending", 4102 + "created_at": "2025-12-28T18:11:10.370113600-05:00", 4103 + "updated_at": "2025-12-28T18:11:10.370113600-05:00", 4104 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 4105 + }, 4106 + { 4107 + "id": 374, 4108 + "change_id": "d41b29e0-cd48-4dac-a6c8-c6179612702e", 4109 + "node_type": "outcome", 4110 + "title": "Chose webextension-polyfill approach. Provides unified browser.* API, Promise-based, future-proof MV3 for both browsers, +20KB but cleaner codebase", 4111 + "description": null, 4112 + "status": "pending", 4113 + "created_at": "2025-12-28T19:04:24.676770900-05:00", 4114 + "updated_at": "2025-12-28T19:04:24.676770900-05:00", 4115 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4116 + }, 4117 + { 4118 + "id": 375, 4119 + "change_id": "5bb34b8b-aec4-4f84-993e-eb9bf7a2d13f", 4120 + "node_type": "action", 4121 + "title": "Installing webextension-polyfill and updating source files to use browser.* API", 4122 + "description": null, 4123 + "status": "completed", 4124 + "created_at": "2025-12-28T19:08:14.642882400-05:00", 4125 + "updated_at": "2025-12-28T19:21:32.531034800-05:00", 4126 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4127 + }, 4128 + { 4129 + "id": 376, 4130 + "change_id": "644181ee-5a44-4967-9657-e9dd5f648c5e", 4131 + "node_type": "outcome", 4132 + "title": "Successfully implemented Firefox compatibility with webextension-polyfill. Both Chrome and Firefox builds compile successfully. Chrome uses service_worker (MV3), Firefox uses scripts array with browser_specific_settings. All chrome.* API calls replaced with browser.* imports.", 4133 + "description": null, 4134 + "status": "completed", 4135 + "created_at": "2025-12-28T19:14:22.309457600-05:00", 4136 + "updated_at": "2025-12-28T19:21:32.658297400-05:00", 4137 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4138 + }, 4139 + { 4140 + "id": 377, 4141 + "change_id": "1dffa024-413f-4a95-b069-66db350abfaa", 4142 + "node_type": "goal", 4143 + "title": "Fix Firefox extension server detection and login check", 4144 + "description": null, 4145 + "status": "completed", 4146 + "created_at": "2025-12-28T20:14:51.646204800-05:00", 4147 + "updated_at": "2025-12-28T20:32:19.249555-05:00", 4148 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85,\"prompt\":\"The extension works in chrome. In firefox, it's failing to detect that the dev server is running and open + logged in on firefox. There's no right-click to inspect on the popup either.\"}" 4149 + }, 4150 + { 4151 + "id": 378, 4152 + "change_id": "9d5626d2-a9ae-42aa-8fda-be3c7528156f", 4153 + "node_type": "observation", 4154 + "title": "Firefox extension debugging differs from Chrome - need to use about:debugging Inspect button or Browser Console, not right-click popup", 4155 + "description": null, 4156 + "status": "pending", 4157 + "created_at": "2025-12-28T20:15:11.710473-05:00", 4158 + "updated_at": "2025-12-28T20:15:11.710473-05:00", 4159 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4160 + }, 4161 + { 4162 + "id": 379, 4163 + "change_id": "7a5af3fe-8567-4f1c-85cd-e47891704974", 4164 + "node_type": "observation", 4165 + "title": "Potential Firefox issues: 1) CORS with credentials:include may be stricter, 2) Cookie partitioning/third-party cookie blocking, 3) Extension needs explicit host_permissions for cookies to work. Firefox manifest has host_permissions but may need additional cookie permissions.", 4166 + "description": null, 4167 + "status": "pending", 4168 + "created_at": "2025-12-28T20:15:31.278249900-05:00", 4169 + "updated_at": "2025-12-28T20:15:31.278249900-05:00", 4170 + "metadata_json": "{\"branch\":\"master\",\"confidence\":85}" 4171 + }, 4172 + { 4173 + "id": 380, 4174 + "change_id": "9c197aae-18d5-46ae-87e7-82c240c8f313", 4175 + "node_type": "action", 4176 + "title": "Adding cookies permission to Firefox manifest for credentials:include support", 4177 + "description": null, 4178 + "status": "pending", 4179 + "created_at": "2025-12-28T20:16:12.019659700-05:00", 4180 + "updated_at": "2025-12-28T20:16:12.019659700-05:00", 4181 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4182 + }, 4183 + { 4184 + "id": 381, 4185 + "change_id": "485a03b0-8a25-4fdf-a8e2-9d3a25c8edf8", 4186 + "node_type": "outcome", 4187 + "title": "Fixed Firefox cookie issue by adding cookies permission to manifest. Firefox requires explicit permission even with host_permissions. Rebuild successful.", 4188 + "description": null, 4189 + "status": "pending", 4190 + "created_at": "2025-12-28T20:16:41.702322300-05:00", 4191 + "updated_at": "2025-12-28T20:16:41.702322300-05:00", 4192 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4193 + }, 4194 + { 4195 + "id": 382, 4196 + "change_id": "35b13d37-0228-435f-a4bc-c5c42811fec3", 4197 + "node_type": "observation", 4198 + "title": "Firefox blocks extension fetch with CORS error despite host_permissions. Server responds 200 but missing Access-Control-Allow-Origin header. Firefox stricter than Chrome on extension CORS.", 4199 + "description": null, 4200 + "status": "pending", 4201 + "created_at": "2025-12-28T20:17:23.414134300-05:00", 4202 + "updated_at": "2025-12-28T20:17:23.414134300-05:00", 4203 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4204 + }, 4205 + { 4206 + "id": 383, 4207 + "change_id": "adc120cd-e56d-400a-9b3e-8207880378c3", 4208 + "node_type": "action", 4209 + "title": "Adding CORS headers to netlify.toml for extension compatibility - wildcard origin with credentials for dev", 4210 + "description": null, 4211 + "status": "pending", 4212 + "created_at": "2025-12-28T20:18:22.172869600-05:00", 4213 + "updated_at": "2025-12-28T20:18:22.172869600-05:00", 4214 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4215 + }, 4216 + { 4217 + "id": 384, 4218 + "change_id": "0f77bfd9-590f-4f1e-be08-78a9deef6d8a", 4219 + "node_type": "outcome", 4220 + "title": "Added CORS headers to netlify.toml for all paths including root and functions. Headers include Access-Control-Allow-Origin:*, Allow-Credentials:true for dev environment. User needs to restart dev server.", 4221 + "description": null, 4222 + "status": "pending", 4223 + "created_at": "2025-12-28T20:19:54.829093600-05:00", 4224 + "updated_at": "2025-12-28T20:19:54.829093600-05:00", 4225 + "metadata_json": "{\"branch\":\"master\",\"confidence\":90}" 4226 + }, 4227 + { 4228 + "id": 385, 4229 + "change_id": "cc0910f0-2381-4aee-bb5d-397cb0f804d1", 4230 + "node_type": "observation", 4231 + "title": "CORS wildcard (*) incompatible with credentials:include. Browser security prevents wildcard CORS with credentialed requests. Extension origins are dynamic (moz-extension://, chrome-extension://). Need to handle CORS in serverless functions by reflecting request origin.", 4232 + "description": null, 4233 + "status": "pending", 4234 + "created_at": "2025-12-28T20:27:31.848523900-05:00", 4235 + "updated_at": "2025-12-28T20:27:31.848523900-05:00", 4236 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4237 + }, 4238 + { 4239 + "id": 386, 4240 + "change_id": "ad4a5ca7-15d1-4776-8ede-6b615613f6e1", 4241 + "node_type": "action", 4242 + "title": "Adding moz-extension:// origin detection to CORS handler for Firefox extension support", 4243 + "description": null, 4244 + "status": "completed", 4245 + "created_at": "2025-12-28T20:28:31.661326900-05:00", 4246 + "updated_at": "2025-12-28T20:32:19.367968600-05:00", 4247 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4248 + }, 4249 + { 4250 + "id": 387, 4251 + "change_id": "cffdee0f-8535-4d88-83ed-fdf6101f7ac3", 4252 + "node_type": "outcome", 4253 + "title": "Fixed Firefox extension CORS by adding moz-extension:// origin detection to response.utils.ts. Reverted netlify.toml changes as functions handle CORS correctly. User needs to restart dev server.", 4254 + "description": null, 4255 + "status": "completed", 4256 + "created_at": "2025-12-28T20:29:39.856303800-05:00", 4257 + "updated_at": "2025-12-28T20:32:19.494690-05:00", 4258 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4259 + }, 4260 + { 4261 + "id": 388, 4262 + "change_id": "0ada864e-be98-4a2f-a14e-ffd3eea9aaa9", 4263 + "node_type": "observation", 4264 + "title": "Health check uses HEAD request to root URL (Vite server), not a Netlify function. Doesn't get CORS headers from getCorsHeaders. Need dedicated health endpoint or change check to use existing function.", 4265 + "description": null, 4266 + "status": "completed", 4267 + "created_at": "2025-12-28T20:37:22.132717600-05:00", 4268 + "updated_at": "2025-12-28T20:38:41.630020900-05:00", 4269 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4270 + }, 4271 + { 4272 + "id": 389, 4273 + "change_id": "f522d5b2-c325-4f34-9f27-b8ea5c50618d", 4274 + "node_type": "outcome", 4275 + "title": "Created /health function endpoint with CORS support. Updated checkServerHealth to use /.netlify/functions/health instead of root URL. Extension rebuilt successfully.", 4276 + "description": null, 4277 + "status": "completed", 4278 + "created_at": "2025-12-28T20:38:19.981309500-05:00", 4279 + "updated_at": "2025-12-28T20:38:41.780183300-05:00", 4280 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4281 + }, 4282 + { 4283 + "id": 390, 4284 + "change_id": "cfdcf45b-47b3-4239-8053-417bd31957ed", 4285 + "node_type": "observation", 4286 + "title": "Server receives session request but returns CORS wildcard (*) instead of extension origin. No session cookie received. Origin header might not be sent by Firefox extension or not detected correctly.", 4287 + "description": null, 4288 + "status": "pending", 4289 + "created_at": "2025-12-28T20:48:12.770638500-05:00", 4290 + "updated_at": "2025-12-28T20:48:12.770638500-05:00", 4291 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4292 + }, 4293 + { 4294 + "id": 391, 4295 + "change_id": "2b53a419-9a47-4285-9a12-9bdfaeeb9ff0", 4296 + "node_type": "observation", 4297 + "title": "Health endpoint gets CORS headers correctly (moz-extension detected). Session endpoint error middleware doesn't pass event to errorResponse, returns wildcard CORS. Need to fix error middleware to pass event.", 4298 + "description": null, 4299 + "status": "completed", 4300 + "created_at": "2025-12-28T20:55:32.024834200-05:00", 4301 + "updated_at": "2025-12-28T21:38:14.729731500-05:00", 4302 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4303 + }, 4304 + { 4305 + "id": 392, 4306 + "change_id": "c941d136-3405-483d-bf34-7fb011f6d072", 4307 + "node_type": "action", 4308 + "title": "Fixed error middleware to pass event to errorResponse for proper CORS headers on errors", 4309 + "description": null, 4310 + "status": "completed", 4311 + "created_at": "2025-12-28T20:56:38.876266200-05:00", 4312 + "updated_at": "2025-12-28T21:38:14.888627800-05:00", 4313 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4314 + }, 4315 + { 4316 + "id": 393, 4317 + "change_id": "aafd9977-8800-4152-9f7f-b817db6df573", 4318 + "node_type": "outcome", 4319 + "title": "Fixed Firefox extension CORS completely. Error middleware now passes event to errorResponse so Firefox extension origin is properly reflected in error responses with credentials. Debug logging removed.", 4320 + "description": null, 4321 + "status": "completed", 4322 + "created_at": "2025-12-28T21:37:22.780953600-05:00", 4323 + "updated_at": "2025-12-28T21:38:15.071425500-05:00", 4324 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4325 + }, 4326 + { 4327 + "id": 394, 4328 + "change_id": "3b0dea7a-c3cd-45a8-ba1a-f1040aa4e1d9", 4329 + "node_type": "observation", 4330 + "title": "CORS fully working - Firefox extension origin properly reflected with credentials. But cookies not sent from extension despite credentials:include. Cookie set in web context not accessible from extension context due to Firefox cookie partitioning.", 4331 + "description": null, 4332 + "status": "pending", 4333 + "created_at": "2025-12-28T21:46:45.822343200-05:00", 4334 + "updated_at": "2025-12-28T21:46:45.822343200-05:00", 4335 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4336 + }, 4337 + { 4338 + "id": 395, 4339 + "change_id": "8a93413f-a09c-4cc1-8693-4fe90dc055c4", 4340 + "node_type": "action", 4341 + "title": "Updated extension checkSession to read cookie via browser.cookies API and pass as query parameter. Workaround for Firefox SameSite=Lax cookie partitioning.", 4342 + "description": null, 4343 + "status": "pending", 4344 + "created_at": "2025-12-28T21:52:22.059862700-05:00", 4345 + "updated_at": "2025-12-28T21:52:22.059862700-05:00", 4346 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 4347 + }, 4348 + { 4349 + "id": 396, 4350 + "change_id": "864dd973-5f15-4e31-a7da-c548dbbe1f0e", 4351 + "node_type": "outcome", 4352 + "title": "Extension now uses browser.cookies.get() API to read session cookie and pass as query parameter. Workaround for Firefox SameSite=Lax cookie partitioning in extensions. Extension rebuilt successfully.", 4353 + "description": null, 4354 + "status": "pending", 4355 + "created_at": "2025-12-28T22:51:31.578965200-05:00", 4356 + "updated_at": "2025-12-28T22:51:31.578965200-05:00", 4357 + "metadata_json": "{\"branch\":\"master\",\"confidence\":95}" 3764 4358 } 3765 4359 ], 3766 4360 "edges": [ ··· 7536 8130 "weight": 1.0, 7537 8131 "rationale": "All updates completed successfully", 7538 8132 "created_at": "2025-12-27T15:59:43.630208500-05:00" 8133 + }, 8134 + { 8135 + "id": 344, 8136 + "from_node_id": 342, 8137 + "to_node_id": 343, 8138 + "from_change_id": "a6d1f3fb-650d-4227-b1dc-ddb24810464c", 8139 + "to_change_id": "9e0fcead-ea30-4b31-974b-4e07f7fc6787", 8140 + "edge_type": "leads_to", 8141 + "weight": 1.0, 8142 + "rationale": "Git commit with all documentation updates", 8143 + "created_at": "2025-12-27T16:02:15.712335700-05:00" 8144 + }, 8145 + { 8146 + "id": 345, 8147 + "from_node_id": 344, 8148 + "to_node_id": 345, 8149 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8150 + "to_change_id": "0ef352ed-538b-4632-8b62-ebb17603f944", 8151 + "edge_type": "leads_to", 8152 + "weight": 1.0, 8153 + "rationale": "Installation step for Tailwind integration", 8154 + "created_at": "2025-12-27T18:00:42.787737600-05:00" 8155 + }, 8156 + { 8157 + "id": 346, 8158 + "from_node_id": 344, 8159 + "to_node_id": 346, 8160 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8161 + "to_change_id": "888e6ad0-5002-4cdb-b35e-f4214ca07dfa", 8162 + "edge_type": "leads_to", 8163 + "weight": 1.0, 8164 + "rationale": "Configuration step for Tailwind", 8165 + "created_at": "2025-12-27T18:01:28.695956-05:00" 8166 + }, 8167 + { 8168 + "id": 347, 8169 + "from_node_id": 344, 8170 + "to_node_id": 347, 8171 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8172 + "to_change_id": "fae7a634-d921-4b6f-9620-0c58d88b863e", 8173 + "edge_type": "leads_to", 8174 + "weight": 1.0, 8175 + "rationale": "Build process integration", 8176 + "created_at": "2025-12-27T18:01:51.815468700-05:00" 8177 + }, 8178 + { 8179 + "id": 348, 8180 + "from_node_id": 344, 8181 + "to_node_id": 348, 8182 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8183 + "to_change_id": "c25a8f4b-8bf1-4a33-bef9-3731dfd83627", 8184 + "edge_type": "leads_to", 8185 + "weight": 1.0, 8186 + "rationale": "CSS conversion step", 8187 + "created_at": "2025-12-27T18:02:43.312580-05:00" 8188 + }, 8189 + { 8190 + "id": 349, 8191 + "from_node_id": 344, 8192 + "to_node_id": 349, 8193 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8194 + "to_change_id": "c65ee3d9-62a0-47aa-870a-f6422ff2536a", 8195 + "edge_type": "leads_to", 8196 + "weight": 1.0, 8197 + "rationale": "HTML conversion step", 8198 + "created_at": "2025-12-27T18:03:01.642571400-05:00" 8199 + }, 8200 + { 8201 + "id": 350, 8202 + "from_node_id": 344, 8203 + "to_node_id": 350, 8204 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8205 + "to_change_id": "8136e615-5baa-4fe5-9a7d-d672ff1a6f85", 8206 + "edge_type": "leads_to", 8207 + "weight": 1.0, 8208 + "rationale": "Final outcome of Tailwind integration", 8209 + "created_at": "2025-12-27T18:07:51.011406300-05:00" 8210 + }, 8211 + { 8212 + "id": 351, 8213 + "from_node_id": 344, 8214 + "to_node_id": 351, 8215 + "from_change_id": "2a06900e-ea62-4adf-81d5-7f0cf1a29b31", 8216 + "to_change_id": "9468bcb3-78ec-4dae-8d8f-968ba6f5b3fe", 8217 + "edge_type": "leads_to", 8218 + "weight": 1.0, 8219 + "rationale": "Git commit for Tailwind integration", 8220 + "created_at": "2025-12-27T18:38:58.347778400-05:00" 8221 + }, 8222 + { 8223 + "id": 352, 8224 + "from_node_id": 352, 8225 + "to_node_id": 353, 8226 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8227 + "to_change_id": "eaed6e9b-9f16-4b45-8783-44ea2ea1f2a9", 8228 + "edge_type": "leads_to", 8229 + "weight": 1.0, 8230 + "rationale": "Initial analysis of issues", 8231 + "created_at": "2025-12-27T22:06:21.516165300-05:00" 8232 + }, 8233 + { 8234 + "id": 353, 8235 + "from_node_id": 352, 8236 + "to_node_id": 354, 8237 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8238 + "to_change_id": "d66fc83e-9737-4047-8ce2-e2ba857aeea9", 8239 + "edge_type": "leads_to", 8240 + "weight": 1.0, 8241 + "rationale": "Need to decide dark mode approach", 8242 + "created_at": "2025-12-27T22:07:03.103941500-05:00" 8243 + }, 8244 + { 8245 + "id": 354, 8246 + "from_node_id": 354, 8247 + "to_node_id": 355, 8248 + "from_change_id": "d66fc83e-9737-4047-8ce2-e2ba857aeea9", 8249 + "to_change_id": "76e2a379-7803-4c82-8013-be6b62f2d360", 8250 + "edge_type": "leads_to", 8251 + "weight": 1.0, 8252 + "rationale": "Decision outcome", 8253 + "created_at": "2025-12-27T22:07:06.239151500-05:00" 8254 + }, 8255 + { 8256 + "id": 355, 8257 + "from_node_id": 352, 8258 + "to_node_id": 356, 8259 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8260 + "to_change_id": "df681aa8-e470-4ead-a0d2-a4095febfa3d", 8261 + "edge_type": "leads_to", 8262 + "weight": 1.0, 8263 + "rationale": "Implementation of dark mode fix", 8264 + "created_at": "2025-12-27T22:07:26.713411300-05:00" 8265 + }, 8266 + { 8267 + "id": 356, 8268 + "from_node_id": 352, 8269 + "to_node_id": 357, 8270 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8271 + "to_change_id": "57060303-5a30-4f11-a752-a02376df5ea7", 8272 + "edge_type": "leads_to", 8273 + "weight": 1.0, 8274 + "rationale": "Implementation of server message fix", 8275 + "created_at": "2025-12-27T22:07:51.662925600-05:00" 8276 + }, 8277 + { 8278 + "id": 357, 8279 + "from_node_id": 352, 8280 + "to_node_id": 358, 8281 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8282 + "to_change_id": "fc211ac7-7a1a-4b69-835a-992c354e8237", 8283 + "edge_type": "leads_to", 8284 + "weight": 1.0, 8285 + "rationale": "Final outcome of fixes", 8286 + "created_at": "2025-12-27T22:09:30.425884400-05:00" 8287 + }, 8288 + { 8289 + "id": 358, 8290 + "from_node_id": 352, 8291 + "to_node_id": 359, 8292 + "from_change_id": "b852ce18-1747-4c26-a65e-acfbbed2b1a5", 8293 + "to_change_id": "4a7d5885-1713-4ba7-ad13-bb12b58c9410", 8294 + "edge_type": "leads_to", 8295 + "weight": 1.0, 8296 + "rationale": "Git commit for fixes", 8297 + "created_at": "2025-12-27T22:10:27.225192300-05:00" 8298 + }, 8299 + { 8300 + "id": 359, 8301 + "from_node_id": 360, 8302 + "to_node_id": 361, 8303 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8304 + "to_change_id": "aecf2327-d20d-4c6c-b6b0-06ccf26a2b27", 8305 + "edge_type": "leads_to", 8306 + "weight": 1.0, 8307 + "rationale": "Root cause analysis", 8308 + "created_at": "2025-12-27T22:23:47.445630900-05:00" 8309 + }, 8310 + { 8311 + "id": 360, 8312 + "from_node_id": 360, 8313 + "to_node_id": 362, 8314 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8315 + "to_change_id": "e897db97-44d8-4993-b4c3-0d829265b2f8", 8316 + "edge_type": "leads_to", 8317 + "weight": 1.0, 8318 + "rationale": "Rebuilt dev version", 8319 + "created_at": "2025-12-27T22:24:19.438433600-05:00" 8320 + }, 8321 + { 8322 + "id": 361, 8323 + "from_node_id": 360, 8324 + "to_node_id": 363, 8325 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8326 + "to_change_id": "2c62bfa3-d148-4448-8c2b-d0cf1e94ceb0", 8327 + "edge_type": "leads_to", 8328 + "weight": 1.0, 8329 + "rationale": "Root cause: CORS configuration", 8330 + "created_at": "2025-12-27T22:24:53.741163700-05:00" 8331 + }, 8332 + { 8333 + "id": 362, 8334 + "from_node_id": 360, 8335 + "to_node_id": 364, 8336 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8337 + "to_change_id": "560d6bea-47ec-408d-919b-15ca7198aac9", 8338 + "edge_type": "leads_to", 8339 + "weight": 1.0, 8340 + "rationale": "Implementation of CORS fix", 8341 + "created_at": "2025-12-27T22:25:24.843330900-05:00" 8342 + }, 8343 + { 8344 + "id": 363, 8345 + "from_node_id": 360, 8346 + "to_node_id": 365, 8347 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8348 + "to_change_id": "3ef0c9e9-aa40-4914-a5f4-32bcfaf68d04", 8349 + "edge_type": "leads_to", 8350 + "weight": 1.0, 8351 + "rationale": "CORS fix completed", 8352 + "created_at": "2025-12-27T22:41:44.160528300-05:00" 8353 + }, 8354 + { 8355 + "id": 364, 8356 + "from_node_id": 360, 8357 + "to_node_id": 366, 8358 + "from_change_id": "706d5a7f-08ed-43f7-aee5-0bed28d9402a", 8359 + "to_change_id": "77b7ed7e-a113-41f6-a677-50d376f3f008", 8360 + "edge_type": "leads_to", 8361 + "weight": 1.0, 8362 + "rationale": "Git commit for CORS fixes", 8363 + "created_at": "2025-12-27T22:42:51.663598100-05:00" 8364 + }, 8365 + { 8366 + "id": 365, 8367 + "from_node_id": 367, 8368 + "to_node_id": 368, 8369 + "from_change_id": "df6abf7a-e7a4-45f3-8485-b933319416d9", 8370 + "to_change_id": "79721edf-aa05-4580-8c28-7d20941ef155", 8371 + "edge_type": "leads_to", 8372 + "weight": 1.0, 8373 + "rationale": "Analysis step for Firefox compatibility", 8374 + "created_at": "2025-12-28T18:10:09.484445500-05:00" 8375 + }, 8376 + { 8377 + "id": 366, 8378 + "from_node_id": 368, 8379 + "to_node_id": 369, 8380 + "from_change_id": "79721edf-aa05-4580-8c28-7d20941ef155", 8381 + "to_change_id": "783841d0-c096-48f6-be18-193a9dcc7d4b", 8382 + "edge_type": "leads_to", 8383 + "weight": 1.0, 8384 + "rationale": "Detailed analysis of compatibility issues", 8385 + "created_at": "2025-12-28T18:10:49.163552300-05:00" 8386 + }, 8387 + { 8388 + "id": 367, 8389 + "from_node_id": 369, 8390 + "to_node_id": 370, 8391 + "from_change_id": "783841d0-c096-48f6-be18-193a9dcc7d4b", 8392 + "to_change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 8393 + "edge_type": "leads_to", 8394 + "weight": 1.0, 8395 + "rationale": "Need to decide implementation strategy", 8396 + "created_at": "2025-12-28T18:10:51.434960600-05:00" 8397 + }, 8398 + { 8399 + "id": 368, 8400 + "from_node_id": 370, 8401 + "to_node_id": 371, 8402 + "from_change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 8403 + "to_change_id": "159906da-984f-4a1d-a1a6-98e0fc0cf369", 8404 + "edge_type": "leads_to", 8405 + "weight": 1.0, 8406 + "rationale": "Option A", 8407 + "created_at": "2025-12-28T18:11:07.060637-05:00" 8408 + }, 8409 + { 8410 + "id": 369, 8411 + "from_node_id": 370, 8412 + "to_node_id": 372, 8413 + "from_change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 8414 + "to_change_id": "df5e42e6-53c1-4b30-8b6f-f2385cd9e247", 8415 + "edge_type": "leads_to", 8416 + "weight": 1.0, 8417 + "rationale": "Option B", 8418 + "created_at": "2025-12-28T18:11:09.223792400-05:00" 8419 + }, 8420 + { 8421 + "id": 370, 8422 + "from_node_id": 370, 8423 + "to_node_id": 373, 8424 + "from_change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 8425 + "to_change_id": "7bb58202-7a9b-4e8b-8b9e-927e5106bce7", 8426 + "edge_type": "leads_to", 8427 + "weight": 1.0, 8428 + "rationale": "Option C", 8429 + "created_at": "2025-12-28T18:11:11.439827800-05:00" 8430 + }, 8431 + { 8432 + "id": 371, 8433 + "from_node_id": 370, 8434 + "to_node_id": 374, 8435 + "from_change_id": "fd2d5b63-c26c-4592-89a6-3ccb4234c3c6", 8436 + "to_change_id": "d41b29e0-cd48-4dac-a6c8-c6179612702e", 8437 + "edge_type": "leads_to", 8438 + "weight": 1.0, 8439 + "rationale": "User selected option 1", 8440 + "created_at": "2025-12-28T19:04:26.708742600-05:00" 8441 + }, 8442 + { 8443 + "id": 372, 8444 + "from_node_id": 374, 8445 + "to_node_id": 375, 8446 + "from_change_id": "d41b29e0-cd48-4dac-a6c8-c6179612702e", 8447 + "to_change_id": "5bb34b8b-aec4-4f84-993e-eb9bf7a2d13f", 8448 + "edge_type": "leads_to", 8449 + "weight": 1.0, 8450 + "rationale": "Implementation based on decision", 8451 + "created_at": "2025-12-28T19:08:16.677078600-05:00" 8452 + }, 8453 + { 8454 + "id": 373, 8455 + "from_node_id": 375, 8456 + "to_node_id": 376, 8457 + "from_change_id": "5bb34b8b-aec4-4f84-993e-eb9bf7a2d13f", 8458 + "to_change_id": "644181ee-5a44-4967-9657-e9dd5f648c5e", 8459 + "edge_type": "leads_to", 8460 + "weight": 1.0, 8461 + "rationale": "Implementation completed successfully", 8462 + "created_at": "2025-12-28T19:14:24.961595600-05:00" 8463 + }, 8464 + { 8465 + "id": 374, 8466 + "from_node_id": 377, 8467 + "to_node_id": 378, 8468 + "from_change_id": "1dffa024-413f-4a95-b069-66db350abfaa", 8469 + "to_change_id": "9d5626d2-a9ae-42aa-8fda-be3c7528156f", 8470 + "edge_type": "leads_to", 8471 + "weight": 1.0, 8472 + "rationale": "First observation about debugging", 8473 + "created_at": "2025-12-28T20:15:13.725635900-05:00" 8474 + }, 8475 + { 8476 + "id": 375, 8477 + "from_node_id": 378, 8478 + "to_node_id": 379, 8479 + "from_change_id": "9d5626d2-a9ae-42aa-8fda-be3c7528156f", 8480 + "to_change_id": "7a5af3fe-8567-4f1c-85cd-e47891704974", 8481 + "edge_type": "leads_to", 8482 + "weight": 1.0, 8483 + "rationale": "Hypothesis about root causes", 8484 + "created_at": "2025-12-28T20:15:33.187041700-05:00" 8485 + }, 8486 + { 8487 + "id": 376, 8488 + "from_node_id": 379, 8489 + "to_node_id": 380, 8490 + "from_change_id": "7a5af3fe-8567-4f1c-85cd-e47891704974", 8491 + "to_change_id": "9c197aae-18d5-46ae-87e7-82c240c8f313", 8492 + "edge_type": "leads_to", 8493 + "weight": 1.0, 8494 + "rationale": "Fix based on hypothesis", 8495 + "created_at": "2025-12-28T20:16:14.104406300-05:00" 8496 + }, 8497 + { 8498 + "id": 377, 8499 + "from_node_id": 380, 8500 + "to_node_id": 381, 8501 + "from_change_id": "9c197aae-18d5-46ae-87e7-82c240c8f313", 8502 + "to_change_id": "485a03b0-8a25-4fdf-a8e2-9d3a25c8edf8", 8503 + "edge_type": "leads_to", 8504 + "weight": 1.0, 8505 + "rationale": "Fix implemented and tested", 8506 + "created_at": "2025-12-28T20:16:43.953511400-05:00" 8507 + }, 8508 + { 8509 + "id": 378, 8510 + "from_node_id": 381, 8511 + "to_node_id": 382, 8512 + "from_change_id": "485a03b0-8a25-4fdf-a8e2-9d3a25c8edf8", 8513 + "to_change_id": "35b13d37-0228-435f-a4bc-c5c42811fec3", 8514 + "edge_type": "leads_to", 8515 + "weight": 1.0, 8516 + "rationale": "Root cause identified from error logs", 8517 + "created_at": "2025-12-28T20:17:25.488041200-05:00" 8518 + }, 8519 + { 8520 + "id": 379, 8521 + "from_node_id": 382, 8522 + "to_node_id": 383, 8523 + "from_change_id": "35b13d37-0228-435f-a4bc-c5c42811fec3", 8524 + "to_change_id": "adc120cd-e56d-400a-9b3e-8207880378c3", 8525 + "edge_type": "leads_to", 8526 + "weight": 1.0, 8527 + "rationale": "Fix for CORS issue", 8528 + "created_at": "2025-12-28T20:19:41.484076700-05:00" 8529 + }, 8530 + { 8531 + "id": 380, 8532 + "from_node_id": 383, 8533 + "to_node_id": 384, 8534 + "from_change_id": "adc120cd-e56d-400a-9b3e-8207880378c3", 8535 + "to_change_id": "0f77bfd9-590f-4f1e-be08-78a9deef6d8a", 8536 + "edge_type": "leads_to", 8537 + "weight": 1.0, 8538 + "rationale": "Implementation complete", 8539 + "created_at": "2025-12-28T20:19:56.872404900-05:00" 8540 + }, 8541 + { 8542 + "id": 381, 8543 + "from_node_id": 384, 8544 + "to_node_id": 385, 8545 + "from_change_id": "0f77bfd9-590f-4f1e-be08-78a9deef6d8a", 8546 + "to_change_id": "cc0910f0-2381-4aee-bb5d-397cb0f804d1", 8547 + "edge_type": "leads_to", 8548 + "weight": 1.0, 8549 + "rationale": "New error reveals real issue", 8550 + "created_at": "2025-12-28T20:27:34.035766400-05:00" 8551 + }, 8552 + { 8553 + "id": 382, 8554 + "from_node_id": 385, 8555 + "to_node_id": 386, 8556 + "from_change_id": "cc0910f0-2381-4aee-bb5d-397cb0f804d1", 8557 + "to_change_id": "ad4a5ca7-15d1-4776-8ede-6b615613f6e1", 8558 + "edge_type": "leads_to", 8559 + "weight": 1.0, 8560 + "rationale": "Fix for Firefox extension origin", 8561 + "created_at": "2025-12-28T20:28:33.839045700-05:00" 8562 + }, 8563 + { 8564 + "id": 383, 8565 + "from_node_id": 386, 8566 + "to_node_id": 387, 8567 + "from_change_id": "ad4a5ca7-15d1-4776-8ede-6b615613f6e1", 8568 + "to_change_id": "cffdee0f-8535-4d88-83ed-fdf6101f7ac3", 8569 + "edge_type": "leads_to", 8570 + "weight": 1.0, 8571 + "rationale": "Complete fix implemented", 8572 + "created_at": "2025-12-28T20:30:09.745415200-05:00" 8573 + }, 8574 + { 8575 + "id": 384, 8576 + "from_node_id": 387, 8577 + "to_node_id": 388, 8578 + "from_change_id": "cffdee0f-8535-4d88-83ed-fdf6101f7ac3", 8579 + "to_change_id": "0ada864e-be98-4a2f-a14e-ffd3eea9aaa9", 8580 + "edge_type": "leads_to", 8581 + "weight": 1.0, 8582 + "rationale": "New issue discovered in health check", 8583 + "created_at": "2025-12-28T20:37:24.355885500-05:00" 8584 + }, 8585 + { 8586 + "id": 385, 8587 + "from_node_id": 388, 8588 + "to_node_id": 389, 8589 + "from_change_id": "0ada864e-be98-4a2f-a14e-ffd3eea9aaa9", 8590 + "to_change_id": "f522d5b2-c325-4f34-9f27-b8ea5c50618d", 8591 + "edge_type": "leads_to", 8592 + "weight": 1.0, 8593 + "rationale": "Fix implemented", 8594 + "created_at": "2025-12-28T20:38:22.044029100-05:00" 8595 + }, 8596 + { 8597 + "id": 386, 8598 + "from_node_id": 389, 8599 + "to_node_id": 390, 8600 + "from_change_id": "f522d5b2-c325-4f34-9f27-b8ea5c50618d", 8601 + "to_change_id": "cfdcf45b-47b3-4239-8053-417bd31957ed", 8602 + "edge_type": "leads_to", 8603 + "weight": 1.0, 8604 + "rationale": "Issue persists - need to debug headers", 8605 + "created_at": "2025-12-28T20:48:14.949702100-05:00" 8606 + }, 8607 + { 8608 + "id": 387, 8609 + "from_node_id": 390, 8610 + "to_node_id": 391, 8611 + "from_change_id": "cfdcf45b-47b3-4239-8053-417bd31957ed", 8612 + "to_change_id": "2b53a419-9a47-4285-9a12-9bdfaeeb9ff0", 8613 + "edge_type": "leads_to", 8614 + "weight": 1.0, 8615 + "rationale": "Root cause identified from debug logs", 8616 + "created_at": "2025-12-28T20:55:34.094943700-05:00" 8617 + }, 8618 + { 8619 + "id": 388, 8620 + "from_node_id": 391, 8621 + "to_node_id": 392, 8622 + "from_change_id": "2b53a419-9a47-4285-9a12-9bdfaeeb9ff0", 8623 + "to_change_id": "c941d136-3405-483d-bf34-7fb011f6d072", 8624 + "edge_type": "leads_to", 8625 + "weight": 1.0, 8626 + "rationale": "Fix implemented", 8627 + "created_at": "2025-12-28T20:57:35.872426900-05:00" 8628 + }, 8629 + { 8630 + "id": 389, 8631 + "from_node_id": 392, 8632 + "to_node_id": 393, 8633 + "from_change_id": "c941d136-3405-483d-bf34-7fb011f6d072", 8634 + "to_change_id": "aafd9977-8800-4152-9f7f-b817db6df573", 8635 + "edge_type": "leads_to", 8636 + "weight": 1.0, 8637 + "rationale": "Complete fix with cleanup", 8638 + "created_at": "2025-12-28T21:37:27.704906300-05:00" 8639 + }, 8640 + { 8641 + "id": 390, 8642 + "from_node_id": 393, 8643 + "to_node_id": 394, 8644 + "from_change_id": "aafd9977-8800-4152-9f7f-b817db6df573", 8645 + "to_change_id": "3b0dea7a-c3cd-45a8-ba1a-f1040aa4e1d9", 8646 + "edge_type": "leads_to", 8647 + "weight": 1.0, 8648 + "rationale": "New issue - cookie partitioning", 8649 + "created_at": "2025-12-28T21:46:48.417911400-05:00" 8650 + }, 8651 + { 8652 + "id": 391, 8653 + "from_node_id": 394, 8654 + "to_node_id": 395, 8655 + "from_change_id": "3b0dea7a-c3cd-45a8-ba1a-f1040aa4e1d9", 8656 + "to_change_id": "8a93413f-a09c-4cc1-8693-4fe90dc055c4", 8657 + "edge_type": "leads_to", 8658 + "weight": 1.0, 8659 + "rationale": "Workaround using browser.cookies API", 8660 + "created_at": "2025-12-28T21:52:52.704792400-05:00" 8661 + }, 8662 + { 8663 + "id": 392, 8664 + "from_node_id": 395, 8665 + "to_node_id": 396, 8666 + "from_change_id": "8a93413f-a09c-4cc1-8693-4fe90dc055c4", 8667 + "to_change_id": "864dd973-5f15-4e31-a7da-c548dbbe1f0e", 8668 + "edge_type": "leads_to", 8669 + "weight": 1.0, 8670 + "rationale": "Complete workaround", 8671 + "created_at": "2025-12-28T22:51:33.159870400-05:00" 7539 8672 } 7540 8673 ] 7541 8674 }
+173
packages/extension/FIREFOX.md
··· 1 + # Firefox Extension Installation Guide 2 + 3 + The ATlast Importer extension now supports both Chrome and Firefox! 4 + 5 + ## Building for Firefox 6 + 7 + The build system automatically creates both Chrome and Firefox versions: 8 + 9 + ```bash 10 + pnpm run build # Development build for both browsers 11 + pnpm run build:prod # Production build for both browsers 12 + ``` 13 + 14 + Output directories: 15 + - `dist/chrome/` - Chrome/Edge version (Manifest V3 with service worker) 16 + - `dist/firefox/` - Firefox version (Manifest V3 with scripts array) 17 + 18 + ## Installing in Firefox (Development) 19 + 20 + ### Option 1: Temporary Installation (for testing) 21 + 22 + 1. Open Firefox 23 + 2. Navigate to `about:debugging#/runtime/this-firefox` 24 + 3. Click "Load Temporary Add-on..." 25 + 4. Navigate to `packages/extension/dist/firefox/` 26 + 5. Select the `manifest.json` file 27 + 28 + **Note:** Temporary extensions are removed when Firefox restarts. 29 + 30 + ### Option 2: Loading from ZIP (for distribution) 31 + 32 + 1. Build the production version: 33 + ```bash 34 + pnpm run build:prod 35 + pnpm run package:firefox 36 + ``` 37 + 38 + 2. This creates `dist/firefox.zip` 39 + 40 + 3. For testing: 41 + - Go to `about:debugging#/runtime/this-firefox` 42 + - Click "Load Temporary Add-on..." 43 + - Select the `firefox.zip` file 44 + 45 + 4. For publishing: 46 + - Submit `firefox.zip` to [addons.mozilla.org](https://addons.mozilla.org/developers/) 47 + 48 + ## Key Differences from Chrome Version 49 + 50 + ### Manifest Differences 51 + 52 + **Chrome (`manifest.chrome.json`):** 53 + ```json 54 + { 55 + "manifest_version": 3, 56 + "background": { 57 + "service_worker": "background/service-worker.js", 58 + "type": "module" 59 + } 60 + } 61 + ``` 62 + 63 + **Firefox (`manifest.firefox.json`):** 64 + ```json 65 + { 66 + "manifest_version": 3, 67 + "background": { 68 + "scripts": ["background/service-worker.js"], 69 + "type": "module" 70 + }, 71 + "browser_specific_settings": { 72 + "gecko": { 73 + "id": "atlast-importer@byarielm.fyi", 74 + "strict_min_version": "109.0" 75 + } 76 + } 77 + } 78 + ``` 79 + 80 + ### Cross-Browser Compatibility 81 + 82 + - All code uses `webextension-polyfill` library 83 + - Chrome-specific `chrome.*` APIs replaced with unified `browser.*` API 84 + - Promise-based instead of callback-based 85 + - Single codebase works across both browsers 86 + 87 + ### Requirements 88 + 89 + - **Firefox:** Version 109+ (for Manifest V3 support) 90 + - **Chrome/Edge:** Latest version 91 + 92 + ## Testing 93 + 94 + After loading the extension in Firefox: 95 + 96 + 1. Navigate to Twitter/X Following page (e.g., `https://twitter.com/username/following`) 97 + 2. Click the extension icon in the toolbar 98 + 3. The popup should show "Ready to scan" state 99 + 4. Click "Start Scan" to scrape usernames 100 + 5. Click "Open on ATlast" to upload results 101 + 102 + ## Debugging 103 + 104 + ### View Console Logs 105 + 106 + **Background Script:** 107 + - Go to `about:debugging#/runtime/this-firefox` 108 + - Find "ATlast Importer" in the list 109 + - Click "Inspect" 110 + 111 + **Popup:** 112 + - Right-click extension icon โ†’ "Inspect Extension" 113 + 114 + **Content Script:** 115 + - Open DevTools on Twitter/X page (F12) 116 + - Look for `[ATlast]` prefixed logs in Console 117 + 118 + ### Common Issues 119 + 120 + 1. **Extension not loading:** 121 + - Check Firefox version is 109+ 122 + - Ensure manifest.json is valid 123 + - Check browser console for errors 124 + 125 + 2. **Scan not starting:** 126 + - Verify you're on Twitter/X Following page 127 + - Check content script is injected (look for console logs) 128 + - Ensure page is fully loaded 129 + 130 + 3. **"Server offline" message:** 131 + - Make sure dev server is running (`netlify dev`) 132 + - Check API URL in extension settings 133 + 134 + ## Packaging for Distribution 135 + 136 + Create production builds for both browsers: 137 + 138 + ```bash 139 + pnpm run package:prod 140 + ``` 141 + 142 + This creates: 143 + - `dist/chrome.zip` - Ready for Chrome Web Store 144 + - `dist/firefox.zip` - Ready for Firefox Add-ons 145 + 146 + ## Development Workflow 147 + 148 + ```bash 149 + # Watch mode (auto-rebuild on changes) 150 + pnpm run dev 151 + 152 + # In Firefox: 153 + # 1. about:debugging โ†’ Reload extension after each rebuild 154 + # 2. Or use web-ext for auto-reload: 155 + 156 + npx web-ext run --source-dir=dist/firefox 157 + ``` 158 + 159 + ## Differences You Might Notice 160 + 161 + 1. **Background page persistence:** 162 + - Chrome: Service worker (non-persistent) 163 + - Firefox: Scripts array (similar behavior in MV3) 164 + 165 + 2. **API behavior:** 166 + - Firefox: Native Promise support 167 + - Chrome: Promises via polyfill 168 + 169 + 3. **Extension ID:** 170 + - Chrome: Auto-generated 171 + - Firefox: Explicitly set as `atlast-importer@byarielm.fyi` 172 + 173 + Both versions use the same source code and should behave identically!
+93 -40
packages/extension/build.js
··· 2 2 import * as fs from 'fs'; 3 3 import * as path from 'path'; 4 4 import { fileURLToPath } from 'url'; 5 + import postcss from 'postcss'; 6 + import tailwindcss from 'tailwindcss'; 7 + import autoprefixer from 'autoprefixer'; 5 8 6 9 const __dirname = path.dirname(fileURLToPath(import.meta.url)); 7 10 ··· 18 21 console.log(`๐Ÿ”— API URL: ${ATLAST_API_URL}`); 19 22 20 23 // Clean dist directory 21 - const distDir = path.join(__dirname, 'dist', 'chrome'); 22 - if (fs.existsSync(distDir)) { 23 - fs.rmSync(distDir, { recursive: true }); 24 + const distBaseDir = path.join(__dirname, 'dist'); 25 + if (fs.existsSync(distBaseDir)) { 26 + fs.rmSync(distBaseDir, { recursive: true }); 24 27 } 25 - fs.mkdirSync(distDir, { recursive: true }); 28 + fs.mkdirSync(distBaseDir, { recursive: true }); 26 29 27 30 // Build configuration base 28 31 const buildConfigBase = { ··· 35 38 '__ATLAST_API_URL__': JSON.stringify(ATLAST_API_URL), 36 39 '__BUILD_MODE__': JSON.stringify(mode), 37 40 }, 41 + // Include webextension-polyfill in the bundle 42 + external: [], 38 43 }; 39 44 40 - // Build scripts 41 - const scripts = [ 42 - { 43 - ...buildConfigBase, 44 - entryPoints: ['src/content/index.ts'], 45 - outfile: path.join(distDir, 'content', 'index.js'), 46 - }, 47 - { 48 - ...buildConfigBase, 49 - entryPoints: ['src/background/service-worker.ts'], 50 - outfile: path.join(distDir, 'background', 'service-worker.js'), 51 - }, 52 - { 53 - ...buildConfigBase, 54 - entryPoints: ['src/popup/popup.ts'], 55 - outfile: path.join(distDir, 'popup', 'popup.js'), 56 - }, 57 - ]; 45 + // Build scripts for a specific browser 46 + function getScripts(browser) { 47 + const distDir = path.join(distBaseDir, browser); 48 + return [ 49 + { 50 + ...buildConfigBase, 51 + entryPoints: ['src/content/index.ts'], 52 + outfile: path.join(distDir, 'content', 'index.js'), 53 + }, 54 + { 55 + ...buildConfigBase, 56 + entryPoints: ['src/background/service-worker.ts'], 57 + outfile: path.join(distDir, 'background', 'service-worker.js'), 58 + }, 59 + { 60 + ...buildConfigBase, 61 + entryPoints: ['src/popup/popup.ts'], 62 + outfile: path.join(distDir, 'popup', 'popup.js'), 63 + }, 64 + ]; 65 + } 58 66 59 67 // Build function 60 68 async function build() { 61 69 try { 62 - console.log('๐Ÿ”จ Building extension...'); 70 + console.log('๐Ÿ”จ Building extension for Chrome and Firefox...'); 63 71 64 - // Build all scripts 65 - for (const config of scripts) { 66 - if (watch) { 67 - const ctx = await esbuild.context(config); 68 - await ctx.watch(); 69 - console.log(`๐Ÿ‘€ Watching ${path.basename(config.entryPoints[0])}...`); 70 - } else { 71 - await esbuild.build(config); 72 - console.log(`โœ… Built ${path.basename(config.entryPoints[0])}`); 72 + const browsers = ['chrome', 'firefox']; 73 + 74 + for (const browser of browsers) { 75 + console.log(`\n๐Ÿ“ฆ Building ${browser} version...`); 76 + const scripts = getScripts(browser); 77 + 78 + // Build all scripts 79 + for (const config of scripts) { 80 + if (watch) { 81 + const ctx = await esbuild.context(config); 82 + await ctx.watch(); 83 + console.log(`๐Ÿ‘€ Watching ${browser}/${path.basename(config.entryPoints[0])}...`); 84 + } else { 85 + await esbuild.build(config); 86 + console.log(`โœ… Built ${browser}/${path.basename(config.entryPoints[0])}`); 87 + } 73 88 } 89 + 90 + // Copy static files 91 + copyStaticFiles(browser); 92 + 93 + // Process CSS with Tailwind 94 + await processCSS(browser); 74 95 } 75 96 76 - // Copy static files 77 - copyStaticFiles(); 78 - 79 97 if (!watch) { 80 - console.log('โœจ Build complete!'); 98 + console.log('\nโœจ Build complete for both browsers!'); 81 99 } 82 100 } catch (error) { 83 101 console.error('โŒ Build failed:', error); ··· 85 103 } 86 104 } 87 105 106 + // Process CSS with PostCSS (Tailwind + Autoprefixer) 107 + async function processCSS(browser) { 108 + const cssPath = path.join(__dirname, 'src/popup/popup.css'); 109 + const distDir = path.join(distBaseDir, browser); 110 + const outputPath = path.join(distDir, 'popup/popup.css'); 111 + 112 + const css = fs.readFileSync(cssPath, 'utf8'); 113 + 114 + // Import cssnano dynamically for production minification 115 + const plugins = [tailwindcss, autoprefixer]; 116 + if (isProd) { 117 + const cssnano = (await import('cssnano')).default; 118 + plugins.push(cssnano); 119 + } 120 + 121 + const result = await postcss(plugins).process(css, { 122 + from: cssPath, 123 + to: outputPath, 124 + }); 125 + 126 + // Create directory if it doesn't exist 127 + const destDir = path.dirname(outputPath); 128 + if (!fs.existsSync(destDir)) { 129 + fs.mkdirSync(destDir, { recursive: true }); 130 + } 131 + 132 + fs.writeFileSync(outputPath, result.css); 133 + console.log('๐ŸŽจ Processed CSS with Tailwind'); 134 + } 135 + 88 136 // Copy static files 89 - function copyStaticFiles() { 137 + function copyStaticFiles(browser) { 138 + const distDir = path.join(distBaseDir, browser); 139 + 90 140 const filesToCopy = [ 91 - { from: 'manifest.json', to: 'manifest.json' }, 141 + { from: `manifest.${browser}.json`, to: 'manifest.json', fallback: 'manifest.json' }, 92 142 { from: 'src/popup/popup.html', to: 'popup/popup.html' }, 93 - { from: 'src/popup/popup.css', to: 'popup/popup.css' }, 94 143 ]; 95 144 96 145 for (const file of filesToCopy) { 97 - const srcPath = path.join(__dirname, file.from); 146 + // Try to use browser-specific file first, fall back to default 147 + let srcPath = path.join(__dirname, file.from); 148 + if (file.fallback && !fs.existsSync(srcPath)) { 149 + srcPath = path.join(__dirname, file.fallback); 150 + } 98 151 const destPath = path.join(distDir, file.to); 99 152 100 153 // Create directory if it doesn't exist
+44
packages/extension/manifest.chrome.json
··· 1 + { 2 + "manifest_version": 3, 3 + "name": "ATlast Importer", 4 + "version": "1.0.0", 5 + "description": "Import your Twitter/X follows to find them on Bluesky", 6 + "permissions": [ 7 + "activeTab", 8 + "storage" 9 + ], 10 + "host_permissions": [ 11 + "https://twitter.com/*", 12 + "https://x.com/*", 13 + "http://127.0.0.1:8888/*", 14 + "http://localhost:8888/*", 15 + "https://atlast.byarielm.fyi/*" 16 + ], 17 + "background": { 18 + "service_worker": "background/service-worker.js", 19 + "type": "module" 20 + }, 21 + "content_scripts": [ 22 + { 23 + "matches": [ 24 + "https://twitter.com/*", 25 + "https://x.com/*" 26 + ], 27 + "js": ["content/index.js"], 28 + "run_at": "document_idle" 29 + } 30 + ], 31 + "action": { 32 + "default_popup": "popup/popup.html", 33 + "default_icon": { 34 + "16": "assets/icon-16.png", 35 + "48": "assets/icon-48.png", 36 + "128": "assets/icon-128.png" 37 + } 38 + }, 39 + "icons": { 40 + "16": "assets/icon-16.png", 41 + "48": "assets/icon-48.png", 42 + "128": "assets/icon-128.png" 43 + } 44 + }
+51
packages/extension/manifest.firefox.json
··· 1 + { 2 + "manifest_version": 3, 3 + "name": "ATlast Importer", 4 + "version": "1.0.0", 5 + "description": "Import your Twitter/X follows to find them on Bluesky", 6 + "permissions": [ 7 + "activeTab", 8 + "storage", 9 + "cookies" 10 + ], 11 + "host_permissions": [ 12 + "https://twitter.com/*", 13 + "https://x.com/*", 14 + "http://127.0.0.1:8888/*", 15 + "http://localhost:8888/*", 16 + "https://atlast.byarielm.fyi/*" 17 + ], 18 + "background": { 19 + "scripts": ["background/service-worker.js"], 20 + "type": "module" 21 + }, 22 + "content_scripts": [ 23 + { 24 + "matches": [ 25 + "https://twitter.com/*", 26 + "https://x.com/*" 27 + ], 28 + "js": ["content/index.js"], 29 + "run_at": "document_idle" 30 + } 31 + ], 32 + "action": { 33 + "default_popup": "popup/popup.html", 34 + "default_icon": { 35 + "16": "assets/icon-16.png", 36 + "48": "assets/icon-48.png", 37 + "128": "assets/icon-128.png" 38 + } 39 + }, 40 + "icons": { 41 + "16": "assets/icon-16.png", 42 + "48": "assets/icon-48.png", 43 + "128": "assets/icon-128.png" 44 + }, 45 + "browser_specific_settings": { 46 + "gecko": { 47 + "id": "atlast-importer@byarielm.fyi", 48 + "strict_min_version": "109.0" 49 + } 50 + } 51 + }
+10 -2
packages/extension/package.json
··· 9 9 "build:prod": "node build.js --prod", 10 10 "dev": "node build.js --watch", 11 11 "package:chrome": "cd dist/chrome && zip -r ../chrome.zip .", 12 - "package:prod": "npm run build:prod && npm run package:chrome" 12 + "package:firefox": "cd dist/firefox && zip -r ../firefox.zip .", 13 + "package:all": "pnpm run package:chrome && pnpm run package:firefox", 14 + "package:prod": "pnpm run build:prod && pnpm run package:all" 13 15 }, 14 16 "dependencies": { 15 - "@atlast/shared": "workspace:*" 17 + "@atlast/shared": "workspace:*", 18 + "webextension-polyfill": "^0.12.0" 16 19 }, 17 20 "devDependencies": { 18 21 "@types/chrome": "^0.0.256", 22 + "@types/webextension-polyfill": "^0.12.4", 23 + "autoprefixer": "^10.4.23", 24 + "cssnano": "^7.1.2", 19 25 "esbuild": "^0.19.11", 26 + "postcss": "^8.5.6", 27 + "tailwindcss": "^3.4.19", 20 28 "typescript": "^5.3.3" 21 29 } 22 30 }
+7
packages/extension/postcss.config.js
··· 1 + export default { 2 + plugins: { 3 + tailwindcss: {}, 4 + autoprefixer: {}, 5 + ...(process.env.NODE_ENV === 'production' ? { cssnano: {} } : {}), 6 + }, 7 + };
+2 -1
packages/extension/src/background/service-worker.ts
··· 1 + import browser from 'webextension-polyfill'; 1 2 import { 2 3 MessageType, 3 4 onMessage, ··· 150 151 /** 151 152 * Log extension installation 152 153 */ 153 - chrome.runtime.onInstalled.addListener((details) => { 154 + browser.runtime.onInstalled.addListener((details) => { 154 155 console.log('[Background] Extension installed:', details.reason); 155 156 156 157 if (details.reason === 'install') {
+36 -9
packages/extension/src/lib/api-client.ts
··· 2 2 * ATlast API client for extension 3 3 */ 4 4 5 + import browser from 'webextension-polyfill'; 6 + 5 7 // These are replaced at build time by esbuild 6 8 declare const __ATLAST_API_URL__: string; 7 9 declare const __BUILD_MODE__: string; ··· 67 69 * Get extension version from manifest 68 70 */ 69 71 export function getExtensionVersion(): string { 70 - return chrome.runtime.getManifest().version; 72 + return browser.runtime.getManifest().version; 71 73 } 72 74 73 75 /** ··· 76 78 */ 77 79 export async function checkServerHealth(): Promise<boolean> { 78 80 try { 79 - // Try to fetch the root URL with a short timeout 81 + // Try to fetch the health endpoint with a short timeout 80 82 const controller = new AbortController(); 81 83 const timeoutId = setTimeout(() => controller.abort(), 3000); 82 84 83 - const response = await fetch(ATLAST_API_URL, { 84 - method: 'HEAD', 85 - signal: controller.signal 85 + const response = await fetch(`${ATLAST_API_URL}/.netlify/functions/health`, { 86 + method: 'GET', 87 + signal: controller.signal, 88 + credentials: 'include', // Include for CORS 86 89 }); 87 90 88 91 clearTimeout(timeoutId); 89 92 90 - // Any response (even 404) means server is running 91 - return true; 93 + // Any successful response means server is running 94 + return response.ok; 92 95 } catch (error) { 93 96 console.error('[API Client] Server health check failed:', error); 94 97 return false; ··· 113 116 avatar?: string; 114 117 } | null> { 115 118 try { 116 - const response = await fetch(`${ATLAST_API_URL}/.netlify/functions/session`, { 119 + // Try to get session cookie using browser.cookies API 120 + // This works around Firefox's cookie partitioning for extensions 121 + let sessionId: string | null = null; 122 + 123 + try { 124 + const cookieName = __BUILD_MODE__ === 'production' ? 'atlast_session' : 'atlast_session_dev'; 125 + const cookie = await browser.cookies.get({ 126 + url: ATLAST_API_URL, 127 + name: cookieName 128 + }); 129 + 130 + if (cookie) { 131 + sessionId = cookie.value; 132 + console.log('[API Client] Found session cookie:', cookieName); 133 + } 134 + } catch (cookieError) { 135 + console.log('[API Client] Could not read cookie:', cookieError); 136 + } 137 + 138 + // Build URL with session parameter if we have one 139 + const url = sessionId 140 + ? `${ATLAST_API_URL}/.netlify/functions/session?session=${sessionId}` 141 + : `${ATLAST_API_URL}/.netlify/functions/session`; 142 + 143 + const response = await fetch(url, { 117 144 method: 'GET', 118 - credentials: 'include', // Include cookies 145 + credentials: 'include', // Include cookies as fallback 119 146 headers: { 120 147 'Accept': 'application/json' 121 148 }
+6 -5
packages/extension/src/lib/messaging.ts
··· 1 + import browser from 'webextension-polyfill'; 1 2 import type { ScraperProgress, ScraperResult } from '../content/scrapers/base-scraper.js'; 2 3 3 4 /** ··· 87 88 * Send message to background script 88 89 */ 89 90 export function sendToBackground<T = any>(message: Message): Promise<T> { 90 - return chrome.runtime.sendMessage(message); 91 + return browser.runtime.sendMessage(message); 91 92 } 92 93 93 94 /** 94 95 * Send message to active tab's content script 95 96 */ 96 97 export async function sendToContent(message: Message): Promise<any> { 97 - const [tab] = await chrome.tabs.query({ active: true, currentWindow: true }); 98 + const [tab] = await browser.tabs.query({ active: true, currentWindow: true }); 98 99 if (!tab.id) { 99 100 throw new Error('No active tab found'); 100 101 } 101 - return chrome.tabs.sendMessage(tab.id, message); 102 + return browser.tabs.sendMessage(tab.id, message); 102 103 } 103 104 104 105 /** 105 106 * Listen for messages 106 107 */ 107 108 export function onMessage( 108 - handler: (message: Message, sender: chrome.runtime.MessageSender) => any | Promise<any> 109 + handler: (message: Message, sender: browser.Runtime.MessageSender) => any | Promise<any> 109 110 ): void { 110 - chrome.runtime.onMessage.addListener((message, sender, sendResponse) => { 111 + browser.runtime.onMessage.addListener((message, sender, sendResponse) => { 111 112 const result = handler(message, sender); 112 113 113 114 // Handle async handlers
+4 -3
packages/extension/src/lib/storage.ts
··· 1 + import browser from 'webextension-polyfill'; 1 2 import type { ExtensionState } from './messaging.js'; 2 3 3 4 /** ··· 11 12 * Get extension state from storage 12 13 */ 13 14 export async function getState(): Promise<ExtensionState> { 14 - const result = await chrome.storage.local.get(STORAGE_KEYS.STATE); 15 + const result = await browser.storage.local.get(STORAGE_KEYS.STATE); 15 16 return result[STORAGE_KEYS.STATE] || { status: 'idle' }; 16 17 } 17 18 ··· 19 20 * Save extension state to storage 20 21 */ 21 22 export async function setState(state: ExtensionState): Promise<void> { 22 - await chrome.storage.local.set({ [STORAGE_KEYS.STATE]: state }); 23 + await browser.storage.local.set({ [STORAGE_KEYS.STATE]: state }); 23 24 } 24 25 25 26 /** 26 27 * Clear extension state 27 28 */ 28 29 export async function clearState(): Promise<void> { 29 - await chrome.storage.local.remove(STORAGE_KEYS.STATE); 30 + await browser.storage.local.remove(STORAGE_KEYS.STATE); 30 31 }
+16 -283
packages/extension/src/popup/popup.css
··· 1 - * { 2 - margin: 0; 3 - padding: 0; 4 - box-sizing: border-box; 5 - } 6 - 7 - code { 8 - background: rgba(0, 0, 0, 0.1); 9 - padding: 4px 8px; 10 - border-radius: 4px; 11 - font-family: 'Courier New', monospace; 12 - font-size: 11px; 13 - display: inline-block; 14 - margin: 8px 0; 15 - } 16 - 17 - @media (prefers-color-scheme: dark) { 18 - code { 19 - background: rgba(255, 255, 255, 0.1); 20 - } 21 - } 22 - 23 - body { 24 - width: 350px; 25 - min-height: 400px; 26 - font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; 27 - color: #1e293b; 28 - background: linear-gradient(135deg, #faf5ff 0%, #ffffff 50%, #ecfeff 100%); 29 - } 30 - 31 - @media (prefers-color-scheme: dark) { 32 - body { 33 - color: #e0f2fe; 34 - background: linear-gradient(135deg, #0f172a 0%, #1e1b4b 50%, #0c4a6e 100%); 35 - } 36 - } 37 - 38 - .container { 39 - display: flex; 40 - flex-direction: column; 41 - min-height: 400px; 42 - } 43 - 44 - header { 45 - background: linear-gradient(to right, #facc15 0%, #f97316 50%, #ec4899 100%); 46 - color: white; 47 - padding: 20px; 48 - text-align: center; 49 - } 50 - 51 - h1 { 52 - font-size: 20px; 53 - font-weight: 700; 54 - margin-bottom: 4px; 55 - } 56 - 57 - .tagline { 58 - font-size: 13px; 59 - opacity: 0.9; 60 - } 61 - 62 - main { 63 - flex: 1; 64 - padding: 24px 20px; 65 - display: flex; 66 - align-items: center; 67 - justify-content: center; 68 - } 69 - 70 - .state { 71 - width: 100%; 72 - text-align: center; 73 - } 74 - 75 - .state.hidden { 76 - display: none; 77 - } 1 + @tailwind base; 2 + @tailwind components; 3 + @tailwind utilities; 78 4 79 - .icon { 80 - font-size: 48px; 81 - margin-bottom: 16px; 82 - } 83 - 84 - .spinner { 85 - animation: spin 2s linear infinite; 86 - } 87 - 5 + /* Custom animations for spinner */ 88 6 @keyframes spin { 89 - from { transform: rotate(0deg); } 90 - to { transform: rotate(360deg); } 91 - } 92 - 93 - .message { 94 - font-size: 16px; 95 - font-weight: 600; 96 - margin-bottom: 12px; 97 - color: #334155; 98 - } 99 - 100 - @media (prefers-color-scheme: dark) { 101 - .message { 102 - color: #e0f2fe; 7 + from { 8 + transform: rotate(0deg); 103 9 } 104 - } 105 - 106 - .hint { 107 - font-size: 13px; 108 - color: #64748b; 109 - margin-top: 8px; 110 - } 111 - 112 - @media (prefers-color-scheme: dark) { 113 - .hint { 114 - color: #94a3b8; 10 + to { 11 + transform: rotate(360deg); 115 12 } 116 13 } 117 14 118 - .btn-primary { 119 - background: #ea580c; 120 - color: white; 121 - border: none; 122 - padding: 12px 24px; 123 - border-radius: 8px; 124 - font-size: 14px; 125 - font-weight: 600; 126 - cursor: pointer; 127 - margin-top: 16px; 128 - width: 100%; 129 - transition: transform 0.2s, box-shadow 0.2s, background-color 0.2s; 130 - } 131 - 132 - .btn-primary:hover { 133 - background: #c2410c; 134 - transform: translateY(-1px); 135 - box-shadow: 0 4px 12px rgba(234, 88, 12, 0.3); 136 - } 137 - 138 - .btn-primary:active { 139 - transform: translateY(0); 140 - } 141 - 142 - .btn-secondary { 143 - background: white; 144 - color: #6b21a8; 145 - border: 2px solid #6b21a8; 146 - padding: 10px 24px; 147 - border-radius: 8px; 148 - font-size: 14px; 149 - font-weight: 600; 150 - cursor: pointer; 151 - margin-top: 16px; 152 - width: 100%; 153 - transition: all 0.2s; 154 - } 155 - 156 - .btn-secondary:hover { 157 - background: #faf5ff; 158 - } 159 - 160 - @media (prefers-color-scheme: dark) { 161 - .btn-secondary { 162 - background: #1e1b4b; 163 - color: #06b6d4; 164 - border-color: #06b6d4; 15 + @keyframes pulse { 16 + 0%, 17 + 100% { 18 + opacity: 1; 165 19 } 166 - 167 - .btn-secondary:hover { 168 - background: #312e81; 20 + 50% { 21 + opacity: 0.7; 169 22 } 170 23 } 171 24 172 - .progress { 173 - margin-top: 20px; 174 - } 175 - 176 - .progress-bar { 177 - width: 100%; 178 - height: 8px; 179 - background: #f0f9ff; 180 - border-radius: 4px; 181 - overflow: hidden; 182 - margin-bottom: 12px; 183 - } 184 - 185 - @media (prefers-color-scheme: dark) { 186 - .progress-bar { 187 - background: #1e293b; 188 - } 25 + .spinner { 26 + animation: spin 2s linear infinite; 189 27 } 190 28 191 29 .progress-fill { 192 - height: 100%; 193 - background: linear-gradient(90deg, #ea580c 0%, #ec4899 100%); 194 - width: 0%; 195 - transition: width 0.3s ease; 196 30 animation: pulse 2s infinite; 197 31 } 198 - 199 - @keyframes pulse { 200 - 0%, 100% { opacity: 1; } 201 - 50% { opacity: 0.7; } 202 - } 203 - 204 - .progress-text { 205 - font-size: 16px; 206 - font-weight: 600; 207 - color: #334155; 208 - } 209 - 210 - @media (prefers-color-scheme: dark) { 211 - .progress-text { 212 - color: #e0f2fe; 213 - } 214 - } 215 - 216 - .status-message { 217 - font-size: 13px; 218 - color: #64748b; 219 - margin-top: 8px; 220 - } 221 - 222 - @media (prefers-color-scheme: dark) { 223 - .status-message { 224 - color: #94a3b8; 225 - } 226 - } 227 - 228 - .count-display { 229 - font-size: 14px; 230 - color: #64748b; 231 - margin-top: 8px; 232 - } 233 - 234 - @media (prefers-color-scheme: dark) { 235 - .count-display { 236 - color: #94a3b8; 237 - } 238 - } 239 - 240 - .count-display strong { 241 - color: #ea580c; 242 - font-size: 18px; 243 - } 244 - 245 - @media (prefers-color-scheme: dark) { 246 - .count-display strong { 247 - color: #fb923c; 248 - } 249 - } 250 - 251 - .error-message { 252 - font-size: 13px; 253 - color: #dc2626; 254 - margin-top: 8px; 255 - padding: 12px; 256 - background: #fee2e2; 257 - border-radius: 6px; 258 - border-left: 3px solid #dc2626; 259 - } 260 - 261 - @media (prefers-color-scheme: dark) { 262 - .error-message { 263 - color: #fca5a5; 264 - background: #450a0a; 265 - border-left-color: #ef4444; 266 - } 267 - } 268 - 269 - footer { 270 - padding: 16px; 271 - text-align: center; 272 - border-top: 1px solid #e0e7ff; 273 - background: white; 274 - } 275 - 276 - @media (prefers-color-scheme: dark) { 277 - footer { 278 - border-top-color: #1e293b; 279 - background: #0f172a; 280 - } 281 - } 282 - 283 - footer a { 284 - color: #ea580c; 285 - text-decoration: none; 286 - font-size: 13px; 287 - font-weight: 500; 288 - } 289 - 290 - @media (prefers-color-scheme: dark) { 291 - footer a { 292 - color: #fb923c; 293 - } 294 - } 295 - 296 - footer a:hover { 297 - text-decoration: underline; 298 - }
+51 -53
packages/extension/src/popup/popup.html
··· 6 6 <title>ATlast Importer</title> 7 7 <link rel="stylesheet" href="popup.css" /> 8 8 </head> 9 - <body> 10 - <div class="container"> 11 - <header> 12 - <h1>ATlast Importer</h1> 13 - <p class="tagline">Find your follows in the ATmosphere</p> 9 + <body class="w-[350px] min-h-[400px] font-sans text-slate-800 dark:text-cyan-50 bg-gradient-to-br from-purple-50 via-white to-cyan-50 dark:from-slate-900 dark:via-purple-950 dark:to-sky-900"> 10 + <div class="flex flex-col min-h-[400px]"> 11 + <header class="bg-firefly-banner text-white p-5 text-center"> 12 + <h1 class="text-xl font-bold mb-1">ATlast Importer</h1> 13 + <p class="text-[13px] opacity-90">Find your follows in the ATmosphere</p> 14 14 </header> 15 15 16 - <main id="app"> 16 + <main id="app" class="flex-1 px-5 py-6 flex items-center justify-center"> 17 17 <!-- Idle state --> 18 - <div id="state-idle" class="state hidden"> 19 - <div class="icon">๐Ÿ”</div> 20 - <p class="message"> 18 + <div id="state-idle" class="w-full text-center hidden"> 19 + <div class="text-5xl mb-4">๐Ÿ”</div> 20 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50"> 21 21 Go to your Twitter/X Following page to start 22 22 </p> 23 - <p class="hint">Visit x.com/yourusername/following</p> 23 + <p class="text-[13px] text-slate-500 dark:text-slate-400 mt-2">Visit x.com/yourusername/following</p> 24 24 </div> 25 25 26 26 <!-- Ready state --> 27 - <div id="state-ready" class="state hidden"> 28 - <div class="icon">โœ…</div> 29 - <p class="message"> 27 + <div id="state-ready" class="w-full text-center hidden"> 28 + <div class="text-5xl mb-4">โœ…</div> 29 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50"> 30 30 Ready to scan <span id="platform-name"></span> 31 31 </p> 32 - <button id="btn-start" class="btn-primary"> 32 + <button id="btn-start" class="w-full bg-orange-600 hover:bg-orange-700 text-white font-semibold py-3 px-6 rounded-lg mt-4 transition-all duration-200 hover:-translate-y-0.5 hover:shadow-lg hover:shadow-orange-600/30 active:translate-y-0"> 33 33 Start Scan 34 34 </button> 35 35 </div> 36 36 37 37 <!-- Scraping state --> 38 - <div id="state-scraping" class="state hidden"> 39 - <div class="icon spinner">โณ</div> 40 - <p class="message">Scanning...</p> 41 - <div class="progress"> 42 - <div class="progress-bar"> 43 - <div id="progress-fill" class="progress-fill"></div> 38 + <div id="state-scraping" class="w-full text-center hidden"> 39 + <div class="text-5xl mb-4 spinner">โณ</div> 40 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Scanning...</p> 41 + <div class="mt-5"> 42 + <div class="w-full h-2 bg-sky-50 dark:bg-slate-800 rounded overflow-hidden mb-3"> 43 + <div id="progress-fill" class="h-full bg-gradient-to-r from-orange-600 to-pink-600 w-0 transition-all duration-300 progress-fill"></div> 44 44 </div> 45 - <p class="progress-text"> 45 + <p class="text-base font-semibold text-slate-700 dark:text-cyan-50"> 46 46 Found <span id="count">0</span> users 47 47 </p> 48 - <p id="status-message" class="status-message"></p> 48 + <p id="status-message" class="text-[13px] text-slate-500 dark:text-slate-400 mt-2"></p> 49 49 </div> 50 50 </div> 51 51 52 52 <!-- Complete state --> 53 - <div id="state-complete" class="state hidden"> 54 - <div class="icon">๐ŸŽ‰</div> 55 - <p class="message">Scan complete!</p> 56 - <p class="count-display"> 57 - Found <strong id="final-count">0</strong> users 53 + <div id="state-complete" class="w-full text-center hidden"> 54 + <div class="text-5xl mb-4">๐ŸŽ‰</div> 55 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Scan complete!</p> 56 + <p class="text-sm text-slate-500 dark:text-slate-400 mt-2"> 57 + Found <strong id="final-count" class="text-orange-600 dark:text-orange-400 text-lg">0</strong> users 58 58 </p> 59 - <button id="btn-upload" class="btn-primary"> 59 + <button id="btn-upload" class="w-full bg-orange-600 hover:bg-orange-700 text-white font-semibold py-3 px-6 rounded-lg mt-4 transition-all duration-200 hover:-translate-y-0.5 hover:shadow-lg hover:shadow-orange-600/30 active:translate-y-0"> 60 60 Open in ATlast 61 61 </button> 62 62 </div> 63 63 64 64 <!-- Uploading state --> 65 - <div id="state-uploading" class="state hidden"> 66 - <div class="icon spinner">๐Ÿ“ค</div> 67 - <p class="message">Uploading to ATlast...</p> 65 + <div id="state-uploading" class="w-full text-center hidden"> 66 + <div class="text-5xl mb-4 spinner">๐Ÿ“ค</div> 67 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Uploading to ATlast...</p> 68 68 </div> 69 69 70 70 <!-- Error state --> 71 - <div id="state-error" class="state hidden"> 72 - <div class="icon">โš ๏ธ</div> 73 - <p class="message">Error</p> 74 - <p id="error-message" class="error-message"></p> 75 - <button id="btn-retry" class="btn-secondary"> 71 + <div id="state-error" class="w-full text-center hidden"> 72 + <div class="text-5xl mb-4">โš ๏ธ</div> 73 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Error</p> 74 + <p id="error-message" class="text-[13px] text-red-600 dark:text-red-400 mt-2 p-3 bg-red-50 dark:bg-red-950/50 rounded border-l-[3px] border-red-600"></p> 75 + <button id="btn-retry" class="w-full bg-white dark:bg-purple-950 text-purple-700 dark:text-cyan-400 border-2 border-purple-700 dark:border-cyan-400 font-semibold py-2.5 px-6 rounded-lg mt-4 transition-all duration-200 hover:bg-purple-50 dark:hover:bg-purple-900"> 76 76 Try Again 77 77 </button> 78 78 </div> 79 79 80 80 <!-- Server offline state --> 81 - <div id="state-offline" class="state hidden"> 82 - <div class="icon">๐Ÿ”Œ</div> 83 - <p class="message">ATlast server not running</p> 84 - <p class="error-message"> 81 + <div id="state-offline" class="w-full text-center hidden"> 82 + <div class="text-5xl mb-4">๐Ÿ”Œ</div> 83 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Server not available</p> 84 + <p id="dev-instructions" class="text-[13px] text-red-600 dark:text-red-400 mt-2 p-3 bg-red-50 dark:bg-red-950/50 rounded border-l-[3px] border-red-600"> 85 85 Start the dev server:<br /> 86 - <code>npx netlify-cli dev --filter @atlast/web</code> 86 + <code class="bg-black/10 dark:bg-white/10 px-2 py-1 rounded font-mono text-[11px] inline-block my-2">npx netlify-cli dev --filter @atlast/web</code> 87 87 </p> 88 - <p class="hint" id="server-url"></p> 89 - <button id="btn-check-server" class="btn-primary"> 88 + <p class="text-[13px] text-slate-500 dark:text-slate-400 mt-2" id="server-url"></p> 89 + <button id="btn-check-server" class="w-full bg-orange-600 hover:bg-orange-700 text-white font-semibold py-3 px-6 rounded-lg mt-4 transition-all duration-200 hover:-translate-y-0.5 hover:shadow-lg hover:shadow-orange-600/30 active:translate-y-0"> 90 90 Check Again 91 91 </button> 92 92 </div> 93 93 94 94 <!-- Not logged in state --> 95 - <div id="state-not-logged-in" class="state hidden"> 96 - <div class="icon">๐Ÿ”</div> 97 - <p class="message">Not logged in to ATlast</p> 98 - <p class="error-message"> 95 + <div id="state-not-logged-in" class="w-full text-center hidden"> 96 + <div class="text-5xl mb-4">๐Ÿ”</div> 97 + <p class="text-base font-semibold mb-3 text-slate-700 dark:text-cyan-50">Not logged in to ATlast</p> 98 + <p class="text-[13px] text-red-600 dark:text-red-400 mt-2 p-3 bg-red-50 dark:bg-red-950/50 rounded border-l-[3px] border-red-600"> 99 99 Please log in to ATlast first, then return here to scan. 100 100 </p> 101 - <button id="btn-open-atlast" class="btn-primary"> 101 + <button id="btn-open-atlast" class="w-full bg-orange-600 hover:bg-orange-700 text-white font-semibold py-3 px-6 rounded-lg mt-4 transition-all duration-200 hover:-translate-y-0.5 hover:shadow-lg hover:shadow-orange-600/30 active:translate-y-0"> 102 102 Open ATlast 103 103 </button> 104 - <button id="btn-retry-login" class="btn-secondary"> 104 + <button id="btn-retry-login" class="w-full bg-white dark:bg-purple-950 text-purple-700 dark:text-cyan-400 border-2 border-purple-700 dark:border-cyan-400 font-semibold py-2.5 px-6 rounded-lg mt-4 transition-all duration-200 hover:bg-purple-50 dark:hover:bg-purple-900"> 105 105 Check Again 106 106 </button> 107 107 </div> 108 108 </main> 109 109 110 - <footer> 111 - <a href="https://atlast.byarielm.fyi" target="_blank" 112 - >atlast.byarielm.fyi</a 113 - > 110 + <footer class="p-4 text-center border-t border-purple-200 dark:border-slate-800 bg-white dark:bg-slate-900"> 111 + <a href="https://atlast.byarielm.fyi" target="_blank" class="text-orange-600 dark:text-orange-400 no-underline text-[13px] font-medium hover:underline">atlast.byarielm.fyi</a> 114 112 </footer> 115 113 </div> 116 114
+22 -4
packages/extension/src/popup/popup.ts
··· 1 + import browser from 'webextension-polyfill'; 1 2 import { 2 3 MessageType, 3 4 sendToBackground, 4 5 sendToContent, 5 6 type ExtensionState 6 7 } from '../lib/messaging.js'; 8 + 9 + // Build mode injected at build time 10 + declare const __BUILD_MODE__: string; 7 11 8 12 /** 9 13 * DOM elements ··· 26 30 statusMessage: document.getElementById('status-message')!, 27 31 errorMessage: document.getElementById('error-message')!, 28 32 serverUrl: document.getElementById('server-url')!, 33 + devInstructions: document.getElementById('dev-instructions')!, 29 34 progressFill: document.getElementById('progress-fill')! as HTMLElement, 30 35 btnStart: document.getElementById('btn-start')! as HTMLButtonElement, 31 36 btnUpload: document.getElementById('btn-upload')! as HTMLButtonElement, ··· 163 168 // Open ATlast at results page with upload data 164 169 const { getApiUrl } = await import('../lib/api-client.js'); 165 170 const resultsUrl = `${getApiUrl()}${response.redirectUrl}`; 166 - chrome.tabs.create({ url: resultsUrl }); 171 + browser.tabs.create({ url: resultsUrl }); 167 172 168 173 } catch (error) { 169 174 console.error('[Popup] Error uploading:', error); ··· 214 219 if (!isOnline) { 215 220 console.log('[Popup] โŒ Server is offline'); 216 221 showState('offline'); 217 - elements.serverUrl.textContent = `Trying to reach: ${getApiUrl()}`; 222 + 223 + // Show appropriate message based on build mode 224 + const apiUrl = getApiUrl(); 225 + const isDev = __BUILD_MODE__ === 'development'; 226 + 227 + // Hide dev instructions in production 228 + if (!isDev) { 229 + elements.devInstructions.classList.add('hidden'); 230 + } 231 + 232 + elements.serverUrl.textContent = isDev 233 + ? `Development server at ${apiUrl}` 234 + : `Cannot reach ${apiUrl}`; 235 + 218 236 return false; 219 237 } 220 238 ··· 264 282 265 283 // Set up login buttons 266 284 elements.btnOpenAtlast.addEventListener('click', () => { 267 - chrome.tabs.create({ url: getApiUrl() }); 285 + browser.tabs.create({ url: getApiUrl() }); 268 286 }); 269 287 270 288 elements.btnRetryLogin.addEventListener('click', async () => { ··· 305 323 }); 306 324 307 325 // Listen for storage changes (when background updates state) 308 - chrome.storage.onChanged.addListener((changes, areaName) => { 326 + browser.storage.onChanged.addListener((changes, areaName) => { 309 327 if (areaName === 'local' && changes.extensionState) { 310 328 const newState = changes.extensionState.newValue; 311 329 console.log('[Popup] ๐Ÿ”„ Storage changed, new state:', newState);
+35
packages/extension/tailwind.config.js
··· 1 + /** @type {import('tailwindcss').Config} */ 2 + export default { 3 + // Use media query dark mode to automatically respect system preference 4 + darkMode: "media", 5 + 6 + // Scan popup HTML and TypeScript files 7 + content: [ 8 + "./src/popup/**/*.{html,ts}", 9 + "./src/content/**/*.ts", 10 + ], 11 + 12 + // Extend with same custom config as web app 13 + theme: { 14 + extend: { 15 + colors: { 16 + firefly: { 17 + glow: "#FCD34D", 18 + amber: "#F59E0B", 19 + orange: "#F97316", 20 + pink: "#EC4899", 21 + cyan: "#10D2F4", 22 + }, 23 + cyan: { 250: "#72EEFD" }, 24 + purple: { 750: "#6A1DD1" }, 25 + yellow: { 650: "#C56508" }, 26 + orange: { 650: "#DF3F00" }, 27 + pink: { 650: "#CD206A" }, 28 + }, 29 + backgroundImage: ({ theme }) => ({ 30 + "firefly-banner": `linear-gradient(to right, ${theme("colors.yellow.400")}, ${theme("colors.orange.500")}, ${theme("colors.pink.600")})`, 31 + "firefly-banner-dark": `linear-gradient(to right, ${theme("colors.yellow.600")}, ${theme("colors.orange.600")}, ${theme("colors.pink.700")})`, 32 + }), 33 + }, 34 + }, 35 + };
+4 -2
packages/functions/src/core/middleware/error.middleware.ts
··· 21 21 } 22 22 23 23 if (error instanceof ApiError) { 24 - return errorResponse(error.message, error.statusCode, error.details); 24 + return errorResponse(error.message, error.statusCode, error.details, event); 25 25 } 26 26 27 27 // Unknown errors ··· 29 29 "Internal server error", 30 30 500, 31 31 error instanceof Error ? error.message : "Unknown error", 32 + event, 32 33 ); 33 34 } 34 35 }; ··· 48 49 console.error("Authenticated handler error:", error); 49 50 50 51 if (error instanceof ApiError) { 51 - return errorResponse(error.message, error.statusCode, error.details); 52 + return errorResponse(error.message, error.statusCode, error.details, event); 52 53 } 53 54 54 55 return errorResponse( 55 56 "Internal server error", 56 57 500, 57 58 error instanceof Error ? error.message : "Unknown error", 59 + event, 58 60 ); 59 61 } 60 62 };
+21
packages/functions/src/health.ts
··· 1 + import { SimpleHandler } from "./core/types/api.types"; 2 + import { successResponse } from "./utils"; 3 + import { withErrorHandling } from "./core/middleware"; 4 + 5 + /** 6 + * Health check endpoint 7 + * Returns 200 OK with server status 8 + */ 9 + const healthHandler: SimpleHandler = async (event) => { 10 + return successResponse( 11 + { 12 + status: "ok", 13 + timestamp: new Date().toISOString(), 14 + }, 15 + 200, 16 + {}, 17 + event 18 + ); 19 + }; 20 + 21 + export const handler = withErrorHandling(healthHandler);
+2 -2
packages/functions/src/session.ts
··· 30 30 return successResponse(cached, 200, { 31 31 "Cache-Control": "private, max-age=300", 32 32 "X-Cache-Status": "HIT", 33 - }); 33 + }, event); 34 34 } 35 35 36 36 const { agent } = await SessionService.getAgentForSession(sessionId, event); ··· 50 50 return successResponse(profileData, 200, { 51 51 "Cache-Control": "private, max-age=300", 52 52 "X-Cache-Status": "MISS", 53 - }); 53 + }, event); 54 54 }; 55 55 56 56 export const handler = withErrorHandling(sessionHandler);
+42 -3
packages/functions/src/utils/response.utils.ts
··· 1 - import { HandlerResponse } from "@netlify/functions"; 1 + import { HandlerResponse, HandlerEvent } from "@netlify/functions"; 2 2 import { ApiResponse } from "../core/types"; 3 3 4 + /** 5 + * Get CORS headers based on request origin 6 + * Supports credentialed requests from extensions and localhost 7 + */ 8 + function getCorsHeaders(event?: HandlerEvent): Record<string, string> { 9 + const origin = event?.headers?.origin || event?.headers?.Origin; 10 + 11 + // Allow all origins for non-credentialed requests (backward compatibility) 12 + if (!origin) { 13 + return { 14 + "Access-Control-Allow-Origin": "*", 15 + }; 16 + } 17 + 18 + // Check if origin is allowed for credentialed requests 19 + const allowedOrigins = [ 20 + 'http://localhost:8888', 21 + 'http://127.0.0.1:8888', 22 + 'https://atlast.byarielm.fyi', 23 + ]; 24 + 25 + const isExtension = origin.startsWith('chrome-extension://') || origin.startsWith('moz-extension://'); 26 + const isAllowedOrigin = allowedOrigins.includes(origin); 27 + 28 + if (isExtension || isAllowedOrigin) { 29 + return { 30 + "Access-Control-Allow-Origin": origin, 31 + "Access-Control-Allow-Credentials": "true", 32 + }; 33 + } 34 + 35 + // Default to wildcard for unknown origins 36 + return { 37 + "Access-Control-Allow-Origin": "*", 38 + }; 39 + } 40 + 4 41 export function successResponse<T>( 5 42 data: T, 6 43 statusCode: number = 200, 7 44 additionalHeaders: Record<string, string> = {}, 45 + event?: HandlerEvent, 8 46 ): HandlerResponse { 9 47 const response: ApiResponse<T> = { 10 48 success: true, ··· 15 53 statusCode, 16 54 headers: { 17 55 "Content-Type": "application/json", 18 - "Access-Control-Allow-Origin": "*", 56 + ...getCorsHeaders(event), 19 57 ...additionalHeaders, 20 58 }, 21 59 body: JSON.stringify(response), ··· 26 64 error: string, 27 65 statusCode: number = 500, 28 66 details?: string, 67 + event?: HandlerEvent, 29 68 ): HandlerResponse { 30 69 const response: ApiResponse = { 31 70 success: false, ··· 37 76 statusCode, 38 77 headers: { 39 78 "Content-Type": "application/json", 40 - "Access-Control-Allow-Origin": "*", 79 + ...getCorsHeaders(event), 41 80 }, 42 81 body: JSON.stringify(response), 43 82 };
+579
pnpm-lock.yaml
··· 114 114 '@atlast/shared': 115 115 specifier: workspace:* 116 116 version: link:../shared 117 + webextension-polyfill: 118 + specifier: ^0.12.0 119 + version: 0.12.0 117 120 devDependencies: 118 121 '@types/chrome': 119 122 specifier: ^0.0.256 120 123 version: 0.0.256 124 + '@types/webextension-polyfill': 125 + specifier: ^0.12.4 126 + version: 0.12.4 127 + autoprefixer: 128 + specifier: ^10.4.23 129 + version: 10.4.23(postcss@8.5.6) 130 + cssnano: 131 + specifier: ^7.1.2 132 + version: 7.1.2(postcss@8.5.6) 121 133 esbuild: 122 134 specifier: ^0.19.11 123 135 version: 0.19.12 136 + postcss: 137 + specifier: ^8.5.6 138 + version: 8.5.6 139 + tailwindcss: 140 + specifier: ^3.4.19 141 + version: 3.4.19 124 142 typescript: 125 143 specifier: ^5.3.3 126 144 version: 5.9.3 ··· 1233 1251 '@types/triple-beam@1.3.5': 1234 1252 resolution: {integrity: sha512-6WaYesThRMCl19iryMYP7/x2OVgCtbIVflDGFpWnb9irXI3UjYE4AzmYuiUKY1AJstGijoY+MgUszMgRxIYTYw==} 1235 1253 1254 + '@types/webextension-polyfill@0.12.4': 1255 + resolution: {integrity: sha512-wK8YdSI0pDiaehSLDIvtvonYmLwUUivg4Z6JCJO8rkyssMAG82cFJgwPK/V7NO61mJBLg/tXeoXQL8AFzpXZmQ==} 1256 + 1236 1257 '@types/yauzl@2.10.3': 1237 1258 resolution: {integrity: sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q==} 1238 1259 ··· 1431 1452 bindings@1.5.0: 1432 1453 resolution: {integrity: sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==} 1433 1454 1455 + boolbase@1.0.0: 1456 + resolution: {integrity: sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww==} 1457 + 1434 1458 brace-expansion@2.0.2: 1435 1459 resolution: {integrity: sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==} 1436 1460 ··· 1471 1495 resolution: {integrity: sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==} 1472 1496 engines: {node: '>=10'} 1473 1497 1498 + caniuse-api@3.0.0: 1499 + resolution: {integrity: sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==} 1500 + 1474 1501 caniuse-lite@1.0.30001761: 1475 1502 resolution: {integrity: sha512-JF9ptu1vP2coz98+5051jZ4PwQgd2ni8A+gYSN7EA7dPKIMf0pDlSUxhdmVOaV3/fYK5uWBkgSXJaRLr4+3A6g==} 1476 1503 ··· 1513 1540 resolution: {integrity: sha512-ezmVcLR3xAVp8kYOm4GS45ZLLgIE6SPAFoduLr6hTDajwb3KZ2F46gulK3XpcwRFb5KKGCSezCBAY4Dw4HsyXA==} 1514 1541 engines: {node: '>=18'} 1515 1542 1543 + colord@2.9.3: 1544 + resolution: {integrity: sha512-jeC1axXpnb0/2nn/Y1LPuLdgXBLH7aDcHu4KEKfqw3CUhX7ZpfBSlPKyqXE6btIgEzfWtrX3/tyBCaCvXvMkOw==} 1545 + 1516 1546 commander@10.0.1: 1517 1547 resolution: {integrity: sha512-y4Mg2tXshplEbSGzx7amzPwKKOCGuoSRP/CjEdwwk0FOGlUbq6lKuoyDZTNZkmxHdJtp54hdfY/JUrdL7Xfdug==} 1518 1548 engines: {node: '>=14'} 1549 + 1550 + commander@11.1.0: 1551 + resolution: {integrity: sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ==} 1552 + engines: {node: '>=16'} 1519 1553 1520 1554 commander@12.1.0: 1521 1555 resolution: {integrity: sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==} ··· 1586 1620 resolution: {integrity: sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==} 1587 1621 engines: {node: '>= 8'} 1588 1622 1623 + css-declaration-sorter@7.3.0: 1624 + resolution: {integrity: sha512-LQF6N/3vkAMYF4xoHLJfG718HRJh34Z8BnNhd6bosOMIVjMlhuZK5++oZa3uYAgrI5+7x2o27gUqTR2U/KjUOQ==} 1625 + engines: {node: ^14 || ^16 || >=18} 1626 + peerDependencies: 1627 + postcss: ^8.0.9 1628 + 1629 + css-select@5.2.2: 1630 + resolution: {integrity: sha512-TizTzUddG/xYLA3NXodFM0fSbNizXjOKhqiQQwvhlspadZokn1KDy0NZFS0wuEubIYAV5/c1/lAr0TaaFXEXzw==} 1631 + 1632 + css-tree@2.2.1: 1633 + resolution: {integrity: sha512-OA0mILzGc1kCOCSJerOeqDxDQ4HOh+G8NbOJFOTgOCzpw7fCBubk0fEyxp8AgOL/jvLgYA/uV0cMbe43ElF1JA==} 1634 + engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0, npm: '>=7.0.0'} 1635 + 1636 + css-tree@3.1.0: 1637 + resolution: {integrity: sha512-0eW44TGN5SQXU1mWSkKwFstI/22X2bG1nYzZTYMAWjylYURhse752YgbE4Cx46AC+bAvI+/dYTPRk1LqSUnu6w==} 1638 + engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0} 1639 + 1640 + css-what@6.2.2: 1641 + resolution: {integrity: sha512-u/O3vwbptzhMs3L1fQE82ZSLHQQfto5gyZzwteVIEyeaY5Fc7R4dapF/BvRoSYFeqfBk4m0V1Vafq5Pjv25wvA==} 1642 + engines: {node: '>= 6'} 1643 + 1589 1644 cssesc@3.0.0: 1590 1645 resolution: {integrity: sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==} 1591 1646 engines: {node: '>=4'} 1592 1647 hasBin: true 1593 1648 1649 + cssnano-preset-default@7.0.10: 1650 + resolution: {integrity: sha512-6ZBjW0Lf1K1Z+0OKUAUpEN62tSXmYChXWi2NAA0afxEVsj9a+MbcB1l5qel6BHJHmULai2fCGRthCeKSFbScpA==} 1651 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 1652 + peerDependencies: 1653 + postcss: ^8.4.32 1654 + 1655 + cssnano-utils@5.0.1: 1656 + resolution: {integrity: sha512-ZIP71eQgG9JwjVZsTPSqhc6GHgEr53uJ7tK5///VfyWj6Xp2DBmixWHqJgPno+PqATzn48pL42ww9x5SSGmhZg==} 1657 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 1658 + peerDependencies: 1659 + postcss: ^8.4.32 1660 + 1661 + cssnano@7.1.2: 1662 + resolution: {integrity: sha512-HYOPBsNvoiFeR1eghKD5C3ASm64v9YVyJB4Ivnl2gqKoQYvjjN/G0rztvKQq8OxocUtC6sjqY8jwYngIB4AByA==} 1663 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 1664 + peerDependencies: 1665 + postcss: ^8.4.32 1666 + 1667 + csso@5.0.5: 1668 + resolution: {integrity: sha512-0LrrStPOdJj+SPCCrGhzryycLjwcgUSHBtxNA8aIDxf0GLsRh1cKYhB00Gd1lDOS4yGH69+SNn13+TWbVHETFQ==} 1669 + engines: {node: ^10 || ^12.20.0 || ^14.13.0 || >=15.0.0, npm: '>=7.0.0'} 1670 + 1594 1671 csstype@3.2.3: 1595 1672 resolution: {integrity: sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==} 1596 1673 ··· 1668 1745 1669 1746 dlv@1.1.3: 1670 1747 resolution: {integrity: sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==} 1748 + 1749 + dom-serializer@2.0.0: 1750 + resolution: {integrity: sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg==} 1751 + 1752 + domelementtype@2.3.0: 1753 + resolution: {integrity: sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw==} 1754 + 1755 + domhandler@5.0.3: 1756 + resolution: {integrity: sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w==} 1757 + engines: {node: '>= 4'} 1758 + 1759 + domutils@3.2.2: 1760 + resolution: {integrity: sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw==} 1671 1761 1672 1762 dot-case@3.0.4: 1673 1763 resolution: {integrity: sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==} ··· 2121 2211 resolution: {integrity: sha512-gvVijfZvn7R+2qyPX8mAuKcFGDf6Nc61GdvGafQsHL0sBIxfKzA+usWn4GFC/bk+QdwPUD4kWFJLhElipq+0VA==} 2122 2212 engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} 2123 2213 2214 + lodash.memoize@4.1.2: 2215 + resolution: {integrity: sha512-t7j+NzmgnQzTAYXcsHYLgimltOV1MXHtlOWf6GjL9Kj8GK5FInw5JotxvbOs+IvV1/Dzo04/fCGfLVs7aXb4Ag==} 2216 + 2217 + lodash.uniq@4.5.0: 2218 + resolution: {integrity: sha512-xfBaXQd9ryd9dlSDvnvI0lvxfLJlYAZzXomUYzLKtUeOQvOP5piqAWuGtrhWeqaXK9hhoM/iyJc5AV+XfsX3HQ==} 2219 + 2124 2220 lodash@4.17.21: 2125 2221 resolution: {integrity: sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==} 2126 2222 ··· 2156 2252 make-dir@3.1.0: 2157 2253 resolution: {integrity: sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==} 2158 2254 engines: {node: '>=8'} 2255 + 2256 + mdn-data@2.0.28: 2257 + resolution: {integrity: sha512-aylIc7Z9y4yzHYAJNuESG3hfhC+0Ibp/MAMiaOZgNv4pmEdFyfZhhhny4MNiAfWdBQ1RQ2mfDWmM1x8SvGyp8g==} 2258 + 2259 + mdn-data@2.12.2: 2260 + resolution: {integrity: sha512-IEn+pegP1aManZuckezWCO+XZQDplx1366JoVhTpMpBB1sPey/SbveZQUosKiKiGYjg1wH4pMlNgXbCiYgihQA==} 2159 2261 2160 2262 merge-options@3.0.4: 2161 2263 resolution: {integrity: sha512-2Sug1+knBjkaMsMgf1ctR1Ujx+Ayku4EdJN4Z+C2+JzoeF7A3OZ9KM2GY0CpQS51NR61LTurMJrRKPhSs3ZRTQ==} ··· 2258 2360 resolution: {integrity: sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ==} 2259 2361 engines: {node: ^12.20.0 || ^14.13.1 || >=16.0.0} 2260 2362 2363 + nth-check@2.1.1: 2364 + resolution: {integrity: sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w==} 2365 + 2261 2366 object-assign@4.1.1: 2262 2367 resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} 2263 2368 engines: {node: '>=0.10.0'} ··· 2394 2499 resolution: {integrity: sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==} 2395 2500 engines: {node: '>=8'} 2396 2501 2502 + postcss-calc@10.1.1: 2503 + resolution: {integrity: sha512-NYEsLHh8DgG/PRH2+G9BTuUdtf9ViS+vdoQ0YA5OQdGsfN4ztiwtDWNtBl9EKeqNMFnIu8IKZ0cLxEQ5r5KVMw==} 2504 + engines: {node: ^18.12 || ^20.9 || >=22.0} 2505 + peerDependencies: 2506 + postcss: ^8.4.38 2507 + 2508 + postcss-colormin@7.0.5: 2509 + resolution: {integrity: sha512-ekIBP/nwzRWhEMmIxHHbXHcMdzd1HIUzBECaj5KEdLz9DVP2HzT065sEhvOx1dkLjYW7jyD0CngThx6bpFi2fA==} 2510 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2511 + peerDependencies: 2512 + postcss: ^8.4.32 2513 + 2514 + postcss-convert-values@7.0.8: 2515 + resolution: {integrity: sha512-+XNKuPfkHTCEo499VzLMYn94TiL3r9YqRE3Ty+jP7UX4qjewUONey1t7CG21lrlTLN07GtGM8MqFVp86D4uKJg==} 2516 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2517 + peerDependencies: 2518 + postcss: ^8.4.32 2519 + 2520 + postcss-discard-comments@7.0.5: 2521 + resolution: {integrity: sha512-IR2Eja8WfYgN5n32vEGSctVQ1+JARfu4UH8M7bgGh1bC+xI/obsPJXaBpQF7MAByvgwZinhpHpdrmXtvVVlKcQ==} 2522 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2523 + peerDependencies: 2524 + postcss: ^8.4.32 2525 + 2526 + postcss-discard-duplicates@7.0.2: 2527 + resolution: {integrity: sha512-eTonaQvPZ/3i1ASDHOKkYwAybiM45zFIc7KXils4mQmHLqIswXD9XNOKEVxtTFnsmwYzF66u4LMgSr0abDlh5w==} 2528 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2529 + peerDependencies: 2530 + postcss: ^8.4.32 2531 + 2532 + postcss-discard-empty@7.0.1: 2533 + resolution: {integrity: sha512-cFrJKZvcg/uxB6Ijr4l6qmn3pXQBna9zyrPC+sK0zjbkDUZew+6xDltSF7OeB7rAtzaaMVYSdbod+sZOCWnMOg==} 2534 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2535 + peerDependencies: 2536 + postcss: ^8.4.32 2537 + 2538 + postcss-discard-overridden@7.0.1: 2539 + resolution: {integrity: sha512-7c3MMjjSZ/qYrx3uc1940GSOzN1Iqjtlqe8uoSg+qdVPYyRb0TILSqqmtlSFuE4mTDECwsm397Ya7iXGzfF7lg==} 2540 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2541 + peerDependencies: 2542 + postcss: ^8.4.32 2543 + 2397 2544 postcss-import@15.1.0: 2398 2545 resolution: {integrity: sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==} 2399 2546 engines: {node: '>=14.0.0'} ··· 2424 2571 yaml: 2425 2572 optional: true 2426 2573 2574 + postcss-merge-longhand@7.0.5: 2575 + resolution: {integrity: sha512-Kpu5v4Ys6QI59FxmxtNB/iHUVDn9Y9sYw66D6+SZoIk4QTz1prC4aYkhIESu+ieG1iylod1f8MILMs1Em3mmIw==} 2576 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2577 + peerDependencies: 2578 + postcss: ^8.4.32 2579 + 2580 + postcss-merge-rules@7.0.7: 2581 + resolution: {integrity: sha512-njWJrd/Ms6XViwowaaCc+/vqhPG3SmXn725AGrnl+BgTuRPEacjiLEaGq16J6XirMJbtKkTwnt67SS+e2WGoew==} 2582 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2583 + peerDependencies: 2584 + postcss: ^8.4.32 2585 + 2586 + postcss-minify-font-values@7.0.1: 2587 + resolution: {integrity: sha512-2m1uiuJeTplll+tq4ENOQSzB8LRnSUChBv7oSyFLsJRtUgAAJGP6LLz0/8lkinTgxrmJSPOEhgY1bMXOQ4ZXhQ==} 2588 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2589 + peerDependencies: 2590 + postcss: ^8.4.32 2591 + 2592 + postcss-minify-gradients@7.0.1: 2593 + resolution: {integrity: sha512-X9JjaysZJwlqNkJbUDgOclyG3jZEpAMOfof6PUZjPnPrePnPG62pS17CjdM32uT1Uq1jFvNSff9l7kNbmMSL2A==} 2594 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2595 + peerDependencies: 2596 + postcss: ^8.4.32 2597 + 2598 + postcss-minify-params@7.0.5: 2599 + resolution: {integrity: sha512-FGK9ky02h6Ighn3UihsyeAH5XmLEE2MSGH5Tc4tXMFtEDx7B+zTG6hD/+/cT+fbF7PbYojsmmWjyTwFwW1JKQQ==} 2600 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2601 + peerDependencies: 2602 + postcss: ^8.4.32 2603 + 2604 + postcss-minify-selectors@7.0.5: 2605 + resolution: {integrity: sha512-x2/IvofHcdIrAm9Q+p06ZD1h6FPcQ32WtCRVodJLDR+WMn8EVHI1kvLxZuGKz/9EY5nAmI6lIQIrpo4tBy5+ug==} 2606 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2607 + peerDependencies: 2608 + postcss: ^8.4.32 2609 + 2427 2610 postcss-nested@6.2.0: 2428 2611 resolution: {integrity: sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==} 2429 2612 engines: {node: '>=12.0'} 2430 2613 peerDependencies: 2431 2614 postcss: ^8.2.14 2432 2615 2616 + postcss-normalize-charset@7.0.1: 2617 + resolution: {integrity: sha512-sn413ofhSQHlZFae//m9FTOfkmiZ+YQXsbosqOWRiVQncU2BA3daX3n0VF3cG6rGLSFVc5Di/yns0dFfh8NFgQ==} 2618 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2619 + peerDependencies: 2620 + postcss: ^8.4.32 2621 + 2622 + postcss-normalize-display-values@7.0.1: 2623 + resolution: {integrity: sha512-E5nnB26XjSYz/mGITm6JgiDpAbVuAkzXwLzRZtts19jHDUBFxZ0BkXAehy0uimrOjYJbocby4FVswA/5noOxrQ==} 2624 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2625 + peerDependencies: 2626 + postcss: ^8.4.32 2627 + 2628 + postcss-normalize-positions@7.0.1: 2629 + resolution: {integrity: sha512-pB/SzrIP2l50ZIYu+yQZyMNmnAcwyYb9R1fVWPRxm4zcUFCY2ign7rcntGFuMXDdd9L2pPNUgoODDk91PzRZuQ==} 2630 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2631 + peerDependencies: 2632 + postcss: ^8.4.32 2633 + 2634 + postcss-normalize-repeat-style@7.0.1: 2635 + resolution: {integrity: sha512-NsSQJ8zj8TIDiF0ig44Byo3Jk9e4gNt9x2VIlJudnQQ5DhWAHJPF4Tr1ITwyHio2BUi/I6Iv0HRO7beHYOloYQ==} 2636 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2637 + peerDependencies: 2638 + postcss: ^8.4.32 2639 + 2640 + postcss-normalize-string@7.0.1: 2641 + resolution: {integrity: sha512-QByrI7hAhsoze992kpbMlJSbZ8FuCEc1OT9EFbZ6HldXNpsdpZr+YXC5di3UEv0+jeZlHbZcoCADgb7a+lPmmQ==} 2642 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2643 + peerDependencies: 2644 + postcss: ^8.4.32 2645 + 2646 + postcss-normalize-timing-functions@7.0.1: 2647 + resolution: {integrity: sha512-bHifyuuSNdKKsnNJ0s8fmfLMlvsQwYVxIoUBnowIVl2ZAdrkYQNGVB4RxjfpvkMjipqvbz0u7feBZybkl/6NJg==} 2648 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2649 + peerDependencies: 2650 + postcss: ^8.4.32 2651 + 2652 + postcss-normalize-unicode@7.0.5: 2653 + resolution: {integrity: sha512-X6BBwiRxVaFHrb2WyBMddIeB5HBjJcAaUHyhLrM2FsxSq5TFqcHSsK7Zu1otag+o0ZphQGJewGH1tAyrD0zX1Q==} 2654 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2655 + peerDependencies: 2656 + postcss: ^8.4.32 2657 + 2658 + postcss-normalize-url@7.0.1: 2659 + resolution: {integrity: sha512-sUcD2cWtyK1AOL/82Fwy1aIVm/wwj5SdZkgZ3QiUzSzQQofrbq15jWJ3BA7Z+yVRwamCjJgZJN0I9IS7c6tgeQ==} 2660 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2661 + peerDependencies: 2662 + postcss: ^8.4.32 2663 + 2664 + postcss-normalize-whitespace@7.0.1: 2665 + resolution: {integrity: sha512-vsbgFHMFQrJBJKrUFJNZ2pgBeBkC2IvvoHjz1to0/0Xk7sII24T0qFOiJzG6Fu3zJoq/0yI4rKWi7WhApW+EFA==} 2666 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2667 + peerDependencies: 2668 + postcss: ^8.4.32 2669 + 2670 + postcss-ordered-values@7.0.2: 2671 + resolution: {integrity: sha512-AMJjt1ECBffF7CEON/Y0rekRLS6KsePU6PRP08UqYW4UGFRnTXNrByUzYK1h8AC7UWTZdQ9O3Oq9kFIhm0SFEw==} 2672 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2673 + peerDependencies: 2674 + postcss: ^8.4.32 2675 + 2676 + postcss-reduce-initial@7.0.5: 2677 + resolution: {integrity: sha512-RHagHLidG8hTZcnr4FpyMB2jtgd/OcyAazjMhoy5qmWJOx1uxKh4ntk0Pb46ajKM0rkf32lRH4C8c9qQiPR6IA==} 2678 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2679 + peerDependencies: 2680 + postcss: ^8.4.32 2681 + 2682 + postcss-reduce-transforms@7.0.1: 2683 + resolution: {integrity: sha512-MhyEbfrm+Mlp/36hvZ9mT9DaO7dbncU0CvWI8V93LRkY6IYlu38OPg3FObnuKTUxJ4qA8HpurdQOo5CyqqO76g==} 2684 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2685 + peerDependencies: 2686 + postcss: ^8.4.32 2687 + 2433 2688 postcss-selector-parser@6.1.2: 2434 2689 resolution: {integrity: sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==} 2435 2690 engines: {node: '>=4'} 2691 + 2692 + postcss-selector-parser@7.1.1: 2693 + resolution: {integrity: sha512-orRsuYpJVw8LdAwqqLykBj9ecS5/cRHlI5+nvTo8LcCKmzDmqVORXtOIYEEQuL9D4BxtA1lm5isAqzQZCoQ6Eg==} 2694 + engines: {node: '>=4'} 2695 + 2696 + postcss-svgo@7.1.0: 2697 + resolution: {integrity: sha512-KnAlfmhtoLz6IuU3Sij2ycusNs4jPW+QoFE5kuuUOK8awR6tMxZQrs5Ey3BUz7nFCzT3eqyFgqkyrHiaU2xx3w==} 2698 + engines: {node: ^18.12.0 || ^20.9.0 || >= 18} 2699 + peerDependencies: 2700 + postcss: ^8.4.32 2701 + 2702 + postcss-unique-selectors@7.0.4: 2703 + resolution: {integrity: sha512-pmlZjsmEAG7cHd7uK3ZiNSW6otSZ13RHuZ/4cDN/bVglS5EpF2r2oxY99SuOHa8m7AWoBCelTS3JPpzsIs8skQ==} 2704 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2705 + peerDependencies: 2706 + postcss: ^8.4.32 2436 2707 2437 2708 postcss-value-parser@4.2.0: 2438 2709 resolution: {integrity: sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==} ··· 2596 2867 resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} 2597 2868 engines: {node: '>=10'} 2598 2869 2870 + sax@1.4.3: 2871 + resolution: {integrity: sha512-yqYn1JhPczigF94DMS+shiDMjDowYO6y9+wB/4WgO0Y19jWYk0lQ4tuG5KI7kj4FTp1wxPj5IFfcrz/s1c3jjQ==} 2872 + 2599 2873 scheduler@0.23.2: 2600 2874 resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==} 2601 2875 ··· 2692 2966 resolution: {integrity: sha512-k55yxKHwaXnpYGsOzg4Vl8+tDrWylxDEpknGjhTiZB8dFRU5rTo9CAzeycivxV3s+zlTKwrs6WxMxR95n26kwg==} 2693 2967 engines: {node: '>=0.10.0'} 2694 2968 2969 + stylehacks@7.0.7: 2970 + resolution: {integrity: sha512-bJkD0JkEtbRrMFtwgpJyBbFIwfDDONQ1Ov3sDLZQP8HuJ73kBOyx66H4bOcAbVWmnfLdvQ0AJwXxOMkpujcO6g==} 2971 + engines: {node: ^18.12.0 || ^20.9.0 || >=22.0} 2972 + peerDependencies: 2973 + postcss: ^8.4.32 2974 + 2695 2975 sucrase@3.35.1: 2696 2976 resolution: {integrity: sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==} 2697 2977 engines: {node: '>=16 || 14 >=14.17'} ··· 2703 2983 2704 2984 svg-parser@2.0.4: 2705 2985 resolution: {integrity: sha512-e4hG1hRwoOdRb37cIMSgzNsxyzKfayW6VOflrwvR+/bzrkyxY/31WkbgnQpgtrNp1SdpJvpUAGTa/ZoiPNDuRQ==} 2986 + 2987 + svgo@4.0.0: 2988 + resolution: {integrity: sha512-VvrHQ+9uniE+Mvx3+C9IEe/lWasXCU0nXMY2kZeLrHNICuRiC8uMPyM14UEaMOFA5mhyQqEkB02VoQ16n3DLaw==} 2989 + engines: {node: '>=16'} 2990 + hasBin: true 2706 2991 2707 2992 tailwindcss@3.4.19: 2708 2993 resolution: {integrity: sha512-3ofp+LL8E+pK/JuPLPggVAIaEuhvIz4qNcf3nA1Xn2o/7fb7s/TYpHhwGDv1ZU3PkBluUVaF8PyCHcm48cKLWQ==} ··· 2868 3153 optional: true 2869 3154 terser: 2870 3155 optional: true 3156 + 3157 + webextension-polyfill@0.12.0: 3158 + resolution: {integrity: sha512-97TBmpoWJEE+3nFBQ4VocyCdLKfw54rFaJ6EVQYLBCXqCIpLSZkwGgASpv4oPt9gdKCJ80RJlcmNzNn008Ag6Q==} 2871 3159 2872 3160 webidl-conversions@3.0.1: 2873 3161 resolution: {integrity: sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==} ··· 3895 4183 csstype: 3.2.3 3896 4184 3897 4185 '@types/triple-beam@1.3.5': {} 4186 + 4187 + '@types/webextension-polyfill@0.12.4': {} 3898 4188 3899 4189 '@types/yauzl@2.10.3': 3900 4190 dependencies: ··· 4125 4415 dependencies: 4126 4416 file-uri-to-path: 1.0.0 4127 4417 4418 + boolbase@1.0.0: {} 4419 + 4128 4420 brace-expansion@2.0.2: 4129 4421 dependencies: 4130 4422 balanced-match: 1.0.2 ··· 4159 4451 camelcase-css@2.0.1: {} 4160 4452 4161 4453 camelcase@6.3.0: {} 4454 + 4455 + caniuse-api@3.0.0: 4456 + dependencies: 4457 + browserslist: 4.28.1 4458 + caniuse-lite: 1.0.30001761 4459 + lodash.memoize: 4.1.2 4460 + lodash.uniq: 4.5.0 4162 4461 4163 4462 caniuse-lite@1.0.30001761: {} 4164 4463 ··· 4207 4506 color-convert: 3.1.3 4208 4507 color-string: 2.1.4 4209 4508 4509 + colord@2.9.3: {} 4510 + 4210 4511 commander@10.0.1: {} 4512 + 4513 + commander@11.1.0: {} 4211 4514 4212 4515 commander@12.1.0: {} 4213 4516 ··· 4267 4570 path-key: 3.1.1 4268 4571 shebang-command: 2.0.0 4269 4572 which: 2.0.2 4573 + 4574 + css-declaration-sorter@7.3.0(postcss@8.5.6): 4575 + dependencies: 4576 + postcss: 8.5.6 4577 + 4578 + css-select@5.2.2: 4579 + dependencies: 4580 + boolbase: 1.0.0 4581 + css-what: 6.2.2 4582 + domhandler: 5.0.3 4583 + domutils: 3.2.2 4584 + nth-check: 2.1.1 4585 + 4586 + css-tree@2.2.1: 4587 + dependencies: 4588 + mdn-data: 2.0.28 4589 + source-map-js: 1.2.1 4590 + 4591 + css-tree@3.1.0: 4592 + dependencies: 4593 + mdn-data: 2.12.2 4594 + source-map-js: 1.2.1 4595 + 4596 + css-what@6.2.2: {} 4270 4597 4271 4598 cssesc@3.0.0: {} 4272 4599 4600 + cssnano-preset-default@7.0.10(postcss@8.5.6): 4601 + dependencies: 4602 + browserslist: 4.28.1 4603 + css-declaration-sorter: 7.3.0(postcss@8.5.6) 4604 + cssnano-utils: 5.0.1(postcss@8.5.6) 4605 + postcss: 8.5.6 4606 + postcss-calc: 10.1.1(postcss@8.5.6) 4607 + postcss-colormin: 7.0.5(postcss@8.5.6) 4608 + postcss-convert-values: 7.0.8(postcss@8.5.6) 4609 + postcss-discard-comments: 7.0.5(postcss@8.5.6) 4610 + postcss-discard-duplicates: 7.0.2(postcss@8.5.6) 4611 + postcss-discard-empty: 7.0.1(postcss@8.5.6) 4612 + postcss-discard-overridden: 7.0.1(postcss@8.5.6) 4613 + postcss-merge-longhand: 7.0.5(postcss@8.5.6) 4614 + postcss-merge-rules: 7.0.7(postcss@8.5.6) 4615 + postcss-minify-font-values: 7.0.1(postcss@8.5.6) 4616 + postcss-minify-gradients: 7.0.1(postcss@8.5.6) 4617 + postcss-minify-params: 7.0.5(postcss@8.5.6) 4618 + postcss-minify-selectors: 7.0.5(postcss@8.5.6) 4619 + postcss-normalize-charset: 7.0.1(postcss@8.5.6) 4620 + postcss-normalize-display-values: 7.0.1(postcss@8.5.6) 4621 + postcss-normalize-positions: 7.0.1(postcss@8.5.6) 4622 + postcss-normalize-repeat-style: 7.0.1(postcss@8.5.6) 4623 + postcss-normalize-string: 7.0.1(postcss@8.5.6) 4624 + postcss-normalize-timing-functions: 7.0.1(postcss@8.5.6) 4625 + postcss-normalize-unicode: 7.0.5(postcss@8.5.6) 4626 + postcss-normalize-url: 7.0.1(postcss@8.5.6) 4627 + postcss-normalize-whitespace: 7.0.1(postcss@8.5.6) 4628 + postcss-ordered-values: 7.0.2(postcss@8.5.6) 4629 + postcss-reduce-initial: 7.0.5(postcss@8.5.6) 4630 + postcss-reduce-transforms: 7.0.1(postcss@8.5.6) 4631 + postcss-svgo: 7.1.0(postcss@8.5.6) 4632 + postcss-unique-selectors: 7.0.4(postcss@8.5.6) 4633 + 4634 + cssnano-utils@5.0.1(postcss@8.5.6): 4635 + dependencies: 4636 + postcss: 8.5.6 4637 + 4638 + cssnano@7.1.2(postcss@8.5.6): 4639 + dependencies: 4640 + cssnano-preset-default: 7.0.10(postcss@8.5.6) 4641 + lilconfig: 3.1.3 4642 + postcss: 8.5.6 4643 + 4644 + csso@5.0.5: 4645 + dependencies: 4646 + css-tree: 2.2.1 4647 + 4273 4648 csstype@3.2.3: {} 4274 4649 4275 4650 date-fns@4.1.0: {} ··· 4349 4724 path-type: 4.0.0 4350 4725 4351 4726 dlv@1.1.3: {} 4727 + 4728 + dom-serializer@2.0.0: 4729 + dependencies: 4730 + domelementtype: 2.3.0 4731 + domhandler: 5.0.3 4732 + entities: 4.5.0 4733 + 4734 + domelementtype@2.3.0: {} 4735 + 4736 + domhandler@5.0.3: 4737 + dependencies: 4738 + domelementtype: 2.3.0 4739 + 4740 + domutils@3.2.2: 4741 + dependencies: 4742 + dom-serializer: 2.0.0 4743 + domelementtype: 2.3.0 4744 + domhandler: 5.0.3 4352 4745 4353 4746 dot-case@3.0.4: 4354 4747 dependencies: ··· 4817 5210 dependencies: 4818 5211 p-locate: 6.0.0 4819 5212 5213 + lodash.memoize@4.1.2: {} 5214 + 5215 + lodash.uniq@4.5.0: {} 5216 + 4820 5217 lodash@4.17.21: {} 4821 5218 4822 5219 logform@2.7.0: ··· 4855 5252 make-dir@3.1.0: 4856 5253 dependencies: 4857 5254 semver: 6.3.1 5255 + 5256 + mdn-data@2.0.28: {} 5257 + 5258 + mdn-data@2.12.2: {} 4858 5259 4859 5260 merge-options@3.0.4: 4860 5261 dependencies: ··· 4941 5342 dependencies: 4942 5343 path-key: 4.0.0 4943 5344 5345 + nth-check@2.1.1: 5346 + dependencies: 5347 + boolbase: 1.0.0 5348 + 4944 5349 object-assign@4.1.1: {} 4945 5350 4946 5351 object-hash@3.0.0: {} ··· 5051 5456 dependencies: 5052 5457 find-up: 4.1.0 5053 5458 5459 + postcss-calc@10.1.1(postcss@8.5.6): 5460 + dependencies: 5461 + postcss: 8.5.6 5462 + postcss-selector-parser: 7.1.1 5463 + postcss-value-parser: 4.2.0 5464 + 5465 + postcss-colormin@7.0.5(postcss@8.5.6): 5466 + dependencies: 5467 + browserslist: 4.28.1 5468 + caniuse-api: 3.0.0 5469 + colord: 2.9.3 5470 + postcss: 8.5.6 5471 + postcss-value-parser: 4.2.0 5472 + 5473 + postcss-convert-values@7.0.8(postcss@8.5.6): 5474 + dependencies: 5475 + browserslist: 4.28.1 5476 + postcss: 8.5.6 5477 + postcss-value-parser: 4.2.0 5478 + 5479 + postcss-discard-comments@7.0.5(postcss@8.5.6): 5480 + dependencies: 5481 + postcss: 8.5.6 5482 + postcss-selector-parser: 7.1.1 5483 + 5484 + postcss-discard-duplicates@7.0.2(postcss@8.5.6): 5485 + dependencies: 5486 + postcss: 8.5.6 5487 + 5488 + postcss-discard-empty@7.0.1(postcss@8.5.6): 5489 + dependencies: 5490 + postcss: 8.5.6 5491 + 5492 + postcss-discard-overridden@7.0.1(postcss@8.5.6): 5493 + dependencies: 5494 + postcss: 8.5.6 5495 + 5054 5496 postcss-import@15.1.0(postcss@8.5.6): 5055 5497 dependencies: 5056 5498 postcss: 8.5.6 ··· 5070 5512 jiti: 1.21.7 5071 5513 postcss: 8.5.6 5072 5514 5515 + postcss-merge-longhand@7.0.5(postcss@8.5.6): 5516 + dependencies: 5517 + postcss: 8.5.6 5518 + postcss-value-parser: 4.2.0 5519 + stylehacks: 7.0.7(postcss@8.5.6) 5520 + 5521 + postcss-merge-rules@7.0.7(postcss@8.5.6): 5522 + dependencies: 5523 + browserslist: 4.28.1 5524 + caniuse-api: 3.0.0 5525 + cssnano-utils: 5.0.1(postcss@8.5.6) 5526 + postcss: 8.5.6 5527 + postcss-selector-parser: 7.1.1 5528 + 5529 + postcss-minify-font-values@7.0.1(postcss@8.5.6): 5530 + dependencies: 5531 + postcss: 8.5.6 5532 + postcss-value-parser: 4.2.0 5533 + 5534 + postcss-minify-gradients@7.0.1(postcss@8.5.6): 5535 + dependencies: 5536 + colord: 2.9.3 5537 + cssnano-utils: 5.0.1(postcss@8.5.6) 5538 + postcss: 8.5.6 5539 + postcss-value-parser: 4.2.0 5540 + 5541 + postcss-minify-params@7.0.5(postcss@8.5.6): 5542 + dependencies: 5543 + browserslist: 4.28.1 5544 + cssnano-utils: 5.0.1(postcss@8.5.6) 5545 + postcss: 8.5.6 5546 + postcss-value-parser: 4.2.0 5547 + 5548 + postcss-minify-selectors@7.0.5(postcss@8.5.6): 5549 + dependencies: 5550 + cssesc: 3.0.0 5551 + postcss: 8.5.6 5552 + postcss-selector-parser: 7.1.1 5553 + 5073 5554 postcss-nested@6.2.0(postcss@8.5.6): 5074 5555 dependencies: 5075 5556 postcss: 8.5.6 5076 5557 postcss-selector-parser: 6.1.2 5077 5558 5559 + postcss-normalize-charset@7.0.1(postcss@8.5.6): 5560 + dependencies: 5561 + postcss: 8.5.6 5562 + 5563 + postcss-normalize-display-values@7.0.1(postcss@8.5.6): 5564 + dependencies: 5565 + postcss: 8.5.6 5566 + postcss-value-parser: 4.2.0 5567 + 5568 + postcss-normalize-positions@7.0.1(postcss@8.5.6): 5569 + dependencies: 5570 + postcss: 8.5.6 5571 + postcss-value-parser: 4.2.0 5572 + 5573 + postcss-normalize-repeat-style@7.0.1(postcss@8.5.6): 5574 + dependencies: 5575 + postcss: 8.5.6 5576 + postcss-value-parser: 4.2.0 5577 + 5578 + postcss-normalize-string@7.0.1(postcss@8.5.6): 5579 + dependencies: 5580 + postcss: 8.5.6 5581 + postcss-value-parser: 4.2.0 5582 + 5583 + postcss-normalize-timing-functions@7.0.1(postcss@8.5.6): 5584 + dependencies: 5585 + postcss: 8.5.6 5586 + postcss-value-parser: 4.2.0 5587 + 5588 + postcss-normalize-unicode@7.0.5(postcss@8.5.6): 5589 + dependencies: 5590 + browserslist: 4.28.1 5591 + postcss: 8.5.6 5592 + postcss-value-parser: 4.2.0 5593 + 5594 + postcss-normalize-url@7.0.1(postcss@8.5.6): 5595 + dependencies: 5596 + postcss: 8.5.6 5597 + postcss-value-parser: 4.2.0 5598 + 5599 + postcss-normalize-whitespace@7.0.1(postcss@8.5.6): 5600 + dependencies: 5601 + postcss: 8.5.6 5602 + postcss-value-parser: 4.2.0 5603 + 5604 + postcss-ordered-values@7.0.2(postcss@8.5.6): 5605 + dependencies: 5606 + cssnano-utils: 5.0.1(postcss@8.5.6) 5607 + postcss: 8.5.6 5608 + postcss-value-parser: 4.2.0 5609 + 5610 + postcss-reduce-initial@7.0.5(postcss@8.5.6): 5611 + dependencies: 5612 + browserslist: 4.28.1 5613 + caniuse-api: 3.0.0 5614 + postcss: 8.5.6 5615 + 5616 + postcss-reduce-transforms@7.0.1(postcss@8.5.6): 5617 + dependencies: 5618 + postcss: 8.5.6 5619 + postcss-value-parser: 4.2.0 5620 + 5078 5621 postcss-selector-parser@6.1.2: 5079 5622 dependencies: 5080 5623 cssesc: 3.0.0 5081 5624 util-deprecate: 1.0.2 5082 5625 5626 + postcss-selector-parser@7.1.1: 5627 + dependencies: 5628 + cssesc: 3.0.0 5629 + util-deprecate: 1.0.2 5630 + 5631 + postcss-svgo@7.1.0(postcss@8.5.6): 5632 + dependencies: 5633 + postcss: 8.5.6 5634 + postcss-value-parser: 4.2.0 5635 + svgo: 4.0.0 5636 + 5637 + postcss-unique-selectors@7.0.4(postcss@8.5.6): 5638 + dependencies: 5639 + postcss: 8.5.6 5640 + postcss-selector-parser: 7.1.1 5641 + 5083 5642 postcss-value-parser@4.2.0: {} 5084 5643 5085 5644 postcss-values-parser@6.0.2(postcss@8.5.6): ··· 5277 5836 safe-buffer@5.2.1: {} 5278 5837 5279 5838 safe-stable-stringify@2.5.0: {} 5839 + 5840 + sax@1.4.3: {} 5280 5841 5281 5842 scheduler@0.23.2: 5282 5843 dependencies: ··· 5373 5934 dependencies: 5374 5935 escape-string-regexp: 1.0.5 5375 5936 5937 + stylehacks@7.0.7(postcss@8.5.6): 5938 + dependencies: 5939 + browserslist: 4.28.1 5940 + postcss: 8.5.6 5941 + postcss-selector-parser: 7.1.1 5942 + 5376 5943 sucrase@3.35.1: 5377 5944 dependencies: 5378 5945 '@jridgewell/gen-mapping': 0.3.13 ··· 5386 5953 supports-preserve-symlinks-flag@1.0.0: {} 5387 5954 5388 5955 svg-parser@2.0.4: {} 5956 + 5957 + svgo@4.0.0: 5958 + dependencies: 5959 + commander: 11.1.0 5960 + css-select: 5.2.2 5961 + css-tree: 3.1.0 5962 + css-what: 6.2.2 5963 + csso: 5.0.5 5964 + picocolors: 1.1.1 5965 + sax: 1.4.3 5389 5966 5390 5967 tailwindcss@3.4.19: 5391 5968 dependencies: ··· 5545 6122 optionalDependencies: 5546 6123 '@types/node': 24.10.4 5547 6124 fsevents: 2.3.3 6125 + 6126 + webextension-polyfill@0.12.0: {} 5548 6127 5549 6128 webidl-conversions@3.0.1: {} 5550 6129