@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
@@ -1,44 +1,797 @@
1
1
  #!/usr/bin/env node
2
2
 
3
3
  /**
4
- * @fileoverview Install AWS CLI.
4
+ * @fileoverview Install AWS CLI v2 across supported platforms.
5
5
  * @module installs/aws-cli
6
+ *
7
+ * AWS CLI v2 is the unified tool for interacting with all AWS services.
8
+ * It bundles its own Python runtime, so no separate Python installation is required.
9
+ *
10
+ * Supported platforms:
11
+ * - macOS: Homebrew (brew install awscli)
12
+ * - Ubuntu/Debian: Snap (snap install aws-cli --classic)
13
+ * - Raspberry Pi OS: Snap (requires 64-bit OS)
14
+ * - Amazon Linux 2023: DNF (pre-installed, upgrade with dnf)
15
+ * - Amazon Linux 2: Manual installer (must remove v1 first)
16
+ * - Windows: Chocolatey (choco install awscli)
17
+ * - WSL (Ubuntu): Snap or manual installer
18
+ * - Git Bash: Uses Windows installation (inherited PATH)
6
19
  */
7
20
 
8
21
  const os = require('../utils/common/os');
22
+ const shell = require('../utils/common/shell');
23
+ const brew = require('../utils/macos/brew');
24
+ const snap = require('../utils/ubuntu/snap');
25
+ const apt = require('../utils/ubuntu/apt');
26
+ const choco = require('../utils/windows/choco');
27
+
28
+ /**
29
+ * The name of the AWS CLI command used to verify installation
30
+ * @constant {string}
31
+ */
32
+ const AWS_CLI_COMMAND = 'aws';
33
+
34
+ /**
35
+ * The snap package name for AWS CLI
36
+ * @constant {string}
37
+ */
38
+ const SNAP_PACKAGE_NAME = 'aws-cli';
39
+
40
+ /**
41
+ * The Homebrew formula name for AWS CLI
42
+ * @constant {string}
43
+ */
44
+ const BREW_FORMULA_NAME = 'awscli';
45
+
46
+ /**
47
+ * The Chocolatey package name for AWS CLI
48
+ * @constant {string}
49
+ */
50
+ const CHOCO_PACKAGE_NAME = 'awscli';
51
+
52
+ /**
53
+ * Check if AWS CLI is already installed by verifying the aws command exists
54
+ * @returns {boolean} True if AWS CLI is installed and accessible
55
+ */
56
+ function isAwsCliInstalled() {
57
+ return shell.commandExists(AWS_CLI_COMMAND);
58
+ }
59
+
60
+ /**
61
+ * Get the installed version of AWS CLI
62
+ * @returns {Promise<string|null>} The version string or null if not installed
63
+ */
64
+ async function getAwsCliVersion() {
65
+ if (!isAwsCliInstalled()) {
66
+ return null;
67
+ }
68
+
69
+ const result = await shell.exec('aws --version');
70
+ if (result.code === 0 && result.stdout) {
71
+ // Output format: "aws-cli/2.32.28 Python/3.11.6 Darwin/23.3.0 source/arm64"
72
+ const match = result.stdout.match(/aws-cli\/(\S+)/);
73
+ return match ? match[1] : null;
74
+ }
75
+ return null;
76
+ }
77
+
78
+ /**
79
+ * Check if the installed AWS CLI is version 2 (not legacy v1)
80
+ * @returns {Promise<boolean>} True if AWS CLI v2 is installed
81
+ */
82
+ async function isVersion2() {
83
+ const version = await getAwsCliVersion();
84
+ if (!version) {
85
+ return false;
86
+ }
87
+ // Version 2.x.x starts with "2."
88
+ return version.startsWith('2.');
89
+ }
90
+
91
+ /**
92
+ * Install AWS CLI on macOS using Homebrew
93
+ *
94
+ * Prerequisites:
95
+ * - macOS 11 (Big Sur) or later
96
+ * - Homebrew package manager installed
97
+ *
98
+ * @returns {Promise<void>}
99
+ */
100
+ async function install_macos() {
101
+ // Check if AWS CLI is already installed
102
+ if (isAwsCliInstalled()) {
103
+ const version = await getAwsCliVersion();
104
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
105
+ return;
106
+ }
107
+
108
+ // Verify Homebrew is available
109
+ if (!brew.isInstalled()) {
110
+ console.log('Homebrew is not installed. Please install Homebrew first.');
111
+ console.log('Run: /bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"');
112
+ return;
113
+ }
114
+
115
+ console.log('Installing AWS CLI via Homebrew...');
116
+ const result = await brew.install(BREW_FORMULA_NAME);
117
+
118
+ if (!result.success) {
119
+ console.log('Failed to install AWS CLI via Homebrew.');
120
+ console.log(result.output);
121
+ return;
122
+ }
123
+
124
+ // Verify the installation succeeded
125
+ if (isAwsCliInstalled()) {
126
+ const version = await getAwsCliVersion();
127
+ console.log(`AWS CLI installed successfully (version ${version}).`);
128
+ } else {
129
+ console.log('Installation completed but AWS CLI command not found.');
130
+ console.log('You may need to restart your terminal or add Homebrew to your PATH.');
131
+ }
132
+ }
133
+
134
+ /**
135
+ * Install AWS CLI on Ubuntu/Debian using Snap or manual installer
136
+ *
137
+ * Tries Snap first (official package), but falls back to manual installer
138
+ * if Snap is unavailable (e.g., in Docker containers or systems without systemd).
139
+ *
140
+ * Prerequisites:
141
+ * - Ubuntu 18.04+ or Debian 10+ (64-bit)
142
+ * - sudo privileges
143
+ * - curl and unzip (auto-installed if missing)
144
+ *
145
+ * @returns {Promise<void>}
146
+ */
147
+ async function install_ubuntu() {
148
+ // Check if AWS CLI is already installed
149
+ if (isAwsCliInstalled()) {
150
+ const version = await getAwsCliVersion();
151
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
152
+ return;
153
+ }
154
+
155
+ // Try Snap first if it's available
156
+ if (snap.isInstalled()) {
157
+ console.log('Installing AWS CLI via Snap...');
158
+ const result = await snap.install(SNAP_PACKAGE_NAME, { classic: true });
159
+
160
+ if (result.success) {
161
+ // Verify the installation succeeded
162
+ const isInstalled = isAwsCliInstalled() || await snap.isSnapInstalled(SNAP_PACKAGE_NAME);
163
+ if (isInstalled) {
164
+ const version = await getAwsCliVersion();
165
+ if (version) {
166
+ console.log(`AWS CLI installed successfully (version ${version}).`);
167
+ } else {
168
+ console.log('AWS CLI installed successfully.');
169
+ console.log('You may need to log out and back in, or add /snap/bin to your PATH.');
170
+ }
171
+ return;
172
+ }
173
+ }
174
+ // Snap installation failed, fall through to manual installer
175
+ console.log('Snap installation failed, using manual installer instead...');
176
+ } else {
177
+ console.log('Snap is not available, using manual installer...');
178
+ }
179
+
180
+ // Manual installer method (works in all environments including Docker)
181
+ console.log('Installing AWS CLI v2 using the official installer...');
182
+
183
+ // Ensure required tools are available
184
+ if (!shell.commandExists('unzip')) {
185
+ console.log('Installing unzip...');
186
+ const unzipResult = await shell.exec('sudo DEBIAN_FRONTEND=noninteractive apt-get update -y && sudo DEBIAN_FRONTEND=noninteractive apt-get install -y unzip');
187
+ if (unzipResult.code !== 0) {
188
+ console.log('Failed to install unzip.');
189
+ console.log(unzipResult.stderr);
190
+ return;
191
+ }
192
+ }
193
+
194
+ if (!shell.commandExists('curl')) {
195
+ console.log('Installing curl...');
196
+ const curlResult = await shell.exec('sudo DEBIAN_FRONTEND=noninteractive apt-get update -y && sudo DEBIAN_FRONTEND=noninteractive apt-get install -y curl');
197
+ if (curlResult.code !== 0) {
198
+ console.log('Failed to install curl.');
199
+ console.log(curlResult.stderr);
200
+ return;
201
+ }
202
+ }
203
+
204
+ // Detect architecture and select the appropriate installer
205
+ const arch = os.getArch();
206
+ let installerUrl;
207
+
208
+ if (arch === 'arm64' || arch === 'aarch64') {
209
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-aarch64.zip';
210
+ console.log('Detected ARM64 architecture');
211
+ } else if (arch === 'x64' || arch === 'x86_64') {
212
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip';
213
+ console.log('Detected x86_64 architecture');
214
+ } else {
215
+ console.log(`Unsupported architecture: ${arch}`);
216
+ console.log('AWS CLI v2 supports x86_64 and aarch64 only.');
217
+ return;
218
+ }
219
+
220
+ // Download the installer
221
+ console.log('Downloading AWS CLI v2 installer...');
222
+ const downloadResult = await shell.exec(
223
+ `curl -s "${installerUrl}" -o "/tmp/awscliv2.zip"`
224
+ );
225
+
226
+ if (downloadResult.code !== 0) {
227
+ console.log('Failed to download AWS CLI installer.');
228
+ console.log(downloadResult.stderr);
229
+ return;
230
+ }
231
+
232
+ // Extract the installer
233
+ console.log('Extracting installer...');
234
+ const extractResult = await shell.exec('unzip -o -q /tmp/awscliv2.zip -d /tmp');
235
+
236
+ if (extractResult.code !== 0) {
237
+ console.log('Failed to extract AWS CLI installer.');
238
+ console.log(extractResult.stderr);
239
+ await shell.exec('rm -f /tmp/awscliv2.zip');
240
+ return;
241
+ }
242
+
243
+ // Run the installer
244
+ console.log('Running installer...');
245
+ const installResult = await shell.exec('sudo /tmp/aws/install');
246
+
247
+ // Clean up
248
+ await shell.exec('rm -rf /tmp/awscliv2.zip /tmp/aws');
249
+
250
+ if (installResult.code !== 0) {
251
+ console.log('Failed to run AWS CLI installer.');
252
+ console.log(installResult.stderr);
253
+ return;
254
+ }
255
+
256
+ // Verify the installation succeeded
257
+ // Check both PATH and the standard installation location
258
+ const awsPath = shell.which('aws') || (await shell.exec('test -f /usr/local/bin/aws && echo "/usr/local/bin/aws"')).stdout.trim();
259
+
260
+ if (awsPath) {
261
+ // Try to get version using the full path
262
+ const versionResult = await shell.exec(`${awsPath} --version 2>&1`);
263
+ if (versionResult.code === 0 && versionResult.stdout) {
264
+ const match = versionResult.stdout.match(/aws-cli\/(\S+)/);
265
+ const version = match ? match[1] : 'unknown';
266
+ console.log(`AWS CLI installed successfully (version ${version}).`);
267
+ } else {
268
+ console.log('AWS CLI installed successfully.');
269
+ }
270
+ } else {
271
+ console.log('Installation completed but AWS CLI command not found.');
272
+ console.log('Try opening a new terminal or adding /usr/local/bin to your PATH.');
273
+ }
274
+ }
275
+
276
+ /**
277
+ * Install AWS CLI on Raspberry Pi OS using Snap
278
+ *
279
+ * Important: AWS CLI v2 requires a 64-bit (aarch64) operating system.
280
+ * The 32-bit version of Raspberry Pi OS is not supported.
281
+ *
282
+ * Prerequisites:
283
+ * - Raspberry Pi OS (64-bit) - ARM64/aarch64 architecture
284
+ * - Raspberry Pi 3B+ or later (64-bit capable hardware)
285
+ * - snapd service installed
286
+ * - sudo privileges
287
+ *
288
+ * @returns {Promise<void>}
289
+ */
290
+ async function install_raspbian() {
291
+ // Check if AWS CLI is already installed
292
+ if (isAwsCliInstalled()) {
293
+ const version = await getAwsCliVersion();
294
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
295
+ return;
296
+ }
297
+
298
+ // Verify 64-bit architecture - AWS CLI v2 does not support 32-bit ARM
299
+ const arch = os.getArch();
300
+ if (arch !== 'arm64') {
301
+ console.log('AWS CLI v2 requires a 64-bit operating system.');
302
+ console.log(`Your system architecture is: ${arch}`);
303
+ console.log('Please install the 64-bit version of Raspberry Pi OS.');
304
+ return;
305
+ }
306
+
307
+ // Check if snap is available, install it if not
308
+ if (!snap.isInstalled()) {
309
+ console.log('Snap is not installed. Installing snapd...');
310
+ const aptResult = await apt.install('snapd');
311
+ if (!aptResult.success) {
312
+ console.log('Failed to install snapd.');
313
+ console.log(aptResult.output);
314
+ return;
315
+ }
316
+ console.log('snapd installed. A reboot may be required before continuing.');
317
+ console.log('Run: sudo reboot');
318
+ console.log('Then run this installer again.');
319
+ return;
320
+ }
321
+
322
+ console.log('Installing AWS CLI via Snap...');
323
+ const result = await snap.install(SNAP_PACKAGE_NAME, { classic: true });
324
+
325
+ if (!result.success) {
326
+ console.log('Failed to install AWS CLI via Snap.');
327
+ console.log(result.output);
328
+ return;
329
+ }
330
+
331
+ // Verify the installation succeeded
332
+ const isInstalled = isAwsCliInstalled() || await snap.isSnapInstalled(SNAP_PACKAGE_NAME);
333
+ if (isInstalled) {
334
+ const version = await getAwsCliVersion();
335
+ if (version) {
336
+ console.log(`AWS CLI installed successfully (version ${version}).`);
337
+ } else {
338
+ console.log('AWS CLI installed successfully.');
339
+ console.log('You may need to log out and back in, or add /snap/bin to your PATH.');
340
+ }
341
+ } else {
342
+ console.log('Installation completed but AWS CLI not found.');
343
+ console.log('Try logging out and back in, or run: export PATH=$PATH:/snap/bin');
344
+ }
345
+ }
346
+
347
+ /**
348
+ * Install AWS CLI on Amazon Linux using DNF/YUM or manual installer
349
+ *
350
+ * Amazon Linux 2023: AWS CLI v2 is pre-installed, this upgrades to latest.
351
+ * Amazon Linux 2: Ships with v1, this removes v1 and installs v2 manually.
352
+ *
353
+ * Prerequisites:
354
+ * - Amazon Linux 2023 or Amazon Linux 2
355
+ * - sudo privileges
356
+ * - For AL2: unzip and curl packages
357
+ *
358
+ * @returns {Promise<void>}
359
+ */
360
+ async function install_amazon_linux() {
361
+ // Check if AWS CLI is already installed and which version
362
+ if (isAwsCliInstalled()) {
363
+ const isV2 = await isVersion2();
364
+ if (isV2) {
365
+ const version = await getAwsCliVersion();
366
+ console.log(`AWS CLI v2 is already installed (version ${version}), skipping...`);
367
+ return;
368
+ }
369
+ // AWS CLI v1 is installed, need to remove it first
370
+ console.log('AWS CLI v1 detected. Removing v1 before installing v2...');
371
+ const removeResult = await shell.exec('sudo yum remove -y awscli');
372
+ if (removeResult.code !== 0) {
373
+ console.log('Failed to remove AWS CLI v1.');
374
+ console.log(removeResult.stderr);
375
+ return;
376
+ }
377
+ // Clear bash hash table to forget old aws location
378
+ await shell.exec('hash -r 2>/dev/null || true');
379
+ }
380
+
381
+ // Detect if we're on Amazon Linux 2023 (has dnf) or Amazon Linux 2 (has yum)
382
+ const hasDnf = shell.commandExists('dnf');
383
+
384
+ if (hasDnf) {
385
+ // Amazon Linux 2023 - AWS CLI v2 should be available via DNF
386
+ console.log('Installing/upgrading AWS CLI via DNF...');
387
+ const result = await shell.exec('sudo dnf install -y awscli');
388
+
389
+ if (result.code !== 0) {
390
+ console.log('Failed to install AWS CLI via DNF.');
391
+ console.log(result.stderr);
392
+ return;
393
+ }
394
+ } else {
395
+ // Amazon Linux 2 - Use the official manual installer
396
+ console.log('Installing AWS CLI v2 using the official installer...');
397
+
398
+ // Ensure required tools are available
399
+ if (!shell.commandExists('unzip')) {
400
+ console.log('Installing unzip...');
401
+ await shell.exec('sudo yum install -y unzip');
402
+ }
403
+
404
+ if (!shell.commandExists('curl')) {
405
+ console.log('Installing curl...');
406
+ await shell.exec('sudo yum install -y curl');
407
+ }
408
+
409
+ // Detect architecture and select the appropriate installer
410
+ const arch = os.getArch();
411
+ let installerUrl;
412
+
413
+ if (arch === 'arm64' || arch === 'aarch64') {
414
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-aarch64.zip';
415
+ console.log('Detected ARM64 architecture');
416
+ } else if (arch === 'x64' || arch === 'x86_64') {
417
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip';
418
+ console.log('Detected x86_64 architecture');
419
+ } else {
420
+ console.log(`Unsupported architecture: ${arch}`);
421
+ console.log('AWS CLI v2 supports x86_64 and aarch64 only.');
422
+ return;
423
+ }
424
+
425
+ // Download the installer
426
+ console.log('Downloading AWS CLI v2 installer...');
427
+ const downloadResult = await shell.exec(
428
+ `curl -s "${installerUrl}" -o "/tmp/awscliv2.zip"`
429
+ );
430
+
431
+ if (downloadResult.code !== 0) {
432
+ console.log('Failed to download AWS CLI installer.');
433
+ console.log(downloadResult.stderr);
434
+ return;
435
+ }
436
+
437
+ // Extract the installer
438
+ console.log('Extracting installer...');
439
+ const extractResult = await shell.exec('unzip -o -q /tmp/awscliv2.zip -d /tmp');
440
+
441
+ if (extractResult.code !== 0) {
442
+ console.log('Failed to extract AWS CLI installer.');
443
+ console.log(extractResult.stderr);
444
+ // Clean up
445
+ await shell.exec('rm -f /tmp/awscliv2.zip');
446
+ return;
447
+ }
448
+
449
+ // Run the installer
450
+ console.log('Running installer...');
451
+ const installResult = await shell.exec('sudo /tmp/aws/install');
452
+
453
+ // Clean up regardless of result
454
+ await shell.exec('rm -rf /tmp/awscliv2.zip /tmp/aws');
455
+
456
+ if (installResult.code !== 0) {
457
+ console.log('Failed to run AWS CLI installer.');
458
+ console.log(installResult.stderr);
459
+ return;
460
+ }
461
+ }
462
+
463
+ // Verify the installation succeeded
464
+ // Clear hash table to pick up new aws location
465
+ await shell.exec('hash -r 2>/dev/null || true');
466
+
467
+ if (isAwsCliInstalled()) {
468
+ const version = await getAwsCliVersion();
469
+ console.log(`AWS CLI installed successfully (version ${version}).`);
470
+ } else {
471
+ console.log('Installation completed but AWS CLI command not found.');
472
+ console.log('Try running: hash -r');
473
+ console.log('Or open a new terminal session.');
474
+ }
475
+ }
476
+
477
+ /**
478
+ * Install AWS CLI on Windows using Chocolatey
479
+ *
480
+ * Prerequisites:
481
+ * - Windows 10 or Windows 11 (64-bit)
482
+ * - Chocolatey package manager installed
483
+ * - Administrator PowerShell or Command Prompt
484
+ *
485
+ * @returns {Promise<void>}
486
+ */
487
+ async function install_windows() {
488
+ // Check if AWS CLI is already installed
489
+ if (isAwsCliInstalled()) {
490
+ const version = await getAwsCliVersion();
491
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
492
+ return;
493
+ }
494
+
495
+ // Verify Chocolatey is available
496
+ if (!choco.isInstalled()) {
497
+ console.log('Chocolatey is not installed. Please install Chocolatey first.');
498
+ console.log('Run in an Administrator PowerShell:');
499
+ console.log("Set-ExecutionPolicy Bypass -Scope Process -Force; [System.Net.ServicePointManager]::SecurityProtocol = [System.Net.ServicePointManager]::SecurityProtocol -bor 3072; iex ((New-Object System.Net.WebClient).DownloadString('https://community.chocolatey.org/install.ps1'))");
500
+ return;
501
+ }
502
+
503
+ console.log('Installing AWS CLI via Chocolatey...');
504
+ const result = await choco.install(CHOCO_PACKAGE_NAME);
505
+
506
+ if (!result.success) {
507
+ console.log('Failed to install AWS CLI via Chocolatey.');
508
+ console.log(result.output);
509
+ return;
510
+ }
511
+
512
+ console.log('AWS CLI installed successfully.');
513
+ console.log('You may need to open a new terminal window for the aws command to be available.');
514
+ }
515
+
516
+ /**
517
+ * Install AWS CLI on Ubuntu running in WSL (Windows Subsystem for Linux)
518
+ *
519
+ * WSL Ubuntu follows the same process as native Ubuntu using Snap.
520
+ * If Snap has issues in WSL, falls back to the manual installer method.
521
+ *
522
+ * Prerequisites:
523
+ * - Windows Subsystem for Linux with Ubuntu installed
524
+ * - WSL 2 recommended for best performance
525
+ * - sudo privileges within WSL
526
+ *
527
+ * @returns {Promise<void>}
528
+ */
529
+ async function install_ubuntu_wsl() {
530
+ // Check if AWS CLI is already installed
531
+ if (isAwsCliInstalled()) {
532
+ const version = await getAwsCliVersion();
533
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
534
+ return;
535
+ }
536
+
537
+ // Try Snap first, fall back to manual installer if needed
538
+ // Snap may have issues in WSL due to systemd requirements
539
+ let useManualInstaller = false;
540
+
541
+ if (!snap.isInstalled()) {
542
+ console.log('Snap is not installed. Installing snapd...');
543
+ const aptResult = await apt.install('snapd');
544
+ if (!aptResult.success) {
545
+ console.log('Failed to install snapd. Using manual installer instead...');
546
+ useManualInstaller = true;
547
+ }
548
+ }
549
+
550
+ if (!useManualInstaller) {
551
+ // Try to set up snapd socket if systemd is available
552
+ const systemdCheck = await shell.exec('systemctl --version 2>/dev/null');
553
+ if (systemdCheck.code === 0) {
554
+ await shell.exec('sudo systemctl enable --now snapd.socket 2>/dev/null');
555
+ await shell.exec('sudo ln -s /var/lib/snapd/snap /snap 2>/dev/null || true');
556
+ }
557
+
558
+ console.log('Installing AWS CLI via Snap...');
559
+ const result = await snap.install(SNAP_PACKAGE_NAME, { classic: true });
560
+
561
+ if (!result.success) {
562
+ console.log('Snap installation failed. Using manual installer instead...');
563
+ useManualInstaller = true;
564
+ }
565
+ }
566
+
567
+ // Manual installer fallback for WSL environments where Snap doesn't work
568
+ if (useManualInstaller) {
569
+ console.log('Installing AWS CLI v2 using the official installer...');
570
+
571
+ // Ensure required tools are available
572
+ if (!shell.commandExists('unzip')) {
573
+ console.log('Installing unzip...');
574
+ await apt.install('unzip');
575
+ }
576
+
577
+ if (!shell.commandExists('curl')) {
578
+ console.log('Installing curl...');
579
+ await apt.install('curl');
580
+ }
581
+
582
+ // Detect architecture and select the appropriate installer
583
+ const arch = os.getArch();
584
+ let installerUrl;
585
+
586
+ if (arch === 'arm64' || arch === 'aarch64') {
587
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-aarch64.zip';
588
+ console.log('Detected ARM64 architecture');
589
+ } else if (arch === 'x64' || arch === 'x86_64') {
590
+ installerUrl = 'https://awscli.amazonaws.com/awscli-exe-linux-x86_64.zip';
591
+ console.log('Detected x86_64 architecture');
592
+ } else {
593
+ console.log(`Unsupported architecture: ${arch}`);
594
+ console.log('AWS CLI v2 supports x86_64 and aarch64 only.');
595
+ return;
596
+ }
597
+
598
+ // Download the installer
599
+ console.log('Downloading AWS CLI v2 installer...');
600
+ const downloadResult = await shell.exec(
601
+ `curl -s "${installerUrl}" -o "/tmp/awscliv2.zip"`
602
+ );
603
+
604
+ if (downloadResult.code !== 0) {
605
+ console.log('Failed to download AWS CLI installer.');
606
+ console.log(downloadResult.stderr);
607
+ return;
608
+ }
609
+
610
+ // Extract the installer
611
+ console.log('Extracting installer...');
612
+ const extractResult = await shell.exec('unzip -o -q /tmp/awscliv2.zip -d /tmp');
613
+
614
+ if (extractResult.code !== 0) {
615
+ console.log('Failed to extract AWS CLI installer.');
616
+ console.log(extractResult.stderr);
617
+ await shell.exec('rm -f /tmp/awscliv2.zip');
618
+ return;
619
+ }
620
+
621
+ // Run the installer
622
+ console.log('Running installer...');
623
+ const installResult = await shell.exec('sudo /tmp/aws/install');
624
+
625
+ // Clean up
626
+ await shell.exec('rm -rf /tmp/awscliv2.zip /tmp/aws');
627
+
628
+ if (installResult.code !== 0) {
629
+ console.log('Failed to run AWS CLI installer.');
630
+ console.log(installResult.stderr);
631
+ return;
632
+ }
633
+ }
634
+
635
+ // Verify the installation succeeded
636
+ if (isAwsCliInstalled()) {
637
+ const version = await getAwsCliVersion();
638
+ console.log(`AWS CLI installed successfully (version ${version}).`);
639
+ } else {
640
+ const snapInstalled = await snap.isSnapInstalled(SNAP_PACKAGE_NAME);
641
+ if (snapInstalled) {
642
+ console.log('AWS CLI installed successfully.');
643
+ console.log('You may need to restart your WSL session or add paths to PATH:');
644
+ console.log('export PATH=$PATH:/snap/bin:/usr/local/bin');
645
+ } else {
646
+ console.log('Installation completed but AWS CLI command not found.');
647
+ console.log('Try opening a new terminal or adding /usr/local/bin to your PATH.');
648
+ }
649
+ }
650
+ }
9
651
 
10
652
  /**
11
- * Install AWS CLI across supported platforms.
653
+ * Install AWS CLI in Git Bash on Windows
654
+ *
655
+ * Git Bash inherits the Windows PATH, so once AWS CLI is installed on Windows
656
+ * via Chocolatey or the MSI installer, it is automatically available in Git Bash.
657
+ * This function calls the Windows installer.
658
+ *
659
+ * Prerequisites:
660
+ * - Windows 10 or Windows 11 (64-bit)
661
+ * - Git Bash installed (comes with Git for Windows)
662
+ * - Chocolatey installed on Windows
663
+ *
664
+ * @returns {Promise<void>}
665
+ */
666
+ async function install_gitbash() {
667
+ // Check if AWS CLI is already available (inherited from Windows PATH)
668
+ if (isAwsCliInstalled()) {
669
+ const version = await getAwsCliVersion();
670
+ console.log(`AWS CLI is already installed (version ${version}), skipping...`);
671
+ return;
672
+ }
673
+
674
+ // Check if Chocolatey is available via Windows path
675
+ const chocoPath = '/c/ProgramData/chocolatey/bin/choco.exe';
676
+ const chocoResult = await shell.exec(`"${chocoPath}" --version 2>/dev/null`);
677
+
678
+ if (chocoResult.code !== 0) {
679
+ console.log('Chocolatey is not installed on Windows.');
680
+ console.log('Please install Chocolatey first, then run this installer again.');
681
+ console.log('Or install AWS CLI manually using the MSI installer from:');
682
+ console.log('https://awscli.amazonaws.com/AWSCLIV2.msi');
683
+ return;
684
+ }
685
+
686
+ console.log('Installing AWS CLI via Windows Chocolatey...');
687
+ const result = await shell.exec(`"${chocoPath}" install awscli -y`);
688
+
689
+ if (result.code !== 0) {
690
+ console.log('Failed to install AWS CLI via Chocolatey.');
691
+ console.log(result.stderr);
692
+ return;
693
+ }
694
+
695
+ console.log('AWS CLI installed successfully.');
696
+ console.log('Please close and reopen Git Bash for the aws command to be available.');
697
+ console.log('If aws is still not found, add this to your ~/.bashrc:');
698
+ console.log('export PATH=$PATH:"/c/Program Files/Amazon/AWSCLIV2"');
699
+ }
700
+
701
+ /**
702
+ * Check if AWS CLI is installed on the current platform.
703
+ *
704
+ * This function checks if the AWS CLI command exists in PATH.
705
+ * It uses the internal isAwsCliInstalled helper.
706
+ *
707
+ * @returns {Promise<boolean>} True if AWS CLI is installed
708
+ */
709
+ async function isInstalled() {
710
+ return isAwsCliInstalled();
711
+ }
712
+
713
+ /**
714
+ * Check if this installer is supported on the current platform.
715
+ *
716
+ * AWS CLI can be installed on:
717
+ * - macOS (via Homebrew)
718
+ * - Ubuntu/Debian (via Snap)
719
+ * - Raspberry Pi OS (via Snap, 64-bit only)
720
+ * - Amazon Linux/RHEL/Fedora (via DNF/YUM or manual installer)
721
+ * - Windows (via Chocolatey)
722
+ * - WSL (via Snap or manual installer)
723
+ *
724
+ * @returns {boolean} True if installation is supported on this platform
725
+ */
726
+ function isEligible() {
727
+ const platform = os.detect();
728
+ return ['macos', 'ubuntu', 'debian', 'raspbian', 'amazon_linux', 'rhel', 'fedora', 'wsl', 'windows'].includes(platform.type);
729
+ }
730
+
731
+ /**
732
+ * Main installation entry point - detects platform and runs appropriate installer
733
+ *
734
+ * Supported platforms:
735
+ * - macos: Uses Homebrew
736
+ * - ubuntu: Uses Snap
737
+ * - debian: Uses Snap (same as ubuntu)
738
+ * - raspbian: Uses Snap (requires 64-bit OS)
739
+ * - amazon_linux: Uses DNF (AL2023) or manual installer (AL2)
740
+ * - rhel: Uses same approach as amazon_linux
741
+ * - fedora: Uses DNF
742
+ * - wsl: Uses Snap or manual installer
743
+ * - windows: Uses Chocolatey
744
+ * - gitbash: Uses Windows Chocolatey via Git Bash
12
745
  *
13
746
  * @returns {Promise<void>}
14
747
  */
15
748
  async function install() {
16
749
  const platform = os.detect();
17
750
 
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 };
751
+ // Map platform types to their installer functions
752
+ const installers = {
753
+ 'macos': install_macos,
754
+ 'ubuntu': install_ubuntu,
755
+ 'debian': install_ubuntu,
756
+ 'raspbian': install_raspbian,
757
+ 'amazon_linux': install_amazon_linux,
758
+ 'rhel': install_amazon_linux,
759
+ 'fedora': install_amazon_linux,
760
+ 'wsl': install_ubuntu_wsl,
761
+ 'windows': install_windows,
762
+ };
763
+
764
+ const installer = installers[platform.type];
765
+
766
+ if (!installer) {
767
+ // For unknown platforms, gracefully inform the user without throwing an error
768
+ console.log(`AWS CLI is not available for ${platform.type}.`);
769
+ return;
770
+ }
771
+
772
+ await installer();
773
+ }
774
+
775
+ module.exports = {
776
+ install,
777
+ isInstalled,
778
+ isEligible,
779
+ install_macos,
780
+ install_ubuntu,
781
+ install_ubuntu_wsl,
782
+ install_raspbian,
783
+ install_amazon_linux,
784
+ install_windows,
785
+ install_gitbash,
786
+ // Export helper functions for potential reuse or testing
787
+ isAwsCliInstalled,
788
+ getAwsCliVersion,
789
+ isVersion2,
790
+ };
41
791
 
42
792
  if (require.main === module) {
43
- install();
793
+ install().catch(err => {
794
+ console.error(err.message);
795
+ process.exit(1);
796
+ });
44
797
  }