@fredlackey/devutils 0.0.13 → 0.0.15

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.
@@ -0,0 +1,588 @@
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * @fileoverview Install Comet Browser - Perplexity AI's AI-native web browser.
5
+ * @module installs/comet-browser
6
+ *
7
+ * Comet Browser is an AI-native web browser developed by Perplexity AI. Built on
8
+ * the Chromium engine, Comet integrates advanced AI capabilities directly into
9
+ * the browsing experience. Key features include:
10
+ * - Built-in AI assistant powered by GPT-4o, Claude, and Perplexity Sonar models
11
+ * - Agentic search capabilities for complex tasks (reservations, shopping, etc.)
12
+ * - Smart tab organization and multilingual content translation
13
+ * - Built-in ad blocker and Chrome extension support
14
+ *
15
+ * PLATFORM SUPPORT:
16
+ * - macOS: Official app via Homebrew cask (Apple Silicon and Intel supported)
17
+ * - Windows: Official app via winget from Perplexity
18
+ * - Ubuntu/Debian: NOT SUPPORTED (no Linux builds available)
19
+ * - Raspberry Pi OS: NOT SUPPORTED (no ARM Linux builds available)
20
+ * - Amazon Linux/RHEL: NOT SUPPORTED (no Linux builds available)
21
+ * - WSL: NOT SUPPORTED (install on Windows host instead)
22
+ * - Git Bash: Installs on Windows host via winget
23
+ *
24
+ * IMPORTANT: Comet Browser is currently only available for macOS and Windows.
25
+ * Linux support (including Ubuntu, Raspberry Pi OS, Amazon Linux, and WSL) is
26
+ * not yet available. Perplexity has indicated future support is planned but
27
+ * no release dates have been announced.
28
+ */
29
+
30
+ const os = require('../utils/common/os');
31
+ const shell = require('../utils/common/shell');
32
+ const brew = require('../utils/macos/brew');
33
+ const macosApps = require('../utils/macos/apps');
34
+ const winget = require('../utils/windows/winget');
35
+ const fs = require('fs');
36
+
37
+ /**
38
+ * Indicates whether this installer requires a desktop environment.
39
+ * Comet Browser is a GUI web browser and requires a display.
40
+ * @type {boolean}
41
+ */
42
+ const REQUIRES_DESKTOP = true;
43
+
44
+ /**
45
+ * The Homebrew cask name for Comet Browser on macOS.
46
+ * Using the cask (not formula) because Comet is a GUI application.
47
+ */
48
+ const HOMEBREW_CASK_NAME = 'comet';
49
+
50
+ /**
51
+ * The winget package ID for Comet Browser on Windows.
52
+ * This is the official Perplexity package in the winget repository.
53
+ */
54
+ const WINGET_PACKAGE_ID = 'Perplexity.Comet';
55
+
56
+ /**
57
+ * Path to Comet Browser application on macOS.
58
+ * Used to verify installation succeeded.
59
+ */
60
+ const MACOS_APP_PATH = '/Applications/Comet.app';
61
+
62
+ /**
63
+ * Path to Comet Browser executable on Windows (per-user installation).
64
+ * Comet typically installs to the user's LocalAppData folder.
65
+ */
66
+ const WINDOWS_COMET_PATH_USER = process.env.LOCALAPPDATA
67
+ ? `${process.env.LOCALAPPDATA}\\Comet\\Comet.exe`
68
+ : null;
69
+
70
+ /**
71
+ * Path to Comet Browser executable on Windows (system-wide installation).
72
+ * Used when installed with --scope machine flag.
73
+ */
74
+ const WINDOWS_COMET_PATH_SYSTEM = 'C:\\Program Files\\Comet\\Comet.exe';
75
+
76
+ /**
77
+ * Check if Comet Browser is installed on macOS by verifying the app bundle exists.
78
+ *
79
+ * On macOS, GUI applications are typically installed as .app bundles in /Applications.
80
+ * We check for the bundle's existence rather than relying on PATH because Comet
81
+ * is a GUI application that does not add itself to the shell PATH.
82
+ *
83
+ * @returns {boolean} True if Comet.app exists in /Applications, false otherwise
84
+ */
85
+ function isCometInstalledMacOS() {
86
+ return fs.existsSync(MACOS_APP_PATH);
87
+ }
88
+
89
+ /**
90
+ * Get the installed version of Comet Browser on macOS.
91
+ *
92
+ * Executes the Comet binary within the app bundle with --version flag.
93
+ * The output format is typically: "Comet 143.2.7499.37648"
94
+ *
95
+ * @returns {Promise<string|null>} Version string (e.g., "143.2.7499.37648") or null if not installed
96
+ */
97
+ async function getCometVersionMacOS() {
98
+ if (!isCometInstalledMacOS()) {
99
+ return null;
100
+ }
101
+
102
+ const cometPath = `${MACOS_APP_PATH}/Contents/MacOS/Comet`;
103
+ const result = await shell.exec(`"${cometPath}" --version`);
104
+
105
+ if (result.code === 0 && result.stdout) {
106
+ // Output format: "Comet 143.2.7499.37648"
107
+ const match = result.stdout.match(/Comet\s+([\d.]+)/);
108
+ return match ? match[1] : null;
109
+ }
110
+ return null;
111
+ }
112
+
113
+ /**
114
+ * Check if Comet Browser is installed on Windows.
115
+ *
116
+ * Checks for the Comet executable at the default installation paths.
117
+ * Comet is typically installed to LocalAppData for per-user installation
118
+ * or to Program Files for system-wide installation.
119
+ *
120
+ * @returns {boolean} True if Comet executable exists, false otherwise
121
+ */
122
+ function isCometInstalledWindows() {
123
+ // Check per-user installation path (most common)
124
+ if (WINDOWS_COMET_PATH_USER && fs.existsSync(WINDOWS_COMET_PATH_USER)) {
125
+ return true;
126
+ }
127
+
128
+ // Check system-wide installation path
129
+ if (fs.existsSync(WINDOWS_COMET_PATH_SYSTEM)) {
130
+ return true;
131
+ }
132
+
133
+ return false;
134
+ }
135
+
136
+ /**
137
+ * Get the installed version of Comet Browser on Windows.
138
+ *
139
+ * Uses winget to query the installed package version. This is more reliable
140
+ * than trying to execute the Comet binary with --version.
141
+ *
142
+ * @returns {Promise<string|null>} Version string or null if not installed
143
+ */
144
+ async function getCometVersionWindows() {
145
+ const version = await winget.getPackageVersion(WINGET_PACKAGE_ID);
146
+ return version;
147
+ }
148
+
149
+ /**
150
+ * Install Comet Browser on macOS using Homebrew.
151
+ *
152
+ * Prerequisites:
153
+ * - macOS 12 (Monterey) or later
154
+ * - Both Apple Silicon (M1/M2/M3) and Intel processors are supported
155
+ * - Homebrew package manager installed
156
+ * - Terminal access
157
+ *
158
+ * The installation uses the Homebrew cask 'comet' which downloads
159
+ * and installs Comet to /Applications/Comet.app.
160
+ *
161
+ * This function is idempotent - it checks if Comet is already installed
162
+ * before attempting installation and skips if already present.
163
+ *
164
+ * @returns {Promise<void>}
165
+ */
166
+ async function install_macos() {
167
+ console.log('Checking if Comet Browser is already installed...');
168
+
169
+ // Check if Comet is already installed via file system check
170
+ if (isCometInstalledMacOS()) {
171
+ const version = await getCometVersionMacOS();
172
+ if (version) {
173
+ console.log(`Comet Browser ${version} is already installed, skipping installation.`);
174
+ } else {
175
+ console.log('Comet Browser is already installed, skipping installation.');
176
+ }
177
+ return;
178
+ }
179
+
180
+ // Also check if the Homebrew cask is installed (Comet may be installed but not detected)
181
+ const caskInstalled = await brew.isCaskInstalled(HOMEBREW_CASK_NAME);
182
+ if (caskInstalled) {
183
+ console.log('Comet Browser is already installed via Homebrew, skipping installation.');
184
+ return;
185
+ }
186
+
187
+ // Verify Homebrew is available before proceeding
188
+ if (!brew.isInstalled()) {
189
+ console.log('Homebrew is not installed. Please install Homebrew first.');
190
+ console.log('Run: dev install homebrew');
191
+ return;
192
+ }
193
+
194
+ console.log('Installing Comet Browser via Homebrew...');
195
+
196
+ // Install Comet Browser cask
197
+ // The --cask flag specifies a graphical application
198
+ const result = await brew.installCask(HOMEBREW_CASK_NAME);
199
+
200
+ if (!result.success) {
201
+ console.log('Failed to install Comet Browser via Homebrew.');
202
+ console.log(result.output);
203
+ console.log('');
204
+ console.log('Troubleshooting:');
205
+ console.log(' 1. Run "brew update && brew cleanup" and retry');
206
+ console.log(' 2. Try manual installation: brew reinstall --cask comet');
207
+ console.log(' 3. Check if macOS Gatekeeper is blocking the app:');
208
+ console.log(' xattr -cr /Applications/Comet.app');
209
+ return;
210
+ }
211
+
212
+ // Verify the installation succeeded
213
+ if (!isCometInstalledMacOS()) {
214
+ console.log('Installation may have failed: Comet Browser was not found.');
215
+ console.log('Please check /Applications folder for Comet.app');
216
+ return;
217
+ }
218
+
219
+ const installedVersion = await getCometVersionMacOS();
220
+ console.log(`Comet Browser ${installedVersion || ''} installed successfully.`);
221
+ console.log('');
222
+ console.log('You can launch Comet from Applications or run:');
223
+ console.log(' open -a "Comet"');
224
+ console.log('');
225
+ console.log('NOTE: On first launch, you may need to sign in to your Perplexity account');
226
+ console.log('to access all AI features.');
227
+ }
228
+
229
+ /**
230
+ * Install Comet Browser on Ubuntu/Debian.
231
+ *
232
+ * IMPORTANT: Comet Browser does NOT support Linux. Perplexity has not released
233
+ * Linux builds for Comet Browser. This limitation applies to all Linux
234
+ * distributions, including Ubuntu, Debian, and their derivatives.
235
+ *
236
+ * This function returns gracefully with an informational message per the
237
+ * project's policy of not throwing errors for unsupported platforms.
238
+ *
239
+ * @returns {Promise<void>}
240
+ */
241
+ async function install_ubuntu() {
242
+ console.log('Comet Browser is not available for Ubuntu.');
243
+ return;
244
+ }
245
+
246
+ /**
247
+ * Install Comet Browser on Ubuntu running in WSL (Windows Subsystem for Linux).
248
+ *
249
+ * IMPORTANT: Comet Browser does NOT support Linux, including WSL environments.
250
+ * Since WSL runs a Linux environment, Comet Browser cannot be installed directly
251
+ * within WSL.
252
+ *
253
+ * For WSL users who want to use Comet Browser, the recommended approach is to
254
+ * install it on the Windows host using the Windows installation method.
255
+ *
256
+ * This function returns gracefully with an informational message.
257
+ *
258
+ * @returns {Promise<void>}
259
+ */
260
+ async function install_ubuntu_wsl() {
261
+ console.log('Comet Browser is not available for WSL.');
262
+ return;
263
+ }
264
+
265
+ /**
266
+ * Install Comet Browser on Raspberry Pi OS.
267
+ *
268
+ * IMPORTANT: Comet Browser does NOT support ARM architecture or Linux.
269
+ * Perplexity only provides builds for x86/x64 Windows and macOS (both Intel
270
+ * and Apple Silicon). This means Comet Browser cannot be installed on any
271
+ * Raspberry Pi device.
272
+ *
273
+ * This function returns gracefully with an informational message.
274
+ *
275
+ * @returns {Promise<void>}
276
+ */
277
+ async function install_raspbian() {
278
+ console.log('Comet Browser is not available for Raspberry Pi OS.');
279
+ return;
280
+ }
281
+
282
+ /**
283
+ * Install Comet Browser on Amazon Linux/RHEL.
284
+ *
285
+ * IMPORTANT: Comet Browser does NOT support Linux. Amazon Linux, RHEL,
286
+ * Fedora, Rocky Linux, and AlmaLinux are all unsupported platforms.
287
+ *
288
+ * This function returns gracefully with an informational message.
289
+ *
290
+ * @returns {Promise<void>}
291
+ */
292
+ async function install_amazon_linux() {
293
+ console.log('Comet Browser is not available for Amazon Linux.');
294
+ return;
295
+ }
296
+
297
+ /**
298
+ * Install Comet Browser on Windows using winget.
299
+ *
300
+ * Prerequisites:
301
+ * - Windows 10 version 1809 or later, or Windows 11
302
+ * - winget package manager (included in Windows 10 2004+ and Windows 11)
303
+ * - Administrator privileges may be required for system-wide installation
304
+ *
305
+ * The winget package 'Perplexity.Comet' installs the official Comet Browser
306
+ * from Perplexity. This function performs a per-user installation by default.
307
+ *
308
+ * This function is idempotent - it checks if Comet is already installed
309
+ * before attempting installation and skips if already present.
310
+ *
311
+ * @returns {Promise<void>}
312
+ */
313
+ async function install_windows() {
314
+ console.log('Checking if Comet Browser is already installed...');
315
+
316
+ // Check if Comet is already installed via file system check
317
+ if (isCometInstalledWindows()) {
318
+ const version = await getCometVersionWindows();
319
+ if (version) {
320
+ console.log(`Comet Browser ${version} is already installed, skipping installation.`);
321
+ } else {
322
+ console.log('Comet Browser is already installed, skipping installation.');
323
+ }
324
+ return;
325
+ }
326
+
327
+ // Also check if winget has the package installed
328
+ const isWingetInstalled = await winget.isPackageInstalled(WINGET_PACKAGE_ID);
329
+ if (isWingetInstalled) {
330
+ console.log('Comet Browser is already installed via winget, skipping installation.');
331
+ return;
332
+ }
333
+
334
+ // Verify winget is available
335
+ if (!winget.isInstalled()) {
336
+ console.log('winget is not available. Please install App Installer from the Microsoft Store.');
337
+ console.log('');
338
+ console.log('To install App Installer:');
339
+ console.log(' 1. Open Microsoft Store');
340
+ console.log(' 2. Search for "App Installer"');
341
+ console.log(' 3. Install or update the app');
342
+ return;
343
+ }
344
+
345
+ console.log('Installing Comet Browser via winget...');
346
+ console.log('This may take a few minutes...');
347
+
348
+ // Install Comet Browser using winget
349
+ // The --silent flag suppresses the installer UI
350
+ // The --accept-* flags prevent license agreement prompts
351
+ const result = await winget.install(WINGET_PACKAGE_ID, {
352
+ silent: true,
353
+ source: 'winget' // Explicitly use winget source, not msstore
354
+ });
355
+
356
+ if (!result.success) {
357
+ console.log('Failed to install Comet Browser via winget.');
358
+ console.log(result.output);
359
+ console.log('');
360
+ console.log('Troubleshooting:');
361
+ console.log(' 1. Ensure you have a stable internet connection');
362
+ console.log(' 2. Try running as Administrator if installation fails');
363
+ console.log(' 3. Try manual installation:');
364
+ console.log(' winget install --id Perplexity.Comet --silent --accept-package-agreements --accept-source-agreements');
365
+ return;
366
+ }
367
+
368
+ // Verify the installation succeeded
369
+ // Note: winget installation may complete before files are fully written
370
+ // so we check both winget registry and file system
371
+ const verified = await winget.isPackageInstalled(WINGET_PACKAGE_ID);
372
+ if (!verified && !isCometInstalledWindows()) {
373
+ console.log('Installation may have failed: Comet Browser package not found after install.');
374
+ return;
375
+ }
376
+
377
+ const installedVersion = await getCometVersionWindows();
378
+ console.log(`Comet Browser ${installedVersion || ''} installed successfully.`);
379
+ console.log('');
380
+ console.log('Comet Browser is now available in your Start Menu.');
381
+ console.log('You may need to open a new terminal to use it from the command line.');
382
+ console.log('');
383
+ console.log('NOTE: On first launch, you may need to sign in to your Perplexity account');
384
+ console.log('to access all AI features.');
385
+ }
386
+
387
+ /**
388
+ * Install Comet Browser from Git Bash on Windows.
389
+ *
390
+ * Git Bash runs within the Windows environment and can access Windows
391
+ * executables including winget.exe. This function installs Comet Browser
392
+ * on the Windows host using winget via PowerShell interop.
393
+ *
394
+ * Prerequisites:
395
+ * - Windows 10 or Windows 11
396
+ * - Git Bash installed (comes with Git for Windows)
397
+ * - winget package manager available on Windows
398
+ *
399
+ * @returns {Promise<void>}
400
+ */
401
+ async function install_gitbash() {
402
+ console.log('Detected Git Bash on Windows.');
403
+ console.log('Installing Comet Browser on the Windows host...');
404
+ console.log('');
405
+
406
+ // Check if Comet is already installed by checking Windows paths
407
+ // Git Bash path format uses forward slashes with drive letter prefix
408
+ const userPath = '/c/Users/' + process.env.USERNAME + '/AppData/Local/Comet/Comet.exe';
409
+ const systemPath = '/c/Program Files/Comet/Comet.exe';
410
+
411
+ if (fs.existsSync(userPath) || fs.existsSync(systemPath)) {
412
+ console.log('Comet Browser is already installed, skipping installation.');
413
+ console.log('');
414
+ console.log('To launch Comet from Git Bash:');
415
+ console.log(' start "" "Comet"');
416
+ return;
417
+ }
418
+
419
+ // Try winget via PowerShell
420
+ console.log('Attempting installation via winget...');
421
+ const wingetResult = await shell.exec(
422
+ 'powershell.exe -NoProfile -Command "winget install --id Perplexity.Comet --silent --accept-package-agreements --accept-source-agreements"'
423
+ );
424
+
425
+ if (wingetResult.code === 0) {
426
+ console.log('Comet Browser installed successfully via winget.');
427
+ console.log('');
428
+ console.log('To launch Comet from Git Bash:');
429
+ console.log(' start "" "Comet"');
430
+ console.log('');
431
+ console.log('NOTE: On first launch, you may need to sign in to your Perplexity account');
432
+ console.log('to access all AI features.');
433
+ return;
434
+ }
435
+
436
+ // winget failed
437
+ console.log('Failed to install Comet Browser.');
438
+ console.log(wingetResult.stdout || wingetResult.stderr);
439
+ console.log('');
440
+ console.log('Troubleshooting:');
441
+ console.log(' 1. Ensure winget is available on your Windows system');
442
+ console.log(' 2. Try running Git Bash as Administrator and retry');
443
+ console.log(' 3. Try installing directly from PowerShell:');
444
+ console.log(' winget install --id Perplexity.Comet --silent --accept-package-agreements --accept-source-agreements');
445
+ }
446
+
447
+ /**
448
+ * Check if Comet Browser is currently installed on the system.
449
+ *
450
+ * This function checks for Comet Browser installation across all supported platforms:
451
+ * - macOS: Checks for Comet.app via Homebrew cask or application bundle
452
+ * - Windows: Checks for Comet.exe at standard installation paths
453
+ * - Git Bash: Checks Windows installation paths
454
+ * - Other platforms: Returns false (not supported)
455
+ *
456
+ * @returns {Promise<boolean>} True if Comet Browser is installed, false otherwise
457
+ */
458
+ async function isInstalled() {
459
+ const platform = os.detect();
460
+
461
+ // macOS: Check if Comet.app exists
462
+ if (platform.type === 'macos') {
463
+ // Check if Comet Browser app bundle exists
464
+ if (isCometInstalledMacOS()) {
465
+ return true;
466
+ }
467
+ // Also check via Homebrew cask
468
+ return await brew.isCaskInstalled(HOMEBREW_CASK_NAME);
469
+ }
470
+
471
+ // Windows: Check via winget or file system
472
+ if (platform.type === 'windows') {
473
+ // Check file system first (faster)
474
+ if (isCometInstalledWindows()) {
475
+ return true;
476
+ }
477
+ // Fall back to winget check
478
+ return await winget.isPackageInstalled(WINGET_PACKAGE_ID);
479
+ }
480
+
481
+ // Git Bash: Check Windows paths
482
+ if (platform.type === 'gitbash') {
483
+ const userPath = '/c/Users/' + process.env.USERNAME + '/AppData/Local/Comet/Comet.exe';
484
+ const systemPath = '/c/Program Files/Comet/Comet.exe';
485
+ return fs.existsSync(userPath) || fs.existsSync(systemPath);
486
+ }
487
+
488
+ // Other platforms (Linux, WSL, Raspberry Pi, Amazon Linux) are not supported
489
+ return false;
490
+ }
491
+
492
+ /**
493
+ * Check if this installer is supported on the current platform.
494
+ *
495
+ * Comet Browser is ONLY supported on macOS and Windows. All Linux platforms
496
+ * (including Ubuntu, Debian, Raspberry Pi OS, Amazon Linux, RHEL, Fedora,
497
+ * and WSL) are NOT supported because Perplexity does not provide Linux builds.
498
+ *
499
+ * @returns {boolean} True if installation is supported on this platform
500
+ */
501
+ function isEligible() {
502
+ const platform = os.detect();
503
+
504
+ // Only macOS and Windows (including Git Bash) are supported
505
+ const supportedPlatforms = ['macos', 'windows', 'gitbash'];
506
+ if (!supportedPlatforms.includes(platform.type)) {
507
+ return false;
508
+ }
509
+
510
+ // This installer requires a desktop environment
511
+ if (REQUIRES_DESKTOP && !os.isDesktopAvailable()) {
512
+ return false;
513
+ }
514
+
515
+ return true;
516
+ }
517
+
518
+ /**
519
+ * Main installation entry point.
520
+ *
521
+ * Detects the current platform using os.detect() and routes to the appropriate
522
+ * platform-specific installer function. For unsupported platforms, returns
523
+ * gracefully with an informational message (no errors thrown).
524
+ *
525
+ * Supported platforms:
526
+ * - macOS: Comet Browser via Homebrew cask
527
+ * - Windows: Comet Browser via winget
528
+ * - Git Bash: Comet Browser on Windows host
529
+ *
530
+ * Unsupported platforms (return gracefully):
531
+ * - Ubuntu/Debian
532
+ * - Raspberry Pi OS
533
+ * - Amazon Linux/RHEL/Fedora
534
+ * - WSL
535
+ *
536
+ * @returns {Promise<void>}
537
+ */
538
+ async function install() {
539
+ const platform = os.detect();
540
+
541
+ // Map platform types to their installer functions
542
+ // This mapping handles aliases (e.g., debian uses the same installer as ubuntu)
543
+ const installers = {
544
+ 'macos': install_macos,
545
+ 'ubuntu': install_ubuntu,
546
+ 'debian': install_ubuntu,
547
+ 'wsl': install_ubuntu_wsl,
548
+ 'raspbian': install_raspbian,
549
+ 'amazon_linux': install_amazon_linux,
550
+ 'rhel': install_amazon_linux,
551
+ 'fedora': install_amazon_linux,
552
+ 'windows': install_windows,
553
+ 'gitbash': install_gitbash
554
+ };
555
+
556
+ const installer = installers[platform.type];
557
+
558
+ if (!installer) {
559
+ // Gracefully handle unsupported platforms without throwing an error
560
+ console.log(`Comet Browser is not available for ${platform.type}.`);
561
+ return;
562
+ }
563
+
564
+ await installer();
565
+ }
566
+
567
+ // Export all functions for use as a module and for testing
568
+ module.exports = {
569
+ REQUIRES_DESKTOP,
570
+ install,
571
+ isInstalled,
572
+ isEligible,
573
+ install_macos,
574
+ install_ubuntu,
575
+ install_ubuntu_wsl,
576
+ install_raspbian,
577
+ install_amazon_linux,
578
+ install_windows,
579
+ install_gitbash
580
+ };
581
+
582
+ // Allow direct execution: node comet-browser.js
583
+ if (require.main === module) {
584
+ install().catch(err => {
585
+ console.error(err.message);
586
+ process.exit(1);
587
+ });
588
+ }