@fredlackey/devutils 0.0.1 → 0.0.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (257) 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
@@ -1,23 +1,577 @@
1
1
  #!/usr/bin/env node
2
2
 
3
3
  /**
4
- * @fileoverview Kill Node Inspector processes.
4
+ * killni - Kill Node Inspector processes
5
+ *
6
+ * Migrated from legacy dotfiles alias.
7
+ * Original:
8
+ * killni() {
9
+ * killni_target='node --debug-brk'
10
+ * ps -ef | grep "$killni_target" | grep -v grep | awk '{print $2}' | xargs kill -9
11
+ * }
12
+ *
13
+ * This script finds and forcibly terminates all Node.js processes that were
14
+ * started with the --debug-brk flag. This flag was used by the legacy Node
15
+ * Inspector debugger to pause execution at the first line of code, waiting
16
+ * for a debugger to attach.
17
+ *
18
+ * Note: The --debug-brk flag is deprecated in modern Node.js versions.
19
+ * Modern Node.js uses --inspect and --inspect-brk instead. This script
20
+ * kills processes using either the legacy or modern debug flags.
21
+ *
5
22
  * @module scripts/killni
6
23
  */
7
24
 
25
+ const os = require('../utils/common/os');
26
+ const { execSync, spawnSync } = require('child_process');
27
+
28
+ /**
29
+ * Helper function to check if a command exists on the system.
30
+ *
31
+ * @param {string} cmd - The command name to check
32
+ * @returns {boolean} True if the command exists, false otherwise
33
+ */
34
+ function isCommandAvailable(cmd) {
35
+ try {
36
+ const checkCmd = process.platform === 'win32' ? `where ${cmd}` : `which ${cmd}`;
37
+ execSync(checkCmd, { stdio: 'ignore' });
38
+ return true;
39
+ } catch {
40
+ return false;
41
+ }
42
+ }
43
+
44
+ /**
45
+ * Pure Node.js implementation that cannot be used for this script.
46
+ *
47
+ * Process enumeration and forceful termination require OS-level commands.
48
+ * Node.js can only terminate processes it has a reference to (via child_process),
49
+ * and process.kill() requires knowing the PID in advance. Enumerating all
50
+ * system processes cannot be done reliably with pure Node.js across platforms.
51
+ *
52
+ * @param {string[]} args - Command line arguments (unused)
53
+ * @returns {Promise<void>}
54
+ * @throws {Error} Always throws - this function should not be called directly
55
+ */
56
+ async function do_killni_nodejs(args) {
57
+ // Process enumeration requires OS-specific commands:
58
+ // - Unix: ps -ef, pgrep, or reading /proc
59
+ // - Windows: tasklist, wmic, or Get-Process
60
+ // There is no cross-platform pure Node.js way to enumerate all processes.
61
+ throw new Error(
62
+ 'do_killni_nodejs should not be called directly. ' +
63
+ 'Process enumeration and termination require OS-specific commands.'
64
+ );
65
+ }
66
+
67
+ /**
68
+ * Kill Node Inspector processes on macOS.
69
+ *
70
+ * Uses the native 'ps' and 'grep' commands to find processes, then sends
71
+ * SIGKILL (-9) to forcibly terminate them. This matches the original
72
+ * dotfiles behavior.
73
+ *
74
+ * Searches for both legacy (--debug-brk) and modern (--inspect-brk) flags.
75
+ *
76
+ * @param {string[]} args - Command line arguments (unused)
77
+ * @returns {Promise<void>}
78
+ */
79
+ async function do_killni_macos(args) {
80
+ // Patterns to search for - both legacy and modern debug flags
81
+ const patterns = ['--debug-brk', '--inspect-brk', '--debug', '--inspect'];
82
+ let totalKilled = 0;
83
+
84
+ for (const pattern of patterns) {
85
+ try {
86
+ // Get list of PIDs matching the pattern
87
+ // Using pgrep if available (cleaner), otherwise fall back to ps | grep
88
+ let pids = [];
89
+
90
+ if (isCommandAvailable('pgrep')) {
91
+ // pgrep -f searches the full command line
92
+ // pgrep returns non-zero if no processes found, which is not an error for us
93
+ try {
94
+ const result = execSync(`pgrep -f "${pattern}"`, { encoding: 'utf8' });
95
+ pids = result.trim().split('\n').filter(pid => pid.length > 0);
96
+ } catch {
97
+ // No processes found - this is fine, continue to next pattern
98
+ pids = [];
99
+ }
100
+ } else {
101
+ // Fallback to ps | grep approach (original dotfiles method)
102
+ try {
103
+ const psOutput = execSync(
104
+ `ps -ef | grep "${pattern}" | grep -v grep | awk '{print $2}'`,
105
+ { encoding: 'utf8' }
106
+ );
107
+ pids = psOutput.trim().split('\n').filter(pid => pid.length > 0);
108
+ } catch {
109
+ pids = [];
110
+ }
111
+ }
112
+
113
+ // Kill each process found
114
+ for (const pid of pids) {
115
+ // Skip our own process
116
+ if (parseInt(pid, 10) === process.pid) {
117
+ continue;
118
+ }
119
+
120
+ try {
121
+ // Use kill -9 for forceful termination (matches original alias)
122
+ execSync(`kill -9 ${pid}`, { stdio: 'ignore' });
123
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
124
+ totalKilled++;
125
+ } catch {
126
+ // Process may have already exited - this is fine
127
+ }
128
+ }
129
+ } catch (error) {
130
+ // Error searching for this pattern - continue to next
131
+ continue;
132
+ }
133
+ }
134
+
135
+ if (totalKilled === 0) {
136
+ console.log('No Node Inspector processes found.');
137
+ } else {
138
+ console.log(`Total processes killed: ${totalKilled}`);
139
+ }
140
+ }
141
+
142
+ /**
143
+ * Kill Node Inspector processes on Ubuntu.
144
+ *
145
+ * Uses pgrep/pkill or ps/grep approach depending on what's available.
146
+ * Linux typically has these tools installed by default.
147
+ *
148
+ * @param {string[]} args - Command line arguments (unused)
149
+ * @returns {Promise<void>}
150
+ */
151
+ async function do_killni_ubuntu(args) {
152
+ // Ubuntu typically has pgrep/pkill available
153
+ const patterns = ['--debug-brk', '--inspect-brk', '--debug', '--inspect'];
154
+ let totalKilled = 0;
155
+
156
+ for (const pattern of patterns) {
157
+ try {
158
+ let pids = [];
159
+
160
+ // Try pgrep first (cleaner)
161
+ if (isCommandAvailable('pgrep')) {
162
+ try {
163
+ const result = execSync(`pgrep -f "${pattern}"`, { encoding: 'utf8' });
164
+ pids = result.trim().split('\n').filter(pid => pid.length > 0);
165
+ } catch {
166
+ pids = [];
167
+ }
168
+ } else {
169
+ // Fallback to ps | grep (original approach)
170
+ try {
171
+ const psOutput = execSync(
172
+ `ps -ef | grep "${pattern}" | grep -v grep | awk '{print $2}'`,
173
+ { encoding: 'utf8' }
174
+ );
175
+ pids = psOutput.trim().split('\n').filter(pid => pid.length > 0);
176
+ } catch {
177
+ pids = [];
178
+ }
179
+ }
180
+
181
+ for (const pid of pids) {
182
+ if (parseInt(pid, 10) === process.pid) {
183
+ continue;
184
+ }
185
+
186
+ try {
187
+ execSync(`kill -9 ${pid}`, { stdio: 'ignore' });
188
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
189
+ totalKilled++;
190
+ } catch {
191
+ // Process may have already exited
192
+ }
193
+ }
194
+ } catch {
195
+ continue;
196
+ }
197
+ }
198
+
199
+ if (totalKilled === 0) {
200
+ console.log('No Node Inspector processes found.');
201
+ } else {
202
+ console.log(`Total processes killed: ${totalKilled}`);
203
+ }
204
+ }
205
+
206
+ /**
207
+ * Kill Node Inspector processes on Raspberry Pi OS.
208
+ *
209
+ * Raspberry Pi OS is Debian-based and has the same tools available as Ubuntu.
210
+ *
211
+ * @param {string[]} args - Command line arguments (unused)
212
+ * @returns {Promise<void>}
213
+ */
214
+ async function do_killni_raspbian(args) {
215
+ // Raspbian has the same tools as Ubuntu/Debian
216
+ return do_killni_ubuntu(args);
217
+ }
218
+
219
+ /**
220
+ * Kill Node Inspector processes on Amazon Linux.
221
+ *
222
+ * Amazon Linux (RHEL-based) has similar process tools to Ubuntu.
223
+ *
224
+ * @param {string[]} args - Command line arguments (unused)
225
+ * @returns {Promise<void>}
226
+ */
227
+ async function do_killni_amazon_linux(args) {
228
+ // Amazon Linux has pgrep/ps available
229
+ return do_killni_ubuntu(args);
230
+ }
231
+
232
+ /**
233
+ * Kill Node Inspector processes on Windows using Command Prompt.
234
+ *
235
+ * Windows uses different tools for process management:
236
+ * - tasklist: List running processes
237
+ * - taskkill: Terminate processes
238
+ * - wmic: Windows Management Instrumentation (more detailed process info)
239
+ *
240
+ * We use wmic to get the command line of each node process, then taskkill
241
+ * to terminate matching processes.
242
+ *
243
+ * @param {string[]} args - Command line arguments (unused)
244
+ * @returns {Promise<void>}
245
+ */
246
+ async function do_killni_cmd(args) {
247
+ const patterns = ['--debug-brk', '--inspect-brk', '--debug', '--inspect'];
248
+ let totalKilled = 0;
249
+
250
+ try {
251
+ // Get all node.exe processes with their command lines using wmic
252
+ // wmic output format: CommandLine=..., ProcessId=...
253
+ let wmicOutput;
254
+ try {
255
+ wmicOutput = execSync(
256
+ 'wmic process where "name=\'node.exe\'" get CommandLine,ProcessId /format:csv',
257
+ { encoding: 'utf8' }
258
+ );
259
+ } catch {
260
+ // No node processes found or wmic failed
261
+ console.log('No Node Inspector processes found.');
262
+ return;
263
+ }
264
+
265
+ // Parse wmic CSV output
266
+ // Format: Node,CommandLine,ProcessId (header on first non-empty line)
267
+ const lines = wmicOutput.trim().split('\n').filter(line => line.trim().length > 0);
268
+
269
+ // Skip header line
270
+ for (let i = 1; i < lines.length; i++) {
271
+ const line = lines[i].trim();
272
+ if (!line) continue;
273
+
274
+ // CSV format: ComputerName,CommandLine,ProcessId
275
+ // CommandLine may contain commas, so we need to be careful
276
+ const parts = line.split(',');
277
+ if (parts.length < 3) continue;
278
+
279
+ // ProcessId is the last element
280
+ const pid = parts[parts.length - 1].trim();
281
+ // CommandLine is everything between first comma and last comma
282
+ const commandLine = parts.slice(1, -1).join(',');
283
+
284
+ // Check if this process matches any of our patterns
285
+ for (const pattern of patterns) {
286
+ if (commandLine.includes(pattern)) {
287
+ // Skip our own process
288
+ if (parseInt(pid, 10) === process.pid) {
289
+ continue;
290
+ }
291
+
292
+ try {
293
+ // Use taskkill with /F for forceful termination (like kill -9)
294
+ execSync(`taskkill /PID ${pid} /F`, { stdio: 'ignore' });
295
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
296
+ totalKilled++;
297
+ } catch {
298
+ // Process may have already exited
299
+ }
300
+ break; // Don't count same process multiple times
301
+ }
302
+ }
303
+ }
304
+ } catch (error) {
305
+ // wmic or taskkill not available - try alternative approach
306
+ console.error('Error: Could not enumerate processes.');
307
+ console.error('This script requires wmic and taskkill commands.');
308
+ process.exit(1);
309
+ }
310
+
311
+ if (totalKilled === 0) {
312
+ console.log('No Node Inspector processes found.');
313
+ } else {
314
+ console.log(`Total processes killed: ${totalKilled}`);
315
+ }
316
+ }
317
+
318
+ /**
319
+ * Kill Node Inspector processes on Windows using PowerShell.
320
+ *
321
+ * PowerShell provides Get-Process and Stop-Process cmdlets which are more
322
+ * powerful than the CMD equivalents. We can filter processes by command line
323
+ * using Get-WmiObject or Get-CimInstance.
324
+ *
325
+ * @param {string[]} args - Command line arguments (unused)
326
+ * @returns {Promise<void>}
327
+ */
328
+ async function do_killni_powershell(args) {
329
+ const patterns = ['--debug-brk', '--inspect-brk', '--debug', '--inspect'];
330
+ let totalKilled = 0;
331
+
332
+ try {
333
+ // Use Get-CimInstance (modern) or Get-WmiObject (legacy) to get process command lines
334
+ // This PowerShell command returns JSON for easier parsing
335
+ const psCommand = `
336
+ Get-CimInstance Win32_Process -Filter "Name='node.exe'" |
337
+ Select-Object ProcessId, CommandLine |
338
+ ConvertTo-Json -Compress
339
+ `.replace(/\n/g, ' ');
340
+
341
+ let output;
342
+ try {
343
+ output = execSync(`powershell -Command "${psCommand}"`, { encoding: 'utf8' });
344
+ } catch {
345
+ console.log('No Node Inspector processes found.');
346
+ return;
347
+ }
348
+
349
+ // Parse JSON output
350
+ let processes;
351
+ try {
352
+ const parsed = JSON.parse(output.trim());
353
+ // PowerShell returns single object (not array) if only one process
354
+ processes = Array.isArray(parsed) ? parsed : [parsed];
355
+ } catch {
356
+ console.log('No Node Inspector processes found.');
357
+ return;
358
+ }
359
+
360
+ for (const proc of processes) {
361
+ if (!proc || !proc.ProcessId || !proc.CommandLine) continue;
362
+
363
+ const pid = proc.ProcessId;
364
+ const commandLine = proc.CommandLine;
365
+
366
+ for (const pattern of patterns) {
367
+ if (commandLine.includes(pattern)) {
368
+ if (pid === process.pid) {
369
+ continue;
370
+ }
371
+
372
+ try {
373
+ // Stop-Process -Force is equivalent to kill -9
374
+ execSync(`powershell -Command "Stop-Process -Id ${pid} -Force"`, { stdio: 'ignore' });
375
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
376
+ totalKilled++;
377
+ } catch {
378
+ // Process may have already exited
379
+ }
380
+ break;
381
+ }
382
+ }
383
+ }
384
+ } catch (error) {
385
+ console.error('Error: Could not enumerate processes.');
386
+ console.error('This script requires PowerShell with Get-CimInstance cmdlet.');
387
+ process.exit(1);
388
+ }
389
+
390
+ if (totalKilled === 0) {
391
+ console.log('No Node Inspector processes found.');
392
+ } else {
393
+ console.log(`Total processes killed: ${totalKilled}`);
394
+ }
395
+ }
396
+
397
+ /**
398
+ * Kill Node Inspector processes from Git Bash on Windows.
399
+ *
400
+ * Git Bash provides Unix-like commands (ps, grep, awk) but they may behave
401
+ * differently than on true Unix systems. We try the Unix approach first,
402
+ * then fall back to Windows commands if needed.
403
+ *
404
+ * @param {string[]} args - Command line arguments (unused)
405
+ * @returns {Promise<void>}
406
+ */
407
+ async function do_killni_gitbash(args) {
408
+ const patterns = ['--debug-brk', '--inspect-brk', '--debug', '--inspect'];
409
+ let totalKilled = 0;
410
+
411
+ // In Git Bash, we can try Unix-style commands first
412
+ // If that fails, fall back to Windows commands
413
+ let useUnixStyle = true;
414
+
415
+ for (const pattern of patterns) {
416
+ try {
417
+ let pids = [];
418
+
419
+ if (useUnixStyle) {
420
+ // Try Unix-style approach (ps in Git Bash)
421
+ try {
422
+ const psOutput = execSync(
423
+ `ps -ef | grep "${pattern}" | grep -v grep | awk '{print $2}'`,
424
+ { encoding: 'utf8', shell: '/bin/bash' }
425
+ );
426
+ pids = psOutput.trim().split('\n').filter(pid => pid.length > 0);
427
+ } catch {
428
+ // Unix-style failed, switch to Windows
429
+ useUnixStyle = false;
430
+ }
431
+ }
432
+
433
+ if (!useUnixStyle || pids.length === 0) {
434
+ // Use Windows tasklist/wmic approach
435
+ try {
436
+ const wmicOutput = execSync(
437
+ 'wmic process where "name=\'node.exe\'" get CommandLine,ProcessId /format:csv',
438
+ { encoding: 'utf8' }
439
+ );
440
+
441
+ const lines = wmicOutput.trim().split('\n').filter(line => line.trim().length > 0);
442
+ for (let i = 1; i < lines.length; i++) {
443
+ const line = lines[i].trim();
444
+ if (!line) continue;
445
+
446
+ const parts = line.split(',');
447
+ if (parts.length < 3) continue;
448
+
449
+ const pid = parts[parts.length - 1].trim();
450
+ const commandLine = parts.slice(1, -1).join(',');
451
+
452
+ if (commandLine.includes(pattern)) {
453
+ if (!pids.includes(pid)) {
454
+ pids.push(pid);
455
+ }
456
+ }
457
+ }
458
+ } catch {
459
+ // Neither approach worked
460
+ continue;
461
+ }
462
+ }
463
+
464
+ for (const pid of pids) {
465
+ if (parseInt(pid, 10) === process.pid) {
466
+ continue;
467
+ }
468
+
469
+ try {
470
+ // In Git Bash, try kill first (Unix-style)
471
+ if (useUnixStyle) {
472
+ execSync(`kill -9 ${pid}`, { stdio: 'ignore', shell: '/bin/bash' });
473
+ } else {
474
+ // Fall back to taskkill
475
+ execSync(`taskkill /PID ${pid} /F`, { stdio: 'ignore' });
476
+ }
477
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
478
+ totalKilled++;
479
+ } catch {
480
+ // Try the other approach
481
+ try {
482
+ if (useUnixStyle) {
483
+ execSync(`taskkill /PID ${pid} /F`, { stdio: 'ignore' });
484
+ } else {
485
+ execSync(`kill -9 ${pid}`, { stdio: 'ignore', shell: '/bin/bash' });
486
+ }
487
+ console.log(`Killed process ${pid} (matched: ${pattern})`);
488
+ totalKilled++;
489
+ } catch {
490
+ // Process may have already exited
491
+ }
492
+ }
493
+ }
494
+ } catch {
495
+ continue;
496
+ }
497
+ }
498
+
499
+ if (totalKilled === 0) {
500
+ console.log('No Node Inspector processes found.');
501
+ } else {
502
+ console.log(`Total processes killed: ${totalKilled}`);
503
+ }
504
+ }
505
+
8
506
  /**
9
- * Finds and kills all Node.js processes running with the
10
- * --debug-brk flag (Node Inspector).
507
+ * Main entry point - detects environment and executes appropriate implementation.
508
+ *
509
+ * The "killni" (Kill Node Inspector) command finds and forcibly terminates all
510
+ * Node.js processes that are running with debug flags (--debug-brk, --inspect-brk,
511
+ * --debug, --inspect). This is useful when:
512
+ *
513
+ * - A debugging session is stuck or unresponsive
514
+ * - Multiple debug sessions were accidentally started
515
+ * - You need to clean up orphaned Node Inspector processes
516
+ * - Port conflicts occur because a debug port is still in use
517
+ *
518
+ * The script searches for both legacy (--debug, --debug-brk) and modern
519
+ * (--inspect, --inspect-brk) debug flags to cover all Node.js versions.
520
+ *
521
+ * WARNING: This command forcibly kills processes (SIGKILL / taskkill /F).
522
+ * Any unsaved work in the terminated processes will be lost.
11
523
  *
12
524
  * @param {string[]} args - Command line arguments (unused)
13
525
  * @returns {Promise<void>}
14
526
  */
15
- async function main(args) {
16
- // TODO: Implement Node Inspector process killer
527
+ async function do_killni(args) {
528
+ const platform = os.detect();
529
+
530
+ const handlers = {
531
+ 'macos': do_killni_macos,
532
+ 'ubuntu': do_killni_ubuntu,
533
+ 'debian': do_killni_ubuntu,
534
+ 'raspbian': do_killni_raspbian,
535
+ 'amazon_linux': do_killni_amazon_linux,
536
+ 'rhel': do_killni_amazon_linux,
537
+ 'fedora': do_killni_ubuntu,
538
+ 'linux': do_killni_ubuntu,
539
+ 'wsl': do_killni_ubuntu,
540
+ 'cmd': do_killni_cmd,
541
+ 'windows': do_killni_cmd,
542
+ 'powershell': do_killni_powershell,
543
+ 'gitbash': do_killni_gitbash
544
+ };
545
+
546
+ const handler = handlers[platform.type];
547
+ if (!handler) {
548
+ console.error(`Platform '${platform.type}' is not supported for this command.`);
549
+ console.error('');
550
+ console.error('Supported platforms:');
551
+ console.error(' - macOS');
552
+ console.error(' - Ubuntu, Debian, and other Linux distributions');
553
+ console.error(' - Raspberry Pi OS');
554
+ console.error(' - Amazon Linux, RHEL, Fedora');
555
+ console.error(' - Windows (CMD, PowerShell, Git Bash)');
556
+ process.exit(1);
557
+ }
558
+
559
+ await handler(args);
17
560
  }
18
561
 
19
- module.exports = { main };
562
+ module.exports = {
563
+ main: do_killni,
564
+ do_killni,
565
+ do_killni_nodejs,
566
+ do_killni_macos,
567
+ do_killni_ubuntu,
568
+ do_killni_raspbian,
569
+ do_killni_amazon_linux,
570
+ do_killni_cmd,
571
+ do_killni_powershell,
572
+ do_killni_gitbash
573
+ };
20
574
 
21
575
  if (require.main === module) {
22
- main(process.argv.slice(2));
576
+ do_killni(process.argv.slice(2));
23
577
  }