@ecmaos/coreutils 0.4.0 → 0.4.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 (171) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/CHANGELOG.md +6 -0
  3. package/dist/commands/awk.d.ts +4 -0
  4. package/dist/commands/awk.d.ts.map +1 -0
  5. package/dist/commands/awk.js +324 -0
  6. package/dist/commands/awk.js.map +1 -0
  7. package/dist/commands/chgrp.d.ts +4 -0
  8. package/dist/commands/chgrp.d.ts.map +1 -0
  9. package/dist/commands/chgrp.js +187 -0
  10. package/dist/commands/chgrp.js.map +1 -0
  11. package/dist/commands/chmod.d.ts.map +1 -1
  12. package/dist/commands/chmod.js +139 -2
  13. package/dist/commands/chmod.js.map +1 -1
  14. package/dist/commands/chown.d.ts +4 -0
  15. package/dist/commands/chown.d.ts.map +1 -0
  16. package/dist/commands/chown.js +257 -0
  17. package/dist/commands/chown.js.map +1 -0
  18. package/dist/commands/cksum.d.ts +4 -0
  19. package/dist/commands/cksum.d.ts.map +1 -0
  20. package/dist/commands/cksum.js +124 -0
  21. package/dist/commands/cksum.js.map +1 -0
  22. package/dist/commands/cmp.d.ts +4 -0
  23. package/dist/commands/cmp.d.ts.map +1 -0
  24. package/dist/commands/cmp.js +120 -0
  25. package/dist/commands/cmp.js.map +1 -0
  26. package/dist/commands/column.d.ts +4 -0
  27. package/dist/commands/column.d.ts.map +1 -0
  28. package/dist/commands/column.js +274 -0
  29. package/dist/commands/column.js.map +1 -0
  30. package/dist/commands/cp.d.ts.map +1 -1
  31. package/dist/commands/cp.js +81 -4
  32. package/dist/commands/cp.js.map +1 -1
  33. package/dist/commands/curl.d.ts +4 -0
  34. package/dist/commands/curl.d.ts.map +1 -0
  35. package/dist/commands/curl.js +238 -0
  36. package/dist/commands/curl.js.map +1 -0
  37. package/dist/commands/du.d.ts +4 -0
  38. package/dist/commands/du.d.ts.map +1 -0
  39. package/dist/commands/du.js +168 -0
  40. package/dist/commands/du.js.map +1 -0
  41. package/dist/commands/echo.d.ts.map +1 -1
  42. package/dist/commands/echo.js +125 -2
  43. package/dist/commands/echo.js.map +1 -1
  44. package/dist/commands/expand.d.ts +4 -0
  45. package/dist/commands/expand.d.ts.map +1 -0
  46. package/dist/commands/expand.js +197 -0
  47. package/dist/commands/expand.js.map +1 -0
  48. package/dist/commands/factor.d.ts +4 -0
  49. package/dist/commands/factor.d.ts.map +1 -0
  50. package/dist/commands/factor.js +141 -0
  51. package/dist/commands/factor.js.map +1 -0
  52. package/dist/commands/fmt.d.ts +4 -0
  53. package/dist/commands/fmt.d.ts.map +1 -0
  54. package/dist/commands/fmt.js +278 -0
  55. package/dist/commands/fmt.js.map +1 -0
  56. package/dist/commands/fold.d.ts +4 -0
  57. package/dist/commands/fold.d.ts.map +1 -0
  58. package/dist/commands/fold.js +253 -0
  59. package/dist/commands/fold.js.map +1 -0
  60. package/dist/commands/groups.d.ts +4 -0
  61. package/dist/commands/groups.d.ts.map +1 -0
  62. package/dist/commands/groups.js +61 -0
  63. package/dist/commands/groups.js.map +1 -0
  64. package/dist/commands/hostname.d.ts +4 -0
  65. package/dist/commands/hostname.d.ts.map +1 -0
  66. package/dist/commands/hostname.js +80 -0
  67. package/dist/commands/hostname.js.map +1 -0
  68. package/dist/commands/mount.d.ts.map +1 -1
  69. package/dist/commands/mount.js +192 -97
  70. package/dist/commands/mount.js.map +1 -1
  71. package/dist/commands/od.d.ts +4 -0
  72. package/dist/commands/od.d.ts.map +1 -0
  73. package/dist/commands/od.js +342 -0
  74. package/dist/commands/od.js.map +1 -0
  75. package/dist/commands/pr.d.ts +4 -0
  76. package/dist/commands/pr.d.ts.map +1 -0
  77. package/dist/commands/pr.js +298 -0
  78. package/dist/commands/pr.js.map +1 -0
  79. package/dist/commands/printf.d.ts +4 -0
  80. package/dist/commands/printf.d.ts.map +1 -0
  81. package/dist/commands/printf.js +271 -0
  82. package/dist/commands/printf.js.map +1 -0
  83. package/dist/commands/readlink.d.ts +4 -0
  84. package/dist/commands/readlink.d.ts.map +1 -0
  85. package/dist/commands/readlink.js +104 -0
  86. package/dist/commands/readlink.js.map +1 -0
  87. package/dist/commands/realpath.d.ts +4 -0
  88. package/dist/commands/realpath.d.ts.map +1 -0
  89. package/dist/commands/realpath.js +111 -0
  90. package/dist/commands/realpath.js.map +1 -0
  91. package/dist/commands/rev.d.ts +4 -0
  92. package/dist/commands/rev.d.ts.map +1 -0
  93. package/dist/commands/rev.js +134 -0
  94. package/dist/commands/rev.js.map +1 -0
  95. package/dist/commands/shuf.d.ts +4 -0
  96. package/dist/commands/shuf.d.ts.map +1 -0
  97. package/dist/commands/shuf.js +221 -0
  98. package/dist/commands/shuf.js.map +1 -0
  99. package/dist/commands/sleep.d.ts +4 -0
  100. package/dist/commands/sleep.d.ts.map +1 -0
  101. package/dist/commands/sleep.js +102 -0
  102. package/dist/commands/sleep.js.map +1 -0
  103. package/dist/commands/strings.d.ts +4 -0
  104. package/dist/commands/strings.d.ts.map +1 -0
  105. package/dist/commands/strings.js +170 -0
  106. package/dist/commands/strings.js.map +1 -0
  107. package/dist/commands/tac.d.ts +4 -0
  108. package/dist/commands/tac.d.ts.map +1 -0
  109. package/dist/commands/tac.js +130 -0
  110. package/dist/commands/tac.js.map +1 -0
  111. package/dist/commands/time.d.ts +4 -0
  112. package/dist/commands/time.d.ts.map +1 -0
  113. package/dist/commands/time.js +126 -0
  114. package/dist/commands/time.js.map +1 -0
  115. package/dist/commands/umount.d.ts.map +1 -1
  116. package/dist/commands/umount.js +2 -3
  117. package/dist/commands/umount.js.map +1 -1
  118. package/dist/commands/uname.d.ts +4 -0
  119. package/dist/commands/uname.d.ts.map +1 -0
  120. package/dist/commands/uname.js +149 -0
  121. package/dist/commands/uname.js.map +1 -0
  122. package/dist/commands/unexpand.d.ts +4 -0
  123. package/dist/commands/unexpand.d.ts.map +1 -0
  124. package/dist/commands/unexpand.js +286 -0
  125. package/dist/commands/unexpand.js.map +1 -0
  126. package/dist/commands/uptime.d.ts +4 -0
  127. package/dist/commands/uptime.d.ts.map +1 -0
  128. package/dist/commands/uptime.js +62 -0
  129. package/dist/commands/uptime.js.map +1 -0
  130. package/dist/commands/yes.d.ts +4 -0
  131. package/dist/commands/yes.d.ts.map +1 -0
  132. package/dist/commands/yes.js +58 -0
  133. package/dist/commands/yes.js.map +1 -0
  134. package/dist/index.d.ts +21 -0
  135. package/dist/index.d.ts.map +1 -1
  136. package/dist/index.js +73 -0
  137. package/dist/index.js.map +1 -1
  138. package/package.json +3 -2
  139. package/src/commands/awk.ts +340 -0
  140. package/src/commands/chmod.ts +141 -2
  141. package/src/commands/chown.ts +321 -0
  142. package/src/commands/cksum.ts +133 -0
  143. package/src/commands/cmp.ts +126 -0
  144. package/src/commands/column.ts +273 -0
  145. package/src/commands/cp.ts +93 -4
  146. package/src/commands/curl.ts +231 -0
  147. package/src/commands/echo.ts +122 -2
  148. package/src/commands/expand.ts +207 -0
  149. package/src/commands/factor.ts +151 -0
  150. package/src/commands/fmt.ts +293 -0
  151. package/src/commands/fold.ts +257 -0
  152. package/src/commands/groups.ts +72 -0
  153. package/src/commands/hostname.ts +81 -0
  154. package/src/commands/mount.ts +208 -99
  155. package/src/commands/od.ts +327 -0
  156. package/src/commands/pr.ts +291 -0
  157. package/src/commands/printf.ts +271 -0
  158. package/src/commands/readlink.ts +102 -0
  159. package/src/commands/realpath.ts +126 -0
  160. package/src/commands/rev.ts +143 -0
  161. package/src/commands/shuf.ts +218 -0
  162. package/src/commands/sleep.ts +109 -0
  163. package/src/commands/strings.ts +176 -0
  164. package/src/commands/tac.ts +138 -0
  165. package/src/commands/time.ts +144 -0
  166. package/src/commands/umount.ts +2 -3
  167. package/src/commands/uname.ts +130 -0
  168. package/src/commands/unexpand.ts +305 -0
  169. package/src/commands/uptime.ts +73 -0
  170. package/src/index.ts +73 -0
  171. package/tsconfig.json +4 -0
@@ -0,0 +1,197 @@
1
+ import path from 'path';
2
+ import { TerminalEvents } from '@ecmaos/types';
3
+ import { TerminalCommand } from '../shared/terminal-command.js';
4
+ import { writelnStderr } from '../shared/helpers.js';
5
+ function printUsage(process, terminal) {
6
+ const usage = `Usage: expand [OPTION]... [FILE]...
7
+ Convert tabs to spaces in each FILE.
8
+
9
+ -t, --tabs=NUMBER have tabs NUMBER characters apart, not 8
10
+ -t, --tabs=LIST use comma separated list of tab positions
11
+ --help display this help and exit`;
12
+ writelnStderr(process, terminal, usage);
13
+ }
14
+ function parseTabStops(tabStr) {
15
+ if (tabStr.includes(',')) {
16
+ const stops = tabStr.split(',').map(s => parseInt(s.trim(), 10)).filter(n => !isNaN(n) && n > 0);
17
+ return stops.length > 0 ? stops : [8];
18
+ }
19
+ const single = parseInt(tabStr, 10);
20
+ return !isNaN(single) && single > 0 ? [single] : [8];
21
+ }
22
+ function expandTabs(line, tabStops) {
23
+ let result = '';
24
+ let column = 0;
25
+ for (let i = 0; i < line.length; i++) {
26
+ const char = line[i];
27
+ if (char === '\t') {
28
+ let nextStop = tabStops[0] ?? 8;
29
+ for (const stop of tabStops) {
30
+ if (stop > column) {
31
+ nextStop = stop;
32
+ break;
33
+ }
34
+ }
35
+ if (nextStop <= column) {
36
+ const lastStop = tabStops[tabStops.length - 1] ?? 8;
37
+ nextStop = lastStop;
38
+ while (nextStop <= column) {
39
+ nextStop += lastStop;
40
+ }
41
+ }
42
+ const spaces = nextStop - column;
43
+ result += ' '.repeat(spaces);
44
+ column = nextStop;
45
+ }
46
+ else {
47
+ result += char;
48
+ if (char === '\n' || char === '\r') {
49
+ column = 0;
50
+ }
51
+ else {
52
+ column++;
53
+ }
54
+ }
55
+ }
56
+ return result;
57
+ }
58
+ export function createCommand(kernel, shell, terminal) {
59
+ return new TerminalCommand({
60
+ command: 'expand',
61
+ description: 'Convert tabs to spaces',
62
+ kernel,
63
+ shell,
64
+ terminal,
65
+ run: async (pid, argv) => {
66
+ const process = kernel.processes.get(pid);
67
+ if (!process)
68
+ return 1;
69
+ if (argv.length > 0 && (argv[0] === '--help' || argv[0] === '-h')) {
70
+ printUsage(process, terminal);
71
+ return 0;
72
+ }
73
+ let tabStops = [8];
74
+ const files = [];
75
+ for (let i = 0; i < argv.length; i++) {
76
+ const arg = argv[i];
77
+ if (!arg)
78
+ continue;
79
+ if (arg === '--help' || arg === '-h') {
80
+ printUsage(process, terminal);
81
+ return 0;
82
+ }
83
+ else if (arg === '-t' || arg === '--tabs') {
84
+ if (i + 1 < argv.length) {
85
+ const tabStr = argv[++i];
86
+ if (tabStr !== undefined) {
87
+ tabStops = parseTabStops(tabStr);
88
+ }
89
+ }
90
+ }
91
+ else if (arg.startsWith('--tabs=')) {
92
+ const tabStr = arg.slice(7);
93
+ tabStops = parseTabStops(tabStr);
94
+ }
95
+ else if (arg.startsWith('-t')) {
96
+ const tabStr = arg.slice(2);
97
+ if (tabStr) {
98
+ tabStops = parseTabStops(tabStr);
99
+ }
100
+ }
101
+ else if (!arg.startsWith('-')) {
102
+ files.push(arg);
103
+ }
104
+ else {
105
+ await writelnStderr(process, terminal, `expand: invalid option -- '${arg.slice(1)}'`);
106
+ await writelnStderr(process, terminal, "Try 'expand --help' for more information.");
107
+ return 1;
108
+ }
109
+ }
110
+ const writer = process.stdout.getWriter();
111
+ try {
112
+ let lines = [];
113
+ if (files.length === 0) {
114
+ if (!process.stdin) {
115
+ return 0;
116
+ }
117
+ const reader = process.stdin.getReader();
118
+ const decoder = new TextDecoder();
119
+ let buffer = '';
120
+ try {
121
+ while (true) {
122
+ const { done, value } = await reader.read();
123
+ if (done)
124
+ break;
125
+ if (value) {
126
+ buffer += decoder.decode(value, { stream: true });
127
+ const newLines = buffer.split('\n');
128
+ buffer = newLines.pop() || '';
129
+ lines.push(...newLines);
130
+ }
131
+ }
132
+ if (buffer) {
133
+ lines.push(buffer);
134
+ }
135
+ }
136
+ finally {
137
+ reader.releaseLock();
138
+ }
139
+ }
140
+ else {
141
+ for (const file of files) {
142
+ const fullPath = path.resolve(shell.cwd, file);
143
+ let interrupted = false;
144
+ const interruptHandler = () => { interrupted = true; };
145
+ kernel.terminal.events.on(TerminalEvents.INTERRUPT, interruptHandler);
146
+ try {
147
+ if (fullPath.startsWith('/dev')) {
148
+ await writelnStderr(process, terminal, `expand: ${file}: cannot process device files`);
149
+ continue;
150
+ }
151
+ const handle = await shell.context.fs.promises.open(fullPath, 'r');
152
+ const stat = await shell.context.fs.promises.stat(fullPath);
153
+ const decoder = new TextDecoder();
154
+ let content = '';
155
+ let bytesRead = 0;
156
+ const chunkSize = 1024;
157
+ while (bytesRead < stat.size) {
158
+ if (interrupted)
159
+ break;
160
+ const data = new Uint8Array(chunkSize);
161
+ const readSize = Math.min(chunkSize, stat.size - bytesRead);
162
+ await handle.read(data, 0, readSize, bytesRead);
163
+ const chunk = data.subarray(0, readSize);
164
+ content += decoder.decode(chunk, { stream: true });
165
+ bytesRead += readSize;
166
+ }
167
+ const fileLines = content.split('\n');
168
+ if (fileLines[fileLines.length - 1] === '') {
169
+ fileLines.pop();
170
+ }
171
+ lines.push(...fileLines);
172
+ }
173
+ catch (error) {
174
+ await writelnStderr(process, terminal, `expand: ${file}: ${error instanceof Error ? error.message : 'Unknown error'}`);
175
+ }
176
+ finally {
177
+ kernel.terminal.events.off(TerminalEvents.INTERRUPT, interruptHandler);
178
+ }
179
+ }
180
+ }
181
+ for (const line of lines) {
182
+ const expanded = expandTabs(line, tabStops);
183
+ await writer.write(new TextEncoder().encode(expanded + '\n'));
184
+ }
185
+ return 0;
186
+ }
187
+ catch (error) {
188
+ await writelnStderr(process, terminal, `expand: ${error instanceof Error ? error.message : 'Unknown error'}`);
189
+ return 1;
190
+ }
191
+ finally {
192
+ writer.releaseLock();
193
+ }
194
+ }
195
+ });
196
+ }
197
+ //# sourceMappingURL=expand.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"expand.js","sourceRoot":"","sources":["../../src/commands/expand.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { Kernel, Shell, Terminal } from '@ecmaos/types';
2
+ import { TerminalCommand } from '../shared/terminal-command.js';
3
+ export declare function createCommand(kernel: Kernel, shell: Shell, terminal: Terminal): TerminalCommand;
4
+ //# sourceMappingURL=factor.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"factor.d.ts","sourceRoot":"","sources":["../../src/commands/factor.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,MAAM,EAAW,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAA;AACrE,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAsC/D,wBAAgB,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,GAAG,eAAe,CA+G/F"}
@@ -0,0 +1,141 @@
1
+ import { TerminalCommand } from '../shared/terminal-command.js';
2
+ import { writelnStderr, writelnStdout } from '../shared/helpers.js';
3
+ function printUsage(process, terminal) {
4
+ const usage = `Usage: factor [NUMBER]...
5
+ Print prime factors of each NUMBER.
6
+
7
+ --help display this help and exit`;
8
+ writelnStderr(process, terminal, usage);
9
+ }
10
+ function factorize(n) {
11
+ if (n < 2) {
12
+ return [n];
13
+ }
14
+ const factors = [];
15
+ let num = n;
16
+ while (num % 2 === 0) {
17
+ factors.push(2);
18
+ num /= 2;
19
+ }
20
+ for (let i = 3; i * i <= num; i += 2) {
21
+ while (num % i === 0) {
22
+ factors.push(i);
23
+ num /= i;
24
+ }
25
+ }
26
+ if (num > 2) {
27
+ factors.push(num);
28
+ }
29
+ return factors;
30
+ }
31
+ export function createCommand(kernel, shell, terminal) {
32
+ return new TerminalCommand({
33
+ command: 'factor',
34
+ description: 'Print prime factors of numbers',
35
+ kernel,
36
+ shell,
37
+ terminal,
38
+ run: async (pid, argv) => {
39
+ const process = kernel.processes.get(pid);
40
+ if (!process)
41
+ return 1;
42
+ if (argv.length > 0 && (argv[0] === '--help' || argv[0] === '-h')) {
43
+ printUsage(process, terminal);
44
+ return 0;
45
+ }
46
+ const numbers = [];
47
+ for (const arg of argv) {
48
+ if (!arg)
49
+ continue;
50
+ if (arg === '--help' || arg === '-h') {
51
+ printUsage(process, terminal);
52
+ return 0;
53
+ }
54
+ else if (!arg.startsWith('-')) {
55
+ numbers.push(arg);
56
+ }
57
+ else {
58
+ await writelnStderr(process, terminal, `factor: invalid option -- '${arg.slice(1)}'`);
59
+ await writelnStderr(process, terminal, "Try 'factor --help' for more information.");
60
+ return 1;
61
+ }
62
+ }
63
+ if (numbers.length === 0) {
64
+ if (!process.stdin) {
65
+ await writelnStderr(process, terminal, 'factor: missing operand');
66
+ await writelnStderr(process, terminal, "Try 'factor --help' for more information.");
67
+ return 1;
68
+ }
69
+ if (process.stdinIsTTY) {
70
+ try {
71
+ while (true) {
72
+ const line = await terminal.readline('', false, true);
73
+ if (!line)
74
+ break;
75
+ const trimmed = line.trim();
76
+ if (trimmed) {
77
+ numbers.push(...trimmed.split(/\s+/));
78
+ }
79
+ else {
80
+ break;
81
+ }
82
+ }
83
+ }
84
+ catch {
85
+ }
86
+ finally {
87
+ terminal.listen();
88
+ }
89
+ }
90
+ else {
91
+ const reader = process.stdin.getReader();
92
+ const decoder = new TextDecoder();
93
+ let buffer = '';
94
+ try {
95
+ while (true) {
96
+ const { done, value } = await reader.read();
97
+ if (done)
98
+ break;
99
+ if (value) {
100
+ buffer += decoder.decode(value, { stream: true });
101
+ const lines = buffer.split('\n');
102
+ buffer = lines.pop() || '';
103
+ for (const line of lines) {
104
+ const trimmed = line.trim();
105
+ if (trimmed) {
106
+ numbers.push(...trimmed.split(/\s+/));
107
+ }
108
+ }
109
+ }
110
+ }
111
+ if (buffer.trim()) {
112
+ numbers.push(...buffer.trim().split(/\s+/));
113
+ }
114
+ }
115
+ finally {
116
+ reader.releaseLock();
117
+ }
118
+ }
119
+ }
120
+ if (numbers.length === 0) {
121
+ await writelnStderr(process, terminal, 'factor: missing operand');
122
+ await writelnStderr(process, terminal, "Try 'factor --help' for more information.");
123
+ return 1;
124
+ }
125
+ let hasError = false;
126
+ for (const numStr of numbers) {
127
+ const num = parseInt(numStr, 10);
128
+ if (isNaN(num) || num < 0) {
129
+ await writelnStderr(process, terminal, `factor: '${numStr}' is not a valid positive integer`);
130
+ hasError = true;
131
+ continue;
132
+ }
133
+ const factors = factorize(num);
134
+ const output = `${num}: ${factors.join(' ')}`;
135
+ await writelnStdout(process, terminal, output);
136
+ }
137
+ return hasError ? 1 : 0;
138
+ }
139
+ });
140
+ }
141
+ //# sourceMappingURL=factor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"factor.js","sourceRoot":"","sources":["../../src/commands/factor.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import type { Kernel, Shell, Terminal } from '@ecmaos/types';
2
+ import { TerminalCommand } from '../shared/terminal-command.js';
3
+ export declare function createCommand(kernel: Kernel, shell: Shell, terminal: Terminal): TerminalCommand;
4
+ //# sourceMappingURL=fmt.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fmt.d.ts","sourceRoot":"","sources":["../../src/commands/fmt.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,EAAW,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAA;AAErE,OAAO,EAAE,eAAe,EAAE,MAAM,+BAA+B,CAAA;AAsH/D,wBAAgB,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,GAAG,eAAe,CA2K/F"}
@@ -0,0 +1,278 @@
1
+ import path from 'path';
2
+ import { TerminalEvents } from '@ecmaos/types';
3
+ import { TerminalCommand } from '../shared/terminal-command.js';
4
+ import { writelnStderr } from '../shared/helpers.js';
5
+ function printUsage(process, terminal) {
6
+ const usage = `Usage: fmt [OPTION]... [FILE]...
7
+ Reformat paragraph text.
8
+
9
+ -w, --width=WIDTH maximum line width (default: 75)
10
+ -s, --split-only split long lines, but do not join short lines
11
+ -u, --uniform-spacing use uniform spacing (one space between words)
12
+ --help display this help and exit`;
13
+ writelnStderr(process, terminal, usage);
14
+ }
15
+ function normalizeWhitespace(text) {
16
+ return text.replace(/\s+/g, ' ').trim();
17
+ }
18
+ function formatText(lines, width, splitOnly, uniformSpacing) {
19
+ const result = [];
20
+ let currentParagraph = [];
21
+ for (const line of lines) {
22
+ const trimmed = line.trim();
23
+ if (trimmed === '') {
24
+ if (currentParagraph.length > 0) {
25
+ const formatted = formatParagraph(currentParagraph, width, splitOnly, uniformSpacing);
26
+ result.push(...formatted);
27
+ currentParagraph = [];
28
+ }
29
+ result.push('');
30
+ }
31
+ else {
32
+ currentParagraph.push(trimmed);
33
+ }
34
+ }
35
+ if (currentParagraph.length > 0) {
36
+ const formatted = formatParagraph(currentParagraph, width, splitOnly, uniformSpacing);
37
+ result.push(...formatted);
38
+ }
39
+ return result;
40
+ }
41
+ function formatParagraph(paragraph, width, splitOnly, uniformSpacing) {
42
+ if (paragraph.length === 0)
43
+ return [];
44
+ let text = paragraph.join(' ');
45
+ if (uniformSpacing) {
46
+ text = normalizeWhitespace(text);
47
+ }
48
+ else {
49
+ text = text.replace(/\s+/g, ' ');
50
+ }
51
+ if (splitOnly) {
52
+ return splitLongLines(text, width);
53
+ }
54
+ const words = text.split(/\s+/).filter(w => w.length > 0);
55
+ if (words.length === 0)
56
+ return [];
57
+ const result = [];
58
+ let currentLine = '';
59
+ for (const word of words) {
60
+ const testLine = currentLine ? `${currentLine} ${word}` : word;
61
+ if (testLine.length <= width) {
62
+ currentLine = testLine;
63
+ }
64
+ else {
65
+ if (currentLine) {
66
+ result.push(currentLine);
67
+ }
68
+ currentLine = word;
69
+ if (currentLine.length > width) {
70
+ const split = splitLongLines(currentLine, width);
71
+ if (split.length > 0) {
72
+ result.push(...split.slice(0, -1));
73
+ currentLine = split[split.length - 1] || word;
74
+ }
75
+ }
76
+ }
77
+ }
78
+ if (currentLine) {
79
+ result.push(currentLine);
80
+ }
81
+ return result;
82
+ }
83
+ function splitLongLines(text, width) {
84
+ if (text.length <= width)
85
+ return [text];
86
+ const result = [];
87
+ let remaining = text;
88
+ while (remaining.length > width) {
89
+ let breakPoint = width;
90
+ const spaceIndex = remaining.lastIndexOf(' ', width);
91
+ if (spaceIndex > 0) {
92
+ breakPoint = spaceIndex;
93
+ }
94
+ result.push(remaining.slice(0, breakPoint).trim());
95
+ remaining = remaining.slice(breakPoint).trim();
96
+ }
97
+ if (remaining.length > 0) {
98
+ result.push(remaining);
99
+ }
100
+ return result;
101
+ }
102
+ export function createCommand(kernel, shell, terminal) {
103
+ return new TerminalCommand({
104
+ command: 'fmt',
105
+ description: 'Reformat paragraph text',
106
+ kernel,
107
+ shell,
108
+ terminal,
109
+ run: async (pid, argv) => {
110
+ const process = kernel.processes.get(pid);
111
+ if (!process)
112
+ return 1;
113
+ if (argv.length > 0 && (argv[0] === '--help' || argv[0] === '-h')) {
114
+ printUsage(process, terminal);
115
+ return 0;
116
+ }
117
+ let width = 75;
118
+ let splitOnly = false;
119
+ let uniformSpacing = false;
120
+ const files = [];
121
+ for (let i = 0; i < argv.length; i++) {
122
+ const arg = argv[i];
123
+ if (!arg)
124
+ continue;
125
+ if (arg === '--help' || arg === '-h') {
126
+ printUsage(process, terminal);
127
+ return 0;
128
+ }
129
+ else if (arg === '-w' || arg === '--width') {
130
+ if (i + 1 < argv.length) {
131
+ const widthStr = argv[++i];
132
+ if (widthStr !== undefined) {
133
+ const parsed = parseInt(widthStr, 10);
134
+ if (!isNaN(parsed) && parsed > 0) {
135
+ width = parsed;
136
+ }
137
+ else {
138
+ await writelnStderr(process, terminal, `fmt: invalid width: ${widthStr}`);
139
+ return 1;
140
+ }
141
+ }
142
+ }
143
+ }
144
+ else if (arg.startsWith('--width=')) {
145
+ const widthStr = arg.slice(8);
146
+ const parsed = parseInt(widthStr, 10);
147
+ if (!isNaN(parsed) && parsed > 0) {
148
+ width = parsed;
149
+ }
150
+ else {
151
+ await writelnStderr(process, terminal, `fmt: invalid width: ${widthStr}`);
152
+ return 1;
153
+ }
154
+ }
155
+ else if (arg.startsWith('-w')) {
156
+ const widthStr = arg.slice(2);
157
+ if (widthStr) {
158
+ const parsed = parseInt(widthStr, 10);
159
+ if (!isNaN(parsed) && parsed > 0) {
160
+ width = parsed;
161
+ }
162
+ else {
163
+ await writelnStderr(process, terminal, `fmt: invalid width: ${widthStr}`);
164
+ return 1;
165
+ }
166
+ }
167
+ }
168
+ else if (arg === '-s' || arg === '--split-only') {
169
+ splitOnly = true;
170
+ }
171
+ else if (arg === '-u' || arg === '--uniform-spacing') {
172
+ uniformSpacing = true;
173
+ }
174
+ else if (arg.startsWith('-')) {
175
+ const flags = arg.slice(1).split('');
176
+ if (flags.includes('s'))
177
+ splitOnly = true;
178
+ if (flags.includes('u'))
179
+ uniformSpacing = true;
180
+ const invalidFlags = flags.filter(f => !['s', 'u'].includes(f));
181
+ if (invalidFlags.length > 0) {
182
+ await writelnStderr(process, terminal, `fmt: invalid option -- '${invalidFlags[0]}'`);
183
+ await writelnStderr(process, terminal, "Try 'fmt --help' for more information.");
184
+ return 1;
185
+ }
186
+ }
187
+ else {
188
+ files.push(arg);
189
+ }
190
+ }
191
+ const writer = process.stdout.getWriter();
192
+ try {
193
+ let lines = [];
194
+ if (files.length === 0) {
195
+ if (!process.stdin) {
196
+ return 0;
197
+ }
198
+ const reader = process.stdin.getReader();
199
+ const decoder = new TextDecoder();
200
+ let buffer = '';
201
+ try {
202
+ while (true) {
203
+ const { done, value } = await reader.read();
204
+ if (done)
205
+ break;
206
+ if (value) {
207
+ buffer += decoder.decode(value, { stream: true });
208
+ const newLines = buffer.split('\n');
209
+ buffer = newLines.pop() || '';
210
+ lines.push(...newLines);
211
+ }
212
+ }
213
+ if (buffer) {
214
+ lines.push(buffer);
215
+ }
216
+ }
217
+ finally {
218
+ reader.releaseLock();
219
+ }
220
+ }
221
+ else {
222
+ for (const file of files) {
223
+ const fullPath = path.resolve(shell.cwd, file);
224
+ let interrupted = false;
225
+ const interruptHandler = () => { interrupted = true; };
226
+ kernel.terminal.events.on(TerminalEvents.INTERRUPT, interruptHandler);
227
+ try {
228
+ if (fullPath.startsWith('/dev')) {
229
+ await writelnStderr(process, terminal, `fmt: ${file}: cannot process device files`);
230
+ continue;
231
+ }
232
+ const handle = await shell.context.fs.promises.open(fullPath, 'r');
233
+ const stat = await shell.context.fs.promises.stat(fullPath);
234
+ const decoder = new TextDecoder();
235
+ let content = '';
236
+ let bytesRead = 0;
237
+ const chunkSize = 1024;
238
+ while (bytesRead < stat.size) {
239
+ if (interrupted)
240
+ break;
241
+ const data = new Uint8Array(chunkSize);
242
+ const readSize = Math.min(chunkSize, stat.size - bytesRead);
243
+ await handle.read(data, 0, readSize, bytesRead);
244
+ const chunk = data.subarray(0, readSize);
245
+ content += decoder.decode(chunk, { stream: true });
246
+ bytesRead += readSize;
247
+ }
248
+ const fileLines = content.split('\n');
249
+ if (fileLines[fileLines.length - 1] === '') {
250
+ fileLines.pop();
251
+ }
252
+ lines.push(...fileLines);
253
+ }
254
+ catch (error) {
255
+ await writelnStderr(process, terminal, `fmt: ${file}: ${error instanceof Error ? error.message : 'Unknown error'}`);
256
+ }
257
+ finally {
258
+ kernel.terminal.events.off(TerminalEvents.INTERRUPT, interruptHandler);
259
+ }
260
+ }
261
+ }
262
+ const formatted = formatText(lines, width, splitOnly, uniformSpacing);
263
+ for (const line of formatted) {
264
+ await writer.write(new TextEncoder().encode(line + '\n'));
265
+ }
266
+ return 0;
267
+ }
268
+ catch (error) {
269
+ await writelnStderr(process, terminal, `fmt: ${error instanceof Error ? error.message : 'Unknown error'}`);
270
+ return 1;
271
+ }
272
+ finally {
273
+ writer.releaseLock();
274
+ }
275
+ }
276
+ });
277
+ }
278
+ //# sourceMappingURL=fmt.js.map