Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: merge cli-ux library with oclif/core #345

Merged
merged 28 commits into from
Jan 26, 2022
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
5ece939
feat: merge cli-ux library with oclif/core
peternhale Jan 14, 2022
936a380
chore: fix relative paths
peternhale Jan 14, 2022
a4a43a4
feat: merge cli-ux library with oclif/core
peternhale Jan 14, 2022
efe5458
v1.1.2-test-02
peternhale Jan 18, 2022
754b256
chore: resolve name collision for core/Config and cli-ux/Config
peternhale Jan 18, 2022
dec7004
chore: revert to latest version number
peternhale Jan 18, 2022
31321d4
chore: fix build error
peternhale Jan 18, 2022
263ff55
chore: fix imports in tests
peternhale Jan 18, 2022
905e5d5
chore: wrap cli-ux exported members in namespace
peternhale Jan 18, 2022
8d2f1ec
chore: remove unneeded import
peternhale Jan 18, 2022
e7eb16d
Revert "chore: remove unneeded import"
peternhale Jan 19, 2022
dab5aaf
Revert "chore: wrap cli-ux exported members in namespace"
peternhale Jan 19, 2022
8ec7303
chore: rework cliux namespace export
peternhale Jan 19, 2022
4811a6c
chore: document migration
peternhale Jan 20, 2022
65a40b4
chore: apply suggestions from code review
peternhale Jan 20, 2022
c7df81d
chore: remove unneeded console.log statement
peternhale Jan 20, 2022
917a4ba
chore: update migration guide
peternhale Jan 20, 2022
37dfcf7
chore: fix up examples
peternhale Jan 20, 2022
86ca405
chore: apply review suggestions
peternhale Jan 20, 2022
5edba2e
chore: adjust exports
peternhale Jan 25, 2022
9806fa8
chore: abandon use of namesapce
peternhale Jan 25, 2022
703286a
v1.1.2-test-07
peternhale Jan 25, 2022
baacd2e
v1.1.2-test-08
peternhale Jan 25, 2022
598bf34
v1.1.2-test-09
peternhale Jan 25, 2022
61eee52
chore: remove use of lodash/castArray
peternhale Jan 25, 2022
9f31ac3
chore: exclude windows latest, see npm/cli#4234
RodEsp Jan 26, 2022
9f2a7c6
fix: integration test command in ci config
RodEsp Jan 26, 2022
b964947
chore: fix integration test job name
RodEsp Jan 26, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -8,3 +8,4 @@
/node_modules
/tmp
/test/tmp
.DS_Store
21 changes: 19 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -6,19 +6,31 @@
"bugs": "https://github.com/oclif/core/issues",
"dependencies": {
"@oclif/linewrap": "^1.0.0",
"@oclif/screen": "^3.0.2",
"ansi-escapes": "^4.3.0",
"ansi-styles": "^4.2.0",
"cardinal": "^2.1.1",
"chalk": "^4.1.2",
"clean-stack": "^3.0.1",
"cli-ux": "^6.0.6",
"cli-progress": "^3.10.0",
"debug": "^4.3.3",
"ejs": "^3.1.6",
"fs-extra": "^9.1.0",
"get-package-type": "^0.1.0",
"globby": "^11.0.4",
"hyperlinker": "^1.0.0",
"indent-string": "^4.0.0",
"is-wsl": "^2.2.0",
"js-yaml": "^3.13.1",
"lodash": "^4.17.21",
"natural-orderby": "^2.0.3",
"object-treeify": "^1.1.4",
"password-prompt": "^1.1.2",
"semver": "^7.3.5",
"string-width": "^4.2.3",
"strip-ansi": "^6.0.1",
"supports-color": "^8.1.1",
"supports-hyperlinks": "^2.2.0",
"tslib": "^2.3.1",
"widest-line": "^3.1.0",
"wrap-ansi": "^7.0.0"
Expand All @@ -28,12 +40,16 @@
"@oclif/plugin-help": "^5.1.7",
"@oclif/plugin-plugins": "^2.0.8",
"@oclif/test": "^1.2.8",
"@types/ansi-styles": "^3.2.1",
"@types/chai": "^4.2.22",
"@types/chai-as-promised": "^7.1.4",
"@types/clean-stack": "^2.1.1",
"@types/cli-progress": "^3.9.2",
"@types/ejs": "^3.1.0",
"@types/fs-extra": "^9.0.13",
"@types/indent-string": "^4.0.1",
"@types/js-yaml": "^3.12.1",
"@types/lodash": "^4.14.117",
"@types/mocha": "^8.2.3",
"@types/nock": "^11.1.0",
"@types/node": "^15.14.9",
Expand All @@ -42,6 +58,7 @@
"@types/semver": "^7.3.9",
"@types/shelljs": "^0.8.10",
"@types/strip-ansi": "^5.2.1",
"@types/supports-color": "^8.1.1",
"@types/wrap-ansi": "^3.0.0",
"chai": "^4.3.4",
"chai-as-promised": "^7.1.1",
Expand All @@ -59,7 +76,7 @@
"shx": "^0.3.3",
"sinon": "^11.1.2",
"ts-node": "^9.1.1",
"typescript": "4.5.2"
"typescript": "4.4.4"
},
"resolutions": {
"@oclif/command": "1.8.9",
Expand Down
201 changes: 201 additions & 0 deletions src/cli-ux/action/base.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,201 @@
import castArray from 'lodash/castArray'
import {inspect} from 'util'

export interface ITask {
action: string;
status: string | undefined;
active: boolean;
}

export type ActionType = 'spinner' | 'simple' | 'debug'

export interface Options {
stdout?: boolean;
}

export class ActionBase {
type!: ActionType

std: 'stdout' | 'stderr' = 'stderr'

protected stdmocks?: ['stdout' | 'stderr', string[]][]

private stdmockOrigs = {
stdout: process.stdout.write,
stderr: process.stderr.write,
}

public start(action: string, status?: string, opts: Options = {}) {
this.std = opts.stdout ? 'stdout' : 'stderr'
const task = {action, status, active: Boolean(this.task && this.task.active)}
this.task = task

this._start()
task.active = true
this._stdout(true)
}

public stop(msg = 'done') {
const task = this.task
if (!task) {
return
}

this._stop(msg)
task.active = false
this.task = undefined
this._stdout(false)
}

private get globals(): { action: { task?: ITask }; output: string | undefined } {
(global as any)['cli-ux'] = (global as any)['cli-ux'] || {}
const globals = (global as any)['cli-ux']
globals.action = globals.action || {}
return globals
}

public get task(): ITask | undefined {
return this.globals.action.task
}

public set task(task: ITask | undefined) {
this.globals.action.task = task
}

protected get output(): string | undefined {
return this.globals.output
}

protected set output(output: string | undefined) {
this.globals.output = output
}

get running(): boolean {
return Boolean(this.task)
}

get status(): string | undefined {
return this.task ? this.task.status : undefined
}

set status(status: string | undefined) {
const task = this.task
if (!task) {
return
}

if (task.status === status) {
return
}

this._updateStatus(status, task.status)
task.status = status
}

public async pauseAsync(fn: () => Promise<any>, icon?: string) {
const task = this.task
const active = task && task.active
if (task && active) {
this._pause(icon)
this._stdout(false)
task.active = false
}

const ret = await fn()
if (task && active) {
this._resume()
}

return ret
}

public pause(fn: () => any, icon?: string) {
const task = this.task
const active = task && task.active
if (task && active) {
this._pause(icon)
this._stdout(false)
task.active = false
}

const ret = fn()
if (task && active) {
this._resume()
}

return ret
}

protected _start() {
throw new Error('not implemented')
}

protected _stop(_: string) {
throw new Error('not implemented')
}

protected _resume() {
if (this.task) this.start(this.task.action, this.task.status)
}

protected _pause(_?: string) {
throw new Error('not implemented')
}

protected _updateStatus(_: string | undefined, __?: string) {}

// mock out stdout/stderr so it doesn't screw up the rendering
protected _stdout(toggle: boolean) {
try {
const outputs: ['stdout', 'stderr'] = ['stdout', 'stderr']
if (toggle) {
if (this.stdmocks) return
this.stdmockOrigs = {
stdout: process.stdout.write,
stderr: process.stderr.write,
}

this.stdmocks = []
for (const std of outputs) {
(process[std] as any).write = (...args: any[]) => {
this.stdmocks!.push([std, args] as ['stdout' | 'stderr', string[]])
}
}
} else {
if (!this.stdmocks) return
// this._write('stderr', '\nresetstdmock\n\n\n')
delete this.stdmocks
for (const std of outputs) process[std].write = this.stdmockOrigs[std] as any
}
} catch (error) {
this._write('stderr', inspect(error))
}
}

// flush mocked stdout/stderr
protected _flushStdout() {
try {
let output = ''
let std: 'stdout' | 'stderr' | undefined
while (this.stdmocks && this.stdmocks.length > 0) {
const cur = this.stdmocks.shift() as ['stdout' | 'stderr', string[]]
std = cur[0]
this._write(std, cur[1])
output += (cur[1][0] as any).toString('utf8')
}
// add newline if there isn't one already
// otherwise we'll just overwrite it when we render

if (output && std && output[output.length - 1] !== '\n') {
this._write(std, '\n')
}
} catch (error) {
this._write('stderr', inspect(error))
}
}

// write to the real stdout/stderr
protected _write(std: 'stdout' | 'stderr', s: string | string[]) {
this.stdmockOrigs[std].apply(process[std], castArray(s) as [string])
}
}
32 changes: 32 additions & 0 deletions src/cli-ux/action/pride-spinner.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
// tslint:disable restrict-plus-operands

import * as chalk from 'chalk'
import * as supportsColor from 'supports-color'

import SpinnerAction from './spinner'

function color(s: string, frameIndex: number): string {
const prideColors = [
chalk.keyword('pink'),
chalk.red,
chalk.keyword('orange'),
chalk.yellow,
chalk.green,
chalk.cyan,
chalk.blue,
chalk.magenta,
]

if (!supportsColor) return s
const has256 = supportsColor.stdout ? supportsColor.stdout.has256 : (process.env.TERM || '').includes('256')
const prideColor = prideColors[frameIndex] || prideColors[0]
return has256 ? prideColor(s) : chalk.magenta(s)
}

export default class PrideSpinnerAction extends SpinnerAction {
protected _frame(): string {
const frame = this.frames[this.frameIndex]
this.frameIndex = ++this.frameIndex % this.frames.length
return color(frame, this.frameIndex)
}
}
44 changes: 44 additions & 0 deletions src/cli-ux/action/simple.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import {ActionBase, ActionType} from './base'

export default class SimpleAction extends ActionBase {
public type: ActionType = 'simple'

protected _start() {
const task = this.task
if (!task) return
this._render(task.action, task.status)
}

protected _pause(icon?: string) {
if (icon) this._updateStatus(icon)
else this._flush()
}

protected _resume() {}

protected _updateStatus(status: string, prevStatus?: string, newline = false) {
const task = this.task
if (!task) return
if (task.active && !prevStatus) this._write(this.std, ` ${status}`)
else this._write(this.std, `${task.action}... ${status}`)
if (newline || !prevStatus) this._flush()
}

protected _stop(status: string) {
const task = this.task
if (!task) return
this._updateStatus(status, task.status, true)
}

private _render(action: string, status?: string) {
const task = this.task
if (!task) return
if (task.active) this._flush()
this._write(this.std, status ? `${action}... ${status}` : `${action}...`)
}

private _flush() {
this._write(this.std, '\n')
this._flushStdout()
}
}
Loading