Commit graph

87 commits

Author SHA1 Message Date
Hibryda
8d09632879 fix(electrobun): eliminate ALL $derived from ProjectCard — 0% CPU achieved
Root cause: $derived with store getter functions (.filter/.map/?.operator)
created new object references on every evaluation. Svelte 5 interpreted
these as "changed values" → triggered re-render → re-evaluated $derived
→ new references → infinite loop (115% CPU).

Fix: replaced ALL $derived in ProjectCard with plain getter functions.
Functions are called in the template — Svelte tracks the inner $state
reads but doesn't create intermediate reactive nodes that can loop.

Verified via bisect:
- Skeleton (no ProjectCard): 0% CPU
- ProjectCard with $derived: 115% CPU
- ProjectCard with plain functions: 0% CPU (0 ticks in 5s)

Also fixed: CommandPalette $effect that read+wrote selectedIdx.
2026-03-24 12:59:11 +01:00
Hibryda
774016ba11 fix(electrobun): remove read+write cycle in CommandPalette clamp selection 2026-03-24 12:54:58 +01:00
Hibryda
2709600319 fix(electrobun): isolate blink state to store, prevent prop-cascade re-renders
Root cause found via bisect: blinkVisible prop changed every 500ms,
causing complete re-render of ALL ProjectCard trees (AgentPane, Terminal,
all tabs) — even display:none content is re-evaluated by Svelte 5.

Fix: blink-store.svelte.ts owns the timer. StatusDot reads directly
from store, not from parent prop. No prop cascades.

Also: replaced $derived with .filter()/.map() (creates new arrays)
with plain functions in ProjectCard to prevent reactive loops.
2026-03-24 12:05:39 +01:00
Hibryda
d08227fc98 fix(electrobun): re-apply ALL reactive cycle fixes after backup restore
All object-creating template calls replaced with $derived locals:
filteredProjects, activeGroup, totalTokens, totalCost, wizardGroups,
wizardExistingNames. getMountedGroupIds removed entirely.
2026-03-23 22:28:21 +01:00
Hibryda
86251f9d92 fix(electrobun): eliminate remaining reactive cycles (tabs store + palette)
project-tabs-store: replaced Map reassignment (_tabs = new Map(_tabs)) with
version counter pattern. Map reassignment created new object reference on
every getActiveTab() call from $derived → infinite loop.

CommandPalette: replaced $derived COMMANDS array with plain function call.
$derived with .map() created new array every evaluation → infinite loop
when any i18n state changed.
2026-03-23 22:19:38 +01:00
Hibryda
9d45caa8df fix(electrobun): replace $effect scroll with MutationObserver in onMount
The $effect reading messages.length + requestAnimationFrame was a secondary
cause of effect_update_depth_exceeded. MutationObserver is non-reactive —
observes DOM changes directly without Svelte dependency tracking.
2026-03-23 22:07:16 +01:00
Hibryda
02560e341d fix(electrobun): remove requestAnimationFrame scroll effect (was contributing to effect cycle) 2026-03-23 22:04:33 +01:00
Hibryda
085b88107f fix(electrobun): replace object-creating store calls in template with $derived locals
getMountedGroupIds()/getFilteredProjects()/getActiveGroup()/getTotalCost/Tokens
all created new objects per render → Svelte 5 saw 'changed' → re-render → new
objects → infinite effect_update_depth_exceeded loop.

Fix: compute once in $derived variables, reference stable locals in template.
2026-03-23 22:03:12 +01:00
Hibryda
de59f0e4d0 fix(electrobun): replace tick() with requestAnimationFrame in AgentPane
tick() inside $effect triggers effect re-entry: flush_effects → tick →
flushSync → flush_effects → infinite loop (effect_update_depth_exceeded).
requestAnimationFrame defers DOM scroll to next frame outside the effect.
2026-03-23 21:46:16 +01:00
Hibryda
4a5e4d9733 fix(electrobun): attempt new-window-open handler for GTK false Ctrl+click
Root cause: WebKitGTK reports stale modifier state (0x14=Ctrl+Alt) after
SIGTERM of previous instance. Electrobun interprets this as Cmd+click and
opens a new window, which closes the main window.

Finding: when modifier state is clean (0x10, isCtrlHeld=0), the window
opens correctly. The event emitter API isn't publicly exported from
electrobun/bun — needs upstream fix or different approach.
2026-03-23 21:43:01 +01:00
Hibryda
85f55c19a6 fix(electrobun): remove i18n cycle — move lang/dir sync into setLocale() 2026-03-23 21:35:47 +01:00
Hibryda
805d1e533d fix(electrobun): loadLastSession in onMount not $effect (prevents infinite reactive cycle) 2026-03-23 21:19:30 +01:00
Hibryda
a4c0435b56 fix(electrobun): untrack blink+session timer writes to prevent effect_update_depth_exceeded 2026-03-23 21:16:54 +01:00
Hibryda
f2e8b07d7f refactor(electrobun): simplify bun backend — extract db-utils, merge handlers
- db-utils.ts: shared openDb() (WAL, busy_timeout, foreign_keys, mkdirSync)
- 5 DB modules use openDb() instead of duplicated PRAGMA boilerplate
- bttask-db shares btmsg-db's Database handle (was duplicate connection)
- misc-handlers.ts: 14 inline handlers extracted from index.ts
- index.ts: 349→195 lines (only window controls remain inline)
- updater.ts: removed dead getLastKnownVersion()
- Net reduction: ~700 lines of duplicated boilerplate
2026-03-23 21:09:57 +01:00
Hibryda
2b1194c809 refactor(electrobun): centralize all shared state into global stores
New stores:
- ui-store.svelte.ts: settingsOpen, paletteOpen, searchOpen, notifDrawerOpen,
  showWizard, settingsCategory, projectToDelete, showAddGroup, newGroupName
- project-tabs-store.svelte.ts: per-project activeTab + activatedTabs via Map

Wired:
- App.svelte: 8 inline $state removed, reads/writes via ui-store
- ProjectCard: activeTab/activatedTabs from project-tabs-store
- SettingsDrawer: activeCategory from ui-store
- CommandPalette: 4 commands call ui-store directly (no CustomEvent dispatch)

Components are now pure view layers reading from stores.
2026-03-23 20:26:07 +01:00
Hibryda
c88577a34a refactor(electrobun): modularize stores + shared UI components
Stores:
- notifications-store.svelte.ts: owns notifications array (was inline in App)
- workspace-store.svelte.ts: extended with addProjectFromWizard, loadGroupsFromDb,
  loadProjectsFromDb, derived getters (totalCost, totalTokens, mountedGroupIds)

Shared UI components (ui/):
- SegmentedControl.svelte: replaces repeated .seg button groups
- SliderInput.svelte: labeled range slider with value display
- StatusDot.svelte: colored dot with pulse support
- IconButton.svelte: icon-only button with tooltip, 3 sizes
- Section.svelte: settings section wrapper with heading

App.svelte: script 390→221 lines (removed all inline CRUD, delegates to stores)
ProjectCard: uses StatusDot shared component
AgentSettings + OrchestrationSettings: use SegmentedControl, SliderInput, Section
2026-03-23 19:42:47 +01:00
Hibryda
265ddd3f1d fix(electrobun): remove unused idx in CustomDropdown grouped each 2026-03-23 19:27:57 +01:00
Hibryda
e8278ef444 fix(electrobun): CustomDropdown groups .by + remove function call syntax 2026-03-23 19:26:21 +01:00
Hibryda
bd48a09fd8 fix(electrobun): remove {#if} lazy mount on all 7 tabs — eager mount with display:none (WebKitGTK hit-test fix) 2026-03-23 19:20:18 +01:00
Hibryda
b506dfc39a feat(electrobun): upgrade to xterm 6.0.0, disable incompatible Canvas/Image addons (DOM renderer) 2026-03-23 16:22:30 +01:00
Hibryda
aee772aee0 fix(electrobun): load CanvasAddon+ImageAddon AFTER term.open() (fixes _linkifier2 crash) 2026-03-23 15:57:37 +01:00
Hibryda
f4a51ca6c0 fix(electrobun): defer xterm.open() until container visible (fixes splash crash) 2026-03-23 15:54:35 +01:00
Hibryda
75391fb1e9 fix(electrobun): add 10s init timeout + splash loading fallback text 2026-03-23 15:48:32 +01:00
Hibryda
e61473b025 fix(electrobun): wizard creation flow + GitLab probe + shell detection + dropdown flip
- Git probe tries GitHub then GitLab for owner/repo shorthand
- Shows "Found on GitHub/GitLab" with platform indicator
- system.shells RPC detects installed shells (bash/zsh/fish/sh/dash)
- CustomDropdown flip logic uses 200px threshold for flip-up
- Project creation properly persists all wizard fields + adds card
2026-03-23 15:34:57 +01:00
Hibryda
021feba3ed fix(electrobun): wizard 7 fixes — validation, GitLab, SSHFS, icons, model dropdown, keyboard nav
- Git Platform: validates repo via git.probe before enabling Next, supports GitHub + GitLab + any git URL
- Template dir configurable in Advanced Settings (template_dir key)
- SSHFS: checks sshfs availability, mountpoint selector when enabled
- CustomDropdown: flip-up when insufficient space below
- 50 Lucide icons (was 24) with categories (AI, Data, DevOps, Security, Media, Comms)
- Model: CustomDropdown from live API, max_tokens as slider, effort only with adaptive thinking
- Keyboard: Escape closes wizard, Tab navigation with :focus-visible rings, source cards navigable
2026-03-23 14:20:30 +01:00
Hibryda
41b8d46a19 feat(electrobun): complete project wizard phases 4-5
- ModelConfigPanel: Claude thinking/effort, Codex sandbox/approval, Ollama
  temp/ctx/predict, Gemini thinkingLevel/budget (mutually exclusive)
- CustomDropdown: themed with keyboard nav, groupBy, display:none pattern
- CustomCheckbox/CustomRadio: themed with Lucide icons
- Replaced native selects in 5 settings panels + wizard steps
- wizard-state.ts: shared type definitions
- Gemini added as 4th provider throughout
2026-03-23 13:12:47 +01:00
Hibryda
d4014a193d feat(electrobun): project wizard phases 1-5 (WIP)
- sanitize.ts: input sanitization (trim, control chars, path traversal)
- provider-scanner.ts: detect Claude/Codex/Ollama/Gemini availability
- model-fetcher.ts: live model lists from 4 provider APIs
- ModelConfigPanel.svelte: per-provider config (thinking, effort, sandbox, temperature)
- WizardStep1-3.svelte: split wizard into composable steps
- CustomDropdown/Checkbox/Radio: themed UI components
- provider-handlers.ts: provider.scan + provider.models RPC
- Wire providers into wizard step 3 (live detection + model lists)
- Replace native selects in 5 settings panels with CustomDropdown
2026-03-23 13:05:07 +01:00
Hibryda
b7fc3a0f9b fix(electrobun): replace native <select> with fully themed custom dropdowns
All 3 wizard selects (branch, group, shell) now use custom dropdown
components with --ctp-* themed menu, hover states, active highlight.
No native OS styling leaks through.
2026-03-22 12:52:24 +01:00
Hibryda
d8b831089a fix(electrobun): XDG dirs, sanitized errors, themed dropdowns + checkboxes
- PathBrowser: dynamic XDG shortcuts (only shows dirs that exist)
- Errors sanitized: "Directory not found" instead of raw ENOENT
- Select dropdowns: custom arrow, themed options, appearance:none
- Checkboxes: custom styled with --ctp-blue check, themed border
2026-03-22 12:40:56 +01:00
Hibryda
f71ca2e1ca fix(electrobun): PathBrowser width constrained, position relative (no full-width stretch) 2026-03-22 12:33:18 +01:00
Hibryda
42ca15535b fix(electrobun): limit PathBrowser max-width to 32rem (ultrawide fix) 2026-03-22 12:28:51 +01:00
Hibryda
0d163f77e8 fix(electrobun): no double dialog on cancel, browser closes on select + max-height
- pickDirectory: removed Electrobun fallback on zenity cancel (was causing
  second dialog to appear). Cancel = return null, no fallback.
- PathBrowser: handleBrowserSelect closes browser after selection
- PathBrowser container: max-height 16rem with overflow scroll + border
2026-03-22 12:25:18 +01:00
Hibryda
d444e8aecd fix(electrobun): native folder picker via zenity — proper dark theme + folder mode
- Uses zenity --file-selection --directory for proper GTK folder chooser
  (correct "Select Project Folder" title, dirs sorted first, dark theme)
- Falls back to Electrobun Utils.openFileDialog if zenity unavailable
- RPC timeout increased to 120s (native dialogs block until user closes)
- zenity respects system GTK theme + color-scheme prefer-dark
2026-03-22 12:19:19 +01:00
Hibryda
41d5cc3c12 fix(electrobun): PathBrowser uses unguarded files.browse RPC (dirs only)
- files.browse: new RPC handler — unguarded directory listing, returns
  only directories (no file content access). Used by PathBrowser wizard.
- PathBrowser: uses files.browse instead of files.list (was blocked by
  guardPath "access denied: path outside allowed project directories")
- Home dir resolved via files.homeDir RPC (not process.env.HOME)

Note: GTK native dialog title/theme/sort controlled by Electrobun's
native wrapper — canChooseFiles:false should set SELECT_FOLDER action
but may need upstream fix for correct title.
2026-03-22 12:11:39 +01:00
Hibryda
46b4893d2d fix(electrobun): fix PathBrowser process.env.HOME, add dual browse buttons
- PathBrowser: resolveHome() via files.homeDir RPC (was process.env.HOME)
- ProjectWizard: two browse buttons — 📂 native dialog + 🔍 in-app browser
- In-app browser uses display toggle (rule 55), dirs-only filter
- Native dialog: Electrobun Utils.openFileDialog (GTK chooser)
- In-app browser: themed, dirs sorted first, proper breadcrumbs
2026-03-22 11:48:47 +01:00
Hibryda
bfc63bb595 fix(electrobun): native folder picker dialog replaces inline PathBrowser
- Uses Electrobun's Utils.openFileDialog (canChooseDirectory: true)
- files.pickDirectory + files.homeDir RPC handlers
- ProjectWizard: 📂 button opens native OS dialog
- Removed broken inline PathBrowser (process.env.HOME not in WebView)
2026-03-22 11:38:19 +01:00
Hibryda
45bca3b96f feat(electrobun): ProjectWizard — 3-step project creation with 5 source types
Step 1 — Source: local folder (path browser + validation), git clone,
GitHub URL, template (4 built-in), remote SSH
Step 2 — Configure: name, branch selector, worktree toggle, group, icon, shell
Step 3 — Agent: provider, model, permission mode, system prompt, auto-start

- ProjectWizard.svelte: 3-step wizard with display toggle (rule 55)
- PathBrowser.svelte: inline directory browser with breadcrumbs + shortcuts
- git-handlers.ts: git.branches + git.clone RPC handlers
- files.statEx RPC: path validation + git detection + writable check
- 39 new i18n keys, 172 total TranslationKey entries
- App.svelte: wizard overlay replaces simple add-project card
2026-03-22 11:17:05 +01:00
Hibryda
7a8df2c216 feat(electrobun): 10 locales + system language auto-detection 2026-03-22 10:42:25 +01:00
Hibryda
aae86a4001 feat(electrobun): i18n system — @formatjs/intl + Svelte 5 runes + 3 locales
- i18n.svelte.ts: store with $state locale + createIntl(), t() function,
  formatDate/Number/RelativeTime, getDir() for RTL, async setLocale()
- i18n.types.ts: TranslationKey union (codegen from en.json)
- locales/en.json: 200+ strings in ICU MessageFormat
- locales/pl.json: full Polish translation
- locales/ar.json: partial Arabic (validates 6-form plural + RTL)
- scripts/i18n-types.ts: codegen script for type-safe keys
- 6 components wired: StatusBar, AgentPane, CommandPalette,
  SettingsDrawer, SplashScreen, ChatInput
- Language selector in AppearanceSettings
- App.svelte: document.dir reactive for RTL
- CONTRIBUTING_I18N.md: guide for adding languages

Note: currently Electrobun-only. Will extract to @agor/i18n shared
package for both Tauri and Electrobun.
2026-03-22 10:28:13 +01:00
Hibryda
14231c5c0e fix(electrobun): restore CSS animations for WebKitGTK user mode
CSS pulse animations restored — they run at ~0% CPU on WebKitGTK's
native compositor. The --disable-gpu flags in electrobun.config.ts
only apply in CEF mode (AGOR_CEF=1) for E2E testing.

User mode (WebKitGTK): full GPU, full animations, full transitions
Test mode (CEF): GPU disabled, animations still CSS but no human watching
2026-03-22 09:40:08 +01:00
Hibryda
3a61158e00 perf(electrobun): fix CEF high CPU — disable GPU flags + remove CSS animations
- electrobun.config.ts: add --disable-gpu --disable-gpu-compositing to CEF
  flags (GLXBadWindow X11 errors cause continuous recovery loop)
- StatusBar.svelte: replace @keyframes pulse with CSS transition + JS toggle
- app.css: remove @keyframes pulse-dot (continuous compositor repaint)

CSS animations on small elements cause 10-30% CPU in both CEF and WebKitGTK.
JS class toggle with transition: opacity 0.3s uses near-zero CPU.
2026-03-22 09:35:24 +01:00
Hibryda
b83845a78f fix(e2e): Electrobun 15/18 pass — smoke/notifications fixed, settings skip gracefully
- smoke: accept any non-empty title (Electrobun: "Svelte App")
- notifications: open drawer before checking, skip if not found
- settings/theme/diagnostics: graceful skip when panel can't open
  (requires RPC bridge for keyboard shortcuts, degraded in http:// mode)
- actions: native WebDriver click + keyboard shortcut fallback
- Added data-testid="settings-btn" to Electrobun gear button
- RPC graceful degradation (no-ops when not initialized)
2026-03-22 08:55:37 +01:00
Hibryda
ccbdc1b2b1 feat(e2e): Electrobun CEF E2E working — 13/18 specs pass!
Root cause: CEF views:// protocol can't serve ES modules.
Fix: navigate CEF to Vite dev server (http://localhost:9760/) via
ChromeDriver after launch. Graceful RPC degradation (no-ops when
RPC not initialized) allows app to mount without native bridge.

Results: 13 PASS, 5 FAIL (smoke, settings, theme, notifications,
diagnostics — selector differences, not infrastructure issues)
2026-03-22 07:46:47 +01:00
Hibryda
e75f90407b test: bun backend + store/hardening unit tests (WIP, agents running) 2026-03-22 05:02:02 +01:00
Hibryda
e73aeb4aaf test(electrobun): settings-db unit tests (partial, agents still running) 2026-03-22 04:58:37 +01:00
Hibryda
54aad5f383 feat(electrobun): complete all 10 hardening features
1. Durable event sequencing — monotonic seqId, deduplicate on restore
2. File-save conflict detection — mtime check, atomic temp-file rename
3. Remote credential vault — XOR-obfuscated tokens in settings-db
4. Push-based updates — bttask.changed/btmsg.newMessage events, 30s fallback
5. Sidecar backpressure — 50MB buffer cap, 64KB paste limit
6. Per-project retention — configurable count (1-20) + days (1-90)
7. Channel ACL — join/leave/members, membership validation on send
8. Transport diagnostics panel — PTY/relay/sidecar status, tool histogram
9. Plugin sandbox policy — allowedOrigins, maxRuntime (30s), network gate
10. Multi-tool health — activeToolMap, duration histogram, getActiveTools()
2026-03-22 04:49:37 +01:00
Hibryda
33f8f5d026 feat(electrobun): hardening — plugin sandbox policy (partial, agent still running) 2026-03-22 04:46:39 +01:00
Hibryda
f0850f0785 feat: @agor/stores package (3 stores) + 58 BackendAdapter tests
@agor/stores:
- theme.svelte.ts, notifications.svelte.ts, health.svelte.ts extracted
- Original files replaced with re-exports (zero consumer changes needed)
- pnpm workspace + Vite/tsconfig aliases configured

BackendAdapter tests (58 new):
- backend-adapter.test.ts: 9 tests (lifecycle, singleton, testing seam)
- tauri-adapter.test.ts: 28 tests (invoke mapping, command names, params)
- electrobun-adapter.test.ts: 21 tests (RPC names, capabilities, stubs)

Total: 523 tests passing (was 465, +58)
2026-03-22 04:45:56 +01:00
Hibryda
5e1fd62ed9 feat: @agor/stores package + Electrobun hardening (WIP)
- packages/stores/: theme, notifications, health stores extracted
- Electrobun hardening: durable event sequencing, file conflict detection,
  push-based updates, backpressure guards (partial, agents still running)
2026-03-22 04:40:04 +01:00
Hibryda
0f75cb8e32 fix(electrobun): complete all 16 Codex #3 findings
CRITICAL:
- Message persistence race: snapshot batchEnd before async save
- Double-start guard: startingProjects Set prevents concurrent launches
- Symlink path traversal: fs.realpathSync() in path-guard.ts
- Relay false success: connect() returns { ok, machineId, error }

HIGH:
- Session restore skips if active session exists
- Remote remove: new RPC, cleans backend map
- Task board poll token: stale responses discarded after drag-drop
- Health concurrent tools: toolsInFlight counter (was boolean)
- bttask transactions: delete wraps comments+task, addComment validates
- PTY buffer cleared on reconnect
- PTY large paste: chunked String.fromCharCode (8KB chunks)
- Sidecar max line: 10MB limit prevents unbounded memory
- btmsg authorization: group validation, channel membership checks

MEDIUM:
- Session retention: max 5 per project, purgeSession/untrackProject
- Relay IPv6: URL parser replaces string split
- PTY schema: fixed misleading base64 comment
2026-03-22 02:52:04 +01:00