Skip to content

Commit 2c4b5e8

Browse files
authored
feat(log): provide custom formatting for every entry kind (#1123)
* feat(log): provide custom formatting for every entry kind continues #1122 * chore: ignore unknown log entries
1 parent 85c260c commit 2c4b5e8

File tree

4 files changed

+79
-33
lines changed

4 files changed

+79
-33
lines changed

.size-limit.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
{
1717
"name": "dts libdefs",
1818
"path": "build/*.d.ts",
19-
"limit": "39.15 kB",
19+
"limit": "39.3 kB",
2020
"brotli": false,
2121
"gzip": false
2222
},
@@ -30,7 +30,7 @@
3030
{
3131
"name": "all",
3232
"path": "build/*",
33-
"limit": "850.8 kB",
33+
"limit": "851.1 kB",
3434
"brotli": false,
3535
"gzip": false
3636
}

docs/configuration.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -108,10 +108,13 @@ Set `log.output` to change the stream.
108108
$.log.output = process.stdout
109109
```
110110

111-
Set `log.formatCmd` to customize the command highlighter:
111+
Set `log.formatters` to customize each log entry kind printing:
112112

113113
```ts
114-
$.log.formatCmd = (cmd: string) => chalk.bgRedBright.black(cmd)
114+
$.log.formatters = {
115+
cmd: (entry: LogEntry) => `CMD: ${entry.cmd}`,
116+
fetch: (entry: LogEntry) => `FETCH: ${entry.url}`
117+
}
115118
```
116119

117120
## `$.timeout`

src/log.ts

+54-28
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,12 @@ export type LogEntry = {
2424
id: string
2525
}
2626
| {
27-
kind: 'stdout' | 'stderr'
27+
kind: 'stdout'
28+
data: Buffer
29+
id: string
30+
}
31+
| {
32+
kind: 'stderr'
2833
data: Buffer
2934
id: string
3035
}
@@ -59,39 +64,60 @@ export type LogEntry = {
5964
}
6065
)
6166

62-
type LogFormatter = (cmd?: string) => string
67+
type LogFormatters = {
68+
[key in LogEntry['kind']]: (
69+
entry: Extract<LogEntry, { kind: key }>
70+
) => string | Buffer
71+
}
72+
73+
const formatters: LogFormatters = {
74+
cmd({ cmd }) {
75+
return formatCmd(cmd)
76+
},
77+
stdout({ data }) {
78+
return data
79+
},
80+
stderr({ data }) {
81+
return data
82+
},
83+
custom({ data }) {
84+
return data
85+
},
86+
fetch(entry) {
87+
const init = entry.init ? ' ' + inspect(entry.init) : ''
88+
return '$ ' + chalk.greenBright('fetch') + ` ${entry.url}${init}\n`
89+
},
90+
cd(entry) {
91+
return '$ ' + chalk.greenBright('cd') + ` ${entry.dir}\n`
92+
},
93+
retry(entry) {
94+
return (
95+
chalk.bgRed.white(' FAIL ') +
96+
` Attempt: ${entry.attempt}${entry.total == Infinity ? '' : `/${entry.total}`}` +
97+
(entry.delay > 0 ? `; next in ${entry.delay}ms` : '') +
98+
'\n'
99+
)
100+
},
101+
end() {
102+
return ''
103+
},
104+
}
105+
63106
type Log = {
64107
(entry: LogEntry): void
65-
formatCmd?: LogFormatter
108+
formatters?: Partial<LogFormatters>
66109
output?: NodeJS.WriteStream
67110
}
111+
68112
export const log: Log = function (entry) {
69113
if (!entry.verbose) return
70114
const stream = log.output || process.stderr
71-
switch (entry.kind) {
72-
case 'cmd':
73-
stream.write((log.formatCmd || formatCmd)(entry.cmd))
74-
break
75-
case 'stdout':
76-
case 'stderr':
77-
case 'custom':
78-
stream.write(entry.data)
79-
break
80-
case 'cd':
81-
stream.write('$ ' + chalk.greenBright('cd') + ` ${entry.dir}\n`)
82-
break
83-
case 'fetch':
84-
const init = entry.init ? ' ' + inspect(entry.init) : ''
85-
stream.write('$ ' + chalk.greenBright('fetch') + ` ${entry.url}${init}\n`)
86-
break
87-
case 'retry':
88-
stream.write(
89-
chalk.bgRed.white(' FAIL ') +
90-
` Attempt: ${entry.attempt}${entry.total == Infinity ? '' : `/${entry.total}`}` +
91-
(entry.delay > 0 ? `; next in ${entry.delay}ms` : '') +
92-
'\n'
93-
)
94-
}
115+
const format = (log.formatters?.[entry.kind] || formatters[entry.kind]) as (
116+
entry: LogEntry
117+
) => string | Buffer
118+
if (!format) return // ignore unknown log entries
119+
const data = format(entry)
120+
stream.write(data)
95121
}
96122

97123
const SYNTAX = '()[]{}<>;:+|&='
@@ -115,7 +141,7 @@ const RESERVED_WORDS = new Set([
115141
'EOF',
116142
])
117143

118-
export const formatCmd: LogFormatter = function (cmd) {
144+
export function formatCmd(cmd: string): string {
119145
if (cmd == undefined) return chalk.grey('undefined')
120146
let q = ''
121147
let out = '$ '

test/log.test.ts

+18-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,10 @@ describe('log', () => {
2727

2828
before(() => (log.output = stream))
2929

30-
after(() => delete log.output)
30+
after(() => {
31+
delete log.output
32+
delete log.formatters
33+
})
3134

3235
beforeEach(() => (data.length = 0))
3336

@@ -88,6 +91,20 @@ describe('log', () => {
8891
'\x1B[41m\x1B[37m FAIL \x1B[39m\x1B[49m Attempt: 1/3; next in 1000ms\n'
8992
)
9093
})
94+
95+
test('formatters', () => {
96+
log.formatters = {
97+
cmd: ({ cmd }) => `CMD: ${cmd}`,
98+
}
99+
100+
log({
101+
kind: 'cmd',
102+
cmd: 'echo hi',
103+
id: '1',
104+
verbose: true,
105+
})
106+
assert.equal(data.join(''), 'CMD: echo hi')
107+
})
91108
})
92109

93110
test('formatCwd()', () => {

0 commit comments

Comments
 (0)