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 = []; } }