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.

157 lines
3.2 KiB

// Native
const EventEmitter = require('events');
// Packages
const ansi = require('ansi-escapes');
const io = require('socket.io-client');
const chalk = require('chalk');
const { compare, deserialize } = require('./logs');
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 = [];
}
}
module.exports = 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/states?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);
// log buffer
this.buf = [];
}
onState(state) {
// console.log(state)
if (!state.id) {
console.error('> Deployment not found');
this.emit('error');
return;
}
if (state.error) {
this.emit('error', state);
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;
}
log = deserialize(log);
const timer = setTimeout(
() => {
this.buf.sort((a, b) => compare(a.log, b.log));
const idx = this.buf.findIndex(b => b.log.id === log.id) + 1;
for (const b of this.buf.slice(0, idx)) {
clearTimeout(b.timer);
this.printLog(b.log);
}
this.buf = this.buf.slice(idx);
},
300
);
this.buf.push({ log, timer });
}
onComplete() {
this.socket.disconnect();
if (this.building) {
this.building = false;
}
this.buf.sort((a, b) => compare(a.log, b.log));
// flush all buffer
for (const b of this.buf) {
clearTimeout(b.timer);
this.printLog(b.log);
}
this.buf = [];
this.emit('close');
}
onSocketError(err) {
if (this.debug) {
console.log(`> [debug] Socket error ${err}\n${err.stack}`);
}
}
printLog(log) {
const data = log.object ? JSON.stringify(log.object) : log.text;
if (log.type === 'command') {
console.log(`${chalk.gray('>')}${data}`);
this.lines.reset();
} else if (log.type === 'stderr') {
data.split('\n').forEach(v => {
if (v.length > 0) {
console.error(chalk.gray(`> ${v}`));
}
});
this.lines.reset();
} else if (log.type === 'stdout') {
data.split('\n').forEach(v => {
if (v.length > 0) {
this.lines.write(`${chalk.gray('>')} ${v}`);
}
});
}
}
};