You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

123 lines
2.4 KiB

import ansi from 'ansi-escapes';
import io from 'socket.io-client';
import chalk from 'chalk';
import EventEmitter from 'events';
export default class Logger extends EventEmitter {
constructor (host, { debug = false, quiet = false } = {}) {
super();
this.host = host;
this.debug = debug;
this.quiet = quiet;
// readyState
this.building = false;
this.socket = io(`https://io.now.sh?host=${host}&v=2`);
this.socket.once('error', this.onSocketError.bind(this));
this.socket.on('state', this.onState.bind(this));
this.socket.on('logs', this.onLog.bind(this));
this.socket.on('backend', this.onComplete.bind(this));
this.lines = new Lines(10);
}
onState (state) {
if (!state.id) {
console.error('> Deployment not found');
this.emit('error');
return;
}
if (state.error) {
console.error('> Deployment error');
this.emit('error');
return;
}
if (state.backend) {
this.onComplete();
return;
}
if (state.logs) {
state.logs.forEach(this.onLog, this);
}
}
onLog (log) {
if (!this.building) {
if (!this.quiet) {
console.log('> Building');
}
this.building = true;
}
if (this.quiet) return;
const data = log.object ? JSON.stringify(log.object) : log.text;
if ('command' === log.type) {
console.log(`${chalk.gray('>')}${data}`);
this.lines.reset();
} else if ('stderr' === log.type) {
data.split('\n').forEach((v) => {
if (v.length) {
console.error(chalk.red(`> ${v}`));
}
});
this.lines.reset();
} else if ('stdout' === log.type) {
data.split('\n').forEach((v) => {
if (v.length) {
this.lines.write(`${chalk.gray('>')} ${v}`);
}
});
}
}
onComplete () {
this.socket.disconnect();
if (this.building) {
this.building = false;
}
this.emit('close');
}
onSocketError (err) {
if (this.debug) {
console.log('> [debug] Socket error', err.stack);
}
}
}
class Lines {
constructor (maxLines = 100) {
this.max = maxLines;
this.buf = [];
}
write (str) {
const { max, buf } = this;
if (buf.length === max) {
process.stdout.write(ansi.eraseLines(max + 1));
buf.shift();
buf.forEach((line) => console.log(line));
}
buf.push(str);
console.log(str);
}
reset () {
this.buf = [];
}
}