@ecmaos/coreutils 0.3.1 → 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.
- package/.turbo/turbo-build.log +1 -1
- package/CHANGELOG.md +48 -0
- package/dist/commands/awk.d.ts +4 -0
- package/dist/commands/awk.d.ts.map +1 -0
- package/dist/commands/awk.js +324 -0
- package/dist/commands/awk.js.map +1 -0
- package/dist/commands/chgrp.d.ts +4 -0
- package/dist/commands/chgrp.d.ts.map +1 -0
- package/dist/commands/chgrp.js +187 -0
- package/dist/commands/chgrp.js.map +1 -0
- package/dist/commands/chmod.d.ts.map +1 -1
- package/dist/commands/chmod.js +139 -2
- package/dist/commands/chmod.js.map +1 -1
- package/dist/commands/chown.d.ts +4 -0
- package/dist/commands/chown.d.ts.map +1 -0
- package/dist/commands/chown.js +257 -0
- package/dist/commands/chown.js.map +1 -0
- package/dist/commands/cksum.d.ts +4 -0
- package/dist/commands/cksum.d.ts.map +1 -0
- package/dist/commands/cksum.js +124 -0
- package/dist/commands/cksum.js.map +1 -0
- package/dist/commands/cmp.d.ts +4 -0
- package/dist/commands/cmp.d.ts.map +1 -0
- package/dist/commands/cmp.js +120 -0
- package/dist/commands/cmp.js.map +1 -0
- package/dist/commands/column.d.ts +4 -0
- package/dist/commands/column.d.ts.map +1 -0
- package/dist/commands/column.js +274 -0
- package/dist/commands/column.js.map +1 -0
- package/dist/commands/cp.d.ts.map +1 -1
- package/dist/commands/cp.js +81 -4
- package/dist/commands/cp.js.map +1 -1
- package/dist/commands/cron.d.ts.map +1 -1
- package/dist/commands/cron.js +116 -23
- package/dist/commands/cron.js.map +1 -1
- package/dist/commands/curl.d.ts +4 -0
- package/dist/commands/curl.d.ts.map +1 -0
- package/dist/commands/curl.js +238 -0
- package/dist/commands/curl.js.map +1 -0
- package/dist/commands/du.d.ts +4 -0
- package/dist/commands/du.d.ts.map +1 -0
- package/dist/commands/du.js +168 -0
- package/dist/commands/du.js.map +1 -0
- package/dist/commands/echo.d.ts.map +1 -1
- package/dist/commands/echo.js +125 -2
- package/dist/commands/echo.js.map +1 -1
- package/dist/commands/env.d.ts +4 -0
- package/dist/commands/env.d.ts.map +1 -0
- package/dist/commands/env.js +129 -0
- package/dist/commands/env.js.map +1 -0
- package/dist/commands/expand.d.ts +4 -0
- package/dist/commands/expand.d.ts.map +1 -0
- package/dist/commands/expand.js +197 -0
- package/dist/commands/expand.js.map +1 -0
- package/dist/commands/factor.d.ts +4 -0
- package/dist/commands/factor.d.ts.map +1 -0
- package/dist/commands/factor.js +141 -0
- package/dist/commands/factor.js.map +1 -0
- package/dist/commands/fmt.d.ts +4 -0
- package/dist/commands/fmt.d.ts.map +1 -0
- package/dist/commands/fmt.js +278 -0
- package/dist/commands/fmt.js.map +1 -0
- package/dist/commands/fold.d.ts +4 -0
- package/dist/commands/fold.d.ts.map +1 -0
- package/dist/commands/fold.js +253 -0
- package/dist/commands/fold.js.map +1 -0
- package/dist/commands/groups.d.ts +4 -0
- package/dist/commands/groups.d.ts.map +1 -0
- package/dist/commands/groups.js +61 -0
- package/dist/commands/groups.js.map +1 -0
- package/dist/commands/head.d.ts.map +1 -1
- package/dist/commands/head.js +184 -77
- package/dist/commands/head.js.map +1 -1
- package/dist/commands/hostname.d.ts +4 -0
- package/dist/commands/hostname.d.ts.map +1 -0
- package/dist/commands/hostname.js +80 -0
- package/dist/commands/hostname.js.map +1 -0
- package/dist/commands/less.d.ts.map +1 -1
- package/dist/commands/less.js +1 -0
- package/dist/commands/less.js.map +1 -1
- package/dist/commands/man.d.ts.map +1 -1
- package/dist/commands/man.js +3 -1
- package/dist/commands/man.js.map +1 -1
- package/dist/commands/mount.d.ts +4 -0
- package/dist/commands/mount.d.ts.map +1 -0
- package/dist/commands/mount.js +1136 -0
- package/dist/commands/mount.js.map +1 -0
- package/dist/commands/od.d.ts +4 -0
- package/dist/commands/od.d.ts.map +1 -0
- package/dist/commands/od.js +342 -0
- package/dist/commands/od.js.map +1 -0
- package/dist/commands/pr.d.ts +4 -0
- package/dist/commands/pr.d.ts.map +1 -0
- package/dist/commands/pr.js +298 -0
- package/dist/commands/pr.js.map +1 -0
- package/dist/commands/printf.d.ts +4 -0
- package/dist/commands/printf.d.ts.map +1 -0
- package/dist/commands/printf.js +271 -0
- package/dist/commands/printf.js.map +1 -0
- package/dist/commands/readlink.d.ts +4 -0
- package/dist/commands/readlink.d.ts.map +1 -0
- package/dist/commands/readlink.js +104 -0
- package/dist/commands/readlink.js.map +1 -0
- package/dist/commands/realpath.d.ts +4 -0
- package/dist/commands/realpath.d.ts.map +1 -0
- package/dist/commands/realpath.js +111 -0
- package/dist/commands/realpath.js.map +1 -0
- package/dist/commands/rev.d.ts +4 -0
- package/dist/commands/rev.d.ts.map +1 -0
- package/dist/commands/rev.js +134 -0
- package/dist/commands/rev.js.map +1 -0
- package/dist/commands/shuf.d.ts +4 -0
- package/dist/commands/shuf.d.ts.map +1 -0
- package/dist/commands/shuf.js +221 -0
- package/dist/commands/shuf.js.map +1 -0
- package/dist/commands/sleep.d.ts +4 -0
- package/dist/commands/sleep.d.ts.map +1 -0
- package/dist/commands/sleep.js +102 -0
- package/dist/commands/sleep.js.map +1 -0
- package/dist/commands/strings.d.ts +4 -0
- package/dist/commands/strings.d.ts.map +1 -0
- package/dist/commands/strings.js +170 -0
- package/dist/commands/strings.js.map +1 -0
- package/dist/commands/tac.d.ts +4 -0
- package/dist/commands/tac.d.ts.map +1 -0
- package/dist/commands/tac.js +130 -0
- package/dist/commands/tac.js.map +1 -0
- package/dist/commands/time.d.ts +4 -0
- package/dist/commands/time.d.ts.map +1 -0
- package/dist/commands/time.js +126 -0
- package/dist/commands/time.js.map +1 -0
- package/dist/commands/umount.d.ts +4 -0
- package/dist/commands/umount.d.ts.map +1 -0
- package/dist/commands/umount.js +103 -0
- package/dist/commands/umount.js.map +1 -0
- package/dist/commands/uname.d.ts +4 -0
- package/dist/commands/uname.d.ts.map +1 -0
- package/dist/commands/uname.js +149 -0
- package/dist/commands/uname.js.map +1 -0
- package/dist/commands/unexpand.d.ts +4 -0
- package/dist/commands/unexpand.d.ts.map +1 -0
- package/dist/commands/unexpand.js +286 -0
- package/dist/commands/unexpand.js.map +1 -0
- package/dist/commands/uptime.d.ts +4 -0
- package/dist/commands/uptime.d.ts.map +1 -0
- package/dist/commands/uptime.js +62 -0
- package/dist/commands/uptime.js.map +1 -0
- package/dist/commands/view.d.ts +1 -0
- package/dist/commands/view.d.ts.map +1 -1
- package/dist/commands/view.js +408 -66
- package/dist/commands/view.js.map +1 -1
- package/dist/commands/yes.d.ts +4 -0
- package/dist/commands/yes.d.ts.map +1 -0
- package/dist/commands/yes.js +58 -0
- package/dist/commands/yes.js.map +1 -0
- package/dist/index.d.ts +24 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +82 -0
- package/dist/index.js.map +1 -1
- package/package.json +12 -3
- package/src/commands/awk.ts +340 -0
- package/src/commands/chmod.ts +141 -2
- package/src/commands/chown.ts +321 -0
- package/src/commands/cksum.ts +133 -0
- package/src/commands/cmp.ts +126 -0
- package/src/commands/column.ts +273 -0
- package/src/commands/cp.ts +93 -4
- package/src/commands/cron.ts +115 -23
- package/src/commands/curl.ts +231 -0
- package/src/commands/echo.ts +122 -2
- package/src/commands/env.ts +143 -0
- package/src/commands/expand.ts +207 -0
- package/src/commands/factor.ts +151 -0
- package/src/commands/fmt.ts +293 -0
- package/src/commands/fold.ts +257 -0
- package/src/commands/groups.ts +72 -0
- package/src/commands/head.ts +176 -77
- package/src/commands/hostname.ts +81 -0
- package/src/commands/less.ts +1 -0
- package/src/commands/man.ts +4 -1
- package/src/commands/mount.ts +1302 -0
- package/src/commands/od.ts +327 -0
- package/src/commands/pr.ts +291 -0
- package/src/commands/printf.ts +271 -0
- package/src/commands/readlink.ts +102 -0
- package/src/commands/realpath.ts +126 -0
- package/src/commands/rev.ts +143 -0
- package/src/commands/shuf.ts +218 -0
- package/src/commands/sleep.ts +109 -0
- package/src/commands/strings.ts +176 -0
- package/src/commands/tac.ts +138 -0
- package/src/commands/time.ts +144 -0
- package/src/commands/umount.ts +116 -0
- package/src/commands/uname.ts +130 -0
- package/src/commands/unexpand.ts +305 -0
- package/src/commands/uptime.ts +73 -0
- package/src/commands/view.ts +463 -73
- package/src/index.ts +82 -0
- 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 @@
|
|
|
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
|