1
- const proggy = require ( 'proggy' )
2
- const { log, output, META } = require ( 'proc-log' )
1
+ const { log, output, input, META } = require ( 'proc-log' )
3
2
const { explain } = require ( './explain-eresolve.js' )
4
3
const { formatWithOptions } = require ( './format' )
5
4
@@ -137,18 +136,17 @@ class Display {
137
136
// Handlers are set immediately so they can buffer all events
138
137
process . on ( 'log' , this . #logHandler)
139
138
process . on ( 'output' , this . #outputHandler)
139
+ process . on ( 'input' , this . #inputHandler)
140
+ this . #progress = new Progress ( { stream : stderr } )
140
141
}
141
142
142
143
off ( ) {
143
144
process . off ( 'log' , this . #logHandler)
144
145
this . #logState. buffer . length = 0
145
-
146
146
process . off ( 'output' , this . #outputHandler)
147
147
this . #outputState. buffer . length = 0
148
-
149
- if ( this . #progress) {
150
- this . #progress. stop ( )
151
- }
148
+ process . off ( 'input' , this . #inputHandler)
149
+ this . #progress. off ( )
152
150
}
153
151
154
152
get chalk ( ) {
@@ -170,7 +168,6 @@ class Display {
170
168
timing,
171
169
unicode,
172
170
} ) {
173
- this . #command = command
174
171
// get createSupportsColor from chalk directly if this lands
175
172
// https://github.com/chalk/chalk/pull/600
176
173
const [ { Chalk } , { createSupportsColor } ] = await Promise . all ( [
@@ -181,17 +178,14 @@ class Display {
181
178
// what it knows about the environment to get color support since we already
182
179
// determined in our definitions that we want to show colors.
183
180
const level = Math . max ( createSupportsColor ( null ) . level , 1 )
184
-
185
181
this . #noColorChalk = new Chalk ( { level : 0 } )
186
-
187
182
this . #stdoutColor = stdoutColor
188
183
this . #stdoutChalk = stdoutColor ? new Chalk ( { level } ) : this . #noColorChalk
189
-
190
184
this . #stderrColor = stderrColor
191
185
this . #stderrChalk = stderrColor ? new Chalk ( { level } ) : this . #noColorChalk
192
-
193
186
this . #logColors = COLOR_PALETTE ( { chalk : this . #stderrChalk } )
194
187
188
+ this . #command = command
195
189
this . #levelIndex = LEVEL_OPTIONS [ loglevel ] . index
196
190
this . #timing = timing
197
191
this . #json = json
@@ -201,104 +195,132 @@ class Display {
201
195
// Emit resume event on the logs which will flush output
202
196
log . resume ( )
203
197
output . flush ( )
204
- this . #startProgress( { progress, unicode } )
198
+ this . #progress. load ( {
199
+ unicode,
200
+ enabled : ! ! progress && ! this . #silent,
201
+ } )
205
202
}
206
203
207
204
// STREAM WRITES
208
205
209
206
// Write formatted and (non-)colorized output to streams
210
- #stdoutWrite ( options , ...args ) {
211
- this . #stdout. write ( formatWithOptions ( { colors : this . #stdoutColor, ...options } , ...args ) )
212
- }
213
-
214
- #stderrWrite ( options , ...args ) {
215
- this . #stderr. write ( formatWithOptions ( { colors : this . #stderrColor, ...options } , ...args ) )
207
+ #write ( stream , options , ...args ) {
208
+ const colors = stream === this . #stdout ? this . #stdoutColor : this . #stderrColor
209
+ const value = formatWithOptions ( { colors, ...options } , ...args )
210
+ this . #progress. write ( ( ) => stream . write ( value ) )
216
211
}
217
212
218
213
// HANDLERS
219
214
220
215
// Arrow function assigned to a private class field so it can be passed
221
216
// directly as a listener and still reference "this"
222
217
#logHandler = withMeta ( ( level , meta , ...args ) => {
223
- if ( level === log . KEYS . resume ) {
224
- this . #logState. buffering = false
225
- this . #logState. buffer . forEach ( ( item ) => this . #tryWriteLog( ...item ) )
226
- this . #logState. buffer . length = 0
227
- return
228
- }
229
-
230
- if ( level === log . KEYS . pause ) {
231
- this . #logState. buffering = true
232
- return
233
- }
234
-
235
- if ( this . #logState. buffering ) {
236
- this . #logState. buffer . push ( [ level , meta , ...args ] )
237
- return
218
+ switch ( level ) {
219
+ case log . KEYS . resume :
220
+ this . #logState. buffering = false
221
+ this . #logState. buffer . forEach ( ( item ) => this . #tryWriteLog( ...item ) )
222
+ this . #logState. buffer . length = 0
223
+ break
224
+
225
+ case log . KEYS . pause :
226
+ this . #logState. buffering = true
227
+ break
228
+
229
+ default :
230
+ if ( this . #logState. buffering ) {
231
+ this . #logState. buffer . push ( [ level , meta , ...args ] )
232
+ } else {
233
+ this . #tryWriteLog( level , meta , ...args )
234
+ }
235
+ break
238
236
}
239
-
240
- this . #tryWriteLog( level , meta , ...args )
241
237
} )
242
238
243
239
// Arrow function assigned to a private class field so it can be passed
244
240
// directly as a listener and still reference "this"
245
241
#outputHandler = withMeta ( ( level , meta , ...args ) => {
246
- if ( level === output . KEYS . flush ) {
247
- this . #outputState. buffering = false
248
-
249
- if ( meta . jsonError && this . #json) {
250
- const json = { }
251
- for ( const item of this . #outputState. buffer ) {
252
- // index 2 skips the level and meta
253
- Object . assign ( json , tryJsonParse ( item [ 2 ] ) )
242
+ switch ( level ) {
243
+ case output . KEYS . flush :
244
+ this . #outputState. buffering = false
245
+ if ( meta . jsonError && this . #json) {
246
+ const json = { }
247
+ for ( const item of this . #outputState. buffer ) {
248
+ // index 2 skips the level and meta
249
+ Object . assign ( json , tryJsonParse ( item [ 2 ] ) )
250
+ }
251
+ this . #writeOutput(
252
+ output . KEYS . standard ,
253
+ meta ,
254
+ JSON . stringify ( { ...json , error : meta . jsonError } , null , 2 )
255
+ )
256
+ } else {
257
+ this . #outputState. buffer . forEach ( ( item ) => this . #writeOutput( ...item ) )
254
258
}
255
- this . #writeOutput(
256
- output . KEYS . standard ,
257
- meta ,
258
- JSON . stringify ( { ...json , error : meta . jsonError } , null , 2 )
259
- )
260
- } else {
261
- this . #outputState. buffer . forEach ( ( item ) => this . #writeOutput( ...item ) )
262
- }
263
-
264
- this . #outputState. buffer . length = 0
265
- return
266
- }
267
-
268
- if ( level === output . KEYS . buffer ) {
269
- this . #outputState. buffer . push ( [ output . KEYS . standard , meta , ...args ] )
270
- return
271
- }
272
-
273
- if ( this . #outputState. buffering ) {
274
- this . #outputState. buffer . push ( [ level , meta , ...args ] )
275
- return
259
+ this . #outputState. buffer . length = 0
260
+ break
261
+
262
+ case output . KEYS . buffer :
263
+ this . #outputState. buffer . push ( [ output . KEYS . standard , meta , ...args ] )
264
+ break
265
+
266
+ default :
267
+ if ( this . #outputState. buffering ) {
268
+ this . #outputState. buffer . push ( [ level , meta , ...args ] )
269
+ } else {
270
+ // HACK: if it looks like the banner and we are in a state where we hide the
271
+ // banner then dont write any output. This hack can be replaced with proc-log.META
272
+ const isBanner = args . length === 1 &&
273
+ typeof args [ 0 ] === 'string' &&
274
+ args [ 0 ] . startsWith ( '\n> ' ) &&
275
+ args [ 0 ] . endsWith ( '\n' )
276
+ const hideBanner = this . #silent || [ 'exec' , 'explore' ] . includes ( this . #command)
277
+ if ( ! ( isBanner && hideBanner ) ) {
278
+ this . #writeOutput( level , meta , ...args )
279
+ }
280
+ }
281
+ break
276
282
}
283
+ } )
277
284
278
- // HACK: if it looks like the banner and we are in a state where we hide the
279
- // banner then dont write any output. This hack can be replaced with proc-log.META
280
- const isBanner = args . length === 1 &&
281
- typeof args [ 0 ] === 'string' &&
282
- args [ 0 ] . startsWith ( '\n> ' ) &&
283
- args [ 0 ] . endsWith ( '\n' )
284
- const hideBanner = this . #silent || [ 'exec' , 'explore' ] . includes ( this . #command)
285
- if ( isBanner && hideBanner ) {
286
- return
285
+ #inputHandler = withMeta ( ( level , meta , ...args ) => {
286
+ switch ( level ) {
287
+ case input . KEYS . start :
288
+ log . pause ( )
289
+ this . #outputState. buffering = true
290
+ this . #progress. off ( )
291
+ break
292
+
293
+ case input . KEYS . end :
294
+ log . resume ( )
295
+ output . flush ( )
296
+ this . #progress. resume ( )
297
+ break
298
+
299
+ case input . KEYS . read : {
300
+ // The convention when calling input.read is to pass in a single fn that returns
301
+ // the promise to await. resolve and reject are provided by proc-log
302
+ const [ res , rej , p ] = args
303
+ return input . start ( ( ) => p ( )
304
+ . then ( res )
305
+ . catch ( rej )
306
+ // Any call to procLog.input.read will render a prompt to the user, so we always
307
+ // add a single newline of output to stdout to move the cursor to the next line
308
+ . finally ( ( ) => output . standard ( '' ) ) )
309
+ }
287
310
}
288
-
289
- this . #writeOutput( level , meta , ...args )
290
311
} )
291
312
292
313
// OUTPUT
293
314
294
315
#writeOutput ( level , meta , ...args ) {
295
- if ( level === output . KEYS . standard ) {
296
- this . #stdoutWrite( { } , ...args )
297
- return
298
- }
299
-
300
- if ( level === output . KEYS . error ) {
301
- this . #stderrWrite( { } , ...args )
316
+ switch ( level ) {
317
+ case output . KEYS . standard :
318
+ this . #write( this . #stdout, { } , ...args )
319
+ break
320
+
321
+ case output . KEYS . error :
322
+ this . #write( this . #stderr, { } , ...args )
323
+ break
302
324
}
303
325
}
304
326
@@ -344,22 +366,118 @@ class Display {
344
366
this . #logColors[ level ] ( level ) ,
345
367
title ? this . #logColors. title ( title ) : null ,
346
368
]
347
- this . #stderrWrite( { prefix } , ...args )
348
- } else if ( this . #progress) {
349
- // TODO: make this display a single log line of filtered messages
369
+ this . #write( this . #stderr, { prefix } , ...args )
370
+ }
371
+ }
372
+ }
373
+
374
+ class Progress {
375
+ // Taken from https://github.com/sindresorhus/cli-spinners
376
+ // MIT License
377
+ // Copyright (c) Sindre Sorhus <sindresorhus@gmail.com> (https://sindresorhus.com)
378
+ static dots = { duration : 80 , frames : [ '⠋' , '⠙' , '⠹' , '⠸' , '⠼' , '⠴' , '⠦' , '⠧' , '⠇' , '⠏' ] }
379
+ static lines = { duration : 130 , frames : [ '-' , '\\' , '|' , '/' ] }
380
+
381
+ #stream
382
+ #spinner
383
+ #enabled = false
384
+
385
+ #frameIndex = 0
386
+ #lastUpdate = 0
387
+ #interval
388
+ #timeout
389
+
390
+ // We are rendering is enabled option is set and we are not waiting for the render timeout
391
+ get #rendering ( ) {
392
+ return this . #enabled && ! this . #timeout
393
+ }
394
+
395
+ // We are spinning if enabled option is set and the render interval has been set
396
+ get #spinning ( ) {
397
+ return this . #enabled && this . #interval
398
+ }
399
+
400
+ constructor ( { stream } ) {
401
+ this . #stream = stream
402
+ }
403
+
404
+ load ( { enabled, unicode } ) {
405
+ this . #enabled = enabled
406
+ this . #spinner = unicode ? Progress . dots : Progress . lines
407
+ // Dont render the spinner for short durations
408
+ this . #render( 200 )
409
+ }
410
+
411
+ off ( ) {
412
+ if ( ! this . #enabled) {
413
+ return
414
+ }
415
+ clearTimeout ( this . #timeout)
416
+ this . #timeout = null
417
+ clearInterval ( this . #interval)
418
+ this . #interval = null
419
+ this . #frameIndex = 0
420
+ this . #lastUpdate = 0
421
+ this . #clearSpinner( )
422
+ }
423
+
424
+ resume ( ) {
425
+ this . #render( )
426
+ }
427
+
428
+ // If we are currenting rendering the spinner we clear it
429
+ // before writing our line and then re-render the spinner after.
430
+ // If not then all we need to do is write the line
431
+ write ( write ) {
432
+ if ( this . #spinning) {
433
+ this . #clearSpinner( )
434
+ }
435
+ write ( )
436
+ if ( this . #spinning) {
437
+ this . #render( )
350
438
}
351
439
}
352
440
353
- // PROGRESS
441
+ #render ( ms ) {
442
+ if ( ms ) {
443
+ this . #timeout = setTimeout ( ( ) => {
444
+ this . #timeout = null
445
+ this . #renderSpinner( )
446
+ } , ms )
447
+ // Make sure this timeout does not keep the process open
448
+ this . #timeout. unref ( )
449
+ } else {
450
+ this . #renderSpinner( )
451
+ }
452
+ }
354
453
355
- #startProgress ( { progress , unicode } ) {
356
- if ( ! progress || this . #silent ) {
454
+ #renderSpinner ( ) {
455
+ if ( ! this . #rendering ) {
357
456
return
358
457
}
359
- this . #progress = proggy . createClient ( { normalize : true } )
360
- // TODO: implement proggy trackers in arborist/doctor
361
- // TODO: listen to progress events here and build progress UI
362
- // TODO: see deprecated gauge package for what unicode chars were used
458
+ // We always attempt to render immediately but we only request to move to the next
459
+ // frame if it has been longer than our spinner frame duration since our last update
460
+ this . #renderFrame( Date . now ( ) - this . #lastUpdate >= this . #spinner. duration )
461
+ clearInterval ( this . #interval)
462
+ this . #interval = setInterval ( ( ) => this . #renderFrame( true ) , this . #spinner. duration )
463
+ }
464
+
465
+ #renderFrame ( next ) {
466
+ if ( next ) {
467
+ this . #lastUpdate = Date . now ( )
468
+ this . #frameIndex++
469
+ if ( this . #frameIndex >= this . #spinner. frames . length ) {
470
+ this . #frameIndex = 0
471
+ }
472
+ }
473
+ this . #clearSpinner( )
474
+ this . #stream. write ( this . #spinner. frames [ this . #frameIndex] )
475
+ }
476
+
477
+ #clearSpinner ( ) {
478
+ // Move to the start of the line and clear the rest of the line
479
+ this . #stream. cursorTo ( 0 )
480
+ this . #stream. clearLine ( 1 )
363
481
}
364
482
}
365
483
0 commit comments