binary-collections 2.0.12 → 2.0.13

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 (231) hide show
  1. package/binaries/binary-executor.cjs +89 -13
  2. package/binaries/clean-nodemodule.cjs +89 -13
  3. package/binaries/clean-nodemodules.cjs +89 -13
  4. package/binaries/dev.cjs +89 -13
  5. package/binaries/empty.cjs +89 -13
  6. package/binaries/git-reduce-size.cjs +89 -13
  7. package/binaries/javakill.cjs +89 -13
  8. package/binaries/kill-process.cjs +89 -13
  9. package/binaries/nodekill.cjs +89 -13
  10. package/binaries/prod.cjs +89 -13
  11. package/binaries/py.cjs +89 -13
  12. package/binaries/rmfind.cjs +89 -13
  13. package/binaries/rmx.cjs +89 -13
  14. package/binaries/submodule-token.cjs +89 -13
  15. package/binaries/test-cjs.cjs +89 -13
  16. package/binaries/test-esm.cjs +89 -13
  17. package/binaries/yarn-clean.cjs +89 -13
  18. package/binaries/yc +22 -1
  19. package/binaries/yc.cjs +89 -13
  20. package/binaries/ycw +256 -0
  21. package/binaries/ycw.cjs +254 -0
  22. package/docs-src/binary-collections.md +34 -0
  23. package/docs-src/changelog.md +26 -0
  24. package/docs-src/clean-github-actions-caches.md +115 -0
  25. package/docs-src/copy-move-file.md +59 -0
  26. package/docs-src/del-gradle.md +17 -0
  27. package/docs-src/del-ps.md +28 -0
  28. package/docs-src/downloader.md +62 -0
  29. package/docs-src/env-helpers.md +29 -0
  30. package/docs-src/find-node-modules.md +17 -0
  31. package/docs-src/free-chatgpt.md +26 -0
  32. package/docs-src/git-diff.md +33 -0
  33. package/docs-src/git-fix.md +34 -0
  34. package/docs-src/git-purge.md +17 -0
  35. package/docs-src/git-reduce-size.md +17 -0
  36. package/docs-src/git-undo.md +21 -0
  37. package/docs-src/kill-night-crows.md +26 -0
  38. package/docs-src/node-cache-cleaner.md +182 -0
  39. package/docs-src/node-executor.md +50 -0
  40. package/docs-src/node-package-packer.md +48 -0
  41. package/docs-src/npm-run-series.md +43 -0
  42. package/docs-src/package-resolutions-updater.md +22 -0
  43. package/docs-src/php-cs-fixer-staged.md +19 -0
  44. package/docs-src/print-directory-tree.md +35 -0
  45. package/docs-src/print-tarball-tree.md +55 -0
  46. package/docs-src/py.md +19 -0
  47. package/docs-src/remove-module.md +32 -0
  48. package/docs-src/rmfind-rmx.md +21 -0
  49. package/docs-src/rmpath.md +38 -0
  50. package/docs-src/run-by-checksum.md +87 -0
  51. package/docs-src/submodule-install.md +31 -0
  52. package/docs-src/submodule-remove.md +22 -0
  53. package/docs-src/submodule-token.md +17 -0
  54. package/docs-src/test-runners.md +21 -0
  55. package/docs-src/yarn-install.md +31 -0
  56. package/docs-src/yarn-reinstall.md +27 -0
  57. package/lib/binary-collections/config.cjs +1 -1
  58. package/lib/binary-collections/config.mjs +1 -1
  59. package/lib/binary-collections/findScript.cjs +35 -10
  60. package/lib/binary-collections/findScript.mjs +2 -2
  61. package/lib/binary-collections/listScript.cjs +35 -10
  62. package/lib/binary-collections/listScript.mjs +2 -2
  63. package/lib/binary-collections.cjs +45 -25
  64. package/lib/binary-collections.mjs +11 -11
  65. package/lib/changelog.cjs +7 -12
  66. package/lib/changelog.mjs +2 -2
  67. package/lib/{chunk-SPTECFE5.mjs → chunk-2MN4VPV2.mjs} +86 -20
  68. package/lib/{chunk-V2IBPCEV.mjs → chunk-6C7KTYGZ.mjs} +3 -1
  69. package/lib/{chunk-5RTXZVCW.mjs → chunk-6RK5UCTP.mjs} +5 -10
  70. package/lib/{chunk-FB2WKVJD.mjs → chunk-CD3HF3LK.mjs} +67 -35
  71. package/lib/{chunk-6PU7BAHB.mjs → chunk-FLYSZFLW.mjs} +1 -1
  72. package/lib/chunk-KLKAIFKI.mjs +40 -0
  73. package/lib/chunk-LVSPEFU2.mjs +86 -0
  74. package/lib/{chunk-2LSRSEXF.mjs → chunk-MGPYPKIE.mjs} +2 -2
  75. package/lib/{chunk-ZOWVMII3.mjs → chunk-NQXUYO67.mjs} +35 -10
  76. package/lib/{chunk-C6D2TTYU.mjs → chunk-OBXLTXFJ.mjs} +4 -2
  77. package/lib/chunk-QD4T255Z.mjs +40 -0
  78. package/lib/{chunk-XPJGCDOD.mjs → chunk-QII2EKCS.mjs} +14 -2
  79. package/lib/chunk-RDGDLSPD.mjs +76 -0
  80. package/lib/{chunk-NCZPTKDV.mjs → chunk-RDN6HF5Z.mjs} +1 -1
  81. package/lib/chunk-RJKTSUAX.mjs +123 -0
  82. package/lib/{chunk-66KDU4TX.mjs → chunk-TBWXE7ST.mjs} +36 -61
  83. package/lib/{chunk-M3YIYRHT.mjs → chunk-UY5VUEA3.mjs} +1 -1
  84. package/lib/chunk-WSHVPGNM.mjs +44 -0
  85. package/lib/{chunk-G5UUEWUO.mjs → chunk-X2B3X7D4.mjs} +1 -1
  86. package/lib/clean-github-actions-caches-cli.cjs +255 -213
  87. package/lib/clean-github-actions-caches-cli.mjs +25 -7
  88. package/lib/clean-github-actions-caches.cjs +231 -35
  89. package/lib/clean-github-actions-caches.d.cts +39 -1
  90. package/lib/clean-github-actions-caches.mjs +2 -1
  91. package/lib/cross-env/command.cjs +2 -2
  92. package/lib/cross-env/command.js +2 -2
  93. package/lib/cross-env/command.mjs +2 -2
  94. package/lib/cross-env/index.cjs +2 -2
  95. package/lib/cross-env/index.mjs +3 -3
  96. package/lib/cross-env/variable.mjs +2 -2
  97. package/lib/del-gradle.cjs +6 -11
  98. package/lib/del-gradle.mjs +1 -1
  99. package/lib/del-node-modules.cjs +185 -3
  100. package/lib/del-node-modules.js +3 -3
  101. package/lib/del-node-modules.mjs +6 -3
  102. package/lib/{del-ps.cjs → del-ps-cli.cjs} +36 -41
  103. package/lib/del-ps-cli.mjs +44 -0
  104. package/lib/del-yarn-caches.cjs +6 -11
  105. package/lib/del-yarn-caches.mjs +1 -1
  106. package/lib/downloader-cli.cjs +256 -0
  107. package/lib/downloader-cli.d.cts +2 -0
  108. package/lib/downloader-cli.mjs +90 -0
  109. package/lib/file/copy-cli.cjs +183 -2
  110. package/lib/file/copy-cli.mjs +6 -2
  111. package/lib/file/move-cli.cjs +183 -2
  112. package/lib/file/move-cli.mjs +6 -2
  113. package/lib/find-node-modules-cli.cjs +1 -1
  114. package/lib/find-node-modules-cli.mjs +1 -1
  115. package/lib/find-node-modules.cjs +1 -1
  116. package/lib/find-node-modules.mjs +1 -1
  117. package/lib/free-chatgpt.cjs +6 -11
  118. package/lib/free-chatgpt.mjs +1 -1
  119. package/lib/git/user-config.cjs +7 -12
  120. package/lib/git/user-config.mjs +2 -2
  121. package/lib/git-diff-cli.cjs +91 -30
  122. package/lib/git-diff-cli.mjs +3 -3
  123. package/lib/git-diff.cjs +91 -30
  124. package/lib/git-diff.js +85 -28
  125. package/lib/git-diff.mjs +3 -3
  126. package/lib/git-fix.cjs +7 -12
  127. package/lib/git-fix.mjs +2 -2
  128. package/lib/git-purge.cjs +7 -12
  129. package/lib/git-purge.mjs +2 -2
  130. package/lib/index.cjs +1 -1
  131. package/lib/index.mjs +1 -1
  132. package/lib/node-cache-cleaner-cli.cjs +185 -2
  133. package/lib/node-cache-cleaner-cli.js +2 -5
  134. package/lib/node-cache-cleaner-cli.mjs +8 -4
  135. package/lib/node-executor.cjs +183 -2
  136. package/lib/node-executor.mjs +5 -2
  137. package/lib/node-package-packer/build-readme.cjs +150 -0
  138. package/lib/node-package-packer/build-readme.d.mts +10 -0
  139. package/lib/node-package-packer/build-readme.mjs +10 -0
  140. package/lib/node-package-packer/build-tarball.cjs +495 -0
  141. package/lib/node-package-packer/build-tarball.d.mts +33 -0
  142. package/lib/node-package-packer/build-tarball.mjs +175 -0
  143. package/lib/node-package-packer-cli.cjs +525 -0
  144. package/lib/node-package-packer-cli.d.mts +1 -0
  145. package/lib/node-package-packer-cli.mjs +34 -0
  146. package/lib/npm-run-series.cjs +7 -12
  147. package/lib/npm-run-series.mjs +2 -2
  148. package/lib/package-resolutions-updater-cli.cjs +73 -76
  149. package/lib/package-resolutions-updater-cli.mjs +4 -3
  150. package/lib/package-resolutions-updater.cjs +71 -74
  151. package/lib/package-resolutions-updater.d.mts +34 -0
  152. package/lib/package-resolutions-updater.mjs +3 -2
  153. package/lib/php-cs-fixer-staged.cjs +1 -1
  154. package/lib/php-cs-fixer-staged.mjs +1 -1
  155. package/lib/print-directory-tree.cjs +16 -18
  156. package/lib/print-directory-tree.mjs +11 -8
  157. package/lib/print-tarball-tree.cjs +262 -0
  158. package/lib/print-tarball-tree.d.mts +1 -0
  159. package/lib/print-tarball-tree.mjs +68 -0
  160. package/lib/ps/index.cjs +10 -10
  161. package/lib/ps/index.mjs +4 -4
  162. package/lib/ps/table-parser.d.ts +3 -4
  163. package/lib/ps/table-parser.js +9 -16
  164. package/lib/remove-module.cjs +17 -22
  165. package/lib/remove-module.mjs +2 -2
  166. package/lib/rm-node-module-cli.cjs +171 -4
  167. package/lib/rm-node-module-cli.mjs +7 -4
  168. package/lib/rmpath-cli.cjs +285 -0
  169. package/lib/rmpath-cli.d.mts +1 -0
  170. package/lib/rmpath-cli.mjs +23 -0
  171. package/lib/rmpath.cjs +6 -217
  172. package/lib/rmpath.mjs +5 -101
  173. package/lib/run-by-checksum/cache.cjs +69 -0
  174. package/lib/run-by-checksum/cache.d.ts +19 -0
  175. package/lib/run-by-checksum/cache.js +50 -0
  176. package/lib/run-by-checksum/cache.mjs +12 -0
  177. package/lib/run-by-checksum/hash.cjs +72 -0
  178. package/lib/run-by-checksum/hash.d.ts +14 -0
  179. package/lib/run-by-checksum/hash.js +85 -0
  180. package/lib/{ps/isWin.mjs → run-by-checksum/hash.mjs} +5 -5
  181. package/lib/run-by-checksum/run.cjs +169 -0
  182. package/lib/run-by-checksum/run.d.ts +22 -0
  183. package/lib/run-by-checksum/run.js +93 -0
  184. package/lib/run-by-checksum/run.mjs +10 -0
  185. package/lib/run-by-checksum-cli.cjs +382 -0
  186. package/lib/run-by-checksum-cli.d.ts +2 -0
  187. package/lib/run-by-checksum-cli.js +43 -0
  188. package/lib/run-by-checksum-cli.mjs +56 -0
  189. package/lib/submodule-install.cjs +8 -13
  190. package/lib/submodule-install.mjs +3 -3
  191. package/lib/submodule-remove-cli.cjs +169 -2
  192. package/lib/submodule-remove-cli.js +2 -2
  193. package/lib/submodule-remove-cli.mjs +5 -2
  194. package/lib/utils/fetchResponse.cjs +24 -0
  195. package/lib/utils/fetchResponse.d.cts +25 -0
  196. package/lib/utils/fetchResponse.mjs +6 -0
  197. package/lib/utils/index.cjs +5 -10
  198. package/lib/utils/index.d.cts +2 -9
  199. package/lib/utils/index.mjs +1 -1
  200. package/lib/utils/isWindows.mjs +3 -1
  201. package/lib/utils/runBash.cjs +1 -1
  202. package/lib/utils/runBash.mjs +1 -1
  203. package/lib/yarn-per-branch-lock-installer.cjs +202 -11
  204. package/lib/yarn-per-branch-lock-installer.mjs +24 -11
  205. package/lib/yarn-reinstall.cjs +6 -11
  206. package/lib/yarn-reinstall.mjs +1 -1
  207. package/package.json +35 -10
  208. package/readme.html +2 -2
  209. package/readme.md +5 -5
  210. package/releases/readme.md +6 -3
  211. package/tmp/test-repo-runChecksum/test-complex-glob/README.md +1 -0
  212. package/tmp/test-repo-runChecksum/test-mixed-args/README.md +1 -0
  213. package/.opencode/package.json +0 -5
  214. package/lib/chunk-6S4NXESK.mjs +0 -26
  215. package/lib/del-ps.js +0 -32
  216. package/lib/del-ps.mjs +0 -43
  217. package/lib/ps/isWin.cjs +0 -26
  218. package/lib/ps/isWin.d.ts +0 -2
  219. package/lib/ps/isWin.js +0 -4
  220. package/test/package.json +0 -20
  221. package/test-project/package.json +0 -22
  222. package/test-project/workspaces/workspace-a/package.json +0 -135
  223. package/test-project/workspaces/workspace-a/test/demo/package.json +0 -25
  224. package/test-project/workspaces/workspace-b/package.json +0 -139
  225. package/test-project/workspaces/workspace-b/test/sample-project/package.json +0 -7
  226. package/test-project/workspaces/workspace-b/themes/hexo-theme-flowbite/package.json +0 -96
  227. package/tmp/rm-node-modules-test-project/package.json +0 -17
  228. package/tmp/rm-node-modules-test-project/packages/workspace-a/package.json +0 -16
  229. package/tmp/rm-node-modules-test-project/packages/workspace-b/package.json +0 -16
  230. package/tmp/test-repo/package.json +0 -17
  231. /package/lib/{del-ps.d.ts → del-ps-cli.d.mts} +0 -0
@@ -12,9 +12,11 @@
12
12
  * - fs:
13
13
  * Used to check whether files exist.
14
14
  */
15
- const { spawnSync } = require("child_process");
16
- const path = require("path");
17
- const fs = require("fs");
15
+ const { spawnSync } = require('child_process');
16
+ const path = require('path');
17
+ const fs = require('fs');
18
+ const which = require('which');
19
+ const { isWindows } = require('../src/utils/isWindows.js');
18
20
 
19
21
  /**
20
22
  * __dirname
@@ -49,7 +51,7 @@ const base = path.basename(__filename, path.extname(__filename));
49
51
  * .sh
50
52
  * executable file without extension
51
53
  */
52
- const candidates = process.platform === "win32" ? [".cmd", ".bat", ".ps1", ".vbs"] : [".sh", ""];
54
+ const candidates = process.platform === 'win32' ? ['.cmd', '.bat', '.ps1', '.vbs'] : ['.sh', ''];
53
55
 
54
56
  /**
55
57
  * Search for the first matching script
@@ -72,13 +74,32 @@ let found = null;
72
74
 
73
75
  for (const ext of candidates) {
74
76
  const script = path.join(binDir, base + ext);
75
-
76
- if (fs.existsSync(script)) {
77
+ const exists = fs.existsSync(script);
78
+ // console.log(`Checking for ${script}: ${exists ? 'found' : 'not found'}`);
79
+ if (exists) {
77
80
  found = script;
78
81
  break;
79
82
  }
80
83
  }
81
84
 
85
+ /**
86
+ * If no matching script was found,
87
+ * try check if `bash` is available and if so, check for a .sh script.
88
+ */
89
+ if (!found) {
90
+ try {
91
+ spawnSync('bash', ['--version'], { stdio: 'ignore' });
92
+ const bashScript = [path.join(binDir, base), path.join(binDir, base + '.sh')].find((script) =>
93
+ fs.existsSync(script)
94
+ );
95
+ if (bashScript) {
96
+ found = bashScript;
97
+ }
98
+ } catch {
99
+ // bash is not available, do nothing
100
+ }
101
+ }
102
+
82
103
  /**
83
104
  * If no matching script was found,
84
105
  * print an error and exit with failure code 1.
@@ -92,8 +113,9 @@ if (!found) {
92
113
  * Detect special script types
93
114
  * that require a shell/interpreter.
94
115
  */
95
- const isPs1 = found.endsWith(".ps1");
96
- const isCmd = found.endsWith(".cmd");
116
+ const isPs1 = found.endsWith('.ps1');
117
+ const isCmd = found.endsWith('.cmd');
118
+ const isUnixShell = found.endsWith('.sh') || path.extname(found) === '';
97
119
 
98
120
  /**
99
121
  * cmd
@@ -124,9 +146,9 @@ let cmd, args;
124
146
  * Forward all user-provided command-line arguments.
125
147
  */
126
148
  if (isPs1) {
127
- cmd = "powershell.exe";
149
+ cmd = 'powershell.exe';
128
150
 
129
- args = ["-NoProfile", "-ExecutionPolicy", "Bypass", "-File", found, ...process.argv.slice(2)];
151
+ args = ['-NoProfile', '-ExecutionPolicy', 'Bypass', '-File', found, ...process.argv.slice(2)];
130
152
 
131
153
  /**
132
154
  * CMD batch files:
@@ -137,9 +159,9 @@ if (isPs1) {
137
159
  * "execute command and terminate"
138
160
  */
139
161
  } else if (isCmd) {
140
- cmd = "cmd.exe";
162
+ cmd = 'cmd.exe';
141
163
 
142
- args = ["/c", found, ...process.argv.slice(2)];
164
+ args = ['/c', found, ...process.argv.slice(2)];
143
165
 
144
166
  /**
145
167
  * Other scripts:
@@ -150,11 +172,65 @@ if (isPs1) {
150
172
  *
151
173
  * These can be executed directly.
152
174
  */
175
+ } else if (isUnixShell) {
176
+ // Capture shebang scripts (no extension) and .sh scripts
177
+ const shebang = fs.readFileSync(found, 'utf8').split('\n')[0].trim();
178
+ const interpreter = shebang.startsWith('#!')
179
+ ? shebang
180
+ .slice(2)
181
+ .trim()
182
+ .replace(/^\/usr\/bin\/env\s+/, '')
183
+ .replace(/^\/bin\/env\s+/, '')
184
+ .replace(/^\/usr\/bin\//, '')
185
+ .replace(/^\/bin\//, '')
186
+ : null;
187
+
188
+ if (interpreter) {
189
+ const resolvedOrNull = which.sync(interpreter, { nothrow: true });
190
+ if (!resolvedOrNull) {
191
+ if (['bash', 'sh'].includes(interpreter) && isWindows()) {
192
+ const locationsToCheck = [
193
+ 'C:\\Program Files\\Git\\usr\\bin',
194
+ 'C:\\Program Files\\Git\\bin',
195
+ 'C:\\Program Files (x86)\\Git\\usr\\bin',
196
+ 'C:\\msys64\\usr\\bin',
197
+ 'C:\\msys64\\bin',
198
+ 'C:\\cygwin64\\bin',
199
+ 'C:\\cygwin\\bin',
200
+ 'C:\\MinGW\\bin',
201
+ 'C:\\MinGW\\msys\\1.0\\bin'
202
+ ];
203
+ for (const location of locationsToCheck) {
204
+ const potentialPath = path.join(location, 'bash.exe');
205
+ if (fs.existsSync(potentialPath)) {
206
+ cmd = potentialPath;
207
+ args = [found, ...process.argv.slice(2)];
208
+ break;
209
+ }
210
+ }
211
+ }
212
+ } else {
213
+ cmd = resolvedOrNull || interpreter;
214
+ args = [found, ...process.argv.slice(2)];
215
+ }
216
+ } else {
217
+ cmd = found;
218
+ args = process.argv.slice(2);
219
+ }
153
220
  } else {
154
221
  cmd = found;
155
222
  args = process.argv.slice(2);
156
223
  }
157
224
 
225
+ console.log(
226
+ `Executing: ${path.isAbsolute(cmd) ? path.relative(process.cwd(), cmd) : cmd} ${args
227
+ .map((f) => {
228
+ if (path.isAbsolute(f)) return path.relative(process.cwd(), f);
229
+ return f;
230
+ })
231
+ .join(' ')}`
232
+ );
233
+
158
234
  /**
159
235
  * Execute the selected script synchronously.
160
236
  *
@@ -165,7 +241,7 @@ if (isPs1) {
165
241
  * spawnSync waits until the child process exits.
166
242
  */
167
243
  const result = spawnSync(cmd, args, {
168
- stdio: "inherit"
244
+ stdio: 'inherit'
169
245
  });
170
246
 
171
247
  /**
package/binaries/prod.cjs CHANGED
@@ -12,9 +12,11 @@
12
12
  * - fs:
13
13
  * Used to check whether files exist.
14
14
  */
15
- const { spawnSync } = require("child_process");
16
- const path = require("path");
17
- const fs = require("fs");
15
+ const { spawnSync } = require('child_process');
16
+ const path = require('path');
17
+ const fs = require('fs');
18
+ const which = require('which');
19
+ const { isWindows } = require('../src/utils/isWindows.js');
18
20
 
19
21
  /**
20
22
  * __dirname
@@ -49,7 +51,7 @@ const base = path.basename(__filename, path.extname(__filename));
49
51
  * .sh
50
52
  * executable file without extension
51
53
  */
52
- const candidates = process.platform === "win32" ? [".cmd", ".bat", ".ps1", ".vbs"] : [".sh", ""];
54
+ const candidates = process.platform === 'win32' ? ['.cmd', '.bat', '.ps1', '.vbs'] : ['.sh', ''];
53
55
 
54
56
  /**
55
57
  * Search for the first matching script
@@ -72,13 +74,32 @@ let found = null;
72
74
 
73
75
  for (const ext of candidates) {
74
76
  const script = path.join(binDir, base + ext);
75
-
76
- if (fs.existsSync(script)) {
77
+ const exists = fs.existsSync(script);
78
+ // console.log(`Checking for ${script}: ${exists ? 'found' : 'not found'}`);
79
+ if (exists) {
77
80
  found = script;
78
81
  break;
79
82
  }
80
83
  }
81
84
 
85
+ /**
86
+ * If no matching script was found,
87
+ * try check if `bash` is available and if so, check for a .sh script.
88
+ */
89
+ if (!found) {
90
+ try {
91
+ spawnSync('bash', ['--version'], { stdio: 'ignore' });
92
+ const bashScript = [path.join(binDir, base), path.join(binDir, base + '.sh')].find((script) =>
93
+ fs.existsSync(script)
94
+ );
95
+ if (bashScript) {
96
+ found = bashScript;
97
+ }
98
+ } catch {
99
+ // bash is not available, do nothing
100
+ }
101
+ }
102
+
82
103
  /**
83
104
  * If no matching script was found,
84
105
  * print an error and exit with failure code 1.
@@ -92,8 +113,9 @@ if (!found) {
92
113
  * Detect special script types
93
114
  * that require a shell/interpreter.
94
115
  */
95
- const isPs1 = found.endsWith(".ps1");
96
- const isCmd = found.endsWith(".cmd");
116
+ const isPs1 = found.endsWith('.ps1');
117
+ const isCmd = found.endsWith('.cmd');
118
+ const isUnixShell = found.endsWith('.sh') || path.extname(found) === '';
97
119
 
98
120
  /**
99
121
  * cmd
@@ -124,9 +146,9 @@ let cmd, args;
124
146
  * Forward all user-provided command-line arguments.
125
147
  */
126
148
  if (isPs1) {
127
- cmd = "powershell.exe";
149
+ cmd = 'powershell.exe';
128
150
 
129
- args = ["-NoProfile", "-ExecutionPolicy", "Bypass", "-File", found, ...process.argv.slice(2)];
151
+ args = ['-NoProfile', '-ExecutionPolicy', 'Bypass', '-File', found, ...process.argv.slice(2)];
130
152
 
131
153
  /**
132
154
  * CMD batch files:
@@ -137,9 +159,9 @@ if (isPs1) {
137
159
  * "execute command and terminate"
138
160
  */
139
161
  } else if (isCmd) {
140
- cmd = "cmd.exe";
162
+ cmd = 'cmd.exe';
141
163
 
142
- args = ["/c", found, ...process.argv.slice(2)];
164
+ args = ['/c', found, ...process.argv.slice(2)];
143
165
 
144
166
  /**
145
167
  * Other scripts:
@@ -150,11 +172,65 @@ if (isPs1) {
150
172
  *
151
173
  * These can be executed directly.
152
174
  */
175
+ } else if (isUnixShell) {
176
+ // Capture shebang scripts (no extension) and .sh scripts
177
+ const shebang = fs.readFileSync(found, 'utf8').split('\n')[0].trim();
178
+ const interpreter = shebang.startsWith('#!')
179
+ ? shebang
180
+ .slice(2)
181
+ .trim()
182
+ .replace(/^\/usr\/bin\/env\s+/, '')
183
+ .replace(/^\/bin\/env\s+/, '')
184
+ .replace(/^\/usr\/bin\//, '')
185
+ .replace(/^\/bin\//, '')
186
+ : null;
187
+
188
+ if (interpreter) {
189
+ const resolvedOrNull = which.sync(interpreter, { nothrow: true });
190
+ if (!resolvedOrNull) {
191
+ if (['bash', 'sh'].includes(interpreter) && isWindows()) {
192
+ const locationsToCheck = [
193
+ 'C:\\Program Files\\Git\\usr\\bin',
194
+ 'C:\\Program Files\\Git\\bin',
195
+ 'C:\\Program Files (x86)\\Git\\usr\\bin',
196
+ 'C:\\msys64\\usr\\bin',
197
+ 'C:\\msys64\\bin',
198
+ 'C:\\cygwin64\\bin',
199
+ 'C:\\cygwin\\bin',
200
+ 'C:\\MinGW\\bin',
201
+ 'C:\\MinGW\\msys\\1.0\\bin'
202
+ ];
203
+ for (const location of locationsToCheck) {
204
+ const potentialPath = path.join(location, 'bash.exe');
205
+ if (fs.existsSync(potentialPath)) {
206
+ cmd = potentialPath;
207
+ args = [found, ...process.argv.slice(2)];
208
+ break;
209
+ }
210
+ }
211
+ }
212
+ } else {
213
+ cmd = resolvedOrNull || interpreter;
214
+ args = [found, ...process.argv.slice(2)];
215
+ }
216
+ } else {
217
+ cmd = found;
218
+ args = process.argv.slice(2);
219
+ }
153
220
  } else {
154
221
  cmd = found;
155
222
  args = process.argv.slice(2);
156
223
  }
157
224
 
225
+ console.log(
226
+ `Executing: ${path.isAbsolute(cmd) ? path.relative(process.cwd(), cmd) : cmd} ${args
227
+ .map((f) => {
228
+ if (path.isAbsolute(f)) return path.relative(process.cwd(), f);
229
+ return f;
230
+ })
231
+ .join(' ')}`
232
+ );
233
+
158
234
  /**
159
235
  * Execute the selected script synchronously.
160
236
  *
@@ -165,7 +241,7 @@ if (isPs1) {
165
241
  * spawnSync waits until the child process exits.
166
242
  */
167
243
  const result = spawnSync(cmd, args, {
168
- stdio: "inherit"
244
+ stdio: 'inherit'
169
245
  });
170
246
 
171
247
  /**
package/binaries/py.cjs CHANGED
@@ -12,9 +12,11 @@
12
12
  * - fs:
13
13
  * Used to check whether files exist.
14
14
  */
15
- const { spawnSync } = require("child_process");
16
- const path = require("path");
17
- const fs = require("fs");
15
+ const { spawnSync } = require('child_process');
16
+ const path = require('path');
17
+ const fs = require('fs');
18
+ const which = require('which');
19
+ const { isWindows } = require('../src/utils/isWindows.js');
18
20
 
19
21
  /**
20
22
  * __dirname
@@ -49,7 +51,7 @@ const base = path.basename(__filename, path.extname(__filename));
49
51
  * .sh
50
52
  * executable file without extension
51
53
  */
52
- const candidates = process.platform === "win32" ? [".cmd", ".bat", ".ps1", ".vbs"] : [".sh", ""];
54
+ const candidates = process.platform === 'win32' ? ['.cmd', '.bat', '.ps1', '.vbs'] : ['.sh', ''];
53
55
 
54
56
  /**
55
57
  * Search for the first matching script
@@ -72,13 +74,32 @@ let found = null;
72
74
 
73
75
  for (const ext of candidates) {
74
76
  const script = path.join(binDir, base + ext);
75
-
76
- if (fs.existsSync(script)) {
77
+ const exists = fs.existsSync(script);
78
+ // console.log(`Checking for ${script}: ${exists ? 'found' : 'not found'}`);
79
+ if (exists) {
77
80
  found = script;
78
81
  break;
79
82
  }
80
83
  }
81
84
 
85
+ /**
86
+ * If no matching script was found,
87
+ * try check if `bash` is available and if so, check for a .sh script.
88
+ */
89
+ if (!found) {
90
+ try {
91
+ spawnSync('bash', ['--version'], { stdio: 'ignore' });
92
+ const bashScript = [path.join(binDir, base), path.join(binDir, base + '.sh')].find((script) =>
93
+ fs.existsSync(script)
94
+ );
95
+ if (bashScript) {
96
+ found = bashScript;
97
+ }
98
+ } catch {
99
+ // bash is not available, do nothing
100
+ }
101
+ }
102
+
82
103
  /**
83
104
  * If no matching script was found,
84
105
  * print an error and exit with failure code 1.
@@ -92,8 +113,9 @@ if (!found) {
92
113
  * Detect special script types
93
114
  * that require a shell/interpreter.
94
115
  */
95
- const isPs1 = found.endsWith(".ps1");
96
- const isCmd = found.endsWith(".cmd");
116
+ const isPs1 = found.endsWith('.ps1');
117
+ const isCmd = found.endsWith('.cmd');
118
+ const isUnixShell = found.endsWith('.sh') || path.extname(found) === '';
97
119
 
98
120
  /**
99
121
  * cmd
@@ -124,9 +146,9 @@ let cmd, args;
124
146
  * Forward all user-provided command-line arguments.
125
147
  */
126
148
  if (isPs1) {
127
- cmd = "powershell.exe";
149
+ cmd = 'powershell.exe';
128
150
 
129
- args = ["-NoProfile", "-ExecutionPolicy", "Bypass", "-File", found, ...process.argv.slice(2)];
151
+ args = ['-NoProfile', '-ExecutionPolicy', 'Bypass', '-File', found, ...process.argv.slice(2)];
130
152
 
131
153
  /**
132
154
  * CMD batch files:
@@ -137,9 +159,9 @@ if (isPs1) {
137
159
  * "execute command and terminate"
138
160
  */
139
161
  } else if (isCmd) {
140
- cmd = "cmd.exe";
162
+ cmd = 'cmd.exe';
141
163
 
142
- args = ["/c", found, ...process.argv.slice(2)];
164
+ args = ['/c', found, ...process.argv.slice(2)];
143
165
 
144
166
  /**
145
167
  * Other scripts:
@@ -150,11 +172,65 @@ if (isPs1) {
150
172
  *
151
173
  * These can be executed directly.
152
174
  */
175
+ } else if (isUnixShell) {
176
+ // Capture shebang scripts (no extension) and .sh scripts
177
+ const shebang = fs.readFileSync(found, 'utf8').split('\n')[0].trim();
178
+ const interpreter = shebang.startsWith('#!')
179
+ ? shebang
180
+ .slice(2)
181
+ .trim()
182
+ .replace(/^\/usr\/bin\/env\s+/, '')
183
+ .replace(/^\/bin\/env\s+/, '')
184
+ .replace(/^\/usr\/bin\//, '')
185
+ .replace(/^\/bin\//, '')
186
+ : null;
187
+
188
+ if (interpreter) {
189
+ const resolvedOrNull = which.sync(interpreter, { nothrow: true });
190
+ if (!resolvedOrNull) {
191
+ if (['bash', 'sh'].includes(interpreter) && isWindows()) {
192
+ const locationsToCheck = [
193
+ 'C:\\Program Files\\Git\\usr\\bin',
194
+ 'C:\\Program Files\\Git\\bin',
195
+ 'C:\\Program Files (x86)\\Git\\usr\\bin',
196
+ 'C:\\msys64\\usr\\bin',
197
+ 'C:\\msys64\\bin',
198
+ 'C:\\cygwin64\\bin',
199
+ 'C:\\cygwin\\bin',
200
+ 'C:\\MinGW\\bin',
201
+ 'C:\\MinGW\\msys\\1.0\\bin'
202
+ ];
203
+ for (const location of locationsToCheck) {
204
+ const potentialPath = path.join(location, 'bash.exe');
205
+ if (fs.existsSync(potentialPath)) {
206
+ cmd = potentialPath;
207
+ args = [found, ...process.argv.slice(2)];
208
+ break;
209
+ }
210
+ }
211
+ }
212
+ } else {
213
+ cmd = resolvedOrNull || interpreter;
214
+ args = [found, ...process.argv.slice(2)];
215
+ }
216
+ } else {
217
+ cmd = found;
218
+ args = process.argv.slice(2);
219
+ }
153
220
  } else {
154
221
  cmd = found;
155
222
  args = process.argv.slice(2);
156
223
  }
157
224
 
225
+ console.log(
226
+ `Executing: ${path.isAbsolute(cmd) ? path.relative(process.cwd(), cmd) : cmd} ${args
227
+ .map((f) => {
228
+ if (path.isAbsolute(f)) return path.relative(process.cwd(), f);
229
+ return f;
230
+ })
231
+ .join(' ')}`
232
+ );
233
+
158
234
  /**
159
235
  * Execute the selected script synchronously.
160
236
  *
@@ -165,7 +241,7 @@ if (isPs1) {
165
241
  * spawnSync waits until the child process exits.
166
242
  */
167
243
  const result = spawnSync(cmd, args, {
168
- stdio: "inherit"
244
+ stdio: 'inherit'
169
245
  });
170
246
 
171
247
  /**
@@ -12,9 +12,11 @@
12
12
  * - fs:
13
13
  * Used to check whether files exist.
14
14
  */
15
- const { spawnSync } = require("child_process");
16
- const path = require("path");
17
- const fs = require("fs");
15
+ const { spawnSync } = require('child_process');
16
+ const path = require('path');
17
+ const fs = require('fs');
18
+ const which = require('which');
19
+ const { isWindows } = require('../src/utils/isWindows.js');
18
20
 
19
21
  /**
20
22
  * __dirname
@@ -49,7 +51,7 @@ const base = path.basename(__filename, path.extname(__filename));
49
51
  * .sh
50
52
  * executable file without extension
51
53
  */
52
- const candidates = process.platform === "win32" ? [".cmd", ".bat", ".ps1", ".vbs"] : [".sh", ""];
54
+ const candidates = process.platform === 'win32' ? ['.cmd', '.bat', '.ps1', '.vbs'] : ['.sh', ''];
53
55
 
54
56
  /**
55
57
  * Search for the first matching script
@@ -72,13 +74,32 @@ let found = null;
72
74
 
73
75
  for (const ext of candidates) {
74
76
  const script = path.join(binDir, base + ext);
75
-
76
- if (fs.existsSync(script)) {
77
+ const exists = fs.existsSync(script);
78
+ // console.log(`Checking for ${script}: ${exists ? 'found' : 'not found'}`);
79
+ if (exists) {
77
80
  found = script;
78
81
  break;
79
82
  }
80
83
  }
81
84
 
85
+ /**
86
+ * If no matching script was found,
87
+ * try check if `bash` is available and if so, check for a .sh script.
88
+ */
89
+ if (!found) {
90
+ try {
91
+ spawnSync('bash', ['--version'], { stdio: 'ignore' });
92
+ const bashScript = [path.join(binDir, base), path.join(binDir, base + '.sh')].find((script) =>
93
+ fs.existsSync(script)
94
+ );
95
+ if (bashScript) {
96
+ found = bashScript;
97
+ }
98
+ } catch {
99
+ // bash is not available, do nothing
100
+ }
101
+ }
102
+
82
103
  /**
83
104
  * If no matching script was found,
84
105
  * print an error and exit with failure code 1.
@@ -92,8 +113,9 @@ if (!found) {
92
113
  * Detect special script types
93
114
  * that require a shell/interpreter.
94
115
  */
95
- const isPs1 = found.endsWith(".ps1");
96
- const isCmd = found.endsWith(".cmd");
116
+ const isPs1 = found.endsWith('.ps1');
117
+ const isCmd = found.endsWith('.cmd');
118
+ const isUnixShell = found.endsWith('.sh') || path.extname(found) === '';
97
119
 
98
120
  /**
99
121
  * cmd
@@ -124,9 +146,9 @@ let cmd, args;
124
146
  * Forward all user-provided command-line arguments.
125
147
  */
126
148
  if (isPs1) {
127
- cmd = "powershell.exe";
149
+ cmd = 'powershell.exe';
128
150
 
129
- args = ["-NoProfile", "-ExecutionPolicy", "Bypass", "-File", found, ...process.argv.slice(2)];
151
+ args = ['-NoProfile', '-ExecutionPolicy', 'Bypass', '-File', found, ...process.argv.slice(2)];
130
152
 
131
153
  /**
132
154
  * CMD batch files:
@@ -137,9 +159,9 @@ if (isPs1) {
137
159
  * "execute command and terminate"
138
160
  */
139
161
  } else if (isCmd) {
140
- cmd = "cmd.exe";
162
+ cmd = 'cmd.exe';
141
163
 
142
- args = ["/c", found, ...process.argv.slice(2)];
164
+ args = ['/c', found, ...process.argv.slice(2)];
143
165
 
144
166
  /**
145
167
  * Other scripts:
@@ -150,11 +172,65 @@ if (isPs1) {
150
172
  *
151
173
  * These can be executed directly.
152
174
  */
175
+ } else if (isUnixShell) {
176
+ // Capture shebang scripts (no extension) and .sh scripts
177
+ const shebang = fs.readFileSync(found, 'utf8').split('\n')[0].trim();
178
+ const interpreter = shebang.startsWith('#!')
179
+ ? shebang
180
+ .slice(2)
181
+ .trim()
182
+ .replace(/^\/usr\/bin\/env\s+/, '')
183
+ .replace(/^\/bin\/env\s+/, '')
184
+ .replace(/^\/usr\/bin\//, '')
185
+ .replace(/^\/bin\//, '')
186
+ : null;
187
+
188
+ if (interpreter) {
189
+ const resolvedOrNull = which.sync(interpreter, { nothrow: true });
190
+ if (!resolvedOrNull) {
191
+ if (['bash', 'sh'].includes(interpreter) && isWindows()) {
192
+ const locationsToCheck = [
193
+ 'C:\\Program Files\\Git\\usr\\bin',
194
+ 'C:\\Program Files\\Git\\bin',
195
+ 'C:\\Program Files (x86)\\Git\\usr\\bin',
196
+ 'C:\\msys64\\usr\\bin',
197
+ 'C:\\msys64\\bin',
198
+ 'C:\\cygwin64\\bin',
199
+ 'C:\\cygwin\\bin',
200
+ 'C:\\MinGW\\bin',
201
+ 'C:\\MinGW\\msys\\1.0\\bin'
202
+ ];
203
+ for (const location of locationsToCheck) {
204
+ const potentialPath = path.join(location, 'bash.exe');
205
+ if (fs.existsSync(potentialPath)) {
206
+ cmd = potentialPath;
207
+ args = [found, ...process.argv.slice(2)];
208
+ break;
209
+ }
210
+ }
211
+ }
212
+ } else {
213
+ cmd = resolvedOrNull || interpreter;
214
+ args = [found, ...process.argv.slice(2)];
215
+ }
216
+ } else {
217
+ cmd = found;
218
+ args = process.argv.slice(2);
219
+ }
153
220
  } else {
154
221
  cmd = found;
155
222
  args = process.argv.slice(2);
156
223
  }
157
224
 
225
+ console.log(
226
+ `Executing: ${path.isAbsolute(cmd) ? path.relative(process.cwd(), cmd) : cmd} ${args
227
+ .map((f) => {
228
+ if (path.isAbsolute(f)) return path.relative(process.cwd(), f);
229
+ return f;
230
+ })
231
+ .join(' ')}`
232
+ );
233
+
158
234
  /**
159
235
  * Execute the selected script synchronously.
160
236
  *
@@ -165,7 +241,7 @@ if (isPs1) {
165
241
  * spawnSync waits until the child process exits.
166
242
  */
167
243
  const result = spawnSync(cmd, args, {
168
- stdio: "inherit"
244
+ stdio: 'inherit'
169
245
  });
170
246
 
171
247
  /**