@fredlackey/devutils 0.0.1 → 0.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (259) hide show
  1. package/README.md +5 -5
  2. package/package.json +1 -1
  3. package/src/commands/install.js +374 -36
  4. package/src/installs/adobe-creative-cloud.js +527 -25
  5. package/src/installs/adobe-creative-cloud.md +605 -0
  6. package/src/installs/appcleaner.js +303 -26
  7. package/src/installs/appcleaner.md +699 -0
  8. package/src/installs/apt-transport-https.js +390 -0
  9. package/src/installs/apt-transport-https.md +678 -0
  10. package/src/installs/atomicparsley.js +624 -26
  11. package/src/installs/atomicparsley.md +795 -0
  12. package/src/installs/aws-cli.js +779 -26
  13. package/src/installs/aws-cli.md +727 -0
  14. package/src/installs/balena-etcher.js +688 -26
  15. package/src/installs/balena-etcher.md +761 -0
  16. package/src/installs/bambu-studio.js +912 -26
  17. package/src/installs/bambu-studio.md +780 -0
  18. package/src/installs/bash-completion.js +554 -23
  19. package/src/installs/bash-completion.md +833 -0
  20. package/src/installs/bash.js +399 -26
  21. package/src/installs/bash.md +993 -0
  22. package/src/installs/beyond-compare.js +585 -26
  23. package/src/installs/beyond-compare.md +813 -0
  24. package/src/installs/build-essential.js +511 -26
  25. package/src/installs/build-essential.md +977 -0
  26. package/src/installs/ca-certificates.js +618 -0
  27. package/src/installs/ca-certificates.md +937 -0
  28. package/src/installs/caffeine.js +490 -26
  29. package/src/installs/caffeine.md +839 -0
  30. package/src/installs/camtasia.js +577 -25
  31. package/src/installs/camtasia.md +762 -0
  32. package/src/installs/chatgpt.js +458 -26
  33. package/src/installs/chatgpt.md +814 -0
  34. package/src/installs/chocolatey.js +447 -0
  35. package/src/installs/chocolatey.md +661 -0
  36. package/src/installs/chrome-canary.js +472 -26
  37. package/src/installs/chrome-canary.md +641 -0
  38. package/src/installs/chromium.js +645 -26
  39. package/src/installs/chromium.md +838 -0
  40. package/src/installs/claude-code.js +558 -26
  41. package/src/installs/claude-code.md +1173 -0
  42. package/src/installs/curl.js +361 -26
  43. package/src/installs/curl.md +714 -0
  44. package/src/installs/cursor.js +561 -26
  45. package/src/installs/cursor.md +970 -0
  46. package/src/installs/dbschema.js +674 -26
  47. package/src/installs/dbschema.md +925 -0
  48. package/src/installs/dependencies.md +435 -0
  49. package/src/installs/development-tools.js +600 -0
  50. package/src/installs/development-tools.md +977 -0
  51. package/src/installs/docker.js +1010 -25
  52. package/src/installs/docker.md +1109 -0
  53. package/src/installs/drawio.js +1001 -26
  54. package/src/installs/drawio.md +795 -0
  55. package/src/installs/elmedia-player.js +328 -25
  56. package/src/installs/elmedia-player.md +556 -0
  57. package/src/installs/ffmpeg.js +870 -25
  58. package/src/installs/ffmpeg.md +852 -0
  59. package/src/installs/file.js +464 -0
  60. package/src/installs/file.md +987 -0
  61. package/src/installs/gemini-cli.js +793 -26
  62. package/src/installs/gemini-cli.md +1153 -0
  63. package/src/installs/git.js +382 -26
  64. package/src/installs/git.md +907 -0
  65. package/src/installs/gitego.js +931 -26
  66. package/src/installs/gitego.md +1172 -0
  67. package/src/installs/go.js +913 -26
  68. package/src/installs/go.md +958 -0
  69. package/src/installs/google-chrome.js +801 -25
  70. package/src/installs/google-chrome.md +862 -0
  71. package/src/installs/gpg.js +412 -73
  72. package/src/installs/gpg.md +1056 -0
  73. package/src/installs/homebrew.js +1015 -26
  74. package/src/installs/homebrew.md +988 -0
  75. package/src/installs/imageoptim.js +950 -26
  76. package/src/installs/imageoptim.md +1119 -0
  77. package/src/installs/installers.json +2297 -0
  78. package/src/installs/jq.js +382 -26
  79. package/src/installs/jq.md +809 -0
  80. package/src/installs/keyboard-maestro.js +701 -26
  81. package/src/installs/keyboard-maestro.md +825 -0
  82. package/src/installs/latex.js +771 -26
  83. package/src/installs/latex.md +1095 -0
  84. package/src/installs/lftp.js +338 -26
  85. package/src/installs/lftp.md +907 -0
  86. package/src/installs/lsb-release.js +346 -0
  87. package/src/installs/lsb-release.md +814 -0
  88. package/src/installs/messenger.js +829 -26
  89. package/src/installs/messenger.md +900 -0
  90. package/src/installs/microsoft-office.js +550 -26
  91. package/src/installs/microsoft-office.md +760 -0
  92. package/src/installs/microsoft-teams.js +782 -25
  93. package/src/installs/microsoft-teams.md +886 -0
  94. package/src/installs/node.js +886 -26
  95. package/src/installs/node.md +1153 -0
  96. package/src/installs/nordpass.js +698 -26
  97. package/src/installs/nordpass.md +921 -0
  98. package/src/installs/nvm.js +977 -26
  99. package/src/installs/nvm.md +1057 -0
  100. package/src/installs/openssh.js +734 -64
  101. package/src/installs/openssh.md +1056 -0
  102. package/src/installs/pandoc.js +644 -26
  103. package/src/installs/pandoc.md +1036 -0
  104. package/src/installs/pinentry.js +492 -26
  105. package/src/installs/pinentry.md +1142 -0
  106. package/src/installs/pngyu.js +851 -26
  107. package/src/installs/pngyu.md +896 -0
  108. package/src/installs/postman.js +781 -26
  109. package/src/installs/postman.md +940 -0
  110. package/src/installs/procps.js +425 -0
  111. package/src/installs/procps.md +851 -0
  112. package/src/installs/safari-tech-preview.js +355 -25
  113. package/src/installs/safari-tech-preview.md +533 -0
  114. package/src/installs/sfnt2woff.js +640 -26
  115. package/src/installs/sfnt2woff.md +795 -0
  116. package/src/installs/shellcheck.js +463 -26
  117. package/src/installs/shellcheck.md +1005 -0
  118. package/src/installs/slack.js +722 -25
  119. package/src/installs/slack.md +865 -0
  120. package/src/installs/snagit.js +566 -25
  121. package/src/installs/snagit.md +844 -0
  122. package/src/installs/software-properties-common.js +372 -0
  123. package/src/installs/software-properties-common.md +805 -0
  124. package/src/installs/spotify.js +858 -25
  125. package/src/installs/spotify.md +901 -0
  126. package/src/installs/studio-3t.js +803 -26
  127. package/src/installs/studio-3t.md +918 -0
  128. package/src/installs/sublime-text.js +780 -25
  129. package/src/installs/sublime-text.md +914 -0
  130. package/src/installs/superwhisper.js +687 -25
  131. package/src/installs/superwhisper.md +630 -0
  132. package/src/installs/tailscale.js +727 -26
  133. package/src/installs/tailscale.md +1100 -0
  134. package/src/installs/tar.js +389 -0
  135. package/src/installs/tar.md +946 -0
  136. package/src/installs/termius.js +780 -26
  137. package/src/installs/termius.md +844 -0
  138. package/src/installs/terraform.js +761 -26
  139. package/src/installs/terraform.md +899 -0
  140. package/src/installs/tidal.js +752 -25
  141. package/src/installs/tidal.md +864 -0
  142. package/src/installs/tmux.js +328 -26
  143. package/src/installs/tmux.md +1030 -0
  144. package/src/installs/tree.js +393 -26
  145. package/src/installs/tree.md +833 -0
  146. package/src/installs/unzip.js +460 -0
  147. package/src/installs/unzip.md +879 -0
  148. package/src/installs/vim.js +403 -26
  149. package/src/installs/vim.md +1040 -0
  150. package/src/installs/vlc.js +803 -26
  151. package/src/installs/vlc.md +927 -0
  152. package/src/installs/vscode.js +825 -26
  153. package/src/installs/vscode.md +1002 -0
  154. package/src/installs/wget.js +415 -0
  155. package/src/installs/wget.md +791 -0
  156. package/src/installs/whatsapp.js +710 -25
  157. package/src/installs/whatsapp.md +854 -0
  158. package/src/installs/winpty.js +352 -0
  159. package/src/installs/winpty.md +620 -0
  160. package/src/installs/woff2.js +535 -26
  161. package/src/installs/woff2.md +977 -0
  162. package/src/installs/wsl.js +572 -0
  163. package/src/installs/wsl.md +699 -0
  164. package/src/installs/xcode-clt.js +520 -0
  165. package/src/installs/xcode-clt.md +351 -0
  166. package/src/installs/xcode.js +542 -26
  167. package/src/installs/xcode.md +573 -0
  168. package/src/installs/yarn.js +806 -26
  169. package/src/installs/yarn.md +1074 -0
  170. package/src/installs/yq.js +636 -26
  171. package/src/installs/yq.md +944 -0
  172. package/src/installs/yt-dlp.js +683 -26
  173. package/src/installs/yt-dlp.md +946 -0
  174. package/src/installs/yum-utils.js +297 -0
  175. package/src/installs/yum-utils.md +648 -0
  176. package/src/installs/zoom.js +740 -25
  177. package/src/installs/zoom.md +884 -0
  178. package/src/scripts/README.md +567 -45
  179. package/src/scripts/STATUS.md +208 -0
  180. package/src/scripts/afk.js +395 -7
  181. package/src/scripts/backup-all.js +731 -9
  182. package/src/scripts/backup-source.js +711 -8
  183. package/src/scripts/brewd.js +373 -7
  184. package/src/scripts/brewi.js +505 -9
  185. package/src/scripts/brewr.js +512 -9
  186. package/src/scripts/brews.js +462 -9
  187. package/src/scripts/brewu.js +488 -7
  188. package/src/scripts/c.js +185 -7
  189. package/src/scripts/ccurl.js +325 -8
  190. package/src/scripts/certbot-crontab-init.js +488 -8
  191. package/src/scripts/certbot-init.js +641 -9
  192. package/src/scripts/ch.js +339 -7
  193. package/src/scripts/claude-danger.js +253 -8
  194. package/src/scripts/clean-dev.js +419 -8
  195. package/src/scripts/clear-dns-cache.js +525 -7
  196. package/src/scripts/clone.js +417 -7
  197. package/src/scripts/code-all.js +420 -7
  198. package/src/scripts/count-files.js +195 -8
  199. package/src/scripts/count-folders.js +195 -8
  200. package/src/scripts/count.js +248 -8
  201. package/src/scripts/d.js +203 -7
  202. package/src/scripts/datauri.js +373 -8
  203. package/src/scripts/delete-files.js +363 -7
  204. package/src/scripts/docker-clean.js +410 -8
  205. package/src/scripts/dp.js +426 -7
  206. package/src/scripts/e.js +375 -9
  207. package/src/scripts/empty-trash.js +497 -7
  208. package/src/scripts/evm.js +428 -9
  209. package/src/scripts/fetch-github-repos.js +441 -10
  210. package/src/scripts/get-channel.js +329 -8
  211. package/src/scripts/get-course.js +384 -11
  212. package/src/scripts/get-dependencies.js +290 -9
  213. package/src/scripts/get-folder.js +783 -10
  214. package/src/scripts/get-tunes.js +411 -10
  215. package/src/scripts/get-video.js +352 -9
  216. package/src/scripts/git-backup.js +561 -9
  217. package/src/scripts/git-clone.js +477 -9
  218. package/src/scripts/git-pup.js +303 -7
  219. package/src/scripts/git-push.js +380 -8
  220. package/src/scripts/h.js +607 -9
  221. package/src/scripts/hide-desktop-icons.js +483 -7
  222. package/src/scripts/hide-hidden-files.js +522 -7
  223. package/src/scripts/install-dependencies-from.js +440 -9
  224. package/src/scripts/ips.js +647 -10
  225. package/src/scripts/iso.js +354 -8
  226. package/src/scripts/killni.js +561 -7
  227. package/src/scripts/ll.js +451 -8
  228. package/src/scripts/local-ip.js +310 -8
  229. package/src/scripts/m.js +508 -8
  230. package/src/scripts/map.js +293 -8
  231. package/src/scripts/mkd.js +287 -7
  232. package/src/scripts/ncu-update-all.js +441 -8
  233. package/src/scripts/nginx-init.js +702 -12
  234. package/src/scripts/npmi.js +366 -7
  235. package/src/scripts/o.js +495 -8
  236. package/src/scripts/org-by-date.js +321 -7
  237. package/src/scripts/p.js +208 -7
  238. package/src/scripts/packages.js +313 -8
  239. package/src/scripts/path.js +209 -7
  240. package/src/scripts/ports.js +582 -8
  241. package/src/scripts/q.js +290 -8
  242. package/src/scripts/refresh-files.js +378 -10
  243. package/src/scripts/remove-smaller-files.js +500 -8
  244. package/src/scripts/rename-files-with-date.js +517 -9
  245. package/src/scripts/resize-image.js +523 -9
  246. package/src/scripts/rm-safe.js +653 -8
  247. package/src/scripts/s.js +525 -9
  248. package/src/scripts/set-git-public.js +349 -7
  249. package/src/scripts/show-desktop-icons.js +459 -7
  250. package/src/scripts/show-hidden-files.js +456 -7
  251. package/src/scripts/tpa.js +265 -8
  252. package/src/scripts/tpo.js +264 -7
  253. package/src/scripts/u.js +489 -7
  254. package/src/scripts/vpush.js +422 -8
  255. package/src/scripts/y.js +267 -7
  256. package/src/utils/common/os.js +94 -2
  257. package/src/utils/ubuntu/apt.js +13 -7
  258. package/src/utils/windows/choco.js +82 -26
  259. package/src/utils/windows/winget.js +89 -27
@@ -3,42 +3,818 @@
3
3
  /**
4
4
  * @fileoverview Install Google Chrome.
5
5
  * @module installs/google-chrome
6
+ *
7
+ * Google Chrome is a fast, secure, and free web browser built by Google.
8
+ * It is the most widely used web browser globally, featuring automatic updates,
9
+ * built-in PDF viewer, password manager, and powerful developer tools.
10
+ *
11
+ * This installer provides:
12
+ * - Google Chrome via Homebrew cask on macOS
13
+ * - Google Chrome via official .deb package on Ubuntu/Debian
14
+ * - Google Chrome via official .rpm package on Amazon Linux/RHEL
15
+ * - Google Chrome via Chocolatey or winget on Windows
16
+ * - Google Chrome within WSL (same as Ubuntu installation)
17
+ * - Google Chrome on Windows host from Git Bash
18
+ *
19
+ * IMPORTANT PLATFORM NOTES:
20
+ * - Raspberry Pi OS: Google Chrome does NOT support ARM architecture.
21
+ * Google only provides x86/x64 builds. The installer will display
22
+ * a graceful message for unsupported platforms.
23
+ * - ARM-based systems: Chrome is only available for x86_64 architecture.
6
24
  */
7
25
 
8
26
  const os = require('../utils/common/os');
27
+ const shell = require('../utils/common/shell');
28
+ const brew = require('../utils/macos/brew');
29
+ const choco = require('../utils/windows/choco');
30
+ const winget = require('../utils/windows/winget');
31
+
32
+ /**
33
+ * Indicates whether this installer requires a desktop environment.
34
+ * Google Chrome is a GUI web browser and requires a display.
35
+ * @type {boolean}
36
+ */
37
+ const REQUIRES_DESKTOP = true;
38
+
39
+ /**
40
+ * The Homebrew cask name for Google Chrome on macOS.
41
+ * Using the cask (not formula) because Chrome is a GUI application.
42
+ */
43
+ const HOMEBREW_CASK_NAME = 'google-chrome';
44
+
45
+ /**
46
+ * The Chocolatey package name for Google Chrome on Windows.
47
+ */
48
+ const CHOCO_PACKAGE_NAME = 'googlechrome';
49
+
50
+ /**
51
+ * The winget package ID for Google Chrome on Windows.
52
+ * Using the full ID ensures we get the correct package.
53
+ */
54
+ const WINGET_PACKAGE_ID = 'Google.Chrome';
55
+
56
+ /**
57
+ * Path to Google Chrome application on macOS.
58
+ * Used to verify installation succeeded.
59
+ */
60
+ const MACOS_APP_PATH = '/Applications/Google Chrome.app';
61
+
62
+ /**
63
+ * Path to Google Chrome executable on Windows (system-wide installation).
64
+ * Used to verify installation and check version.
65
+ */
66
+ const WINDOWS_CHROME_PATH = 'C:\\Program Files\\Google\\Chrome\\Application\\chrome.exe';
67
+
68
+ /**
69
+ * URL to download the official Google Chrome .deb package.
70
+ * This package also sets up the Google repository for automatic updates.
71
+ */
72
+ const CHROME_DEB_URL = 'https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb';
73
+
74
+ /**
75
+ * URL to download the official Google Chrome .rpm package.
76
+ * Used for Amazon Linux and RHEL installations.
77
+ */
78
+ const CHROME_RPM_URL = 'https://dl.google.com/linux/direct/google-chrome-stable_current_x86_64.rpm';
79
+
80
+ /**
81
+ * Check if Google Chrome is installed on macOS by verifying the app bundle exists.
82
+ *
83
+ * On macOS, GUI applications are typically installed as .app bundles in /Applications.
84
+ * We check for the bundle's existence rather than relying on PATH because Chrome
85
+ * is a GUI application that may not add itself to the shell PATH.
86
+ *
87
+ * @returns {boolean} True if Google Chrome.app exists in /Applications, false otherwise
88
+ */
89
+ function isChromeInstalledMacOS() {
90
+ const fs = require('fs');
91
+ return fs.existsSync(MACOS_APP_PATH);
92
+ }
93
+
94
+ /**
95
+ * Get the installed version of Google Chrome on macOS.
96
+ *
97
+ * Executes the Chrome binary within the app bundle with --version flag.
98
+ * The output format is: "Google Chrome 143.0.7499.170"
99
+ *
100
+ * @returns {Promise<string|null>} Version string (e.g., "143.0.7499.170") or null if not installed
101
+ */
102
+ async function getChromeVersionMacOS() {
103
+ if (!isChromeInstalledMacOS()) {
104
+ return null;
105
+ }
106
+
107
+ const chromePath = `${MACOS_APP_PATH}/Contents/MacOS/Google Chrome`;
108
+ const result = await shell.exec(`"${chromePath}" --version`);
109
+
110
+ if (result.code === 0 && result.stdout) {
111
+ // Output format: "Google Chrome 143.0.7499.170"
112
+ const match = result.stdout.match(/Google Chrome\s+([\d.]+)/);
113
+ return match ? match[1] : null;
114
+ }
115
+ return null;
116
+ }
117
+
118
+ /**
119
+ * Check if Google Chrome is installed on Linux by verifying the google-chrome command exists.
120
+ *
121
+ * When installed from Google's official .deb or .rpm package, Chrome adds the
122
+ * 'google-chrome' or 'google-chrome-stable' command to /usr/bin.
123
+ *
124
+ * @returns {boolean} True if the google-chrome command is available, false otherwise
125
+ */
126
+ function isChromeInstalledLinux() {
127
+ return shell.commandExists('google-chrome') || shell.commandExists('google-chrome-stable');
128
+ }
129
+
130
+ /**
131
+ * Get the installed version of Google Chrome on Linux.
132
+ *
133
+ * Tries both 'google-chrome' and 'google-chrome-stable' commands since
134
+ * the command name varies between distributions.
135
+ *
136
+ * @returns {Promise<string|null>} Version string or null if not installed
137
+ */
138
+ async function getChromeVersionLinux() {
139
+ // Try google-chrome first (common symlink name)
140
+ if (shell.commandExists('google-chrome')) {
141
+ const result = await shell.exec('google-chrome --version');
142
+ if (result.code === 0 && result.stdout) {
143
+ const match = result.stdout.match(/Google Chrome\s+([\d.]+)/);
144
+ return match ? match[1] : null;
145
+ }
146
+ }
147
+
148
+ // Fall back to google-chrome-stable (actual binary name)
149
+ if (shell.commandExists('google-chrome-stable')) {
150
+ const result = await shell.exec('google-chrome-stable --version');
151
+ if (result.code === 0 && result.stdout) {
152
+ const match = result.stdout.match(/Google Chrome\s+([\d.]+)/);
153
+ return match ? match[1] : null;
154
+ }
155
+ }
156
+
157
+ return null;
158
+ }
159
+
160
+ /**
161
+ * Check if Google Chrome is installed on Windows.
162
+ *
163
+ * Checks for the Chrome executable at the default installation path.
164
+ * Chrome is typically installed to Program Files for system-wide installation
165
+ * or to LocalAppData for per-user installation.
166
+ *
167
+ * @returns {boolean} True if Chrome executable exists, false otherwise
168
+ */
169
+ function isChromeInstalledWindows() {
170
+ const fs = require('fs');
171
+
172
+ // Check system-wide installation path
173
+ if (fs.existsSync(WINDOWS_CHROME_PATH)) {
174
+ return true;
175
+ }
176
+
177
+ // Check per-user installation path
178
+ const localAppData = process.env.LOCALAPPDATA;
179
+ if (localAppData) {
180
+ const userPath = `${localAppData}\\Google\\Chrome\\Application\\chrome.exe`;
181
+ if (fs.existsSync(userPath)) {
182
+ return true;
183
+ }
184
+ }
185
+
186
+ return false;
187
+ }
188
+
189
+ /**
190
+ * Get the installed version of Google Chrome on Windows.
191
+ *
192
+ * Executes the Chrome executable with --version flag to retrieve the version.
193
+ *
194
+ * @returns {Promise<string|null>} Version string or null if not installed
195
+ */
196
+ async function getChromeVersionWindows() {
197
+ const fs = require('fs');
198
+
199
+ // Determine which path has Chrome installed
200
+ let chromePath = null;
201
+ if (fs.existsSync(WINDOWS_CHROME_PATH)) {
202
+ chromePath = WINDOWS_CHROME_PATH;
203
+ } else {
204
+ const localAppData = process.env.LOCALAPPDATA;
205
+ if (localAppData) {
206
+ const userPath = `${localAppData}\\Google\\Chrome\\Application\\chrome.exe`;
207
+ if (fs.existsSync(userPath)) {
208
+ chromePath = userPath;
209
+ }
210
+ }
211
+ }
212
+
213
+ if (!chromePath) {
214
+ return null;
215
+ }
216
+
217
+ const result = await shell.exec(`"${chromePath}" --version`);
218
+ if (result.code === 0 && result.stdout) {
219
+ const match = result.stdout.match(/Google Chrome\s+([\d.]+)/);
220
+ return match ? match[1] : null;
221
+ }
222
+ return null;
223
+ }
224
+
225
+ /**
226
+ * Install Google Chrome on macOS using Homebrew.
227
+ *
228
+ * Prerequisites:
229
+ * - macOS 12 (Monterey) or later
230
+ * - Homebrew package manager installed
231
+ * - Terminal access
232
+ *
233
+ * The installation uses the Homebrew cask 'google-chrome' which downloads
234
+ * and installs Chrome to /Applications/Google Chrome.app.
235
+ *
236
+ * This function is idempotent - it checks if Chrome is already installed
237
+ * before attempting installation and skips if already present.
238
+ *
239
+ * @returns {Promise<void>}
240
+ * @throws {Error} If Homebrew is not installed or installation fails
241
+ */
242
+ async function install_macos() {
243
+ console.log('Checking if Google Chrome is already installed...');
244
+
245
+ // Check if Chrome is already installed via file system check
246
+ if (isChromeInstalledMacOS()) {
247
+ const version = await getChromeVersionMacOS();
248
+ if (version) {
249
+ console.log(`Google Chrome ${version} is already installed, skipping installation.`);
250
+ } else {
251
+ console.log('Google Chrome is already installed, skipping installation.');
252
+ }
253
+ return;
254
+ }
255
+
256
+ // Also check if the Homebrew cask is installed (Chrome may be installed but not detected)
257
+ const caskInstalled = await brew.isCaskInstalled(HOMEBREW_CASK_NAME);
258
+ if (caskInstalled) {
259
+ console.log('Google Chrome is already installed via Homebrew, skipping installation.');
260
+ return;
261
+ }
262
+
263
+ // Verify Homebrew is available before proceeding
264
+ if (!brew.isInstalled()) {
265
+ throw new Error(
266
+ 'Homebrew is not installed. Please install Homebrew first using:\n' +
267
+ ' dev install homebrew\n' +
268
+ 'Then retry installing Google Chrome.'
269
+ );
270
+ }
271
+
272
+ console.log('Installing Google Chrome via Homebrew...');
273
+
274
+ // Install Google Chrome cask using the --quiet flag for cleaner output
275
+ const result = await brew.installCask(HOMEBREW_CASK_NAME);
276
+
277
+ if (!result.success) {
278
+ throw new Error(
279
+ `Failed to install Google Chrome via Homebrew.\n` +
280
+ `Output: ${result.output}\n\n` +
281
+ `Troubleshooting:\n` +
282
+ ` 1. Run 'brew update && brew cleanup' and retry\n` +
283
+ ` 2. Try manual installation: brew reinstall --cask google-chrome\n` +
284
+ ` 3. Check if macOS Gatekeeper is blocking the app:\n` +
285
+ ` xattr -cr /Applications/Google\\ Chrome.app`
286
+ );
287
+ }
288
+
289
+ // Verify the installation succeeded
290
+ if (!isChromeInstalledMacOS()) {
291
+ throw new Error(
292
+ 'Installation appeared to complete but Google Chrome was not found.\n\n' +
293
+ 'Please check /Applications folder for Google Chrome.app'
294
+ );
295
+ }
296
+
297
+ const installedVersion = await getChromeVersionMacOS();
298
+ console.log(`Google Chrome ${installedVersion || ''} installed successfully.`);
299
+ console.log('');
300
+ console.log('You can launch Chrome from Applications or run:');
301
+ console.log(' open -a "Google Chrome"');
302
+ }
303
+
304
+ /**
305
+ * Install Google Chrome on Ubuntu/Debian using APT.
306
+ *
307
+ * Prerequisites:
308
+ * - Ubuntu 18.04 or later, or Debian 10 or later (64-bit only)
309
+ * - sudo privileges
310
+ * - wget installed (used to download the .deb package)
311
+ *
312
+ * This function downloads Google's official .deb package and installs it.
313
+ * The .deb package automatically adds Google's APT repository for future updates.
314
+ *
315
+ * IMPORTANT: Google Chrome is NOT available in Ubuntu's default repositories.
316
+ * Always use Google's official package to ensure you get the latest version
317
+ * with security updates.
318
+ *
319
+ * This function is idempotent - it checks if Chrome is already installed
320
+ * before attempting installation.
321
+ *
322
+ * @returns {Promise<void>}
323
+ * @throws {Error} If installation fails
324
+ */
325
+ async function install_ubuntu() {
326
+ console.log('Checking if Google Chrome is already installed...');
327
+
328
+ // Check if Chrome is already installed
329
+ if (isChromeInstalledLinux()) {
330
+ const version = await getChromeVersionLinux();
331
+ if (version) {
332
+ console.log(`Google Chrome ${version} is already installed, skipping installation.`);
333
+ } else {
334
+ console.log('Google Chrome is already installed, skipping installation.');
335
+ }
336
+ return;
337
+ }
338
+
339
+ console.log('Installing Google Chrome from Google\'s official repository...');
340
+
341
+ // Step 1: Download the .deb package
342
+ // Using wget with -q for quiet mode (suitable for automation)
343
+ console.log('Downloading Google Chrome package...');
344
+ const downloadResult = await shell.exec(
345
+ `wget -q ${CHROME_DEB_URL} -O /tmp/google-chrome-stable_current_amd64.deb`
346
+ );
347
+
348
+ if (downloadResult.code !== 0) {
349
+ throw new Error(
350
+ `Failed to download Google Chrome package.\n` +
351
+ `Error: ${downloadResult.stderr}\n\n` +
352
+ `Troubleshooting:\n` +
353
+ ` 1. Check your internet connection\n` +
354
+ ` 2. Ensure wget is installed: sudo apt-get install wget\n` +
355
+ ` 3. Try downloading manually: wget ${CHROME_DEB_URL}`
356
+ );
357
+ }
358
+
359
+ // Step 2: Install the package using apt-get
360
+ // DEBIAN_FRONTEND=noninteractive prevents interactive prompts during installation
361
+ console.log('Installing Google Chrome...');
362
+ const installResult = await shell.exec(
363
+ 'sudo DEBIAN_FRONTEND=noninteractive apt-get install -y /tmp/google-chrome-stable_current_amd64.deb'
364
+ );
365
+
366
+ // Clean up the downloaded .deb file regardless of installation result
367
+ await shell.exec('rm -f /tmp/google-chrome-stable_current_amd64.deb');
368
+
369
+ if (installResult.code !== 0) {
370
+ // Try to fix broken dependencies and retry
371
+ console.log('Attempting to fix broken dependencies...');
372
+ const fixResult = await shell.exec('sudo DEBIAN_FRONTEND=noninteractive apt-get install -y -f');
373
+
374
+ if (fixResult.code !== 0) {
375
+ throw new Error(
376
+ `Failed to install Google Chrome.\n` +
377
+ `Error: ${installResult.stderr}\n\n` +
378
+ `Troubleshooting:\n` +
379
+ ` 1. Run: sudo apt-get update\n` +
380
+ ` 2. Install missing dependencies: sudo apt-get install -y -f\n` +
381
+ ` 3. Try again: sudo apt-get install -y ./google-chrome-stable_current_amd64.deb`
382
+ );
383
+ }
384
+ }
385
+
386
+ // Verify installation
387
+ if (!isChromeInstalledLinux()) {
388
+ throw new Error(
389
+ 'Installation appeared to complete but Google Chrome was not found.\n\n' +
390
+ 'Please try running: google-chrome --version'
391
+ );
392
+ }
393
+
394
+ const installedVersion = await getChromeVersionLinux();
395
+ console.log(`Google Chrome ${installedVersion || ''} installed successfully.`);
396
+ console.log('');
397
+ console.log('Launch Chrome with:');
398
+ console.log(' google-chrome &');
399
+ console.log('');
400
+ console.log('NOTE: Chrome will automatically update via apt-get upgrade.');
401
+ }
9
402
 
10
403
  /**
11
- * Install Google Chrome across supported platforms.
404
+ * Install Google Chrome on Ubuntu running in WSL (Windows Subsystem for Linux).
405
+ *
406
+ * Prerequisites:
407
+ * - Windows 10 version 2004 or higher, or Windows 11
408
+ * - WSL 2 with Ubuntu distribution installed
409
+ * - WSLg enabled (included by default in Windows 11 and Windows 10 21H2+)
410
+ * - sudo privileges within WSL
411
+ *
412
+ * This function installs Chrome using the same method as Ubuntu since
413
+ * WSL Ubuntu is functionally identical to native Ubuntu for package management.
414
+ * However, running Chrome requires WSLg for GUI support.
415
+ *
416
+ * @returns {Promise<void>}
417
+ * @throws {Error} If installation fails
418
+ */
419
+ async function install_ubuntu_wsl() {
420
+ console.log('Detected Ubuntu running in WSL (Windows Subsystem for Linux).');
421
+ console.log('');
422
+
423
+ // Use the same installation process as Ubuntu
424
+ await install_ubuntu();
425
+
426
+ // Add WSL-specific post-installation notes
427
+ console.log('');
428
+ console.log('WSL-SPECIFIC NOTES:');
429
+ console.log('');
430
+ console.log('1. Running GUI apps in WSL requires WSLg (Windows 11 / Windows 10 21H2+).');
431
+ console.log(' If Chrome fails to launch, ensure WSLg is enabled.');
432
+ console.log('');
433
+ console.log('2. If you see sandbox errors, try:');
434
+ console.log(' google-chrome --no-sandbox');
435
+ console.log('');
436
+ console.log('3. For headless automation in WSL:');
437
+ console.log(' google-chrome --headless --disable-gpu --dump-dom https://example.com');
438
+ }
439
+
440
+ /**
441
+ * Install Google Chrome on Raspberry Pi OS.
442
+ *
443
+ * IMPORTANT: Google Chrome does NOT support ARM architecture.
444
+ * Google only provides x86/x64 builds of Chrome, and Raspberry Pi uses ARM processors.
445
+ * This means Google Chrome cannot be installed natively on any Raspberry Pi device.
446
+ *
447
+ * This function gracefully informs the user that Chrome is not available
448
+ * on this platform without throwing an error or suggesting alternatives.
449
+ *
450
+ * @returns {Promise<void>}
451
+ */
452
+ async function install_raspbian() {
453
+ console.log('Google Chrome is not available for Raspberry Pi OS.');
454
+ return;
455
+ }
456
+
457
+ /**
458
+ * Install Google Chrome on Amazon Linux using DNF/YUM.
459
+ *
460
+ * Prerequisites:
461
+ * - Amazon Linux 2023 (AL2023) recommended
462
+ * - Amazon Linux 2 is supported but reached end of standard support
463
+ * - sudo privileges
464
+ * - x86_64 architecture (ARM/Graviton instances are not supported)
465
+ *
466
+ * This function downloads and installs Google's official RPM package.
467
+ * Amazon Linux 2023 uses DNF, while Amazon Linux 2 uses YUM.
468
+ *
469
+ * NOTE: Google does not officially support Chrome on Amazon Linux, but the
470
+ * Chrome RPM package installs successfully on both AL2023 and AL2.
471
+ *
472
+ * @returns {Promise<void>}
473
+ * @throws {Error} If installation fails
474
+ */
475
+ async function install_amazon_linux() {
476
+ console.log('Checking if Google Chrome is already installed...');
477
+
478
+ // Check if Chrome is already installed
479
+ if (isChromeInstalledLinux()) {
480
+ const version = await getChromeVersionLinux();
481
+ if (version) {
482
+ console.log(`Google Chrome ${version} is already installed, skipping installation.`);
483
+ } else {
484
+ console.log('Google Chrome is already installed, skipping installation.');
485
+ }
486
+ return;
487
+ }
488
+
489
+ // Detect package manager (dnf for AL2023, yum for AL2)
490
+ const hasDnf = shell.commandExists('dnf');
491
+ const hasYum = shell.commandExists('yum');
492
+ const packageManager = hasDnf ? 'dnf' : (hasYum ? 'yum' : null);
493
+
494
+ if (!packageManager) {
495
+ throw new Error(
496
+ 'Neither dnf nor yum package manager found.\n' +
497
+ 'This installer supports Amazon Linux 2023 (dnf) and Amazon Linux 2 (yum).'
498
+ );
499
+ }
500
+
501
+ console.log(`Detected package manager: ${packageManager}`);
502
+ console.log('Installing Google Chrome from Google\'s official RPM package...');
503
+
504
+ // Install Chrome directly from Google's RPM URL
505
+ // This method also installs required dependencies automatically
506
+ const installResult = await shell.exec(
507
+ `sudo ${packageManager} install -y ${CHROME_RPM_URL}`
508
+ );
509
+
510
+ if (installResult.code !== 0) {
511
+ throw new Error(
512
+ `Failed to install Google Chrome.\n` +
513
+ `Error: ${installResult.stderr}\n\n` +
514
+ `Troubleshooting:\n` +
515
+ ` 1. Install common dependencies first:\n` +
516
+ ` sudo ${packageManager} install -y libXcomposite libXdamage libXrandr libgbm libxkbcommon pango alsa-lib atk at-spi2-atk cups-libs libdrm mesa-libgbm\n` +
517
+ ` 2. Then retry: sudo ${packageManager} install -y ${CHROME_RPM_URL}`
518
+ );
519
+ }
520
+
521
+ // Verify installation
522
+ if (!isChromeInstalledLinux()) {
523
+ throw new Error(
524
+ 'Installation appeared to complete but Google Chrome was not found.\n\n' +
525
+ 'Please try running: google-chrome-stable --version'
526
+ );
527
+ }
528
+
529
+ const installedVersion = await getChromeVersionLinux();
530
+ console.log(`Google Chrome ${installedVersion || ''} installed successfully.`);
531
+ console.log('');
532
+ console.log('NOTE: On EC2 instances, you may need to run Chrome with additional flags:');
533
+ console.log(' google-chrome-stable --no-sandbox --headless');
534
+ }
535
+
536
+ /**
537
+ * Install Google Chrome on Windows using Chocolatey or winget.
538
+ *
539
+ * Prerequisites:
540
+ * - Windows 10 version 1809 or later, or Windows 11
541
+ * - Administrator privileges
542
+ * - Chocolatey or winget package manager installed
543
+ *
544
+ * This function prefers winget if available (built into modern Windows),
545
+ * falling back to Chocolatey if winget is not present.
546
+ *
547
+ * @returns {Promise<void>}
548
+ * @throws {Error} If no package manager is available or installation fails
549
+ */
550
+ async function install_windows() {
551
+ console.log('Checking if Google Chrome is already installed...');
552
+
553
+ // Check if Chrome is already installed
554
+ if (isChromeInstalledWindows()) {
555
+ const version = await getChromeVersionWindows();
556
+ if (version) {
557
+ console.log(`Google Chrome ${version} is already installed, skipping installation.`);
558
+ } else {
559
+ console.log('Google Chrome is already installed, skipping installation.');
560
+ }
561
+ return;
562
+ }
563
+
564
+ // Prefer winget if available (modern Windows includes it by default)
565
+ if (winget.isInstalled()) {
566
+ console.log('Installing Google Chrome via winget...');
567
+
568
+ const result = await winget.install(WINGET_PACKAGE_ID, {
569
+ silent: true
570
+ });
571
+
572
+ if (result.success) {
573
+ console.log('Google Chrome installed successfully via winget.');
574
+ console.log('');
575
+ console.log('NOTE: Chrome is now available in your Start Menu.');
576
+ console.log('You may need to open a new terminal to use it from the command line.');
577
+ return;
578
+ }
579
+
580
+ // winget failed, try Chocolatey as fallback
581
+ console.log('winget installation failed, trying Chocolatey...');
582
+ }
583
+
584
+ // Try Chocolatey if available
585
+ if (choco.isInstalled()) {
586
+ console.log('Installing Google Chrome via Chocolatey...');
587
+
588
+ const result = await choco.install(CHOCO_PACKAGE_NAME);
589
+
590
+ if (!result.success) {
591
+ throw new Error(
592
+ `Failed to install Google Chrome via Chocolatey.\n` +
593
+ `Output: ${result.output}\n\n` +
594
+ `Troubleshooting:\n` +
595
+ ` 1. Ensure you are running as Administrator\n` +
596
+ ` 2. Try manual installation: choco install googlechrome -y --force`
597
+ );
598
+ }
599
+
600
+ console.log('Google Chrome installed successfully via Chocolatey.');
601
+ console.log('');
602
+ console.log('NOTE: Chrome is now available in your Start Menu.');
603
+ console.log('You may need to open a new terminal to use it from the command line.');
604
+ return;
605
+ }
606
+
607
+ // Neither package manager is available
608
+ throw new Error(
609
+ 'Neither winget nor Chocolatey package manager is available.\n\n' +
610
+ 'Please install one of the following:\n' +
611
+ ' - winget: Included in Windows 10 (2004+) and Windows 11\n' +
612
+ ' - Chocolatey: https://chocolatey.org/install\n\n' +
613
+ 'After installing a package manager, retry: dev install google-chrome'
614
+ );
615
+ }
616
+
617
+ /**
618
+ * Install Google Chrome from Git Bash on Windows.
619
+ *
620
+ * Git Bash runs within Windows, so this function installs Google Chrome
621
+ * on the Windows host using PowerShell interop with Chocolatey or winget.
622
+ *
623
+ * Prerequisites:
624
+ * - Windows 10 or Windows 11
625
+ * - Git Bash installed (comes with Git for Windows)
626
+ * - Chocolatey or winget available on Windows
627
+ *
628
+ * @returns {Promise<void>}
629
+ * @throws {Error} If installation fails
630
+ */
631
+ async function install_gitbash() {
632
+ console.log('Detected Git Bash on Windows.');
633
+ console.log('Installing Google Chrome on the Windows host...');
634
+ console.log('');
635
+
636
+ // Check if Chrome is already installed by checking the Windows path
637
+ const fs = require('fs');
638
+
639
+ // Git Bash path format uses forward slashes with drive letter prefix
640
+ const windowsPath = '/c/Program Files/Google/Chrome/Application/chrome.exe';
641
+ const exists = fs.existsSync(windowsPath);
642
+
643
+ if (exists) {
644
+ console.log('Google Chrome is already installed, skipping installation.');
645
+ console.log('');
646
+ console.log('To use Chrome from Git Bash:');
647
+ console.log(' "/c/Program Files/Google/Chrome/Application/chrome.exe" --version');
648
+ return;
649
+ }
650
+
651
+ // Try winget first via PowerShell
652
+ console.log('Attempting installation via winget...');
653
+ const wingetResult = await shell.exec(
654
+ 'powershell.exe -NoProfile -Command "winget install --id Google.Chrome --silent --accept-package-agreements --accept-source-agreements"'
655
+ );
656
+
657
+ if (wingetResult.code === 0) {
658
+ console.log('Google Chrome installed successfully via winget.');
659
+ console.log('');
660
+ console.log('To use Chrome from Git Bash:');
661
+ console.log(' "/c/Program Files/Google/Chrome/Application/chrome.exe" --version');
662
+ return;
663
+ }
664
+
665
+ // Fall back to Chocolatey
666
+ console.log('winget installation failed, trying Chocolatey...');
667
+ const chocoResult = await shell.exec(
668
+ 'powershell.exe -NoProfile -Command "choco install googlechrome -y"'
669
+ );
670
+
671
+ if (chocoResult.code !== 0) {
672
+ throw new Error(
673
+ `Failed to install Google Chrome.\n` +
674
+ `Output: ${chocoResult.stdout || chocoResult.stderr}\n\n` +
675
+ `Troubleshooting:\n` +
676
+ ` 1. Ensure winget or Chocolatey is installed on Windows\n` +
677
+ ` 2. Run Git Bash as Administrator and retry\n` +
678
+ ` 3. Try installing directly from PowerShell:\n` +
679
+ ` winget install --id Google.Chrome --silent`
680
+ );
681
+ }
682
+
683
+ console.log('Google Chrome installed successfully via Chocolatey.');
684
+ console.log('');
685
+ console.log('To use Chrome from Git Bash:');
686
+ console.log(' "/c/Program Files/Google/Chrome/Application/chrome.exe" --version');
687
+ }
688
+
689
+ /**
690
+ * Check if Google Chrome is currently installed on the system.
691
+ *
692
+ * This function checks for Google Chrome installation across all supported platforms:
693
+ * - macOS: Checks for Google Chrome.app via Homebrew cask or application bundle
694
+ * - Windows: Checks for Chrome.exe at standard installation paths
695
+ * - Linux: Checks if google-chrome or google-chrome-stable command exists
696
+ *
697
+ * @returns {Promise<boolean>} True if Google Chrome is installed, false otherwise
698
+ */
699
+ async function isInstalled() {
700
+ const platform = os.detect();
701
+
702
+ if (platform.type === 'macos') {
703
+ // Check if Google Chrome app bundle exists
704
+ if (isChromeInstalledMacOS()) {
705
+ return true;
706
+ }
707
+ // Also check via Homebrew cask
708
+ return await brew.isCaskInstalled(HOMEBREW_CASK_NAME);
709
+ }
710
+
711
+ if (platform.type === 'windows' || platform.type === 'gitbash') {
712
+ return isChromeInstalledWindows();
713
+ }
714
+
715
+ if (platform.type === 'raspbian') {
716
+ // Chrome is not available for Raspberry Pi
717
+ return false;
718
+ }
719
+
720
+ // Linux platforms: Check if google-chrome command exists
721
+ return isChromeInstalledLinux();
722
+ }
723
+
724
+ /**
725
+ * Check if this installer is supported on the current platform.
726
+ *
727
+ * Google Chrome is NOT supported on ARM-based platforms (Raspberry Pi)
728
+ * and requires a desktop environment since it is a GUI application.
729
+ *
730
+ * @returns {boolean} True if installation is supported on this platform
731
+ */
732
+ function isEligible() {
733
+ const platform = os.detect();
734
+
735
+ // First check if the platform is supported
736
+ // Chrome is NOT available for Raspberry Pi (ARM architecture)
737
+ const supportedPlatforms = ['macos', 'ubuntu', 'debian', 'wsl', 'amazon_linux', 'rhel', 'fedora', 'windows', 'gitbash'];
738
+ if (!supportedPlatforms.includes(platform.type)) {
739
+ return false;
740
+ }
741
+
742
+ // This installer requires a desktop environment
743
+ if (REQUIRES_DESKTOP && !os.isDesktopAvailable()) {
744
+ return false;
745
+ }
746
+
747
+ return true;
748
+ }
749
+
750
+ /**
751
+ * Main installation entry point.
752
+ *
753
+ * Detects the current platform using os.detect() and routes to the appropriate
754
+ * platform-specific installer function. Handles platform aliases to ensure
755
+ * all supported distributions use the correct installation method.
756
+ *
757
+ * Supported platforms:
758
+ * - macOS: Google Chrome via Homebrew cask
759
+ * - Ubuntu/Debian: Google Chrome via official .deb package
760
+ * - Amazon Linux/RHEL: Google Chrome via official .rpm package
761
+ * - Windows: Google Chrome via winget or Chocolatey
762
+ * - WSL (Ubuntu): Google Chrome within WSL environment
763
+ * - Git Bash: Google Chrome on Windows host
764
+ *
765
+ * Unsupported platforms:
766
+ * - Raspberry Pi OS (ARM architecture not supported by Chrome)
767
+ * - Any other ARM-based Linux distributions
12
768
  *
13
769
  * @returns {Promise<void>}
14
770
  */
15
771
  async function install() {
16
772
  const platform = os.detect();
17
773
 
18
- switch (platform.type) {
19
- case 'macos':
20
- // TODO: Implement macOS installation
21
- break;
22
- case 'debian':
23
- // TODO: Implement Debian/Ubuntu installation
24
- break;
25
- case 'rhel':
26
- // TODO: Implement RHEL/Amazon Linux installation
27
- break;
28
- case 'windows-wsl':
29
- // TODO: Implement WSL installation
30
- break;
31
- case 'windows':
32
- // TODO: Implement Windows installation
33
- break;
34
- default:
35
- console.error(`Unsupported platform: ${platform.type}`);
36
- process.exit(1);
37
- }
38
- }
39
-
40
- module.exports = { install };
774
+ // Map platform types to their installer functions
775
+ // This mapping handles aliases (e.g., debian uses the same installer as ubuntu)
776
+ const installers = {
777
+ 'macos': install_macos,
778
+ 'ubuntu': install_ubuntu,
779
+ 'debian': install_ubuntu,
780
+ 'wsl': install_ubuntu_wsl,
781
+ 'raspbian': install_raspbian,
782
+ 'amazon_linux': install_amazon_linux,
783
+ 'rhel': install_amazon_linux,
784
+ 'fedora': install_amazon_linux,
785
+ 'windows': install_windows,
786
+ 'gitbash': install_gitbash
787
+ };
788
+
789
+ const installer = installers[platform.type];
790
+
791
+ if (!installer) {
792
+ console.log(`Google Chrome is not available for ${platform.type}.`);
793
+ return;
794
+ }
795
+
796
+ await installer();
797
+ }
798
+
799
+ // Export all functions for use as a module and for testing
800
+ module.exports = {
801
+ REQUIRES_DESKTOP,
802
+ install,
803
+ isInstalled,
804
+ isEligible,
805
+ install_macos,
806
+ install_ubuntu,
807
+ install_ubuntu_wsl,
808
+ install_raspbian,
809
+ install_amazon_linux,
810
+ install_windows,
811
+ install_gitbash
812
+ };
41
813
 
814
+ // Allow direct execution: node google-chrome.js
42
815
  if (require.main === module) {
43
- install();
816
+ install().catch(err => {
817
+ console.error(err.message);
818
+ process.exit(1);
819
+ });
44
820
  }