brew-tui 0.5.0 → 0.5.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +25 -9
- package/build/{brewbar-installer-ZEMXNDHP.js → brewbar-installer-VVZRCP5K.js} +2 -2
- package/build/{chunk-KDJNCZD7.js → chunk-3BPHXV35.js} +21 -13
- package/build/chunk-3BPHXV35.js.map +1 -0
- package/build/index.js +26 -18
- package/build/index.js.map +1 -1
- package/package.json +1 -1
- package/build/chunk-KDJNCZD7.js.map +0 -1
- /package/build/{brewbar-installer-ZEMXNDHP.js.map → brewbar-installer-VVZRCP5K.js.map} +0 -0
package/README.md
CHANGED
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
[](https://nodejs.org/)
|
|
7
7
|
[](LICENSE)
|
|
8
8
|
[](https://github.com/MoLinesGitHub/homebrew-tap)
|
|
9
|
-
[]()
|
|
10
10
|
|
|
11
11
|
A keyboard-driven terminal UI for Homebrew, with a native macOS menu bar companion that watches updates in the background. No daemons, no middleware — both tools call `brew` directly.
|
|
12
12
|
|
|
@@ -62,17 +62,26 @@ npx brew-tui
|
|
|
62
62
|
| **Doctor** | Run `brew doctor` and see warnings at a glance |
|
|
63
63
|
| **Package Info** | Detailed view with dependencies, caveats, and quick actions |
|
|
64
64
|
|
|
65
|
-
### Pro Features —
|
|
65
|
+
### Pro Features — €9,95/mo or €82/year (-31%)
|
|
66
66
|
|
|
67
67
|
| Feature | What it solves |
|
|
68
68
|
|---------|----------------|
|
|
69
|
+
| **Smart Rollback** | Auto-snapshots after every install/upgrade/uninstall/pin; revert with bottle/versioned/pin strategies. Press `R` from a flagged CVE to jump straight to the rollback plan |
|
|
70
|
+
| **Cross-machine Sync** | iCloud Drive backend, AES-256-GCM encrypted client-side. Brewfile and snapshots stay aligned across all your Macs with interactive conflict resolution |
|
|
71
|
+
| **CVE Real-time** | BrewBar polls [OSV.dev](https://osv.dev) hourly. Critical/high CVEs trigger native macOS notifications and a badge count |
|
|
72
|
+
| **Declarative Brewfile** | YAML desired state, drift score 0-100, interactive reconciliation. Closer to a lightweight Nix-flake than `brew bundle` |
|
|
73
|
+
| **Impact Analysis** | Pre-upgrade risk panel (low/medium/high) with dependency tree and reverse-deps that will be affected, surfaced before each upgrade |
|
|
69
74
|
| **Profiles** | Replicate your exact setup on a new Mac in one command |
|
|
70
75
|
| **Smart Cleanup** | Reclaim gigabytes by listing orphans ranked by size |
|
|
71
76
|
| **Action History** | "What did I install last week?" — answered |
|
|
72
|
-
| **Security Audit** |
|
|
77
|
+
| **Security Audit** | Cross-checks every installed package against [OSV.dev](https://osv.dev) live |
|
|
73
78
|
| **BrewBar** | A menu bar app that watches your packages while you sleep — auto-installs and auto-launches the moment you go Pro |
|
|
74
79
|
|
|
75
|
-
|
|
80
|
+
### Team Tier — €8/seat/mo or €81,60/seat/year (-15%, min 3 seats)
|
|
81
|
+
|
|
82
|
+
Everything in Pro plus **Team Compliance** — admin defines a central PolicyFile (JSON) listing required and forbidden packages and required taps. Each Mac on the team gets a 0-100 compliance score, severity-graded violations and an automatic remediation plan. Useful for onboarding, security audits and keeping every developer's environment aligned.
|
|
83
|
+
|
|
84
|
+
[See pricing →](https://molinesdesigns.com/brewtui/)
|
|
76
85
|
|
|
77
86
|
---
|
|
78
87
|
|
|
@@ -175,7 +184,7 @@ Views (React/Ink) --> Stores (Zustand) --> brew-api --> Parsers --> brew CLI (sp
|
|
|
175
184
|
- ESM-only, TypeScript strict mode, built with [tsup](https://github.com/egoist/tsup)
|
|
176
185
|
- All streaming operations (install, upgrade) use AsyncGenerators yielding lines in real time
|
|
177
186
|
- Package names validated via regex before passing to `spawn` (no shell injection)
|
|
178
|
-
-
|
|
187
|
+
- 211 tests across 20 suites (Vitest)
|
|
179
188
|
|
|
180
189
|
---
|
|
181
190
|
|
|
@@ -194,16 +203,23 @@ Views (React/Ink) --> Stores (Zustand) --> brew-api --> Parsers --> brew CLI (sp
|
|
|
194
203
|
|
|
195
204
|
```
|
|
196
205
|
src/
|
|
197
|
-
views/ #
|
|
198
|
-
stores/ # Zustand stores (brew, navigation, license, modal)
|
|
206
|
+
views/ # 16 React/Ink views (incl. rollback, brewfile, sync, compliance)
|
|
207
|
+
stores/ # Zustand stores (brew, navigation, license, modal, rollback, sync, compliance, ...)
|
|
199
208
|
components/ # Shared UI (StatusBadge, ResultBanner, SelectableRow, ...)
|
|
200
209
|
hooks/ # useKeyboard, useBrewStream, useDebounce
|
|
201
210
|
lib/
|
|
202
211
|
license/ # Polar API, AES encryption, anti-tamper, canary
|
|
203
|
-
security/ # OSV vulnerability scanning
|
|
212
|
+
security/ # OSV vulnerability scanning (Pro)
|
|
204
213
|
profiles/ # Profile export/import (Pro)
|
|
205
214
|
cleanup/ # Orphan detection (Pro)
|
|
206
215
|
history/ # Action logging (Pro)
|
|
216
|
+
rollback/ # Snapshot-based rollback engine (Pro)
|
|
217
|
+
state-snapshot/# Periodic Brew state snapshots (Pro)
|
|
218
|
+
diff-engine/ # Snapshot diff for rollback and sync (Pro)
|
|
219
|
+
impact/ # Pre-upgrade impact analysis (Pro)
|
|
220
|
+
brewfile/ # Declarative YAML Brewfile + drift score (Pro)
|
|
221
|
+
sync/ # iCloud-backed cross-machine sync, AES-256-GCM (Pro)
|
|
222
|
+
compliance/ # Team policy enforcement (Team)
|
|
207
223
|
parsers/ # JSON and text parsers for brew output
|
|
208
224
|
i18n/ # English + Spanish translations
|
|
209
225
|
utils/ # Colors, spacing, logger, formatting
|
|
@@ -220,7 +236,7 @@ cd Brew-TUI
|
|
|
220
236
|
npm install
|
|
221
237
|
npm run dev # Run with tsx (requires interactive TTY)
|
|
222
238
|
npm run typecheck # tsc --noEmit
|
|
223
|
-
npm run test # vitest (
|
|
239
|
+
npm run test # vitest (211 tests)
|
|
224
240
|
npm run lint # eslint
|
|
225
241
|
npm run build # Production bundle via tsup
|
|
226
242
|
```
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import {
|
|
2
2
|
fetchWithTimeout,
|
|
3
3
|
t
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-3BPHXV35.js";
|
|
5
5
|
import "./chunk-KDHEUNRI.js";
|
|
6
6
|
|
|
7
7
|
// src/lib/brewbar-installer.ts
|
|
@@ -125,4 +125,4 @@ export {
|
|
|
125
125
|
launchBrewBar,
|
|
126
126
|
uninstallBrewBar
|
|
127
127
|
};
|
|
128
|
-
//# sourceMappingURL=brewbar-installer-
|
|
128
|
+
//# sourceMappingURL=brewbar-installer-VVZRCP5K.js.map
|
|
@@ -259,14 +259,14 @@ var en = {
|
|
|
259
259
|
account_emailLabel: "Email:",
|
|
260
260
|
account_nameLabel: "Name:",
|
|
261
261
|
account_planLabel: "Plan:",
|
|
262
|
-
account_monthlyPrice: "9\u20AC/month",
|
|
263
|
-
|
|
262
|
+
account_monthlyPrice: "9.95\u20AC/month",
|
|
263
|
+
account_yearlyPrice: "82\u20AC/year",
|
|
264
264
|
account_keyLabel: "Key:",
|
|
265
265
|
account_expiresLabel: "Expires:",
|
|
266
266
|
account_activatedLabel: "Activated:",
|
|
267
267
|
account_upgradeTitle: "Upgrade to Brew-TUI Pro",
|
|
268
268
|
account_unlockDesc: "Unlock Profiles, Smart Cleanup, History, Security Audit, and BrewBar (macOS menu bar companion).",
|
|
269
|
-
account_pricing: "9\u20AC/month or
|
|
269
|
+
account_pricing: "9.95\u20AC/month or 82\u20AC/year (save 31%)",
|
|
270
270
|
account_runActivate: "Run:",
|
|
271
271
|
account_activateCmd: "brew-tui activate <key>",
|
|
272
272
|
account_licenseExpired: "Your license has expired. Renew to continue using Pro features.",
|
|
@@ -290,12 +290,16 @@ var en = {
|
|
|
290
290
|
upgrade_historyDesc: "Track every install, uninstall, and upgrade with timestamps. Search and filter your package management history.",
|
|
291
291
|
upgrade_security: "Security Audit",
|
|
292
292
|
upgrade_securityDesc: "Scan installed packages against known vulnerabilities (CVEs). See severity levels, affected versions, and available fixes.",
|
|
293
|
-
upgrade_pricing: "9\u20AC/month or
|
|
293
|
+
upgrade_pricing: "9.95\u20AC/month or 82\u20AC/year (save 31%)",
|
|
294
|
+
upgrade_teamFeature: "{{title}} \u2014 Team tier",
|
|
295
|
+
upgrade_teamPricing: "8\u20AC/seat/month or 81.60\u20AC/seat/year (save 15%) \u2014 from 3 seats",
|
|
294
296
|
upgrade_buyAt: "Buy at:",
|
|
295
|
-
upgrade_buyUrl: "https://buy.polar.sh/
|
|
297
|
+
upgrade_buyUrl: "https://buy.polar.sh/polar_cl_yQsiUeDelyyEQznbWffD1j77JAyP24ra7iEVQ22PA4h",
|
|
298
|
+
upgrade_buyUrlTeam: "https://buy.polar.sh/polar_cl_CO6xqSzKgFiQJwXnhZYGqisOP04Wspi0KKZSn38NjFZ?quantity=3",
|
|
296
299
|
upgrade_activateWith: "Then activate with:",
|
|
297
300
|
upgrade_activateCmd: "brew-tui activate <your-license-key>",
|
|
298
|
-
upgrade_proLabel: "Brew-TUI Pro \u2014 9\u20AC/month or
|
|
301
|
+
upgrade_proLabel: "Brew-TUI Pro \u2014 9.95\u20AC/month or 82\u20AC/year \u2014 Includes BrewBar for macOS",
|
|
302
|
+
upgrade_teamLabel: "Brew-TUI Team \u2014 8\u20AC/seat/month \u2014 Includes everything in Pro plus Compliance",
|
|
299
303
|
// ── Progress Log ──
|
|
300
304
|
progress_noOutput: "No output yet",
|
|
301
305
|
// ── Search Input ──
|
|
@@ -733,14 +737,14 @@ var es = {
|
|
|
733
737
|
account_emailLabel: "Email:",
|
|
734
738
|
account_nameLabel: "Nombre:",
|
|
735
739
|
account_planLabel: "Plan:",
|
|
736
|
-
account_monthlyPrice: "9\u20AC/mes",
|
|
737
|
-
|
|
740
|
+
account_monthlyPrice: "9,95\u20AC/mes",
|
|
741
|
+
account_yearlyPrice: "82\u20AC/a\xF1o",
|
|
738
742
|
account_keyLabel: "Clave:",
|
|
739
743
|
account_expiresLabel: "Expira:",
|
|
740
744
|
account_activatedLabel: "Activado:",
|
|
741
745
|
account_upgradeTitle: "Actualiza a Brew-TUI Pro",
|
|
742
746
|
account_unlockDesc: "Desbloquea Perfiles, Limpieza Inteligente, Historial, Auditor\xEDa de Seguridad y BrewBar (barra de men\xFA macOS).",
|
|
743
|
-
account_pricing: "9\u20AC/mes o
|
|
747
|
+
account_pricing: "9,95\u20AC/mes o 82\u20AC/a\xF1o (ahorra 31%)",
|
|
744
748
|
account_runActivate: "Ejecuta:",
|
|
745
749
|
account_activateCmd: "brew-tui activate <clave>",
|
|
746
750
|
account_licenseExpired: "Tu licencia ha expirado. Renueva para seguir usando las funciones Pro.",
|
|
@@ -764,12 +768,16 @@ var es = {
|
|
|
764
768
|
upgrade_historyDesc: "Rastrea cada instalaci\xF3n, desinstalaci\xF3n y actualizaci\xF3n con marcas de tiempo. Busca y filtra tu historial de gesti\xF3n de paquetes.",
|
|
765
769
|
upgrade_security: "Auditor\xEDa de Seguridad",
|
|
766
770
|
upgrade_securityDesc: "Escanea paquetes instalados contra vulnerabilidades conocidas (CVEs). Ve niveles de severidad, versiones afectadas y correcciones disponibles.",
|
|
767
|
-
upgrade_pricing: "9\u20AC/mes o
|
|
771
|
+
upgrade_pricing: "9,95\u20AC/mes o 82\u20AC/a\xF1o (ahorra 31%)",
|
|
772
|
+
upgrade_teamFeature: "{{title}} \u2014 Tier Team",
|
|
773
|
+
upgrade_teamPricing: "8\u20AC/seat/mes o 81,60\u20AC/seat/a\xF1o (ahorra 15%) \u2014 desde 3 seats",
|
|
768
774
|
upgrade_buyAt: "Compra en:",
|
|
769
|
-
upgrade_buyUrl: "https://buy.polar.sh/
|
|
775
|
+
upgrade_buyUrl: "https://buy.polar.sh/polar_cl_yQsiUeDelyyEQznbWffD1j77JAyP24ra7iEVQ22PA4h",
|
|
776
|
+
upgrade_buyUrlTeam: "https://buy.polar.sh/polar_cl_CO6xqSzKgFiQJwXnhZYGqisOP04Wspi0KKZSn38NjFZ?quantity=3",
|
|
770
777
|
upgrade_activateWith: "Luego activa con:",
|
|
771
778
|
upgrade_activateCmd: "brew-tui activate <tu-clave-de-licencia>",
|
|
772
|
-
upgrade_proLabel: "Brew-TUI Pro \u2014 9\u20AC/mes o
|
|
779
|
+
upgrade_proLabel: "Brew-TUI Pro \u2014 9,95\u20AC/mes o 82\u20AC/a\xF1o \u2014 Incluye BrewBar para macOS",
|
|
780
|
+
upgrade_teamLabel: "Brew-TUI Team \u2014 8\u20AC/seat/mes \u2014 Incluye todo Pro m\xE1s Compliance",
|
|
773
781
|
// ── Progress Log ──
|
|
774
782
|
progress_noOutput: "Sin salida a\xFAn",
|
|
775
783
|
// ── Search Input ──
|
|
@@ -1001,4 +1009,4 @@ export {
|
|
|
1001
1009
|
tp,
|
|
1002
1010
|
fetchWithTimeout
|
|
1003
1011
|
};
|
|
1004
|
-
//# sourceMappingURL=chunk-
|
|
1012
|
+
//# sourceMappingURL=chunk-3BPHXV35.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/i18n/index.ts","../src/i18n/en.ts","../src/i18n/es.ts","../src/lib/fetch-timeout.ts"],"sourcesContent":["import { create } from 'zustand';\nimport en from './en.js';\nimport es from './es.js';\nimport type { Translations, TranslationKey } from './en.js';\n\nexport type Locale = 'en' | 'es';\n\nconst locales: Record<Locale, Translations> = { en, es };\n\nfunction isLocale(s: string): s is Locale {\n return s === 'en' || s === 'es';\n}\n\nfunction detectLocale(): Locale {\n const flag = process.argv.find((a) => a.startsWith('--lang='));\n if (flag) {\n const code = flag.split('=')[1];\n if (code && isLocale(code)) return code;\n }\n const env = process.env.LANG ?? process.env.LC_ALL ?? process.env.LC_MESSAGES ?? '';\n const prefix = env.split(/[_.]/)[0] ?? '';\n if (isLocale(prefix)) return prefix;\n return 'en';\n}\n\n// ── Locale store ──\ninterface LocaleState {\n locale: Locale;\n setLocale: (locale: Locale) => void;\n}\n\nexport const useLocaleStore = create<LocaleState>((set) => ({\n locale: detectLocale(),\n setLocale: (locale) => set({ locale }),\n}));\n\nexport function getLocale(): Locale {\n return useLocaleStore.getState().locale;\n}\n\n// ── Translation function ──\nexport function t(key: TranslationKey, values?: Record<string, string | number>): string {\n const locale = useLocaleStore.getState().locale;\n let text: string = locales[locale][key] ?? locales['en'][key] ?? key;\n if (values) {\n for (const [k, v] of Object.entries(values)) {\n text = text.replaceAll(`{{${k}}}`, String(v));\n }\n }\n return text;\n}\n\n// ── Plural helper ──\nexport function tp(baseKey: string, count: number, values?: Record<string, string | number>): string {\n const suffix = count === 1 ? '_one' : '_other';\n return t(`${baseKey}${suffix}` as TranslationKey, { count, ...values });\n}\n\nexport type { TranslationKey, Translations };\n","const en = {\n // ── App chrome ──\n app_title: 'Brew-TUI',\n pro_badge: 'PRO',\n app_version: 'Brew-TUI v{{version}}',\n\n // ── View labels (header tab bar) ──\n view_dashboard: 'Dashboard',\n view_installed: 'Installed',\n view_search: 'Search',\n view_outdated: 'Outdated',\n view_packageInfo: 'Pkg Info',\n view_services: 'Services',\n view_doctor: 'Doctor',\n view_profiles: 'Profiles',\n view_smartCleanup: 'Cleanup',\n view_history: 'History',\n view_rollback: 'Rollback',\n view_brewfile: 'Brewfile',\n view_sync: 'Sync',\n view_securityAudit: 'Security',\n view_account: 'Account',\n\n // ── Keyboard hint actions ──\n hint_navigate: 'navigate',\n hint_next: 'next',\n hint_quit: 'quit',\n hint_filter: 'filter',\n hint_info: 'info',\n hint_toggle: 'toggle',\n hint_typeToSearch: 'type to search',\n hint_install: 'install',\n hint_uninstall: 'uninstall',\n hint_upgrade: 'upgrade',\n hint_upgradeAll: 'upgrade all',\n hint_back: 'back',\n hint_start: 'start',\n hint_stop: 'stop',\n hint_restart: 'restart',\n hint_refresh: 'refresh',\n hint_new: 'new',\n hint_details: 'details',\n hint_import: 'import',\n hint_delete: 'delete',\n hint_clean: 'clean',\n hint_all: 'all',\n hint_search: 'search',\n hint_clear: 'clear',\n hint_scan: 'scan',\n hint_expand: 'expand',\n hint_cancel: 'cancel',\n hint_force: 'force uninstall (ignore deps)',\n hint_rescan: 'rescan',\n hint_deactivate: 'deactivate',\n hint_promo: 'promo code',\n hint_importProfile: 'import this profile',\n hint_lang: 'lang',\n hint_replay: 'replay',\n hint_edit: 'edit',\n hint_pin: 'pin/unpin',\n hint_rollback_confirm: 'rollback',\n hint_add: 'add',\n hint_reconcile: 'reconcile',\n hint_export: 'export',\n hint_select: 'select',\n hint_sync: 'sync now',\n hint_conflict: 'resolve',\n hint_rollback: 'rollback',\n hint_check_compliance: 'check compliance',\n\n // ── Loading / progress ──\n loading_default: 'Loading...',\n loading_fetchingBrew: 'Fetching Homebrew data...',\n loading_installed: 'Loading installed packages...',\n loading_outdated: 'Checking for outdated packages...',\n loading_services: 'Loading services...',\n loading_doctor: 'Running brew doctor... (this may take a moment)',\n loading_profiles: 'Loading profiles...',\n loading_cleanup: 'Analyzing packages... (checking disk usage)',\n loading_history: 'Loading history...',\n loading_security: 'Scanning packages against OSV vulnerability database...',\n loading_searching: 'Searching...',\n loading_package: 'Loading {{name}}...',\n\n // ── Confirm dialog ──\n confirm_yes: '[Y]es',\n confirm_no: '[N]o',\n\n // ── Error ──\n error_prefix: 'Error: ',\n\n // ── Common ──\n common_andMore: '...and {{count}} more',\n common_exit: '(exit {{code}})',\n common_yes: 'yes',\n common_no: 'no',\n\n // ── Relative time ──\n time_justNow: 'just now',\n time_minutesAgo: '{{n}}m ago',\n time_hoursAgo: '{{n}}h ago',\n time_daysAgo: '{{n}}d ago',\n time_monthsAgo: '{{n}}mo ago',\n\n // ── Badges ──\n badge_outdated: 'outdated',\n badge_pinned: 'pinned',\n badge_kegOnly: 'keg-only',\n badge_dep: 'dep',\n badge_installed: 'installed',\n badge_deprecated: 'deprecated',\n badge_ok: 'ok',\n badge_fail: 'fail',\n badge_error: 'error',\n\n // ── Dashboard ──\n dashboard_pro_status: 'Pro Status',\n dashboard_security: 'Security',\n dashboard_cves: '{{count}} CVEs',\n dashboard_no_cves: 'No CVEs',\n dashboard_brewfile: 'Brewfile',\n dashboard_sync: 'Sync',\n dashboard_sync_never: 'Never synced',\n dashboard_sync_ago: '{{time}} ago',\n dashboard_compliance: 'Compliance',\n dashboard_compliance_violations: '{{count}} violations',\n dashboard_compliance_ok: 'OK',\n dashboard_overview: 'Overview',\n dashboard_formulae: 'Formulae',\n dashboard_casks: 'Casks',\n dashboard_outdated: 'Outdated',\n dashboard_services: 'Services',\n dashboard_systemInfo: 'System Info',\n dashboard_homebrew: 'Homebrew:',\n dashboard_prefix: 'Prefix:',\n dashboard_updated: 'Updated:',\n dashboard_outdatedPackages: 'Outdated Packages',\n dashboard_serviceErrors: 'Service Errors',\n dashboard_partialData: 'Some Homebrew sections failed to load:',\n dashboard_statError: 'ERR',\n\n // ── Installed ──\n installed_formulaeCount: 'Formulae ({{count}})',\n installed_casksCount: 'Casks ({{count}})',\n installed_filterDisplay: 'Filter: \"{{query}}\" ({{count}} matches)',\n installed_noPackages: 'No packages found',\n installed_confirmUninstall: 'Uninstall {{name}}?',\n\n // ── Search ──\n search_placeholder: 'Search Homebrew packages... (enter to search)',\n search_resultsFor: 'Results for',\n search_escToClear: '(esc to clear)',\n search_installing: 'Installing package...',\n search_installComplete: 'Installation complete!',\n search_confirmInstall: 'Install {{name}}?',\n search_formulaeHeader: '=== Formulae ({{count}})',\n search_casksHeader: '=== Casks ({{count}})',\n search_noResults: 'No results found',\n search_minChars: 'Type at least 2 characters to search.',\n\n // ── Outdated ──\n outdated_title: 'Outdated Packages ({{count}})',\n outdated_upgrading: 'Upgrading...',\n outdated_upgradeComplete: 'Upgrade complete!',\n outdated_pressRefresh: '(press r to refresh)',\n outdated_upToDate: 'Everything is up to date!',\n outdated_confirmAll: 'Upgrade all {{count}} packages?',\n outdated_confirmSingle: 'Upgrade {{name}}?',\n outdated_pinned: '[pinned]',\n\n // ── Package Info ──\n pkgInfo_noPackage: 'No package selected. Go to Installed and press Enter on a package.',\n pkgInfo_notFound: 'Package not found',\n pkgInfo_installing: 'Installing {{name}}...',\n pkgInfo_uninstalling: 'Uninstalling {{name}}...',\n pkgInfo_upgrading: 'Upgrading {{name}}...',\n pkgInfo_done: 'Done!',\n pkgInfo_confirmInstall: 'install {{name}}?',\n pkgInfo_confirmUninstall: 'uninstall {{name}}?',\n pkgInfo_confirmUpgrade: 'upgrade {{name}}?',\n pkgInfo_details: 'Details',\n pkgInfo_homepage: 'Homepage:',\n pkgInfo_license: 'License:',\n pkgInfo_tap: 'Tap:',\n pkgInfo_stable: 'Stable:',\n pkgInfo_installed: 'Installed:',\n pkgInfo_bottle: 'Bottle:',\n pkgInfo_onRequest: 'On request:',\n pkgInfo_noDependency: 'no (dependency)',\n pkgInfo_dependencies: 'Dependencies ({{count}})',\n pkgInfo_caveats: 'Caveats',\n\n // ── Services ──\n services_title: 'Homebrew Services',\n services_titleCount: 'Homebrew Services ({{count}})',\n services_noServices: 'No services found',\n services_name: 'Name',\n services_status: 'Status',\n services_user: 'User',\n services_processing: 'Processing...',\n services_confirmStop: 'Stop service {{name}}?',\n services_confirmRestart: 'Restart service {{name}}?',\n\n // ── Doctor ──\n doctor_title: 'Homebrew Doctor',\n doctor_clean: 'Your system is ready to brew.',\n doctor_warningsNotCaptured: 'Doctor finished with warnings but none were captured.',\n\n // ── Profiles ──\n profiles_title: 'Package Profiles ({{count}})',\n profiles_importTitle: 'Importing profile...',\n profiles_importComplete: 'Import complete. Press any key.',\n profiles_importPartial: 'Import finished with errors. Check the log above.',\n profiles_createName: 'Create Profile \\u2014 Name:',\n profiles_namePlaceholder: 'e.g. work, personal, project-x',\n profiles_createDesc: 'Create Profile \"{{name}}\" \\u2014 Description:',\n profiles_descPlaceholder: 'Brief description of this setup',\n profiles_created: 'Created: {{date}}',\n profiles_formulaeCount: 'Formulae ({{count}})',\n profiles_casksCount: 'Casks ({{count}})',\n profiles_confirmDelete: 'Delete profile \"{{name}}\"?',\n profiles_noProfiles: 'No profiles saved yet.',\n profiles_press: 'Press',\n profiles_exportHint: 'to export your current setup as a profile.',\n profiles_editName: 'Edit Profile \\u2014 Name:',\n profiles_editDesc: 'Edit Profile \"{{name}}\" \\u2014 Description:',\n\n // ── Smart Cleanup ──\n cleanup_title: 'Smart Cleanup',\n cleanup_cleaning: 'Cleaning up...',\n cleanup_complete: 'Cleanup complete! Press r to re-analyze.',\n cleanup_orphans: 'Orphans',\n cleanup_reclaimable: 'Reclaimable',\n cleanup_selected: 'Selected',\n cleanup_confirmUninstall: 'Uninstall {{count}} packages?',\n cleanup_confirmForce: 'Some packages have dependencies. Force uninstall {{count}} packages? (ignores dependencies)',\n cleanup_depError: 'Some packages could not be removed due to dependencies.',\n cleanup_systemClean: 'No orphaned packages found. Your system is clean!',\n\n // ── History ──\n history_title: 'Action History ({{count}})',\n history_filterLabel: 'filter: {{filter}}',\n history_searchPlaceholder: 'Search packages...',\n history_confirmClear: 'Clear all {{count}} history entries?',\n history_noEntries: 'No history entries',\n history_noEntriesFor: 'No history entries for \"{{filter}}\"',\n history_all: '(all)',\n history_actionInstall: 'install',\n history_actionUninstall: 'uninstall',\n history_actionUpgrade: 'upgrade',\n history_actionUpgradeAll: 'upgrade-all',\n history_confirmReplay: 'Re-run: {{action}} {{name}}?',\n history_replayAll: 'Re-run: upgrade all packages?',\n\n // ── Security Audit ──\n security_rollback_hint: 'R: open Rollback for version restoration',\n security_title: 'Security Audit',\n security_scanned: 'Scanned',\n security_vulnerable: 'Vulnerable',\n security_critical: 'Critical',\n security_high: 'High',\n security_medium: 'Medium',\n security_noVulns: 'No known vulnerabilities found in your installed packages!',\n security_fixedIn: 'Fixed in: {{version}}',\n security_confirmUpgrade: 'Upgrade {{name}} to fix vulnerabilities?',\n\n // ── Account ──\n account_title: 'Account & License',\n account_confirmDeactivate: 'Deactivate your Pro license on this machine?',\n account_statusLabel: 'Status:',\n account_pro: '[Pro]',\n account_free: '[Free]',\n account_expired: '[Expired]',\n account_validating: '[Validating...]',\n account_emailLabel: 'Email:',\n account_nameLabel: 'Name:',\n account_planLabel: 'Plan:',\n account_monthlyPrice: '9.95\\u20AC/month',\n account_yearlyPrice: '82\\u20AC/year',\n account_keyLabel: 'Key:',\n account_expiresLabel: 'Expires:',\n account_activatedLabel: 'Activated:',\n account_upgradeTitle: 'Upgrade to Brew-TUI Pro',\n account_unlockDesc: 'Unlock Profiles, Smart Cleanup, History, Security Audit, and BrewBar (macOS menu bar companion).',\n account_pricing: '9.95\\u20AC/month or 82\\u20AC/year (save 31%)',\n account_runActivate: 'Run:',\n account_activateCmd: 'brew-tui activate <key>',\n account_licenseExpired: 'Your license has expired. Renew to continue using Pro features.',\n account_deactivating: 'Deactivating...',\n account_loading: 'Loading license status...',\n account_promoTitle: 'Redeem Promo Code',\n account_promoLabel: 'Code:',\n account_promoValidating: 'Validating promo code...',\n account_promoSuccess: 'Promo code redeemed! Pro access until {{expires}}.',\n account_promoInvalid: 'Invalid or expired promo code.',\n account_promoError: 'Could not validate promo code. Check your connection.',\n account_promoEsc: 'esc: cancel',\n account_promoHint: 'p: redeem promo code',\n\n // ── Upgrade Prompt ──\n upgrade_proFeature: '{{title}} \\u2014 Pro Feature',\n upgrade_profiles: 'Package Profiles',\n upgrade_profilesDesc: 'Export and import your Homebrew setup across machines. Save named profiles for work, personal, or project-specific configurations.',\n upgrade_cleanup: 'Smart Cleanup',\n upgrade_cleanupDesc: 'Find orphaned packages, analyze disk usage per package, and reclaim disk space with one-click intelligent cleanup.',\n upgrade_history: 'Action History',\n upgrade_historyDesc: 'Track every install, uninstall, and upgrade with timestamps. Search and filter your package management history.',\n upgrade_security: 'Security Audit',\n upgrade_securityDesc: 'Scan installed packages against known vulnerabilities (CVEs). See severity levels, affected versions, and available fixes.',\n upgrade_pricing: '9.95\\u20AC/month or 82\\u20AC/year (save 31%)',\n upgrade_teamFeature: '{{title}} \\u2014 Team tier',\n upgrade_teamPricing: '8\\u20AC/seat/month or 81.60\\u20AC/seat/year (save 15%) \\u2014 from 3 seats',\n upgrade_buyAt: 'Buy at:',\n upgrade_buyUrl: 'https://buy.polar.sh/polar_cl_yQsiUeDelyyEQznbWffD1j77JAyP24ra7iEVQ22PA4h',\n upgrade_buyUrlTeam: 'https://buy.polar.sh/polar_cl_CO6xqSzKgFiQJwXnhZYGqisOP04Wspi0KKZSn38NjFZ?quantity=3',\n upgrade_activateWith: 'Then activate with:',\n upgrade_activateCmd: 'brew-tui activate <your-license-key>',\n upgrade_proLabel: 'Brew-TUI Pro \\u2014 9.95\\u20AC/month or 82\\u20AC/year \\u2014 Includes BrewBar for macOS',\n upgrade_teamLabel: 'Brew-TUI Team \\u2014 8\\u20AC/seat/month \\u2014 Includes everything in Pro plus Compliance',\n\n // ── Progress Log ──\n progress_noOutput: 'No output yet',\n\n // ── Search Input ──\n searchInput_placeholder: 'Type to filter...',\n\n // ── Profile Manager ──\n profileMgr_tapping: 'Tapping {{name}}...',\n profileMgr_installing: 'Installing {{name}}...',\n profileMgr_installingCask: 'Installing cask {{name}}...',\n profileMgr_importDone: 'Done! Installed {{count}} packages.',\n\n // ── CLI ──\n cli_usageActivate: 'Usage: brew-tui activate <license-key>',\n cli_activated: '\\u2714 Pro activated for {{email}}',\n cli_plan: ' Plan: {{plan}}',\n cli_expires: ' Expires: {{date}}',\n cli_activationFailed: '\\u2718 Activation failed: {{error}}',\n cli_noLicense: 'No active license found.',\n cli_deactivated: '\\u2714 License deactivated.',\n cli_planFree: 'Plan: Free',\n cli_planPro: 'Plan: Pro',\n cli_planExpired: 'Plan: Expired',\n cli_confirmDeactivate: 'Deactivate your Pro license on this machine? (y/N): ',\n cli_deactivateCancelled: 'Deactivation cancelled.',\n cli_upgradeHint: 'Run `brew-tui activate <key>` to upgrade to Pro.',\n cli_revalidateHint: 'Run `brew-tui revalidate` to refresh your current license.',\n cli_email: 'Email: {{email}}',\n cli_status: 'Status: {{status}}',\n cli_revalidated: '\\u2714 License revalidated.',\n cli_revalidateGrace: '\\u26A0 Could not reach the server. Your current license remains usable within the offline grace period.',\n cli_revalidateFailed: '\\u2718 License revalidation failed. Renew your subscription or activate a valid key.',\n cli_rateLimited: 'Too many activation attempts. Try again in {{minutes}} minutes.',\n cli_cooldown: 'Please wait before trying again.',\n cli_brewbarInstalling: 'Downloading BrewBar...',\n cli_brewbarInstalled: '\\u2714 BrewBar installed to /Applications/BrewBar.app',\n cli_brewbarAlreadyInstalled: 'BrewBar is already installed. Use --force to reinstall.',\n cli_brewbarUninstalled: '\\u2714 BrewBar removed from /Applications.',\n cli_brewbarNotInstalled: 'BrewBar is not installed.',\n cli_brewbarProRequired: '\\u2718 BrewBar requires a Pro license.\\n Run: brew-tui activate <key>',\n cli_brewbarRevalidateRequired: '\\u2718 BrewBar requires a valid Pro license.\\n Run: brew-tui revalidate',\n cli_brewbarMacOnly: '\\u2718 BrewBar is only available on macOS.',\n cli_brewbarDownloadFailed: '\\u2718 Failed to download BrewBar: {{error}}',\n cli_brewbarAutoFailed: '\\u26A0 BrewBar auto-launch failed: {{error}}',\n cli_deactivateRemoteFailed: '\\u26A0 Warning: Could not reach the server to deactivate remotely. The license was removed locally but may still count as active.',\n\n // ── License degradation (Layer 15) ──\n license_offlineWarning: 'Your license has not been validated for {{days}} days. Please connect to the internet.',\n\n // ── Plurals ──\n plural_vulns_one: '({{count}} vuln)',\n plural_vulns_other: '({{count}} vulns)',\n plural_warnings_one: '{{count}} warning',\n plural_warnings_other: '{{count}} warnings',\n\n // ── Scroll indicators ──\n scroll_moreAbove: '\\u2191 {{count}} more',\n scroll_moreBelow: '\\u2193 {{count}} more',\n\n // ── SCR-001: Cleanup warning ──\n cleanup_warning_system_tools: 'Warning: detected orphans may include dependencies of tools not managed by Homebrew. Review the list before proceeding.',\n\n // ── SCR-002: Installed column headers ──\n installed_col_package: 'Package',\n installed_col_version: 'Version',\n installed_col_status: 'Status',\n\n // ── SCR-003: Search failed ──\n search_failed: 'Search failed',\n\n // ── SCR-007: Deactivate failed ──\n deactivate_failed: 'Deactivation failed',\n\n // ── ACC-005: Version labels ──\n version_installed: 'installed:',\n version_available: 'available:',\n\n // ── SCR-006: Upgrade-all replay warning ──\n upgrade_all_warning: 'Note: this will upgrade all currently outdated packages, which may differ from the original set.',\n\n // ── SEG-007: Delete account ──\n delete_account_confirm: 'Delete all Brew-TUI data (~/.brew-tui)? This removes your license, profiles, and history. This cannot be undone.',\n delete_account_success: 'All Brew-TUI data has been removed.',\n\n // ── SCR-012: Upgrade All packages list ──\n outdated_upgradeAllList: 'Packages to upgrade: {{list}}',\n\n // ── SCR-005: Profile import summary ──\n profiles_importSummary: 'This profile contains {{formulae}} formulae and {{casks}} casks. Continue?',\n\n // ── SCR-017: Network error ��─\n security_networkError: 'Could not reach OSV.dev vulnerability database. Check your internet connection.',\n\n // ── ARQ-004: Dashboard last updated ──\n dashboard_lastUpdated: 'Last updated: {{time}}',\n\n // ── SCR-014: Services last error ──\n services_lastError: 'Last error: {{error}}',\n\n // ── SCR-010: Generic network error ──\n error_network: 'Network error: unable to reach the server.',\n\n // ── ARQ-005: Security cache ──\n security_cachedResults: 'Showing cached results ({{time}} ago). Press r to rescan.',\n\n // ── Impact Analysis ──\n impact_brewfile_hint: '\\u{1F4A1} Add to Brewfile to pin this version before upgrading',\n impact_analyzing: 'Analyzing upgrade impact...',\n impact_high: 'HIGH RISK',\n impact_medium: 'MEDIUM RISK',\n impact_low: 'LOW RISK',\n impact_affects: 'affects {{count}} installed packages',\n impact_usedBy: 'Used by: {{packages}}',\n impact_hint: 'Select package to see upgrade impact',\n impact_reason_critical_package: 'Critical system package',\n impact_reason_major_bump: 'Major version change',\n impact_reason_many_deps: '{{count}} packages depend on this',\n\n // ── Rollback ──\n rollback_title: 'Rollback \\u2014 Restore Previous State',\n rollback_no_snapshots: 'No snapshots available. Snapshots are captured automatically after each operation.',\n rollback_select_snapshot: 'Select a snapshot to restore',\n rollback_snapshot_label: '{{label}} \\u2014 {{date}}',\n rollback_snapshot_auto: 'Auto',\n rollback_diff_empty: 'No changes detected between this snapshot and current state',\n rollback_confirm: 'Roll back {{count}} package(s) to this state?',\n rollback_strategy_bottle: 'from bottle cache',\n rollback_strategy_versioned: 'from versioned formula',\n rollback_strategy_pin: 'pin only (version not restorable)',\n rollback_strategy_unavailable: 'cannot restore',\n rollback_executing: 'Rolling back...',\n rollback_success: 'Rollback completed',\n rollback_error: 'Rollback failed: {{error}}',\n rollback_item_downgrade: '{{name}}: {{from}} \\u2192 {{to}}',\n rollback_item_remove: 'Remove: {{name}}',\n rollback_item_install: 'Install: {{name}} {{version}}',\n rollback_warning_cask: 'Casks will be pinned only (version restoration not supported)',\n rollback_capturing: 'Capturing current snapshot...',\n\n // ── Sync ──\n sync_title: 'Cross-machine Sync',\n sync_disabled: 'Sync is not configured. Press s to set up iCloud sync.',\n sync_status_ok: 'In sync',\n sync_status_drift: '{{count}} change(s) from other machines',\n sync_status_conflict: '{{count}} conflict(s) need resolution',\n sync_last_sync: 'Last sync: {{date}}',\n sync_machine: 'This machine: {{name}}',\n sync_other_machines: 'Other machines: {{names}}',\n sync_syncing: 'Syncing...',\n sync_success: 'Sync complete',\n sync_error: 'Sync failed: {{error}}',\n sync_conflict_title: 'Conflict: {{package}}',\n sync_conflict_local: 'This machine: {{version}}',\n sync_conflict_remote: '{{machine}}: {{version}}',\n sync_conflict_use_local: 'Keep local',\n sync_conflict_use_remote: 'Use remote',\n sync_setup_icloud: 'Setting up iCloud sync...',\n sync_no_icloud: 'iCloud Drive not found. Ensure iCloud Drive is enabled in System Settings.',\n\n // ── Upgrade Prompt — Sync ──\n upgrade_sync: 'Cross-machine Sync',\n upgrade_syncDesc: 'Keep your Homebrew setup in sync across multiple Macs via iCloud Drive. Automatically merge installations and resolve conflicts.',\n\n // ── Brewfile ──\n brewfile_title: 'Declarative Brewfile',\n brewfile_compliant: 'compliant',\n brewfile_no_brewfile: 'No Brewfile found. Press n to create one from your current installation.',\n brewfile_create_name: 'Brewfile name (Enter to confirm):',\n brewfile_created: 'Brewfile created: {{name}}',\n brewfile_drift_missing: '{{count}} packages missing',\n brewfile_drift_extra: '{{count}} extra packages',\n brewfile_drift_wrong: '{{count}} wrong versions',\n brewfile_reconciling: 'Reconciling...',\n brewfile_reconcile_success: 'Reconciliation complete',\n brewfile_reconcile_error: 'Reconciliation failed: {{error}}',\n brewfile_exported: 'Exported to {{path}}',\n brewfile_formulae_count: '{{count}} formulae',\n brewfile_casks_count: '{{count}} casks',\n brewfile_strict_mode: 'Strict mode',\n\n // ── Compliance ──\n view_compliance: 'Compliance',\n compliance_title: 'Team Compliance',\n compliance_no_policy: 'No policy loaded. Press i to import a policy file.',\n compliance_score: '{{score}}% compliant',\n compliance_violations: '{{count}} violation(s)',\n compliance_ok: 'Fully compliant',\n compliance_import_prompt: 'Policy file path (Enter to confirm):',\n compliance_import_error: 'Failed to load policy: {{error}}',\n compliance_violation_missing: 'Missing: {{name}} (required)',\n compliance_violation_forbidden: 'Forbidden: {{name}} — {{reason}}',\n compliance_violation_version: 'Wrong version: {{name}} (required {{required}}, installed {{installed}})',\n compliance_violation_extra: 'Extra package: {{name}}',\n compliance_remediating: 'Remediating...',\n compliance_remediate_success: 'Remediation complete',\n compliance_remediate_error: 'Remediation failed: {{error}}',\n compliance_export_done: 'Report exported to {{path}}',\n compliance_machine: 'Machine: {{name}}',\n compliance_policy_name: 'Policy: {{name}}',\n compliance_policy_by: 'Maintained by: {{maintainer}}',\n\n // ── Upgrade Prompt — Compliance ──\n upgrade_compliance: 'Team Compliance',\n upgrade_complianceDesc: 'Enforce package policies across your team. Define required, forbidden, and version-pinned packages and automatically remediate deviations.',\n};\n\nexport default en;\nexport type Translations = { [K in keyof typeof en]: string };\nexport type TranslationKey = keyof typeof en;\n","import type { Translations } from './en.js';\n\nconst es: Translations = {\n // ── App chrome ──\n app_title: 'Brew-TUI',\n pro_badge: 'PRO',\n app_version: 'Brew-TUI v{{version}}',\n\n // ── View labels (header tab bar) ──\n view_dashboard: 'Inicio',\n view_installed: 'Instalados',\n view_search: 'Buscar',\n view_outdated: 'Desactual.',\n view_packageInfo: 'Info Paq.',\n view_services: 'Servicios',\n view_doctor: 'Doctor',\n view_profiles: 'Perfiles',\n view_smartCleanup: 'Limpieza',\n view_history: 'Historial',\n view_rollback: 'Rollback',\n view_brewfile: 'Brewfile',\n view_sync: 'Sync',\n view_securityAudit: 'Seguridad',\n view_account: 'Cuenta',\n\n // ── Keyboard hint actions ──\n hint_navigate: 'navegar',\n hint_next: 'siguiente',\n hint_quit: 'salir',\n hint_filter: 'filtrar',\n hint_info: 'info',\n hint_toggle: 'cambiar',\n hint_typeToSearch: 'escriba para buscar',\n hint_install: 'instalar',\n hint_uninstall: 'desinstalar',\n hint_upgrade: 'actualizar',\n hint_upgradeAll: 'actualizar todo',\n hint_back: 'volver',\n hint_start: 'iniciar',\n hint_stop: 'detener',\n hint_restart: 'reiniciar',\n hint_refresh: 'refrescar',\n hint_new: 'nuevo',\n hint_details: 'detalles',\n hint_import: 'importar',\n hint_delete: 'eliminar',\n hint_clean: 'limpiar',\n hint_all: 'todo',\n hint_search: 'buscar',\n hint_clear: 'borrar',\n hint_scan: 'escanear',\n hint_expand: 'expandir',\n hint_cancel: 'cancelar',\n hint_force: 'forzar (ignorar deps)',\n hint_rescan: 're-escanear',\n hint_deactivate: 'desactivar',\n hint_promo: 'c\\u00F3digo promo',\n hint_importProfile: 'importar este perfil',\n hint_lang: 'idioma',\n hint_replay: 'repetir',\n hint_edit: 'editar',\n hint_pin: 'fijar/desfijar',\n hint_rollback_confirm: 'revertir',\n hint_add: 'a\\u00F1adir',\n hint_reconcile: 'reconciliar',\n hint_export: 'exportar',\n hint_select: 'seleccionar',\n hint_sync: 'sincronizar',\n hint_conflict: 'resolver',\n hint_rollback: 'revertir',\n hint_check_compliance: 'verificar conformidad',\n\n // ── Loading / progress ──\n loading_default: 'Cargando...',\n loading_fetchingBrew: 'Obteniendo datos de Homebrew...',\n loading_installed: 'Cargando paquetes instalados...',\n loading_outdated: 'Buscando paquetes desactualizados...',\n loading_services: 'Cargando servicios...',\n loading_doctor: 'Ejecutando brew doctor... (puede tardar un momento)',\n loading_profiles: 'Cargando perfiles...',\n loading_cleanup: 'Analizando paquetes... (verificando uso de disco)',\n loading_history: 'Cargando historial...',\n loading_security: 'Escaneando paquetes en la base de datos de vulnerabilidades OSV...',\n loading_searching: 'Buscando...',\n loading_package: 'Cargando {{name}}...',\n\n // ── Confirm dialog ──\n confirm_yes: '[S]\\u00ED',\n confirm_no: '[N]o',\n\n // ── Error ──\n error_prefix: 'Error: ',\n\n // ── Common ──\n common_andMore: '...y {{count}} m\\u00E1s',\n common_exit: '(salida {{code}})',\n common_yes: 's\\u00ED',\n common_no: 'no',\n\n // ── Relative time ──\n time_justNow: 'ahora',\n time_minutesAgo: 'hace {{n}}m',\n time_hoursAgo: 'hace {{n}}h',\n time_daysAgo: 'hace {{n}}d',\n time_monthsAgo: 'hace {{n}}me',\n\n // ── Badges ──\n badge_outdated: 'desactualizado',\n badge_pinned: 'fijado',\n badge_kegOnly: 'keg-only',\n badge_dep: 'dep',\n badge_installed: 'instalado',\n badge_deprecated: 'obsoleto',\n badge_ok: 'ok',\n badge_fail: 'fallo',\n badge_error: 'error',\n\n // ── Dashboard ──\n dashboard_pro_status: 'Estado Pro',\n dashboard_security: 'Seguridad',\n dashboard_cves: '{{count}} CVEs',\n dashboard_no_cves: 'Sin CVEs',\n dashboard_brewfile: 'Brewfile',\n dashboard_sync: 'Sync',\n dashboard_sync_never: 'Nunca sincronizado',\n dashboard_sync_ago: 'hace {{time}}',\n dashboard_compliance: 'Conformidad',\n dashboard_compliance_violations: '{{count}} violaciones',\n dashboard_compliance_ok: 'OK',\n dashboard_overview: 'Resumen',\n dashboard_formulae: 'Formulae',\n dashboard_casks: 'Casks',\n dashboard_outdated: 'Desactualizados',\n dashboard_services: 'Servicios',\n dashboard_systemInfo: 'Info del Sistema',\n dashboard_homebrew: 'Homebrew:',\n dashboard_prefix: 'Prefijo:',\n dashboard_updated: 'Actualizado:',\n dashboard_outdatedPackages: 'Paquetes Desactualizados',\n dashboard_serviceErrors: 'Errores de Servicios',\n dashboard_partialData: 'Algunas secciones de Homebrew no pudieron cargarse:',\n dashboard_statError: 'ERR',\n\n // ── Installed ──\n installed_formulaeCount: 'Formulae ({{count}})',\n installed_casksCount: 'Casks ({{count}})',\n installed_filterDisplay: 'Filtro: \"{{query}}\" ({{count}} coincidencias)',\n installed_noPackages: 'No se encontraron paquetes',\n installed_confirmUninstall: '\\u00BFDesinstalar {{name}}?',\n\n // ── Search ──\n search_placeholder: 'Buscar paquetes Homebrew... (enter para buscar)',\n search_resultsFor: 'Resultados para',\n search_escToClear: '(esc para limpiar)',\n search_installing: 'Instalando paquete...',\n search_installComplete: '\\u00A1Instalaci\\u00F3n completa!',\n search_confirmInstall: '\\u00BFInstalar {{name}}?',\n search_formulaeHeader: '=== Formulae ({{count}})',\n search_casksHeader: '=== Casks ({{count}})',\n search_noResults: 'Sin resultados',\n search_minChars: 'Escribe al menos 2 caracteres para buscar.',\n\n // ── Outdated ──\n outdated_title: 'Paquetes Desactualizados ({{count}})',\n outdated_upgrading: 'Actualizando...',\n outdated_upgradeComplete: '\\u00A1Actualizaci\\u00F3n completa!',\n outdated_pressRefresh: '(presiona r para refrescar)',\n outdated_upToDate: '\\u00A1Todo est\\u00E1 al d\\u00EDa!',\n outdated_confirmAll: '\\u00BFActualizar los {{count}} paquetes?',\n outdated_confirmSingle: '\\u00BFActualizar {{name}}?',\n outdated_pinned: '[fijado]',\n\n // ── Package Info ──\n pkgInfo_noPackage: 'Ning\\u00FAn paquete seleccionado. Ve a Instalados y presiona Enter en un paquete.',\n pkgInfo_notFound: 'Paquete no encontrado',\n pkgInfo_installing: 'Instalando {{name}}...',\n pkgInfo_uninstalling: 'Desinstalando {{name}}...',\n pkgInfo_upgrading: 'Actualizando {{name}}...',\n pkgInfo_done: '\\u00A1Listo!',\n pkgInfo_confirmInstall: '\\u00BFinstalar {{name}}?',\n pkgInfo_confirmUninstall: '\\u00BFdesinstalar {{name}}?',\n pkgInfo_confirmUpgrade: '\\u00BFactualizar {{name}}?',\n pkgInfo_details: 'Detalles',\n pkgInfo_homepage: 'Web:',\n pkgInfo_license: 'Licencia:',\n pkgInfo_tap: 'Tap:',\n pkgInfo_stable: 'Estable:',\n pkgInfo_installed: 'Instalado:',\n pkgInfo_bottle: 'Bottle:',\n pkgInfo_onRequest: 'Por solicitud:',\n pkgInfo_noDependency: 'no (dependencia)',\n pkgInfo_dependencies: 'Dependencias ({{count}})',\n pkgInfo_caveats: 'Advertencias',\n\n // ── Services ──\n services_title: 'Servicios Homebrew',\n services_titleCount: 'Servicios Homebrew ({{count}})',\n services_noServices: 'No se encontraron servicios',\n services_name: 'Nombre',\n services_status: 'Estado',\n services_user: 'Usuario',\n services_processing: 'Procesando...',\n services_confirmStop: '\\u00BFDetener servicio {{name}}?',\n services_confirmRestart: '\\u00BFReiniciar servicio {{name}}?',\n\n // ── Doctor ──\n doctor_title: 'Homebrew Doctor',\n doctor_clean: 'Tu sistema est\\u00E1 listo para brew.',\n doctor_warningsNotCaptured: 'Doctor termin\\u00F3 con advertencias pero no se capturaron.',\n\n // ── Profiles ──\n profiles_title: 'Perfiles de Paquetes ({{count}})',\n profiles_importTitle: 'Importando perfil...',\n profiles_importComplete: 'Importaci\\u00F3n completa. Presiona cualquier tecla.',\n profiles_importPartial: 'Importaci\\u00F3n finalizada con errores. Revisa el registro arriba.',\n profiles_createName: 'Crear Perfil \\u2014 Nombre:',\n profiles_namePlaceholder: 'ej. trabajo, personal, proyecto-x',\n profiles_createDesc: 'Crear Perfil \"{{name}}\" \\u2014 Descripci\\u00F3n:',\n profiles_descPlaceholder: 'Breve descripci\\u00F3n de esta configuraci\\u00F3n',\n profiles_created: 'Creado: {{date}}',\n profiles_formulaeCount: 'Formulae ({{count}})',\n profiles_casksCount: 'Casks ({{count}})',\n profiles_confirmDelete: '\\u00BFEliminar perfil \"{{name}}\"?',\n profiles_noProfiles: 'A\\u00FAn no hay perfiles guardados.',\n profiles_press: 'Presiona',\n profiles_exportHint: 'para exportar tu configuraci\\u00F3n actual como perfil.',\n profiles_editName: 'Editar Perfil \\u2014 Nombre:',\n profiles_editDesc: 'Editar Perfil \"{{name}}\" \\u2014 Descripci\\u00F3n:',\n\n // ── Smart Cleanup ──\n cleanup_title: 'Limpieza Inteligente',\n cleanup_cleaning: 'Limpiando...',\n cleanup_complete: '\\u00A1Limpieza completa! Presiona r para re-analizar.',\n cleanup_orphans: 'Hu\\u00E9rfanos',\n cleanup_reclaimable: 'Recuperable',\n cleanup_selected: 'Seleccionados',\n cleanup_confirmUninstall: '\\u00BFDesinstalar {{count}} paquetes?',\n cleanup_confirmForce: 'Algunos paquetes tienen dependencias. \\u00BFForzar desinstalaci\\u00F3n de {{count}} paquetes? (ignora dependencias)',\n cleanup_depError: 'Algunos paquetes no pudieron eliminarse por dependencias.',\n cleanup_systemClean: '\\u00A1No se encontraron paquetes hu\\u00E9rfanos. Tu sistema est\\u00E1 limpio!',\n\n // ── History ──\n history_title: 'Historial de Acciones ({{count}})',\n history_filterLabel: 'filtro: {{filter}}',\n history_searchPlaceholder: 'Buscar paquetes...',\n history_confirmClear: '\\u00BFBorrar las {{count}} entradas del historial?',\n history_noEntries: 'Sin entradas en el historial',\n history_noEntriesFor: 'Sin entradas en el historial para \"{{filter}}\"',\n history_all: '(todos)',\n history_actionInstall: 'instalar',\n history_actionUninstall: 'desinstalar',\n history_actionUpgrade: 'actualizar',\n history_actionUpgradeAll: 'actualizar-todo',\n history_confirmReplay: '\\u00BFRe-ejecutar: {{action}} {{name}}?',\n history_replayAll: '\\u00BFRe-ejecutar: actualizar todos los paquetes?',\n\n // ── Security Audit ──\n security_title: 'Auditor\\u00EDa de Seguridad',\n security_scanned: 'Escaneados',\n security_vulnerable: 'Vulnerables',\n security_critical: 'Cr\\u00EDticos',\n security_high: 'Altos',\n security_medium: 'Medios',\n security_noVulns: '\\u00A1No se encontraron vulnerabilidades conocidas en tus paquetes instalados!',\n security_fixedIn: 'Corregido en: {{version}}',\n security_confirmUpgrade: '\\u00BFActualizar {{name}} para corregir vulnerabilidades?',\n\n // ── Account ──\n account_title: 'Cuenta y Licencia',\n account_confirmDeactivate: '\\u00BFDesactivar tu licencia Pro en esta m\\u00E1quina?',\n account_statusLabel: 'Estado:',\n account_pro: '[Pro]',\n account_free: '[Gratis]',\n account_expired: '[Expirada]',\n account_validating: '[Validando...]',\n account_emailLabel: 'Email:',\n account_nameLabel: 'Nombre:',\n account_planLabel: 'Plan:',\n account_monthlyPrice: '9,95\\u20AC/mes',\n account_yearlyPrice: '82\\u20AC/a\\u00F1o',\n account_keyLabel: 'Clave:',\n account_expiresLabel: 'Expira:',\n account_activatedLabel: 'Activado:',\n account_upgradeTitle: 'Actualiza a Brew-TUI Pro',\n account_unlockDesc: 'Desbloquea Perfiles, Limpieza Inteligente, Historial, Auditor\\u00EDa de Seguridad y BrewBar (barra de men\\u00FA macOS).',\n account_pricing: '9,95\\u20AC/mes o 82\\u20AC/a\\u00F1o (ahorra 31%)',\n account_runActivate: 'Ejecuta:',\n account_activateCmd: 'brew-tui activate <clave>',\n account_licenseExpired: 'Tu licencia ha expirado. Renueva para seguir usando las funciones Pro.',\n account_deactivating: 'Desactivando...',\n account_loading: 'Cargando estado de la licencia...',\n account_promoTitle: 'Canjear C\\u00F3digo Promocional',\n account_promoLabel: 'C\\u00F3digo:',\n account_promoValidating: 'Validando c\\u00F3digo promocional...',\n account_promoSuccess: '\\u00A1C\\u00F3digo canjeado! Acceso Pro hasta {{expires}}.',\n account_promoInvalid: 'C\\u00F3digo inv\\u00E1lido o expirado.',\n account_promoError: 'No se pudo validar el c\\u00F3digo. Comprueba tu conexi\\u00F3n.',\n account_promoEsc: 'esc: cancelar',\n account_promoHint: 'p: canjear c\\u00F3digo promocional',\n\n // ── Upgrade Prompt ──\n upgrade_proFeature: '{{title}} \\u2014 Funci\\u00F3n Pro',\n upgrade_profiles: 'Perfiles de Paquetes',\n upgrade_profilesDesc: 'Exporta e importa tu configuraci\\u00F3n de Homebrew entre m\\u00E1quinas. Guarda perfiles con nombre para trabajo, personal o proyectos espec\\u00EDficos.',\n upgrade_cleanup: 'Limpieza Inteligente',\n upgrade_cleanupDesc: 'Encuentra paquetes hu\\u00E9rfanos, analiza uso de disco por paquete y recupera espacio con limpieza inteligente de un clic.',\n upgrade_history: 'Historial de Acciones',\n upgrade_historyDesc: 'Rastrea cada instalaci\\u00F3n, desinstalaci\\u00F3n y actualizaci\\u00F3n con marcas de tiempo. Busca y filtra tu historial de gesti\\u00F3n de paquetes.',\n upgrade_security: 'Auditor\\u00EDa de Seguridad',\n upgrade_securityDesc: 'Escanea paquetes instalados contra vulnerabilidades conocidas (CVEs). Ve niveles de severidad, versiones afectadas y correcciones disponibles.',\n upgrade_pricing: '9,95\\u20AC/mes o 82\\u20AC/a\\u00F1o (ahorra 31%)',\n upgrade_teamFeature: '{{title}} \\u2014 Tier Team',\n upgrade_teamPricing: '8\\u20AC/seat/mes o 81,60\\u20AC/seat/a\\u00F1o (ahorra 15%) \\u2014 desde 3 seats',\n upgrade_buyAt: 'Compra en:',\n upgrade_buyUrl: 'https://buy.polar.sh/polar_cl_yQsiUeDelyyEQznbWffD1j77JAyP24ra7iEVQ22PA4h',\n upgrade_buyUrlTeam: 'https://buy.polar.sh/polar_cl_CO6xqSzKgFiQJwXnhZYGqisOP04Wspi0KKZSn38NjFZ?quantity=3',\n upgrade_activateWith: 'Luego activa con:',\n upgrade_activateCmd: 'brew-tui activate <tu-clave-de-licencia>',\n upgrade_proLabel: 'Brew-TUI Pro \\u2014 9,95\\u20AC/mes o 82\\u20AC/a\\u00F1o \\u2014 Incluye BrewBar para macOS',\n upgrade_teamLabel: 'Brew-TUI Team \\u2014 8\\u20AC/seat/mes \\u2014 Incluye todo Pro m\\u00E1s Compliance',\n\n // ── Progress Log ──\n progress_noOutput: 'Sin salida a\\u00FAn',\n\n // ── Search Input ──\n searchInput_placeholder: 'Escriba para filtrar...',\n\n // ── Profile Manager ──\n profileMgr_tapping: 'A\\u00F1adiendo tap {{name}}...',\n profileMgr_installing: 'Instalando {{name}}...',\n profileMgr_installingCask: 'Instalando cask {{name}}...',\n profileMgr_importDone: '\\u00A1Listo! {{count}} paquetes instalados.',\n\n // ── CLI ──\n cli_usageActivate: 'Uso: brew-tui activate <clave-de-licencia>',\n cli_activated: '\\u2714 Pro activado para {{email}}',\n cli_plan: ' Plan: {{plan}}',\n cli_expires: ' Expira: {{date}}',\n cli_activationFailed: '\\u2718 Activaci\\u00F3n fallida: {{error}}',\n cli_noLicense: 'No se encontr\\u00F3 licencia activa.',\n cli_deactivated: '\\u2714 Licencia desactivada.',\n cli_planFree: 'Plan: Gratis',\n cli_planPro: 'Plan: Pro',\n cli_planExpired: 'Plan: Expirada',\n cli_confirmDeactivate: '\\u00BFDesactivar tu licencia Pro en esta m\\u00E1quina? (s/N): ',\n cli_deactivateCancelled: 'Desactivaci\\u00F3n cancelada.',\n cli_upgradeHint: 'Ejecuta `brew-tui activate <clave>` para actualizar a Pro.',\n cli_revalidateHint: 'Ejecuta `brew-tui revalidate` para refrescar tu licencia actual.',\n cli_email: 'Email: {{email}}',\n cli_status: 'Estado: {{status}}',\n cli_revalidated: '\\u2714 Licencia revalidada.',\n cli_revalidateGrace: '\\u26A0 No se pudo contactar al servidor. Tu licencia actual sigue siendo usable dentro del periodo de gracia offline.',\n cli_revalidateFailed: '\\u2718 La revalidaci\\u00F3n de la licencia fall\\u00F3. Renueva tu suscripci\\u00F3n o activa una clave v\\u00E1lida.',\n cli_rateLimited: 'Demasiados intentos de activaci\\u00F3n. Int\\u00E9ntalo en {{minutes}} minutos.',\n cli_cooldown: 'Por favor espera antes de intentar de nuevo.',\n cli_brewbarInstalling: 'Descargando BrewBar...',\n cli_brewbarInstalled: '\\u2714 BrewBar instalado en /Applications/BrewBar.app',\n cli_brewbarAlreadyInstalled: 'BrewBar ya est\\u00E1 instalado. Usa --force para reinstalar.',\n cli_brewbarUninstalled: '\\u2714 BrewBar eliminado de /Applications.',\n cli_brewbarNotInstalled: 'BrewBar no est\\u00E1 instalado.',\n cli_brewbarProRequired: '\\u2718 BrewBar requiere una licencia Pro.\\n Ejecuta: brew-tui activate <clave>',\n cli_brewbarRevalidateRequired: '\\u2718 BrewBar requiere una licencia Pro v\\u00E1lida.\\n Ejecuta: brew-tui revalidate',\n cli_brewbarMacOnly: '\\u2718 BrewBar solo est\\u00E1 disponible en macOS.',\n cli_brewbarDownloadFailed: '\\u2718 Error al descargar BrewBar: {{error}}',\n cli_brewbarAutoFailed: '\\u26A0 No se pudo lanzar BrewBar automáticamente: {{error}}',\n cli_deactivateRemoteFailed: '\\u26A0 Advertencia: No se pudo contactar al servidor para desactivar remotamente. La licencia se elimin\\u00F3 localmente pero puede seguir contando como activa.',\n\n // ── License degradation (Layer 15) ──\n license_offlineWarning: 'Tu licencia no se ha validado en {{days}} d\\u00EDas. Por favor con\\u00E9ctate a internet.',\n\n // ── Plurals ──\n plural_vulns_one: '({{count}} vuln)',\n plural_vulns_other: '({{count}} vulns)',\n plural_warnings_one: '{{count}} advertencia',\n plural_warnings_other: '{{count}} advertencias',\n\n // ── Scroll indicators ──\n scroll_moreAbove: '\\u2191 {{count}} m\\u00E1s',\n scroll_moreBelow: '\\u2193 {{count}} m\\u00E1s',\n\n // ── SCR-001: Cleanup warning ──\n cleanup_warning_system_tools: 'Advertencia: los hu\\u00E9rfanos detectados pueden incluir dependencias de herramientas no gestionadas por Homebrew. Revisa la lista antes de continuar.',\n\n // ── SCR-002: Installed column headers ──\n installed_col_package: 'Paquete',\n installed_col_version: 'Versi\\u00F3n',\n installed_col_status: 'Estado',\n\n // ── SCR-003: Search failed ──\n search_failed: 'B\\u00FAsqueda fallida',\n\n // ── SCR-007: Deactivate failed ──\n deactivate_failed: 'Error al desactivar',\n\n // ── ACC-005: Version labels ──\n version_installed: 'instalado:',\n version_available: 'disponible:',\n\n // ── SCR-006: Upgrade-all replay warning ──\n upgrade_all_warning: 'Nota: esto actualizar\\u00E1 todos los paquetes desactualizados actualmente, que pueden diferir del conjunto original.',\n\n // ── SEG-007: Delete account ──\n delete_account_confirm: '\\u00BFEliminar todos los datos de Brew-TUI (~/.brew-tui)? Esto elimina tu licencia, perfiles e historial. Esta acci\\u00F3n no se puede deshacer.',\n delete_account_success: 'Todos los datos de Brew-TUI han sido eliminados.',\n\n // ── SCR-012: Upgrade All packages list ──\n outdated_upgradeAllList: 'Paquetes a actualizar: {{list}}',\n\n // ── SCR-005: Profile import summary ──\n profiles_importSummary: 'Este perfil contiene {{formulae}} formulae y {{casks}} casks. \\u00BFContinuar?',\n\n // ── SCR-017: Network error ──\n security_networkError: 'No se pudo conectar con la base de datos de vulnerabilidades OSV.dev. Verifica tu conexi\\u00F3n a internet.',\n\n // ── ARQ-004: Dashboard last updated ──\n dashboard_lastUpdated: '\\u00DAltima actualizaci\\u00F3n: {{time}}',\n\n // ── SCR-014: Services last error ──\n services_lastError: '\\u00DAltimo error: {{error}}',\n\n // ── SCR-010: Generic network error ──\n error_network: 'Error de red: no se puede conectar con el servidor.',\n\n // ── ARQ-005: Security cache ──\n security_rollback_hint: 'R: abrir Rollback para restaurar versiones',\n security_cachedResults: 'Mostrando resultados en cach\\u00E9 (hace {{time}}). Presiona r para re-escanear.',\n\n // ── Impact Analysis ──\n impact_brewfile_hint: '\\u{1F4A1} Añade al Brewfile para fijar esta versión antes de actualizar',\n impact_analyzing: 'Analizando impacto de actualizaci\\u00F3n...',\n impact_high: 'RIESGO ALTO',\n impact_medium: 'RIESGO MEDIO',\n impact_low: 'RIESGO BAJO',\n impact_affects: 'afecta {{count}} paquetes instalados',\n impact_usedBy: 'Usado por: {{packages}}',\n impact_hint: 'Selecciona un paquete para ver el impacto',\n impact_reason_critical_package: 'Paquete cr\\u00EDtico del sistema',\n impact_reason_major_bump: 'Cambio de versi\\u00F3n mayor',\n impact_reason_many_deps: '{{count}} paquetes dependen de este',\n\n // ── Rollback ──\n rollback_title: 'Rollback \\u2014 Restaurar Estado Anterior',\n rollback_no_snapshots: 'No hay snapshots disponibles. Los snapshots se capturan autom\\u00E1ticamente tras cada operaci\\u00F3n.',\n rollback_select_snapshot: 'Selecciona un snapshot para restaurar',\n rollback_snapshot_label: '{{label}} \\u2014 {{date}}',\n rollback_snapshot_auto: 'Auto',\n rollback_diff_empty: 'No se detectaron cambios entre este snapshot y el estado actual',\n rollback_confirm: '\\u00BFRevertir {{count}} paquete(s) a este estado?',\n rollback_strategy_bottle: 'desde cach\\u00E9 de bottle',\n rollback_strategy_versioned: 'desde formula versionada',\n rollback_strategy_pin: 'solo fijar (versi\\u00F3n no restaurable)',\n rollback_strategy_unavailable: 'no se puede restaurar',\n rollback_executing: 'Revirtiendo...',\n rollback_success: 'Rollback completado',\n rollback_error: 'Rollback fallido: {{error}}',\n rollback_item_downgrade: '{{name}}: {{from}} \\u2192 {{to}}',\n rollback_item_remove: 'Eliminar: {{name}}',\n rollback_item_install: 'Instalar: {{name}} {{version}}',\n rollback_warning_cask: 'Los Casks solo se fijar\\u00E1n (restauraci\\u00F3n de versi\\u00F3n no disponible)',\n rollback_capturing: 'Capturando snapshot actual...',\n\n // ── Sync ──\n sync_title: 'Sincronización entre máquinas',\n sync_disabled: 'Sync no está configurado. Pulsa s para configurar iCloud sync.',\n sync_status_ok: 'Sincronizado',\n sync_status_drift: '{{count}} cambio(s) desde otras máquinas',\n sync_status_conflict: '{{count}} conflicto(s) requieren resolución',\n sync_last_sync: 'Última sync: {{date}}',\n sync_machine: 'Esta máquina: {{name}}',\n sync_other_machines: 'Otras máquinas: {{names}}',\n sync_syncing: 'Sincronizando...',\n sync_success: 'Sincronización completa',\n sync_error: 'Sync fallida: {{error}}',\n sync_conflict_title: 'Conflicto: {{package}}',\n sync_conflict_local: 'Esta máquina: {{version}}',\n sync_conflict_remote: '{{machine}}: {{version}}',\n sync_conflict_use_local: 'Mantener local',\n sync_conflict_use_remote: 'Usar remoto',\n sync_setup_icloud: 'Configurando iCloud sync...',\n sync_no_icloud: 'iCloud Drive no encontrado. Asegúrate de que iCloud Drive está activado en Ajustes del Sistema.',\n\n // ── Upgrade Prompt — Sync ──\n upgrade_sync: 'Sincronización entre máquinas',\n upgrade_syncDesc: 'Mantén tu configuración de Homebrew sincronizada entre varios Macs mediante iCloud Drive. Fusiona instalaciones automáticamente y resuelve conflictos.',\n\n // ── Brewfile ──\n brewfile_title: 'Brewfile Declarativo',\n brewfile_compliant: 'en conformidad',\n brewfile_no_brewfile: 'No se encontró Brewfile. Pulsa n para crear uno desde tu instalación actual.',\n brewfile_create_name: 'Nombre del Brewfile (Enter para confirmar):',\n brewfile_created: 'Brewfile creado: {{name}}',\n brewfile_drift_missing: '{{count}} paquetes faltantes',\n brewfile_drift_extra: '{{count}} paquetes extra',\n brewfile_drift_wrong: '{{count}} versiones incorrectas',\n brewfile_reconciling: 'Reconciliando...',\n brewfile_reconcile_success: 'Reconciliación completa',\n brewfile_reconcile_error: 'Reconciliación fallida: {{error}}',\n brewfile_exported: 'Exportado a {{path}}',\n brewfile_formulae_count: '{{count}} formulae',\n brewfile_casks_count: '{{count}} casks',\n brewfile_strict_mode: 'Modo estricto',\n\n // ── Compliance ──\n view_compliance: 'Compliance',\n compliance_title: 'Conformidad de Equipo',\n compliance_no_policy: 'No hay política cargada. Pulsa i para importar un archivo de política.',\n compliance_score: '{{score}}% en conformidad',\n compliance_violations: '{{count}} violación(es)',\n compliance_ok: 'Totalmente en conformidad',\n compliance_import_prompt: 'Ruta del archivo de política (Enter para confirmar):',\n compliance_import_error: 'Error al cargar la política: {{error}}',\n compliance_violation_missing: 'Faltante: {{name}} (requerido)',\n compliance_violation_forbidden: 'Prohibido: {{name}} — {{reason}}',\n compliance_violation_version: 'Versión incorrecta: {{name}} (requerida {{required}}, instalada {{installed}})',\n compliance_violation_extra: 'Paquete extra: {{name}}',\n compliance_remediating: 'Remediando...',\n compliance_remediate_success: 'Remediación completa',\n compliance_remediate_error: 'Remediación fallida: {{error}}',\n compliance_export_done: 'Informe exportado a {{path}}',\n compliance_machine: 'Máquina: {{name}}',\n compliance_policy_name: 'Política: {{name}}',\n compliance_policy_by: 'Mantenida por: {{maintainer}}',\n\n // ── Upgrade Prompt — Compliance ──\n upgrade_compliance: 'Conformidad de Equipo',\n upgrade_complianceDesc: 'Aplica políticas de paquetes en todo tu equipo. Define paquetes requeridos, prohibidos y con versión fijada, y remedia desviaciones automáticamente.',\n};\n\nexport default es;\n","import { logger } from '../utils/logger.js';\n\nexport function fetchWithTimeout(url: string, options: RequestInit = {}, timeoutMs = 15_000): Promise<Response> {\n return fetch(url, { ...options, signal: AbortSignal.timeout(timeoutMs) });\n}\n\n/**\n * Wrap an async function with debug-level latency logging.\n */\nexport function timed<T>(label: string, fn: () => Promise<T>): Promise<T> {\n const start = Date.now();\n return fn().finally(() => logger.debug(`${label} took ${Date.now() - start}ms`));\n}\n"],"mappings":";AAAA,SAAS,cAAc;;;ACAvB,IAAM,KAAK;AAAA;AAAA,EAET,WAAW;AAAA,EACX,WAAW;AAAA,EACX,aAAa;AAAA;AAAA,EAGb,gBAAgB;AAAA,EAChB,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,eAAe;AAAA,EACf,aAAa;AAAA,EACb,eAAe;AAAA,EACf,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,eAAe;AAAA,EACf,eAAe;AAAA,EACf,WAAW;AAAA,EACX,oBAAoB;AAAA,EACpB,cAAc;AAAA;AAAA,EAGd,eAAe;AAAA,EACf,WAAW;AAAA,EACX,WAAW;AAAA,EACX,aAAa;AAAA,EACb,WAAW;AAAA,EACX,aAAa;AAAA,EACb,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,cAAc;AAAA,EACd,cAAc;AAAA,EACd,UAAU;AAAA,EACV,cAAc;AAAA,EACd,aAAa;AAAA,EACb,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,UAAU;AAAA,EACV,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,aAAa;AAAA,EACb,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,aAAa;AAAA,EACb,iBAAiB;AAAA,EACjB,YAAY;AAAA,EACZ,oBAAoB;AAAA,EACpB,WAAW;AAAA,EACX,aAAa;AAAA,EACb,WAAW;AAAA,EACX,UAAU;AAAA,EACV,uBAAuB;AAAA,EACvB,UAAU;AAAA,EACV,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,aAAa;AAAA,EACb,WAAW;AAAA,EACX,eAAe;AAAA,EACf,eAAe;AAAA,EACf,uBAAuB;AAAA;AAAA,EAGvB,iBAAiB;AAAA,EACjB,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,gBAAgB;AAAA,EAChB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA,EACnB,iBAAiB;AAAA;AAAA,EAGjB,aAAa;AAAA,EACb,YAAY;AAAA;AAAA,EAGZ,cAAc;AAAA;AAAA,EAGd,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,WAAW;AAAA;AAAA,EAGX,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,eAAe;AAAA,EACf,cAAc;AAAA,EACd,gBAAgB;AAAA;AAAA,EAGhB,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,eAAe;AAAA,EACf,WAAW;AAAA,EACX,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,aAAa;AAAA;AAAA,EAGb,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,oBAAoB;AAAA,EACpB,gBAAgB;AAAA,EAChB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,iCAAiC;AAAA,EACjC,yBAAyB;AAAA,EACzB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA,EACnB,4BAA4B;AAAA,EAC5B,yBAAyB;AAAA,EACzB,uBAAuB;AAAA,EACvB,qBAAqB;AAAA;AAAA,EAGrB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,4BAA4B;AAAA;AAAA,EAG5B,oBAAoB;AAAA,EACpB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,wBAAwB;AAAA,EACxB,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA;AAAA,EAGjB,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,0BAA0B;AAAA,EAC1B,uBAAuB;AAAA,EACvB,mBAAmB;AAAA,EACnB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,iBAAiB;AAAA;AAAA,EAGjB,mBAAmB;AAAA,EACnB,kBAAkB;AAAA,EAClB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,wBAAwB;AAAA,EACxB,0BAA0B;AAAA,EAC1B,wBAAwB;AAAA,EACxB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,aAAa;AAAA,EACb,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA;AAAA,EAGjB,gBAAgB;AAAA,EAChB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,eAAe;AAAA,EACf,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA;AAAA,EAGzB,cAAc;AAAA,EACd,cAAc;AAAA,EACd,4BAA4B;AAAA;AAAA,EAG5B,gBAAgB;AAAA,EAChB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,0BAA0B;AAAA,EAC1B,qBAAqB;AAAA,EACrB,0BAA0B;AAAA,EAC1B,kBAAkB;AAAA,EAClB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,gBAAgB;AAAA,EAChB,qBAAqB;AAAA,EACrB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA;AAAA,EAGnB,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,0BAA0B;AAAA,EAC1B,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,qBAAqB;AAAA;AAAA,EAGrB,eAAe;AAAA,EACf,qBAAqB;AAAA,EACrB,2BAA2B;AAAA,EAC3B,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,aAAa;AAAA,EACb,uBAAuB;AAAA,EACvB,yBAAyB;AAAA,EACzB,uBAAuB;AAAA,EACvB,0BAA0B;AAAA,EAC1B,uBAAuB;AAAA,EACvB,mBAAmB;AAAA;AAAA,EAGnB,wBAAwB;AAAA,EACxB,gBAAgB;AAAA,EAChB,kBAAkB;AAAA,EAClB,qBAAqB;AAAA,EACrB,mBAAmB;AAAA,EACnB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,yBAAyB;AAAA;AAAA,EAGzB,eAAe;AAAA,EACf,2BAA2B;AAAA,EAC3B,qBAAqB;AAAA,EACrB,aAAa;AAAA,EACb,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA;AAAA,EAGnB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,eAAe;AAAA,EACf,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA;AAAA,EAGnB,mBAAmB;AAAA;AAAA,EAGnB,yBAAyB;AAAA;AAAA,EAGzB,oBAAoB;AAAA,EACpB,uBAAuB;AAAA,EACvB,2BAA2B;AAAA,EAC3B,uBAAuB;AAAA;AAAA,EAGvB,mBAAmB;AAAA,EACnB,eAAe;AAAA,EACf,UAAU;AAAA,EACV,aAAa;AAAA,EACb,sBAAsB;AAAA,EACtB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,cAAc;AAAA,EACd,aAAa;AAAA,EACb,iBAAiB;AAAA,EACjB,uBAAuB;AAAA,EACvB,yBAAyB;AAAA,EACzB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,cAAc;AAAA,EACd,uBAAuB;AAAA,EACvB,sBAAsB;AAAA,EACtB,6BAA6B;AAAA,EAC7B,wBAAwB;AAAA,EACxB,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,+BAA+B;AAAA,EAC/B,oBAAoB;AAAA,EACpB,2BAA2B;AAAA,EAC3B,uBAAuB;AAAA,EACvB,4BAA4B;AAAA;AAAA,EAG5B,wBAAwB;AAAA;AAAA,EAGxB,kBAAkB;AAAA,EAClB,oBAAoB;AAAA,EACpB,qBAAqB;AAAA,EACrB,uBAAuB;AAAA;AAAA,EAGvB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA;AAAA,EAGlB,8BAA8B;AAAA;AAAA,EAG9B,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,sBAAsB;AAAA;AAAA,EAGtB,eAAe;AAAA;AAAA,EAGf,mBAAmB;AAAA;AAAA,EAGnB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA;AAAA,EAGnB,qBAAqB;AAAA;AAAA,EAGrB,wBAAwB;AAAA,EACxB,wBAAwB;AAAA;AAAA,EAGxB,yBAAyB;AAAA;AAAA,EAGzB,wBAAwB;AAAA;AAAA,EAGxB,uBAAuB;AAAA;AAAA,EAGvB,uBAAuB;AAAA;AAAA,EAGvB,oBAAoB;AAAA;AAAA,EAGpB,eAAe;AAAA;AAAA,EAGf,wBAAwB;AAAA;AAAA,EAGxB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,aAAa;AAAA,EACb,eAAe;AAAA,EACf,YAAY;AAAA,EACZ,gBAAgB;AAAA,EAChB,eAAe;AAAA,EACf,aAAa;AAAA,EACb,gCAAgC;AAAA,EAChC,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA;AAAA,EAGzB,gBAAgB;AAAA,EAChB,uBAAuB;AAAA,EACvB,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,0BAA0B;AAAA,EAC1B,6BAA6B;AAAA,EAC7B,uBAAuB;AAAA,EACvB,+BAA+B;AAAA,EAC/B,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,gBAAgB;AAAA,EAChB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,oBAAoB;AAAA;AAAA,EAGpB,YAAY;AAAA,EACZ,eAAe;AAAA,EACf,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,qBAAqB;AAAA,EACrB,cAAc;AAAA,EACd,cAAc;AAAA,EACd,YAAY;AAAA,EACZ,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,0BAA0B;AAAA,EAC1B,mBAAmB;AAAA,EACnB,gBAAgB;AAAA;AAAA,EAGhB,cAAc;AAAA,EACd,kBAAkB;AAAA;AAAA,EAGlB,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,4BAA4B;AAAA,EAC5B,0BAA0B;AAAA,EAC1B,mBAAmB;AAAA,EACnB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA;AAAA,EAGtB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,uBAAuB;AAAA,EACvB,eAAe;AAAA,EACf,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,8BAA8B;AAAA,EAC9B,gCAAgC;AAAA,EAChC,8BAA8B;AAAA,EAC9B,4BAA4B;AAAA,EAC5B,wBAAwB;AAAA,EACxB,8BAA8B;AAAA,EAC9B,4BAA4B;AAAA,EAC5B,wBAAwB;AAAA,EACxB,oBAAoB;AAAA,EACpB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA;AAAA,EAGtB,oBAAoB;AAAA,EACpB,wBAAwB;AAC1B;AAEA,IAAO,aAAQ;;;AC5gBf,IAAM,KAAmB;AAAA;AAAA,EAEvB,WAAW;AAAA,EACX,WAAW;AAAA,EACX,aAAa;AAAA;AAAA,EAGb,gBAAgB;AAAA,EAChB,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,eAAe;AAAA,EACf,aAAa;AAAA,EACb,eAAe;AAAA,EACf,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,eAAe;AAAA,EACf,eAAe;AAAA,EACf,WAAW;AAAA,EACX,oBAAoB;AAAA,EACpB,cAAc;AAAA;AAAA,EAGd,eAAe;AAAA,EACf,WAAW;AAAA,EACX,WAAW;AAAA,EACX,aAAa;AAAA,EACb,WAAW;AAAA,EACX,aAAa;AAAA,EACb,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,cAAc;AAAA,EACd,cAAc;AAAA,EACd,UAAU;AAAA,EACV,cAAc;AAAA,EACd,aAAa;AAAA,EACb,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,UAAU;AAAA,EACV,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,WAAW;AAAA,EACX,aAAa;AAAA,EACb,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,aAAa;AAAA,EACb,iBAAiB;AAAA,EACjB,YAAY;AAAA,EACZ,oBAAoB;AAAA,EACpB,WAAW;AAAA,EACX,aAAa;AAAA,EACb,WAAW;AAAA,EACX,UAAU;AAAA,EACV,uBAAuB;AAAA,EACvB,UAAU;AAAA,EACV,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,aAAa;AAAA,EACb,WAAW;AAAA,EACX,eAAe;AAAA,EACf,eAAe;AAAA,EACf,uBAAuB;AAAA;AAAA,EAGvB,iBAAiB;AAAA,EACjB,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,gBAAgB;AAAA,EAChB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA,EACnB,iBAAiB;AAAA;AAAA,EAGjB,aAAa;AAAA,EACb,YAAY;AAAA;AAAA,EAGZ,cAAc;AAAA;AAAA,EAGd,gBAAgB;AAAA,EAChB,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,WAAW;AAAA;AAAA,EAGX,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,eAAe;AAAA,EACf,cAAc;AAAA,EACd,gBAAgB;AAAA;AAAA,EAGhB,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,eAAe;AAAA,EACf,WAAW;AAAA,EACX,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,aAAa;AAAA;AAAA,EAGb,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,oBAAoB;AAAA,EACpB,gBAAgB;AAAA,EAChB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,iCAAiC;AAAA,EACjC,yBAAyB;AAAA,EACzB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA,EACnB,4BAA4B;AAAA,EAC5B,yBAAyB;AAAA,EACzB,uBAAuB;AAAA,EACvB,qBAAqB;AAAA;AAAA,EAGrB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,4BAA4B;AAAA;AAAA,EAG5B,oBAAoB;AAAA,EACpB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,wBAAwB;AAAA,EACxB,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA;AAAA,EAGjB,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,0BAA0B;AAAA,EAC1B,uBAAuB;AAAA,EACvB,mBAAmB;AAAA,EACnB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,iBAAiB;AAAA;AAAA,EAGjB,mBAAmB;AAAA,EACnB,kBAAkB;AAAA,EAClB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,cAAc;AAAA,EACd,wBAAwB;AAAA,EACxB,0BAA0B;AAAA,EAC1B,wBAAwB;AAAA,EACxB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,aAAa;AAAA,EACb,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA;AAAA,EAGjB,gBAAgB;AAAA,EAChB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,eAAe;AAAA,EACf,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA;AAAA,EAGzB,cAAc;AAAA,EACd,cAAc;AAAA,EACd,4BAA4B;AAAA;AAAA,EAG5B,gBAAgB;AAAA,EAChB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,0BAA0B;AAAA,EAC1B,qBAAqB;AAAA,EACrB,0BAA0B;AAAA,EAC1B,kBAAkB;AAAA,EAClB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,gBAAgB;AAAA,EAChB,qBAAqB;AAAA,EACrB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA;AAAA,EAGnB,eAAe;AAAA,EACf,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,0BAA0B;AAAA,EAC1B,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,qBAAqB;AAAA;AAAA,EAGrB,eAAe;AAAA,EACf,qBAAqB;AAAA,EACrB,2BAA2B;AAAA,EAC3B,sBAAsB;AAAA,EACtB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,aAAa;AAAA,EACb,uBAAuB;AAAA,EACvB,yBAAyB;AAAA,EACzB,uBAAuB;AAAA,EACvB,0BAA0B;AAAA,EAC1B,uBAAuB;AAAA,EACvB,mBAAmB;AAAA;AAAA,EAGnB,gBAAgB;AAAA,EAChB,kBAAkB;AAAA,EAClB,qBAAqB;AAAA,EACrB,mBAAmB;AAAA,EACnB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA,EAClB,yBAAyB;AAAA;AAAA,EAGzB,eAAe;AAAA,EACf,2BAA2B;AAAA,EAC3B,qBAAqB;AAAA,EACrB,aAAa;AAAA,EACb,cAAc;AAAA,EACd,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,oBAAoB;AAAA,EACpB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA;AAAA,EAGnB,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,eAAe;AAAA,EACf,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,mBAAmB;AAAA;AAAA,EAGnB,mBAAmB;AAAA;AAAA,EAGnB,yBAAyB;AAAA;AAAA,EAGzB,oBAAoB;AAAA,EACpB,uBAAuB;AAAA,EACvB,2BAA2B;AAAA,EAC3B,uBAAuB;AAAA;AAAA,EAGvB,mBAAmB;AAAA,EACnB,eAAe;AAAA,EACf,UAAU;AAAA,EACV,aAAa;AAAA,EACb,sBAAsB;AAAA,EACtB,eAAe;AAAA,EACf,iBAAiB;AAAA,EACjB,cAAc;AAAA,EACd,aAAa;AAAA,EACb,iBAAiB;AAAA,EACjB,uBAAuB;AAAA,EACvB,yBAAyB;AAAA,EACzB,iBAAiB;AAAA,EACjB,oBAAoB;AAAA,EACpB,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,iBAAiB;AAAA,EACjB,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,iBAAiB;AAAA,EACjB,cAAc;AAAA,EACd,uBAAuB;AAAA,EACvB,sBAAsB;AAAA,EACtB,6BAA6B;AAAA,EAC7B,wBAAwB;AAAA,EACxB,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,+BAA+B;AAAA,EAC/B,oBAAoB;AAAA,EACpB,2BAA2B;AAAA,EAC3B,uBAAuB;AAAA,EACvB,4BAA4B;AAAA;AAAA,EAG5B,wBAAwB;AAAA;AAAA,EAGxB,kBAAkB;AAAA,EAClB,oBAAoB;AAAA,EACpB,qBAAqB;AAAA,EACrB,uBAAuB;AAAA;AAAA,EAGvB,kBAAkB;AAAA,EAClB,kBAAkB;AAAA;AAAA,EAGlB,8BAA8B;AAAA;AAAA,EAG9B,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,sBAAsB;AAAA;AAAA,EAGtB,eAAe;AAAA;AAAA,EAGf,mBAAmB;AAAA;AAAA,EAGnB,mBAAmB;AAAA,EACnB,mBAAmB;AAAA;AAAA,EAGnB,qBAAqB;AAAA;AAAA,EAGrB,wBAAwB;AAAA,EACxB,wBAAwB;AAAA;AAAA,EAGxB,yBAAyB;AAAA;AAAA,EAGzB,wBAAwB;AAAA;AAAA,EAGxB,uBAAuB;AAAA;AAAA,EAGvB,uBAAuB;AAAA;AAAA,EAGvB,oBAAoB;AAAA;AAAA,EAGpB,eAAe;AAAA;AAAA,EAGf,wBAAwB;AAAA,EACxB,wBAAwB;AAAA;AAAA,EAGxB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,aAAa;AAAA,EACb,eAAe;AAAA,EACf,YAAY;AAAA,EACZ,gBAAgB;AAAA,EAChB,eAAe;AAAA,EACf,aAAa;AAAA,EACb,gCAAgC;AAAA,EAChC,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA;AAAA,EAGzB,gBAAgB;AAAA,EAChB,uBAAuB;AAAA,EACvB,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,wBAAwB;AAAA,EACxB,qBAAqB;AAAA,EACrB,kBAAkB;AAAA,EAClB,0BAA0B;AAAA,EAC1B,6BAA6B;AAAA,EAC7B,uBAAuB;AAAA,EACvB,+BAA+B;AAAA,EAC/B,oBAAoB;AAAA,EACpB,kBAAkB;AAAA,EAClB,gBAAgB;AAAA,EAChB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,uBAAuB;AAAA,EACvB,uBAAuB;AAAA,EACvB,oBAAoB;AAAA;AAAA,EAGpB,YAAY;AAAA,EACZ,eAAe;AAAA,EACf,gBAAgB;AAAA,EAChB,mBAAmB;AAAA,EACnB,sBAAsB;AAAA,EACtB,gBAAgB;AAAA,EAChB,cAAc;AAAA,EACd,qBAAqB;AAAA,EACrB,cAAc;AAAA,EACd,cAAc;AAAA,EACd,YAAY;AAAA,EACZ,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,yBAAyB;AAAA,EACzB,0BAA0B;AAAA,EAC1B,mBAAmB;AAAA,EACnB,gBAAgB;AAAA;AAAA,EAGhB,cAAc;AAAA,EACd,kBAAkB;AAAA;AAAA,EAGlB,gBAAgB;AAAA,EAChB,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA,EACtB,4BAA4B;AAAA,EAC5B,0BAA0B;AAAA,EAC1B,mBAAmB;AAAA,EACnB,yBAAyB;AAAA,EACzB,sBAAsB;AAAA,EACtB,sBAAsB;AAAA;AAAA,EAGtB,iBAAiB;AAAA,EACjB,kBAAkB;AAAA,EAClB,sBAAsB;AAAA,EACtB,kBAAkB;AAAA,EAClB,uBAAuB;AAAA,EACvB,eAAe;AAAA,EACf,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,8BAA8B;AAAA,EAC9B,gCAAgC;AAAA,EAChC,8BAA8B;AAAA,EAC9B,4BAA4B;AAAA,EAC5B,wBAAwB;AAAA,EACxB,8BAA8B;AAAA,EAC9B,4BAA4B;AAAA,EAC5B,wBAAwB;AAAA,EACxB,oBAAoB;AAAA,EACpB,wBAAwB;AAAA,EACxB,sBAAsB;AAAA;AAAA,EAGtB,oBAAoB;AAAA,EACpB,wBAAwB;AAC1B;AAEA,IAAO,aAAQ;;;AFzgBf,IAAM,UAAwC,EAAE,gBAAI,eAAG;AAEvD,SAAS,SAAS,GAAwB;AACxC,SAAO,MAAM,QAAQ,MAAM;AAC7B;AAEA,SAAS,eAAuB;AAC9B,QAAM,OAAO,QAAQ,KAAK,KAAK,CAAC,MAAM,EAAE,WAAW,SAAS,CAAC;AAC7D,MAAI,MAAM;AACR,UAAM,OAAO,KAAK,MAAM,GAAG,EAAE,CAAC;AAC9B,QAAI,QAAQ,SAAS,IAAI,EAAG,QAAO;AAAA,EACrC;AACA,QAAM,MAAM,QAAQ,IAAI,QAAQ,QAAQ,IAAI,UAAU,QAAQ,IAAI,eAAe;AACjF,QAAM,SAAS,IAAI,MAAM,MAAM,EAAE,CAAC,KAAK;AACvC,MAAI,SAAS,MAAM,EAAG,QAAO;AAC7B,SAAO;AACT;AAQO,IAAM,iBAAiB,OAAoB,CAAC,SAAS;AAAA,EAC1D,QAAQ,aAAa;AAAA,EACrB,WAAW,CAAC,WAAW,IAAI,EAAE,OAAO,CAAC;AACvC,EAAE;AAEK,SAAS,YAAoB;AAClC,SAAO,eAAe,SAAS,EAAE;AACnC;AAGO,SAAS,EAAE,KAAqB,QAAkD;AACvF,QAAM,SAAS,eAAe,SAAS,EAAE;AACzC,MAAI,OAAe,QAAQ,MAAM,EAAE,GAAG,KAAK,QAAQ,IAAI,EAAE,GAAG,KAAK;AACjE,MAAI,QAAQ;AACV,eAAW,CAAC,GAAG,CAAC,KAAK,OAAO,QAAQ,MAAM,GAAG;AAC3C,aAAO,KAAK,WAAW,KAAK,CAAC,MAAM,OAAO,CAAC,CAAC;AAAA,IAC9C;AAAA,EACF;AACA,SAAO;AACT;AAGO,SAAS,GAAG,SAAiB,OAAe,QAAkD;AACnG,QAAM,SAAS,UAAU,IAAI,SAAS;AACtC,SAAO,EAAE,GAAG,OAAO,GAAG,MAAM,IAAsB,EAAE,OAAO,GAAG,OAAO,CAAC;AACxE;;;AGtDO,SAAS,iBAAiB,KAAa,UAAuB,CAAC,GAAG,YAAY,MAA2B;AAC9G,SAAO,MAAM,KAAK,EAAE,GAAG,SAAS,QAAQ,YAAY,QAAQ,SAAS,EAAE,CAAC;AAC1E;","names":[]}
|
package/build/index.js
CHANGED
|
@@ -45,7 +45,7 @@ import {
|
|
|
45
45
|
t,
|
|
46
46
|
tp,
|
|
47
47
|
useLocaleStore
|
|
48
|
-
} from "./chunk-
|
|
48
|
+
} from "./chunk-3BPHXV35.js";
|
|
49
49
|
import {
|
|
50
50
|
logger
|
|
51
51
|
} from "./chunk-KDHEUNRI.js";
|
|
@@ -719,6 +719,10 @@ function validateLicenseKey(key) {
|
|
|
719
719
|
throw new Error("Invalid license key format");
|
|
720
720
|
}
|
|
721
721
|
}
|
|
722
|
+
function detectPlan(key) {
|
|
723
|
+
const upper = key.toUpperCase();
|
|
724
|
+
return upper.startsWith("BTUI-T-") || upper.startsWith("BTUI-T_") ? "team" : "pro";
|
|
725
|
+
}
|
|
722
726
|
async function activate(key) {
|
|
723
727
|
validateLicenseKey(key);
|
|
724
728
|
checkRateLimit();
|
|
@@ -734,7 +738,7 @@ async function activate(key) {
|
|
|
734
738
|
status: "active",
|
|
735
739
|
customerEmail: res.meta.customer_email,
|
|
736
740
|
customerName: res.meta.customer_name,
|
|
737
|
-
plan:
|
|
741
|
+
plan: detectPlan(key),
|
|
738
742
|
activatedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
739
743
|
expiresAt: res.license_key.expires_at,
|
|
740
744
|
lastValidatedAt: (/* @__PURE__ */ new Date()).toISOString()
|
|
@@ -847,7 +851,7 @@ var useLicenseStore = create2((set, get) => ({
|
|
|
847
851
|
set({ status: "expired", license, degradation: "expired" });
|
|
848
852
|
return;
|
|
849
853
|
}
|
|
850
|
-
set({ status:
|
|
854
|
+
set({ status: license.plan, license, degradation: level });
|
|
851
855
|
if (needsRevalidation(license)) {
|
|
852
856
|
if (!_revalidatingPromise) {
|
|
853
857
|
_revalidatingPromise = doRevalidation(license, set).finally(() => {
|
|
@@ -859,7 +863,8 @@ var useLicenseStore = create2((set, get) => ({
|
|
|
859
863
|
if (_revalidationInterval) clearInterval(_revalidationInterval);
|
|
860
864
|
_revalidationInterval = setInterval(() => {
|
|
861
865
|
const current = get().license;
|
|
862
|
-
|
|
866
|
+
const status = get().status;
|
|
867
|
+
if (!current || status !== "pro" && status !== "team") return;
|
|
863
868
|
if (!needsRevalidation(current)) return;
|
|
864
869
|
if (_revalidatingPromise) return;
|
|
865
870
|
_revalidatingPromise = doRevalidation(current, set).finally(() => {
|
|
@@ -872,7 +877,7 @@ var useLicenseStore = create2((set, get) => ({
|
|
|
872
877
|
set({ error: null });
|
|
873
878
|
try {
|
|
874
879
|
const license = await activate(key);
|
|
875
|
-
set({ status:
|
|
880
|
+
set({ status: license.plan, license, degradation: "none" });
|
|
876
881
|
return true;
|
|
877
882
|
} catch (err) {
|
|
878
883
|
const msg = err instanceof Error ? err.message : String(err);
|
|
@@ -891,15 +896,13 @@ var useLicenseStore = create2((set, get) => ({
|
|
|
891
896
|
}
|
|
892
897
|
set({ status: "free", license: null, degradation: "none", error: null });
|
|
893
898
|
},
|
|
894
|
-
// Team is a superset of Pro — team users have full Pro access plus team features
|
|
899
|
+
// Team is a superset of Pro — team users have full Pro access plus team features.
|
|
900
|
+
// Pro users do NOT get Team features (Compliance) without paying for the Team tier.
|
|
895
901
|
isPro: () => {
|
|
896
902
|
const s = get().status;
|
|
897
903
|
return s === "pro" || s === "team";
|
|
898
904
|
},
|
|
899
|
-
isTeam: () =>
|
|
900
|
-
const s = get().status;
|
|
901
|
-
return s === "team" || s === "pro";
|
|
902
|
-
}
|
|
905
|
+
isTeam: () => get().status === "team"
|
|
903
906
|
}));
|
|
904
907
|
|
|
905
908
|
// src/hooks/use-keyboard.ts
|
|
@@ -987,6 +990,11 @@ function UpgradePrompt({ viewId }) {
|
|
|
987
990
|
const keys = FEATURE_KEYS[viewId];
|
|
988
991
|
if (!keys) return null;
|
|
989
992
|
const title = t(keys.title);
|
|
993
|
+
const team = isTeamView(viewId);
|
|
994
|
+
const headerKey = team ? "upgrade_teamFeature" : "upgrade_proFeature";
|
|
995
|
+
const pricingKey = team ? "upgrade_teamPricing" : "upgrade_pricing";
|
|
996
|
+
const buyUrlKey = team ? "upgrade_buyUrlTeam" : "upgrade_buyUrl";
|
|
997
|
+
const labelKey = team ? "upgrade_teamLabel" : "upgrade_proLabel";
|
|
990
998
|
return /* @__PURE__ */ jsx5(Box4, { flexDirection: "column", alignItems: "center", paddingY: 2, children: /* @__PURE__ */ jsxs4(
|
|
991
999
|
Box4,
|
|
992
1000
|
{
|
|
@@ -1001,18 +1009,18 @@ function UpgradePrompt({ viewId }) {
|
|
|
1001
1009
|
/* @__PURE__ */ jsxs4(Text4, { bold: true, color: COLORS.brand, children: [
|
|
1002
1010
|
"\u2B50",
|
|
1003
1011
|
" ",
|
|
1004
|
-
t(
|
|
1012
|
+
t(headerKey, { title })
|
|
1005
1013
|
] }),
|
|
1006
1014
|
/* @__PURE__ */ jsx5(Text4, { children: " " }),
|
|
1007
1015
|
/* @__PURE__ */ jsx5(Text4, { color: COLORS.text, wrap: "wrap", children: t(keys.desc) }),
|
|
1008
1016
|
/* @__PURE__ */ jsx5(Text4, { children: " " }),
|
|
1009
1017
|
/* @__PURE__ */ jsxs4(Box4, { flexDirection: "column", alignItems: "center", children: [
|
|
1010
|
-
/* @__PURE__ */ jsx5(Text4, { color: COLORS.info, bold: true, children: t(
|
|
1018
|
+
/* @__PURE__ */ jsx5(Text4, { color: COLORS.info, bold: true, children: t(pricingKey) }),
|
|
1011
1019
|
/* @__PURE__ */ jsx5(Text4, { children: " " }),
|
|
1012
1020
|
/* @__PURE__ */ jsx5(Text4, { color: COLORS.muted, children: t("upgrade_buyAt") }),
|
|
1013
1021
|
/* @__PURE__ */ jsxs4(Text4, { color: COLORS.sky, bold: true, children: [
|
|
1014
1022
|
" ",
|
|
1015
|
-
t(
|
|
1023
|
+
t(buyUrlKey)
|
|
1016
1024
|
] }),
|
|
1017
1025
|
/* @__PURE__ */ jsx5(Text4, { children: " " }),
|
|
1018
1026
|
/* @__PURE__ */ jsx5(Text4, { color: COLORS.muted, children: t("upgrade_activateWith") }),
|
|
@@ -1021,7 +1029,7 @@ function UpgradePrompt({ viewId }) {
|
|
|
1021
1029
|
t("upgrade_activateCmd")
|
|
1022
1030
|
] }),
|
|
1023
1031
|
/* @__PURE__ */ jsx5(Text4, { children: " " }),
|
|
1024
|
-
/* @__PURE__ */ jsx5(Text4, { color: COLORS.brand, children: t(
|
|
1032
|
+
/* @__PURE__ */ jsx5(Text4, { color: COLORS.brand, children: t(labelKey) })
|
|
1025
1033
|
] })
|
|
1026
1034
|
]
|
|
1027
1035
|
}
|
|
@@ -4597,7 +4605,7 @@ function AccountView() {
|
|
|
4597
4605
|
/* @__PURE__ */ jsx30(Box28, { marginTop: 1, children: /* @__PURE__ */ jsxs29(Text29, { color: COLORS.textSecondary, children: [
|
|
4598
4606
|
status === "pro" ? `d ${t("hint_deactivate")}` : "",
|
|
4599
4607
|
" ",
|
|
4600
|
-
t("app_version", { version: "0.5.
|
|
4608
|
+
t("app_version", { version: "0.5.2" })
|
|
4601
4609
|
] }) })
|
|
4602
4610
|
] });
|
|
4603
4611
|
}
|
|
@@ -5981,7 +5989,7 @@ Snapshots: ${snapshots.length} (latest: ${latest ? formatDate(latest.capturedAt)
|
|
|
5981
5989
|
if (command === "install-brewbar") {
|
|
5982
5990
|
await useLicenseStore.getState().initialize();
|
|
5983
5991
|
const isPro = useLicenseStore.getState().isPro();
|
|
5984
|
-
const { installBrewBar } = await import("./brewbar-installer-
|
|
5992
|
+
const { installBrewBar } = await import("./brewbar-installer-VVZRCP5K.js");
|
|
5985
5993
|
try {
|
|
5986
5994
|
await installBrewBar(isPro, arg === "--force");
|
|
5987
5995
|
console.log(t("cli_brewbarInstalled"));
|
|
@@ -5992,7 +6000,7 @@ Snapshots: ${snapshots.length} (latest: ${latest ? formatDate(latest.capturedAt)
|
|
|
5992
6000
|
return;
|
|
5993
6001
|
}
|
|
5994
6002
|
if (command === "uninstall-brewbar") {
|
|
5995
|
-
const { uninstallBrewBar } = await import("./brewbar-installer-
|
|
6003
|
+
const { uninstallBrewBar } = await import("./brewbar-installer-VVZRCP5K.js");
|
|
5996
6004
|
try {
|
|
5997
6005
|
await uninstallBrewBar();
|
|
5998
6006
|
console.log(t("cli_brewbarUninstalled"));
|
|
@@ -6023,7 +6031,7 @@ async function ensureBrewBarRunning() {
|
|
|
6023
6031
|
if (process.platform !== "darwin") return;
|
|
6024
6032
|
await useLicenseStore.getState().initialize();
|
|
6025
6033
|
if (!useLicenseStore.getState().isPro()) return;
|
|
6026
|
-
const { isBrewBarInstalled, installBrewBar, launchBrewBar } = await import("./brewbar-installer-
|
|
6034
|
+
const { isBrewBarInstalled, installBrewBar, launchBrewBar } = await import("./brewbar-installer-VVZRCP5K.js");
|
|
6027
6035
|
try {
|
|
6028
6036
|
if (!await isBrewBarInstalled()) {
|
|
6029
6037
|
console.log(t("cli_brewbarInstalling"));
|