@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,579 @@
1
1
  #!/usr/bin/env node
2
2
 
3
3
  /**
4
- * @fileoverview Install Cursor.
4
+ * @fileoverview Install Cursor - AI-powered code editor built on VS Code.
5
5
  * @module installs/cursor
6
+ *
7
+ * Cursor is available for macOS, Windows, and Linux (including Raspberry Pi OS
8
+ * and Amazon Linux/RHEL). This installer handles platform detection and uses
9
+ * the appropriate package manager or direct download for each platform.
10
+ *
11
+ * Supported platforms:
12
+ * - macOS: Homebrew cask
13
+ * - Ubuntu/Debian: Direct .deb download from cursor.com
14
+ * - Raspberry Pi OS: Direct .deb download (ARM64 only, requires 64-bit OS)
15
+ * - Amazon Linux/RHEL: Direct .rpm download from cursor.com
16
+ * - Windows: winget (preferred) or Chocolatey
17
+ * - WSL: Installs on Windows host via winget
18
+ * - Git Bash: Installs on Windows host via winget
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 apt = require('../utils/ubuntu/apt');
25
+ const winget = require('../utils/windows/winget');
26
+ const macosApps = require('../utils/macos/apps');
27
+
28
+ /**
29
+ * Whether this installer requires a desktop environment to function.
30
+ * Cursor is a GUI code editor.
31
+ */
32
+ const REQUIRES_DESKTOP = true;
33
+
34
+ /**
35
+ * The name of the Cursor application bundle on macOS
36
+ * Used for checking if Cursor is already installed
37
+ */
38
+ const MACOS_APP_NAME = 'Cursor';
39
+
40
+ /**
41
+ * The Homebrew cask name for Cursor
42
+ */
43
+ const HOMEBREW_CASK_NAME = 'cursor';
44
+
45
+ /**
46
+ * The winget package ID for Cursor
47
+ */
48
+ const WINGET_PACKAGE_ID = 'Anysphere.Cursor';
49
+
50
+ /**
51
+ * Download URLs for Cursor packages by platform
52
+ * These URLs redirect to the latest stable release
53
+ */
54
+ const DOWNLOAD_URLS = {
55
+ 'deb-x64': 'https://www.cursor.com/api/download?platform=linux-deb-x64&releaseTrack=stable',
56
+ 'deb-arm64': 'https://www.cursor.com/api/download?platform=linux-deb-arm64&releaseTrack=stable',
57
+ 'rpm-x64': 'https://www.cursor.com/api/download?platform=linux-rpm-x64&releaseTrack=stable',
58
+ 'rpm-arm64': 'https://www.cursor.com/api/download?platform=linux-rpm-arm64&releaseTrack=stable'
59
+ };
60
+
61
+ /**
62
+ * Install Cursor on macOS using Homebrew
63
+ *
64
+ * This function checks if Cursor is already installed by looking for
65
+ * /Applications/Cursor.app. If not installed, it uses Homebrew to install
66
+ * the Cursor cask.
67
+ *
68
+ * @returns {Promise<void>}
69
+ */
70
+ async function install_macos() {
71
+ // Check if Cursor is already installed by looking for the .app bundle
72
+ const isInstalled = macosApps.isAppInstalled(MACOS_APP_NAME);
73
+ if (isInstalled) {
74
+ console.log('Cursor is already installed.');
75
+ return;
76
+ }
77
+
78
+ // Verify Homebrew is available before proceeding
79
+ if (!brew.isInstalled()) {
80
+ console.log('Homebrew is required to install Cursor on macOS.');
81
+ console.log('Please install Homebrew first: https://brew.sh');
82
+ return;
83
+ }
84
+
85
+ console.log('Installing Cursor via Homebrew...');
86
+
87
+ // Install using the Homebrew cask
88
+ const result = await brew.installCask(HOMEBREW_CASK_NAME);
89
+
90
+ if (!result.success) {
91
+ console.log('Failed to install Cursor.');
92
+ console.log(result.output);
93
+ return;
94
+ }
95
+
96
+ // Verify installation succeeded by checking for the app bundle
97
+ const verifyInstalled = macosApps.isAppInstalled(MACOS_APP_NAME);
98
+ if (verifyInstalled) {
99
+ console.log('Cursor installed successfully.');
100
+ } else {
101
+ console.log('Installation completed but Cursor.app was not found in /Applications.');
102
+ console.log('You may need to run the installer again or install manually.');
103
+ }
104
+ }
105
+
106
+ /**
107
+ * Install Cursor on Ubuntu/Debian using direct .deb download
108
+ *
109
+ * This function downloads the latest Cursor .deb package from the official
110
+ * download URL and installs it using apt-get. It automatically detects
111
+ * whether the system is x64 or ARM64 and downloads the appropriate package.
112
+ *
113
+ * @returns {Promise<void>}
114
+ */
115
+ async function install_ubuntu() {
116
+ // Check if Cursor is already installed by checking for the cursor command
117
+ const isInstalled = shell.commandExists('cursor');
118
+ if (isInstalled) {
119
+ console.log('Cursor is already installed.');
120
+ return;
121
+ }
122
+
123
+ // Determine the correct package URL based on system architecture
124
+ const arch = os.getArch();
125
+ let downloadUrl;
126
+
127
+ if (arch === 'arm64') {
128
+ downloadUrl = DOWNLOAD_URLS['deb-arm64'];
129
+ } else if (arch === 'x64') {
130
+ downloadUrl = DOWNLOAD_URLS['deb-x64'];
131
+ } else {
132
+ console.log(`Cursor is not available for ${arch} architecture on Ubuntu/Debian.`);
133
+ return;
134
+ }
135
+
136
+ console.log('Downloading Cursor .deb package...');
137
+
138
+ // Download the .deb package to a temporary location
139
+ const downloadResult = await shell.exec(
140
+ `curl -fsSL "${downloadUrl}" -o /tmp/cursor.deb`
141
+ );
142
+
143
+ if (downloadResult.code !== 0) {
144
+ console.log('Failed to download Cursor package.');
145
+ console.log(downloadResult.stderr);
146
+ return;
147
+ }
148
+
149
+ console.log('Installing Cursor...');
150
+
151
+ // Install the downloaded package using apt-get
152
+ // DEBIAN_FRONTEND=noninteractive ensures no prompts during installation
153
+ const installResult = await shell.exec(
154
+ 'sudo DEBIAN_FRONTEND=noninteractive apt-get install -y /tmp/cursor.deb'
155
+ );
156
+
157
+ // Clean up the downloaded package regardless of installation result
158
+ await shell.exec('rm -f /tmp/cursor.deb');
159
+
160
+ if (installResult.code !== 0) {
161
+ console.log('Failed to install Cursor.');
162
+ console.log(installResult.stderr);
163
+ return;
164
+ }
165
+
166
+ // Verify installation succeeded
167
+ const verifyInstalled = shell.commandExists('cursor');
168
+ if (verifyInstalled) {
169
+ console.log('Cursor installed successfully.');
170
+ } else {
171
+ console.log('Installation completed but cursor command was not found.');
172
+ console.log('You may need to restart your terminal or install the shell command from within Cursor.');
173
+ }
174
+ }
175
+
176
+ /**
177
+ * Install Cursor on Raspberry Pi OS using direct .deb download
178
+ *
179
+ * This function is similar to install_ubuntu but specifically handles
180
+ * Raspberry Pi OS. It requires a 64-bit ARM64 system - Cursor does not
181
+ * support 32-bit ARM (armv7l) systems.
182
+ *
183
+ * @returns {Promise<void>}
184
+ */
185
+ async function install_raspbian() {
186
+ // Check if Cursor is already installed
187
+ const isInstalled = shell.commandExists('cursor');
188
+ if (isInstalled) {
189
+ console.log('Cursor is already installed.');
190
+ return;
191
+ }
192
+
193
+ // Raspberry Pi requires 64-bit ARM64 architecture
194
+ // 32-bit Raspberry Pi OS (armv7l) is not supported
195
+ const arch = os.getArch();
196
+ if (arch !== 'arm64') {
197
+ console.log('Cursor requires 64-bit Raspberry Pi OS (ARM64/aarch64).');
198
+ console.log(`Your system architecture is ${arch}.`);
199
+ console.log('Please install 64-bit Raspberry Pi OS to use Cursor.');
200
+ return;
201
+ }
202
+
203
+ console.log('Downloading Cursor .deb package for ARM64...');
204
+
205
+ // Download the ARM64 .deb package
206
+ const downloadResult = await shell.exec(
207
+ `curl -fsSL "${DOWNLOAD_URLS['deb-arm64']}" -o /tmp/cursor.deb`
208
+ );
209
+
210
+ if (downloadResult.code !== 0) {
211
+ console.log('Failed to download Cursor package.');
212
+ console.log(downloadResult.stderr);
213
+ return;
214
+ }
215
+
216
+ console.log('Installing Cursor...');
217
+
218
+ // Install the downloaded package
219
+ const installResult = await shell.exec(
220
+ 'sudo DEBIAN_FRONTEND=noninteractive apt-get install -y /tmp/cursor.deb'
221
+ );
222
+
223
+ // Clean up the downloaded package
224
+ await shell.exec('rm -f /tmp/cursor.deb');
225
+
226
+ if (installResult.code !== 0) {
227
+ console.log('Failed to install Cursor.');
228
+ console.log(installResult.stderr);
229
+ return;
230
+ }
231
+
232
+ // Verify installation
233
+ const verifyInstalled = shell.commandExists('cursor');
234
+ if (verifyInstalled) {
235
+ console.log('Cursor installed successfully.');
236
+ } else {
237
+ console.log('Installation completed but cursor command was not found.');
238
+ console.log('You may need to restart your terminal or install the shell command from within Cursor.');
239
+ }
240
+ }
241
+
242
+ /**
243
+ * Install Cursor on Amazon Linux/RHEL using direct .rpm download
244
+ *
245
+ * This function downloads the latest Cursor .rpm package and installs it
246
+ * using dnf (preferred) or yum. It supports both x64 and ARM64 architectures.
247
+ *
248
+ * @returns {Promise<void>}
249
+ */
250
+ async function install_amazon_linux() {
251
+ // Check if Cursor is already installed
252
+ const isInstalled = shell.commandExists('cursor');
253
+ if (isInstalled) {
254
+ console.log('Cursor is already installed.');
255
+ return;
256
+ }
257
+
258
+ // Determine the correct package URL based on system architecture
259
+ const arch = os.getArch();
260
+ let downloadUrl;
261
+
262
+ if (arch === 'arm64') {
263
+ downloadUrl = DOWNLOAD_URLS['rpm-arm64'];
264
+ } else if (arch === 'x64') {
265
+ downloadUrl = DOWNLOAD_URLS['rpm-x64'];
266
+ } else {
267
+ console.log(`Cursor is not available for ${arch} architecture on Amazon Linux/RHEL.`);
268
+ return;
269
+ }
270
+
271
+ console.log('Downloading Cursor .rpm package...');
272
+
273
+ // Download the .rpm package
274
+ const downloadResult = await shell.exec(
275
+ `curl -fsSL "${downloadUrl}" -o /tmp/cursor.rpm`
276
+ );
277
+
278
+ if (downloadResult.code !== 0) {
279
+ console.log('Failed to download Cursor package.');
280
+ console.log(downloadResult.stderr);
281
+ return;
282
+ }
283
+
284
+ console.log('Installing Cursor...');
285
+
286
+ // Detect package manager: prefer dnf over yum
287
+ // dnf is the default on Amazon Linux 2023, RHEL 8+, and Fedora
288
+ // yum is used on Amazon Linux 2 and older RHEL versions
289
+ const useDnf = shell.commandExists('dnf');
290
+ const packageManager = useDnf ? 'dnf' : 'yum';
291
+
292
+ const installResult = await shell.exec(
293
+ `sudo ${packageManager} install -y /tmp/cursor.rpm`
294
+ );
295
+
296
+ // Clean up the downloaded package
297
+ await shell.exec('rm -f /tmp/cursor.rpm');
298
+
299
+ if (installResult.code !== 0) {
300
+ console.log('Failed to install Cursor.');
301
+ console.log(installResult.stderr);
302
+ return;
303
+ }
304
+
305
+ // Verify installation
306
+ const verifyInstalled = shell.commandExists('cursor');
307
+ if (verifyInstalled) {
308
+ console.log('Cursor installed successfully.');
309
+ } else {
310
+ console.log('Installation completed but cursor command was not found.');
311
+ console.log('You may need to restart your terminal or install the shell command from within Cursor.');
312
+ }
313
+ }
314
+
315
+ /**
316
+ * Install Cursor on Windows using winget (preferred) or Chocolatey
317
+ *
318
+ * This function uses winget to install Cursor on Windows. Winget is the
319
+ * preferred package manager as it is built into Windows 10 1809+ and
320
+ * Windows 11. If winget is not available, it will inform the user.
321
+ *
322
+ * @returns {Promise<void>}
323
+ */
324
+ async function install_windows() {
325
+ // Check if Cursor is already installed via winget
326
+ const isInstalledViaWinget = await winget.isPackageInstalled(WINGET_PACKAGE_ID);
327
+ if (isInstalledViaWinget) {
328
+ console.log('Cursor is already installed.');
329
+ return;
330
+ }
331
+
332
+ // Also check if cursor command exists (may have been installed manually)
333
+ const cursorCommandExists = shell.commandExists('cursor');
334
+ if (cursorCommandExists) {
335
+ console.log('Cursor is already installed.');
336
+ return;
337
+ }
338
+
339
+ // Verify winget is available
340
+ if (!winget.isInstalled()) {
341
+ console.log('winget is required to install Cursor on Windows.');
342
+ console.log('winget is included with Windows 10 version 1809 and later.');
343
+ console.log('You can also install it from the Microsoft Store (App Installer).');
344
+ return;
345
+ }
346
+
347
+ console.log('Installing Cursor via winget...');
348
+
349
+ // Install Cursor using winget with silent flags and auto-accept agreements
350
+ const result = await winget.install(WINGET_PACKAGE_ID);
351
+
352
+ if (!result.success) {
353
+ console.log('Failed to install Cursor.');
354
+ console.log(result.output);
355
+ return;
356
+ }
357
+
358
+ console.log('Cursor installed successfully.');
359
+ console.log('You may need to restart your terminal for the cursor command to be available.');
360
+ }
361
+
362
+ /**
363
+ * Install Cursor on Ubuntu running in WSL (Windows Subsystem for Linux)
364
+ *
365
+ * For WSL, Cursor should be installed on the Windows host and then connected
366
+ * to WSL using Cursor's remote development capabilities. This function installs
367
+ * Cursor on the Windows host via winget.
368
+ *
369
+ * @returns {Promise<void>}
370
+ */
371
+ async function install_ubuntu_wsl() {
372
+ // Check if Cursor is already installed on Windows host
373
+ // We use winget.exe to query from within WSL
374
+ const checkResult = await shell.exec(
375
+ `winget.exe list --exact --id "${WINGET_PACKAGE_ID}" 2>/dev/null`
376
+ );
377
+
378
+ if (checkResult.code === 0 && checkResult.stdout.includes(WINGET_PACKAGE_ID)) {
379
+ console.log('Cursor is already installed on Windows.');
380
+ console.log('To connect to WSL, open Cursor on Windows and use "Connect to WSL" from the Remote menu.');
381
+ return;
382
+ }
383
+
384
+ // Verify winget is accessible from WSL
385
+ const wingetExists = shell.commandExists('winget.exe');
386
+ if (!wingetExists) {
387
+ console.log('winget is required to install Cursor from WSL.');
388
+ console.log('winget should be available if you are running Windows 10 1809+ or Windows 11.');
389
+ return;
390
+ }
391
+
392
+ console.log('Installing Cursor on Windows host via winget...');
393
+
394
+ // Install Cursor on Windows using winget.exe from within WSL
395
+ const installResult = await shell.exec(
396
+ `winget.exe install --id "${WINGET_PACKAGE_ID}" --silent --accept-package-agreements --accept-source-agreements`
397
+ );
398
+
399
+ if (installResult.code !== 0) {
400
+ console.log('Failed to install Cursor.');
401
+ console.log(installResult.stderr);
402
+ return;
403
+ }
404
+
405
+ console.log('Cursor installed successfully on Windows.');
406
+ console.log('To use Cursor with WSL:');
407
+ console.log(' 1. Open Cursor on Windows');
408
+ console.log(' 2. Click the Remote icon in the bottom-left corner');
409
+ console.log(' 3. Select "Connect to WSL" and choose your distribution');
410
+ }
411
+
412
+ /**
413
+ * Install Cursor on Windows via Git Bash
414
+ *
415
+ * Git Bash runs on Windows, so we install Cursor as a Windows application
416
+ * using winget. The cursor command will then be available in Git Bash.
417
+ *
418
+ * @returns {Promise<void>}
419
+ */
420
+ async function install_gitbash() {
421
+ // Check if Cursor is already installed
422
+ // Try winget first (works in Git Bash as winget.exe)
423
+ const checkResult = await shell.exec(
424
+ `winget.exe list --exact --id "${WINGET_PACKAGE_ID}" 2>/dev/null`
425
+ );
426
+
427
+ if (checkResult.code === 0 && checkResult.stdout.includes(WINGET_PACKAGE_ID)) {
428
+ console.log('Cursor is already installed.');
429
+ return;
430
+ }
431
+
432
+ // Verify winget is accessible
433
+ const wingetExists = shell.commandExists('winget.exe');
434
+ if (!wingetExists) {
435
+ console.log('winget is required to install Cursor.');
436
+ console.log('winget should be available if you are running Windows 10 1809+ or Windows 11.');
437
+ return;
438
+ }
439
+
440
+ console.log('Installing Cursor via winget...');
441
+
442
+ // Install Cursor using winget.exe
443
+ const installResult = await shell.exec(
444
+ `winget.exe install --id "${WINGET_PACKAGE_ID}" --silent --accept-package-agreements --accept-source-agreements`
445
+ );
446
+
447
+ if (installResult.code !== 0) {
448
+ console.log('Failed to install Cursor.');
449
+ console.log(installResult.stderr);
450
+ return;
451
+ }
452
+
453
+ console.log('Cursor installed successfully.');
454
+ console.log('You may need to restart Git Bash for the cursor command to be available.');
455
+ }
9
456
 
10
457
  /**
11
- * Install Cursor across supported platforms.
458
+ * Check if Cursor is installed on the current platform.
459
+ *
460
+ * This function performs platform-specific checks to determine if Cursor
461
+ * is already installed:
462
+ * - macOS: Checks for Cursor.app in /Applications
463
+ * - Windows/Git Bash/WSL: Checks for winget package
464
+ * - Ubuntu/Debian/Raspberry Pi/Amazon Linux: Checks for cursor command
465
+ *
466
+ * @returns {Promise<boolean>} True if Cursor is installed
467
+ */
468
+ async function isInstalled() {
469
+ const platform = os.detect();
470
+
471
+ // macOS: Check for Cursor.app
472
+ if (platform.type === 'macos') {
473
+ return macosApps.isAppInstalled(MACOS_APP_NAME);
474
+ }
475
+
476
+ // Windows: Check via winget
477
+ if (platform.type === 'windows') {
478
+ return await winget.isPackageInstalled(WINGET_PACKAGE_ID);
479
+ }
480
+
481
+ // WSL and Git Bash: Check via winget.exe
482
+ if (platform.type === 'wsl' || platform.type === 'gitbash') {
483
+ const checkResult = await shell.exec(
484
+ `winget.exe list --exact --id "${WINGET_PACKAGE_ID}" 2>/dev/null`
485
+ );
486
+ return checkResult.code === 0 && checkResult.stdout.includes(WINGET_PACKAGE_ID);
487
+ }
488
+
489
+ // Ubuntu/Debian/Raspberry Pi/Amazon Linux: Check for cursor command
490
+ if (['ubuntu', 'debian', 'raspbian', 'amazon_linux', 'rhel', 'fedora'].includes(platform.type)) {
491
+ return shell.commandExists('cursor');
492
+ }
493
+
494
+ return false;
495
+ }
496
+
497
+ /**
498
+ * Check if this installer is supported on the current platform.
499
+ *
500
+ * Cursor can be installed on:
501
+ * - macOS (via Homebrew cask)
502
+ * - Ubuntu/Debian (via direct .deb download)
503
+ * - Raspberry Pi OS (via direct .deb download, ARM64 only)
504
+ * - Amazon Linux/RHEL/Fedora (via direct .rpm download)
505
+ * - Windows (via winget)
506
+ * - WSL (installs on Windows host via winget)
507
+ * - Git Bash (installs on Windows host via winget)
508
+ *
509
+ * @returns {boolean} True if installation is supported on this platform
510
+ */
511
+ function isEligible() {
512
+ const platform = os.detect();
513
+ const supportedPlatforms = ['macos', 'ubuntu', 'debian', 'wsl', 'raspbian', 'amazon_linux', 'rhel', 'fedora', 'windows', 'gitbash'];
514
+ if (!supportedPlatforms.includes(platform.type)) {
515
+ return false;
516
+ }
517
+ if (REQUIRES_DESKTOP && !os.isDesktopAvailable()) {
518
+ return false;
519
+ }
520
+ return true;
521
+ }
522
+
523
+ /**
524
+ * Main installation entry point - detects platform and runs appropriate installer
525
+ *
526
+ * This function uses os.detect() to determine the current platform and
527
+ * delegates to the appropriate platform-specific installation function.
12
528
  *
13
529
  * @returns {Promise<void>}
14
530
  */
15
531
  async function install() {
16
532
  const platform = os.detect();
17
533
 
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 };
534
+ // Map platform types to their installer functions
535
+ // Some platforms share installers (e.g., debian uses ubuntu installer)
536
+ const installers = {
537
+ 'macos': install_macos,
538
+ 'ubuntu': install_ubuntu,
539
+ 'debian': install_ubuntu,
540
+ 'wsl': install_ubuntu_wsl,
541
+ 'raspbian': install_raspbian,
542
+ 'amazon_linux': install_amazon_linux,
543
+ 'rhel': install_amazon_linux,
544
+ 'fedora': install_amazon_linux,
545
+ 'windows': install_windows
546
+ };
547
+
548
+ const installer = installers[platform.type];
549
+
550
+ if (!installer) {
551
+ // For unsupported platforms, log a friendly message and return gracefully
552
+ // Do NOT throw an error - per project requirements
553
+ console.log(`Cursor is not available for ${platform.type}.`);
554
+ return;
555
+ }
556
+
557
+ await installer();
558
+ }
559
+
560
+ module.exports = {
561
+ REQUIRES_DESKTOP,
562
+ install,
563
+ isInstalled,
564
+ isEligible,
565
+ install_macos,
566
+ install_ubuntu,
567
+ install_ubuntu_wsl,
568
+ install_raspbian,
569
+ install_amazon_linux,
570
+ install_windows,
571
+ install_gitbash
572
+ };
41
573
 
42
574
  if (require.main === module) {
43
- install();
575
+ install().catch(err => {
576
+ console.error(err.message);
577
+ process.exit(1);
578
+ });
44
579
  }