@fredlackey/devutils 0.0.19 → 0.1.0

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 (122) hide show
  1. package/README.md +223 -32
  2. package/package.json +7 -5
  3. package/src/api/loader.js +229 -0
  4. package/src/api/registry.json +62 -0
  5. package/src/cli.js +305 -0
  6. package/src/commands/ai/index.js +16 -0
  7. package/src/commands/ai/launch.js +112 -0
  8. package/src/commands/ai/list.js +54 -0
  9. package/src/commands/ai/resume.js +70 -0
  10. package/src/commands/ai/sessions.js +121 -0
  11. package/src/commands/ai/set.js +131 -0
  12. package/src/commands/ai/show.js +74 -0
  13. package/src/commands/ai/tools.js +46 -0
  14. package/src/commands/alias/add.js +93 -0
  15. package/src/commands/alias/helpers.js +107 -0
  16. package/src/commands/alias/index.js +14 -0
  17. package/src/commands/alias/list.js +55 -0
  18. package/src/commands/alias/remove.js +62 -0
  19. package/src/commands/alias/sync.js +109 -0
  20. package/src/commands/api/disable.js +73 -0
  21. package/src/commands/api/enable.js +148 -0
  22. package/src/commands/api/index.js +15 -0
  23. package/src/commands/api/list.js +66 -0
  24. package/src/commands/api/update.js +87 -0
  25. package/src/commands/auth/index.js +15 -0
  26. package/src/commands/auth/list.js +49 -0
  27. package/src/commands/auth/login.js +384 -0
  28. package/src/commands/auth/logout.js +111 -0
  29. package/src/commands/auth/refresh.js +184 -0
  30. package/src/commands/auth/services.js +169 -0
  31. package/src/commands/auth/status.js +104 -0
  32. package/src/commands/config/export.js +224 -0
  33. package/src/commands/config/get.js +52 -0
  34. package/src/commands/config/import.js +308 -0
  35. package/src/commands/config/index.js +17 -0
  36. package/src/commands/config/init.js +143 -0
  37. package/src/commands/config/reset.js +57 -0
  38. package/src/commands/config/set.js +93 -0
  39. package/src/commands/config/show.js +35 -0
  40. package/src/commands/help.js +338 -0
  41. package/src/commands/identity/add.js +133 -0
  42. package/src/commands/identity/index.js +17 -0
  43. package/src/commands/identity/link.js +76 -0
  44. package/src/commands/identity/list.js +48 -0
  45. package/src/commands/identity/remove.js +72 -0
  46. package/src/commands/identity/show.js +65 -0
  47. package/src/commands/identity/sync.js +172 -0
  48. package/src/commands/identity/unlink.js +57 -0
  49. package/src/commands/ignore/add.js +165 -0
  50. package/src/commands/ignore/index.js +14 -0
  51. package/src/commands/ignore/list.js +89 -0
  52. package/src/commands/ignore/markers.js +43 -0
  53. package/src/commands/ignore/remove.js +164 -0
  54. package/src/commands/ignore/show.js +169 -0
  55. package/src/commands/machine/detect.js +122 -0
  56. package/src/commands/machine/index.js +14 -0
  57. package/src/commands/machine/list.js +74 -0
  58. package/src/commands/machine/set.js +106 -0
  59. package/src/commands/machine/show.js +35 -0
  60. package/src/commands/schema.js +152 -0
  61. package/src/commands/search/collections.js +134 -0
  62. package/src/commands/search/get.js +71 -0
  63. package/src/commands/search/index-cmd.js +54 -0
  64. package/src/commands/search/index.js +21 -0
  65. package/src/commands/search/keyword.js +60 -0
  66. package/src/commands/search/qmd.js +70 -0
  67. package/src/commands/search/query.js +64 -0
  68. package/src/commands/search/semantic.js +62 -0
  69. package/src/commands/search/status.js +46 -0
  70. package/src/commands/status.js +276 -0
  71. package/src/commands/tools/check.js +79 -0
  72. package/src/commands/tools/index.js +14 -0
  73. package/src/commands/tools/install.js +110 -0
  74. package/src/commands/tools/list.js +91 -0
  75. package/src/commands/tools/search.js +60 -0
  76. package/src/commands/update.js +113 -0
  77. package/src/commands/util/add.js +151 -0
  78. package/src/commands/util/index.js +15 -0
  79. package/src/commands/util/list.js +97 -0
  80. package/src/commands/util/remove.js +76 -0
  81. package/src/commands/util/run.js +79 -0
  82. package/src/commands/util/show.js +67 -0
  83. package/src/commands/version.js +33 -0
  84. package/src/installers/_template.js +104 -0
  85. package/src/installers/git.js +150 -0
  86. package/src/installers/homebrew.js +190 -0
  87. package/src/installers/node.js +223 -0
  88. package/src/installers/registry.json +29 -0
  89. package/src/lib/config.js +125 -0
  90. package/src/lib/detect.js +74 -0
  91. package/src/lib/errors.js +114 -0
  92. package/src/lib/github.js +315 -0
  93. package/src/lib/installer.js +225 -0
  94. package/src/lib/output.js +239 -0
  95. package/src/lib/platform.js +112 -0
  96. package/src/lib/platforms/amazon-linux.js +41 -0
  97. package/src/lib/platforms/gitbash.js +46 -0
  98. package/src/lib/platforms/macos.js +45 -0
  99. package/src/lib/platforms/raspbian.js +41 -0
  100. package/src/lib/platforms/ubuntu.js +39 -0
  101. package/src/lib/platforms/windows.js +45 -0
  102. package/src/lib/prompt.js +161 -0
  103. package/src/lib/schema.js +211 -0
  104. package/src/lib/shell.js +75 -0
  105. package/src/patterns/gitignore/claude-code.txt +25 -0
  106. package/src/patterns/gitignore/docker.txt +15 -0
  107. package/src/patterns/gitignore/go.txt +24 -0
  108. package/src/patterns/gitignore/java.txt +38 -0
  109. package/src/patterns/gitignore/jetbrains.txt +26 -0
  110. package/src/patterns/gitignore/linux.txt +18 -0
  111. package/src/patterns/gitignore/macos.txt +27 -0
  112. package/src/patterns/gitignore/node.txt +51 -0
  113. package/src/patterns/gitignore/python.txt +55 -0
  114. package/src/patterns/gitignore/rust.txt +14 -0
  115. package/src/patterns/gitignore/terraform.txt +30 -0
  116. package/src/patterns/gitignore/vscode.txt +15 -0
  117. package/src/patterns/gitignore/windows.txt +25 -0
  118. package/src/utils/clone/index.js +165 -0
  119. package/src/utils/git-push/index.js +230 -0
  120. package/src/utils/git-status/index.js +116 -0
  121. package/src/utils/git-status/unix.sh +75 -0
  122. package/src/utils/registry.json +41 -0
@@ -0,0 +1,225 @@
1
+ 'use strict';
2
+
3
+ const path = require('path');
4
+ const fs = require('fs');
5
+
6
+ /**
7
+ * Cached registry data. Set on first call to loadRegistry().
8
+ * The registry doesn't change during a CLI run, so we only read it once.
9
+ * @type {Array<object>|null}
10
+ */
11
+ let registryCache = null;
12
+
13
+ /**
14
+ * Reads and parses the installer registry (src/installers/registry.json).
15
+ * Returns the tools array. Results are cached after the first call.
16
+ *
17
+ * @returns {Array<object>} The array of tool entries from the registry.
18
+ */
19
+ function loadRegistry() {
20
+ if (registryCache) {
21
+ return registryCache;
22
+ }
23
+
24
+ const registryPath = path.join(__dirname, '..', 'installers', 'registry.json');
25
+ const raw = fs.readFileSync(registryPath, 'utf8');
26
+ const data = JSON.parse(raw);
27
+ registryCache = data.tools || [];
28
+ return registryCache;
29
+ }
30
+
31
+ /**
32
+ * Looks up a tool by name in the registry.
33
+ * Comparison is case-insensitive.
34
+ *
35
+ * @param {string} name - The tool name to look up (e.g. 'git', 'Node').
36
+ * @returns {object|null} The tool entry, or null if not found.
37
+ */
38
+ function findTool(name) {
39
+ if (!name) {
40
+ return null;
41
+ }
42
+
43
+ const tools = loadRegistry();
44
+ const lower = name.toLowerCase();
45
+ return tools.find(t => t.name.toLowerCase() === lower) || null;
46
+ }
47
+
48
+ /**
49
+ * Loads the installer module for a given tool entry.
50
+ * Validates that the file exists and exports the required functions (isInstalled, install).
51
+ *
52
+ * @param {object} toolEntry - A tool entry from the registry (must have an `installer` field).
53
+ * @returns {object} The installer module (with isInstalled, install, etc.).
54
+ * @throws {Error} If the installer file is missing or doesn't export required functions.
55
+ */
56
+ function loadInstaller(toolEntry) {
57
+ if (!toolEntry || !toolEntry.installer) {
58
+ throw new Error('Invalid tool entry: missing installer field.');
59
+ }
60
+
61
+ const installerPath = path.join(__dirname, '..', 'installers', toolEntry.installer);
62
+
63
+ if (!fs.existsSync(installerPath)) {
64
+ throw new Error(
65
+ `Installer file not found: ${toolEntry.installer}. ` +
66
+ `Expected at ${installerPath}`
67
+ );
68
+ }
69
+
70
+ const mod = require(installerPath);
71
+
72
+ if (typeof mod.isInstalled !== 'function') {
73
+ throw new Error(
74
+ `Installer "${toolEntry.installer}" does not export an isInstalled() function.`
75
+ );
76
+ }
77
+
78
+ if (typeof mod.install !== 'function') {
79
+ throw new Error(
80
+ `Installer "${toolEntry.installer}" does not export an install() function.`
81
+ );
82
+ }
83
+
84
+ return mod;
85
+ }
86
+
87
+ /**
88
+ * Resolves the full dependency chain for a tool, in installation order.
89
+ * If tool A depends on B, and B depends on C, returns ['C', 'B', 'A'].
90
+ * Detects circular dependencies and throws if found.
91
+ *
92
+ * @param {string} toolName - The name of the tool to resolve.
93
+ * @param {Set<string>} [visited] - Already-visited tool names (used internally for recursion).
94
+ * @param {string[]} [chain] - Current dependency chain (used internally for circular detection).
95
+ * @returns {string[]} Flat, ordered array of tool names to install.
96
+ * @throws {Error} If a circular dependency is detected or a tool is unknown.
97
+ */
98
+ function resolveDependencies(toolName, visited = new Set(), chain = []) {
99
+ if (chain.includes(toolName)) {
100
+ throw new Error(
101
+ `Circular dependency detected: ${chain.join(' -> ')} -> ${toolName}`
102
+ );
103
+ }
104
+
105
+ if (visited.has(toolName)) {
106
+ return [];
107
+ }
108
+
109
+ const tool = findTool(toolName);
110
+ if (!tool) {
111
+ throw new Error(`Unknown tool: ${toolName}`);
112
+ }
113
+
114
+ chain.push(toolName);
115
+ visited.add(toolName);
116
+
117
+ const result = [];
118
+ for (const dep of tool.dependencies) {
119
+ result.push(...resolveDependencies(dep, visited, [...chain]));
120
+ }
121
+ result.push(toolName);
122
+ return result;
123
+ }
124
+
125
+ /**
126
+ * Checks if a tool is installed on the current system.
127
+ * Loads the tool's installer and calls its isInstalled() function.
128
+ *
129
+ * @param {string} toolName - The name of the tool to check.
130
+ * @param {object} context - The CLI context object.
131
+ * @returns {Promise<boolean>} true if the tool is installed.
132
+ * @throws {Error} If the tool is unknown or the installer can't be loaded.
133
+ */
134
+ async function checkInstalled(toolName, context) {
135
+ const tool = findTool(toolName);
136
+ if (!tool) {
137
+ throw new Error(`Unknown tool: ${toolName}`);
138
+ }
139
+
140
+ const mod = loadInstaller(tool);
141
+ return mod.isInstalled(context);
142
+ }
143
+
144
+ /**
145
+ * Installs a tool, including any dependencies, using the platform-appropriate method.
146
+ *
147
+ * Steps:
148
+ * 1. Looks up the tool in the registry. Throws if not found.
149
+ * 2. Checks if the current platform is supported. Throws if not.
150
+ * 3. Resolves the dependency chain.
151
+ * 4. For each tool in the chain, checks if it's already installed.
152
+ * If not, loads the installer and calls install(context).
153
+ * 5. Returns a summary object.
154
+ *
155
+ * @param {string} toolName - The name of the tool to install.
156
+ * @param {object} context - The CLI context object.
157
+ * @returns {Promise<{ tool: string, alreadyInstalled: boolean, dependenciesInstalled: string[], installed: boolean }>}
158
+ * @throws {Error} If the tool is unknown, unsupported on the current platform, or installation fails.
159
+ */
160
+ async function installTool(toolName, context) {
161
+ const tool = findTool(toolName);
162
+ if (!tool) {
163
+ throw new Error(`Tool '${toolName}' not found in registry.`);
164
+ }
165
+
166
+ // Check platform support
167
+ const platformType = context.platform.detect().type;
168
+ if (!tool.platforms.includes(platformType)) {
169
+ throw new Error(
170
+ `Tool '${toolName}' is not supported on ${platformType}. ` +
171
+ `Supported platforms: ${tool.platforms.join(', ')}`
172
+ );
173
+ }
174
+
175
+ // Resolve dependencies
176
+ const chain = resolveDependencies(toolName);
177
+ const dependenciesInstalled = [];
178
+ let alreadyInstalled = false;
179
+ let installed = false;
180
+
181
+ for (const depName of chain) {
182
+ const depTool = findTool(depName);
183
+ const depMod = loadInstaller(depTool);
184
+ const isAlready = await depMod.isInstalled(context);
185
+
186
+ if (isAlready) {
187
+ if (depName === toolName) {
188
+ alreadyInstalled = true;
189
+ }
190
+ continue;
191
+ }
192
+
193
+ // Check platform support for the dependency too
194
+ if (!depTool.platforms.includes(platformType)) {
195
+ throw new Error(
196
+ `Dependency '${depName}' is not supported on ${platformType}. ` +
197
+ `Cannot install '${toolName}'.`
198
+ );
199
+ }
200
+
201
+ await depMod.install(context);
202
+
203
+ if (depName === toolName) {
204
+ installed = true;
205
+ } else {
206
+ dependenciesInstalled.push(depName);
207
+ }
208
+ }
209
+
210
+ return {
211
+ tool: toolName,
212
+ alreadyInstalled,
213
+ dependenciesInstalled,
214
+ installed,
215
+ };
216
+ }
217
+
218
+ module.exports = {
219
+ loadRegistry,
220
+ findTool,
221
+ loadInstaller,
222
+ resolveDependencies,
223
+ checkInstalled,
224
+ installTool,
225
+ };
@@ -0,0 +1,239 @@
1
+ 'use strict';
2
+
3
+ /**
4
+ * Renders data as JSON. Pretty-prints for TTY callers, compact for machines.
5
+ * @param {*} data - The data to render.
6
+ * @param {string} caller - The caller context ('ai', 'ci', 'tty', 'pipe').
7
+ * @returns {string}
8
+ */
9
+ function renderJson(data, caller) {
10
+ if (caller === 'tty') {
11
+ return JSON.stringify(data, null, 2);
12
+ }
13
+ return JSON.stringify(data);
14
+ }
15
+
16
+ /**
17
+ * Renders an array of objects as an aligned text table.
18
+ * @param {Array<object>|object} data - The data to render. Single objects are wrapped in an array.
19
+ * @returns {string}
20
+ */
21
+ function renderTable(data) {
22
+ if (!Array.isArray(data)) data = [data];
23
+ if (data.length === 0) return '(no data)';
24
+
25
+ const keys = Object.keys(data[0]);
26
+
27
+ // Calculate column widths
28
+ const widths = {};
29
+ for (const key of keys) {
30
+ widths[key] = key.length;
31
+ }
32
+ for (const row of data) {
33
+ for (const key of keys) {
34
+ const val = String(row[key] === undefined ? '' : row[key]);
35
+ widths[key] = Math.max(widths[key], val.length);
36
+ }
37
+ }
38
+
39
+ // Build header
40
+ const header = keys.map(k => k.padEnd(widths[k])).join(' ');
41
+ const separator = keys.map(k => '-'.repeat(widths[k])).join(' ');
42
+
43
+ // Build rows
44
+ const rows = data.map(row => {
45
+ return keys.map(k => {
46
+ const val = String(row[k] === undefined ? '' : row[k]);
47
+ return val.padEnd(widths[k]);
48
+ }).join(' ');
49
+ });
50
+
51
+ return [header, separator, ...rows].join('\n');
52
+ }
53
+
54
+ /**
55
+ * Formats a value for YAML output.
56
+ * Quotes strings that could be misinterpreted by YAML parsers.
57
+ * @param {*} value - The value to format.
58
+ * @returns {string}
59
+ */
60
+ function formatYamlValue(value) {
61
+ if (value === null || value === undefined) return 'null';
62
+ if (typeof value === 'boolean') return value ? 'true' : 'false';
63
+ if (typeof value === 'number') return String(value);
64
+ if (typeof value === 'string') {
65
+ if (/[:#\[\]{}&*!|>'"%@`]/.test(value) || value === '' || value === 'true' || value === 'false' || value === 'null') {
66
+ return `"${value.replace(/"/g, '\\"')}"`;
67
+ }
68
+ return value;
69
+ }
70
+ return String(value);
71
+ }
72
+
73
+ /**
74
+ * Renders data as simple YAML.
75
+ * Handles nested objects, arrays, strings, numbers, booleans, and null.
76
+ * @param {*} data - The data to render.
77
+ * @param {number} [indent=0] - Current indentation level.
78
+ * @returns {string}
79
+ */
80
+ function renderYaml(data, indent = 0) {
81
+ const prefix = ' '.repeat(indent);
82
+
83
+ if (Array.isArray(data)) {
84
+ if (data.length === 0) return `${prefix}[]`;
85
+ return data.map(item => {
86
+ if (typeof item === 'object' && item !== null) {
87
+ const inner = renderYaml(item, indent + 1);
88
+ return `${prefix}-\n${inner}`;
89
+ }
90
+ return `${prefix}- ${formatYamlValue(item)}`;
91
+ }).join('\n');
92
+ }
93
+
94
+ if (typeof data === 'object' && data !== null) {
95
+ const entries = Object.entries(data);
96
+ if (entries.length === 0) return `${prefix}{}`;
97
+ return entries.map(([key, value]) => {
98
+ if (typeof value === 'object' && value !== null) {
99
+ return `${prefix}${key}:\n${renderYaml(value, indent + 1)}`;
100
+ }
101
+ return `${prefix}${key}: ${formatYamlValue(value)}`;
102
+ }).join('\n');
103
+ }
104
+
105
+ return `${prefix}${formatYamlValue(data)}`;
106
+ }
107
+
108
+ /**
109
+ * Renders an array of objects as CSV with proper escaping.
110
+ * @param {Array<object>|object} data - The data to render.
111
+ * @returns {string}
112
+ */
113
+ function renderCsv(data) {
114
+ if (!Array.isArray(data)) data = [data];
115
+ if (data.length === 0) return '';
116
+
117
+ const keys = Object.keys(data[0]);
118
+
119
+ function escapeCsvField(value) {
120
+ const str = String(value === undefined || value === null ? '' : value);
121
+ if (str.includes(',') || str.includes('"') || str.includes('\n')) {
122
+ return `"${str.replace(/"/g, '""')}"`;
123
+ }
124
+ return str;
125
+ }
126
+
127
+ const header = keys.map(escapeCsvField).join(',');
128
+ const rows = data.map(row => {
129
+ return keys.map(k => escapeCsvField(row[k])).join(',');
130
+ });
131
+
132
+ return [header, ...rows].join('\n');
133
+ }
134
+
135
+ /**
136
+ * Renders data in the specified format.
137
+ * Falls back to JSON for unknown formats.
138
+ *
139
+ * @param {*} data - The data to render.
140
+ * @param {string} format - The output format ('json', 'table', 'yaml', 'csv').
141
+ * @param {string} [caller] - The caller context (affects JSON pretty-printing).
142
+ * @returns {string}
143
+ */
144
+ function render(data, format, caller) {
145
+ switch (format) {
146
+ case 'json':
147
+ return renderJson(data, caller);
148
+ case 'table':
149
+ return renderTable(data);
150
+ case 'yaml':
151
+ return renderYaml(data);
152
+ case 'csv':
153
+ return renderCsv(data);
154
+ default:
155
+ return renderJson(data, caller);
156
+ }
157
+ }
158
+
159
+ /**
160
+ * Alias for render(). Some command stories use context.output.print(data).
161
+ */
162
+ const print = render;
163
+
164
+ /**
165
+ * Writes a plain text informational message to stdout.
166
+ * Does not go through the format rendering pipeline.
167
+ * @param {string} message - The message to write.
168
+ */
169
+ function info(message) {
170
+ process.stdout.write(message + '\n');
171
+ }
172
+
173
+ /**
174
+ * Writes a plain text error message to stderr.
175
+ * Does not go through the format rendering pipeline.
176
+ * @param {string} message - The message to write.
177
+ */
178
+ function error(message) {
179
+ process.stderr.write(message + '\n');
180
+ }
181
+
182
+ /**
183
+ * Renders an error object for output.
184
+ * Shows a human-readable message for table format, JSON for everything else.
185
+ *
186
+ * @param {object} errorObj - The error object (typically { error: { code, message, service } }).
187
+ * @param {string} format - The output format.
188
+ * @returns {string}
189
+ */
190
+ function renderError(errorObj, format) {
191
+ if (format === 'table') {
192
+ const msg = errorObj.error ? errorObj.error.message : JSON.stringify(errorObj);
193
+ return `Error: ${msg}`;
194
+ }
195
+ return JSON.stringify(errorObj);
196
+ }
197
+
198
+ /**
199
+ * Creates a pre-configured formatter bound to a specific format and caller.
200
+ * Commands use this so they don't have to pass format/caller on every call.
201
+ *
202
+ * @param {{ format: string, caller: string }} context - The format and caller context.
203
+ * @returns {object} An object with out(), err(), render(), print(), info(), error(), renderError() methods.
204
+ */
205
+ function createFormatter(context) {
206
+ const { format, caller } = context;
207
+
208
+ function _render(data) {
209
+ return render(data, format, caller);
210
+ }
211
+
212
+ function _info(message) {
213
+ process.stdout.write(message + '\n');
214
+ }
215
+
216
+ function _error(message) {
217
+ process.stderr.write(message + '\n');
218
+ }
219
+
220
+ return {
221
+ out(data) {
222
+ const output = render(data, format, caller);
223
+ process.stdout.write(output + '\n');
224
+ },
225
+ err(errorObj) {
226
+ const output = renderError(errorObj, format);
227
+ process.stderr.write(output + '\n');
228
+ },
229
+ render: _render,
230
+ print: _render,
231
+ info: _info,
232
+ error: _error,
233
+ renderError(errorObj) {
234
+ return renderError(errorObj, format);
235
+ },
236
+ };
237
+ }
238
+
239
+ module.exports = { render, print, renderError, createFormatter, info, error };
@@ -0,0 +1,112 @@
1
+ 'use strict';
2
+
3
+ const fs = require('fs');
4
+
5
+ /**
6
+ * Cached detection result. Set on first call to detect().
7
+ * The OS doesn't change while the process is running, so we only detect once.
8
+ * @type {{ type: string, arch: string, packageManager: string|null }|null}
9
+ */
10
+ let cached = null;
11
+
12
+ /**
13
+ * Lazy-loading lookup for platform helper modules.
14
+ * Each entry is a function that requires the platform file only when called.
15
+ * Other modules should never import platform files directly -- always go
16
+ * through platform.js using getHelper().
17
+ */
18
+ const helpers = {
19
+ 'macos': () => require('./platforms/macos'),
20
+ 'ubuntu': () => require('./platforms/ubuntu'),
21
+ 'raspbian': () => require('./platforms/raspbian'),
22
+ 'amazon-linux': () => require('./platforms/amazon-linux'),
23
+ 'windows': () => require('./platforms/windows'),
24
+ 'gitbash': () => require('./platforms/gitbash'),
25
+ };
26
+
27
+ /**
28
+ * Reads /etc/os-release and returns the value of the ID= line.
29
+ * This tells us which Linux distro is running (e.g. 'ubuntu', 'raspbian', 'amzn').
30
+ * @returns {string|null} The distro ID in lowercase, or null if it can't be determined.
31
+ */
32
+ function getLinuxDistroId() {
33
+ try {
34
+ const content = fs.readFileSync('/etc/os-release', 'utf8');
35
+ const match = content.match(/^ID=["']?([^"'\n]+)["']?/m);
36
+ return match ? match[1].toLowerCase() : null;
37
+ } catch {
38
+ return null;
39
+ }
40
+ }
41
+
42
+ /**
43
+ * Detects the current platform and returns an object describing the OS type,
44
+ * CPU architecture, and default package manager.
45
+ *
46
+ * Results are cached after the first call. Calling detect() multiple times
47
+ * returns the same object reference.
48
+ *
49
+ * @returns {{ type: string, arch: string, packageManager: string|null }}
50
+ * - type: 'macos', 'ubuntu', 'raspbian', 'amazon-linux', 'windows', 'gitbash', or 'linux' (unknown)
51
+ * - arch: CPU architecture string from process.arch (e.g. 'x64', 'arm64')
52
+ * - packageManager: 'brew', 'apt', 'dnf', 'yum', 'choco', 'manual', or null
53
+ */
54
+ function detect() {
55
+ if (cached) {
56
+ return cached;
57
+ }
58
+
59
+ const arch = process.arch;
60
+ let type = 'linux';
61
+ let packageManager = null;
62
+
63
+ if (process.platform === 'darwin') {
64
+ type = 'macos';
65
+ packageManager = 'brew';
66
+ } else if (process.platform === 'win32') {
67
+ // Git Bash sets the MSYSTEM environment variable (e.g. 'MINGW64').
68
+ // If it's set, we're running inside Git Bash. Otherwise, native Windows.
69
+ if (process.env.MSYSTEM) {
70
+ type = 'gitbash';
71
+ packageManager = 'manual';
72
+ } else {
73
+ type = 'windows';
74
+ packageManager = 'choco';
75
+ }
76
+ } else if (process.platform === 'linux') {
77
+ const distroId = getLinuxDistroId();
78
+
79
+ if (distroId === 'ubuntu') {
80
+ type = 'ubuntu';
81
+ packageManager = 'apt';
82
+ } else if (distroId === 'raspbian') {
83
+ type = 'raspbian';
84
+ packageManager = 'apt';
85
+ } else if (distroId === 'amzn') {
86
+ type = 'amazon-linux';
87
+ // Use dnf if available (AL2023+), otherwise fall back to yum
88
+ packageManager = fs.existsSync('/usr/bin/dnf') ? 'dnf' : 'yum';
89
+ } else {
90
+ // Unknown Linux distro -- don't crash, just report what we know
91
+ type = 'linux';
92
+ packageManager = null;
93
+ }
94
+ }
95
+
96
+ cached = { type, arch, packageManager };
97
+ return cached;
98
+ }
99
+
100
+ /**
101
+ * Returns the platform-specific helper module for the detected platform.
102
+ * The helper module exports functions like isInstalled(), getAppPaths(), etc.
103
+ *
104
+ * @returns {object|null} The helper module, or null if the platform has no helper.
105
+ */
106
+ function getHelper() {
107
+ const { type } = detect();
108
+ const loader = helpers[type];
109
+ return loader ? loader() : null;
110
+ }
111
+
112
+ module.exports = { detect, getHelper };
@@ -0,0 +1,41 @@
1
+ 'use strict';
2
+
3
+ const fs = require('fs');
4
+ const { execFileSync } = require('child_process');
5
+
6
+ /**
7
+ * Platform name identifier.
8
+ * @type {string}
9
+ */
10
+ const name = 'amazon-linux';
11
+
12
+ /**
13
+ * Default package manager for Amazon Linux.
14
+ * Uses dnf on newer versions (AL2023+), falls back to yum on older versions.
15
+ * @type {string}
16
+ */
17
+ const packageManager = fs.existsSync('/usr/bin/dnf') ? 'dnf' : 'yum';
18
+
19
+ /**
20
+ * Checks if a binary is available on the system PATH.
21
+ * @param {string} binary - The name of the binary to look for (e.g. 'node', 'git').
22
+ * @returns {boolean} True if the binary exists on the PATH, false otherwise.
23
+ */
24
+ function isInstalled(binary) {
25
+ try {
26
+ execFileSync('which', [binary], { stdio: 'ignore' });
27
+ return true;
28
+ } catch {
29
+ return false;
30
+ }
31
+ }
32
+
33
+ /**
34
+ * Returns common application directories for Amazon Linux.
35
+ * @returns {string[]} Array of directory paths where applications are typically installed.
36
+ */
37
+ function getAppPaths() {
38
+ return ['/usr/bin', '/usr/local/bin'];
39
+ }
40
+
41
+ module.exports = { name, packageManager, isInstalled, getAppPaths };
@@ -0,0 +1,46 @@
1
+ 'use strict';
2
+
3
+ const { execFileSync } = require('child_process');
4
+
5
+ /**
6
+ * Platform name identifier.
7
+ * @type {string}
8
+ */
9
+ const name = 'gitbash';
10
+
11
+ /**
12
+ * Package manager for Git Bash.
13
+ * Git Bash doesn't have its own package manager, so this is 'manual'.
14
+ * @type {string}
15
+ */
16
+ const packageManager = 'manual';
17
+
18
+ /**
19
+ * Checks if a binary is available on the system PATH.
20
+ * Git Bash provides a "which" command, so we use that instead of "where".
21
+ * @param {string} binary - The name of the binary to look for (e.g. 'node', 'git').
22
+ * @returns {boolean} True if the binary exists on the PATH, false otherwise.
23
+ */
24
+ function isInstalled(binary) {
25
+ try {
26
+ execFileSync('which', [binary], { stdio: 'ignore' });
27
+ return true;
28
+ } catch {
29
+ return false;
30
+ }
31
+ }
32
+
33
+ /**
34
+ * Returns common application directories for Git Bash.
35
+ * Git Bash runs on top of Windows, so the paths are the same as Windows.
36
+ * @returns {string[]} Array of directory paths where applications are typically installed.
37
+ */
38
+ function getAppPaths() {
39
+ return [
40
+ process.env.ProgramFiles,
41
+ process.env['ProgramFiles(x86)'],
42
+ process.env.LOCALAPPDATA,
43
+ ].filter(Boolean);
44
+ }
45
+
46
+ module.exports = { name, packageManager, isInstalled, getAppPaths };
@@ -0,0 +1,45 @@
1
+ 'use strict';
2
+
3
+ const os = require('os');
4
+ const path = require('path');
5
+ const { execFileSync } = require('child_process');
6
+
7
+ /**
8
+ * Platform name identifier.
9
+ * @type {string}
10
+ */
11
+ const name = 'macos';
12
+
13
+ /**
14
+ * Default package manager for macOS.
15
+ * @type {string}
16
+ */
17
+ const packageManager = 'brew';
18
+
19
+ /**
20
+ * Checks if a binary is available on the system PATH.
21
+ * @param {string} binary - The name of the binary to look for (e.g. 'node', 'git').
22
+ * @returns {boolean} True if the binary exists on the PATH, false otherwise.
23
+ */
24
+ function isInstalled(binary) {
25
+ try {
26
+ execFileSync('which', [binary], { stdio: 'ignore' });
27
+ return true;
28
+ } catch {
29
+ return false;
30
+ }
31
+ }
32
+
33
+ /**
34
+ * Returns common application directories for macOS.
35
+ * @returns {string[]} Array of directory paths where applications are typically installed.
36
+ */
37
+ function getAppPaths() {
38
+ return [
39
+ '/Applications',
40
+ '/usr/local/bin',
41
+ path.join(os.homedir(), 'Applications'),
42
+ ];
43
+ }
44
+
45
+ module.exports = { name, packageManager, isInstalled, getAppPaths };