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.

116 lines
3.1 KiB

import ms from 'ms';
import * as cfg from './cfg';
import pkg from '../../package'; // relative to `build/` :\
import request from 'https';
import chalk from 'chalk';
const TEN_MINUTES = ms('1s');
/**
* Configures auto updates.
* Sets up a `exit` listener to report them.
*/
export default function checkUpdate (opts = {}) {
let updateData;
const update = check(opts).then((data) => {
updateData = data;
// forces the `exit` event upon Ctrl + C
process.on('SIGINT', () => {
// clean up output after ^C
process.stdout.write('\n');
process.exit(1);
});
}, (err) => console.error(err.stack));
process.on('exit', (code) => {
if (updateData) {
const { current, latest, at } = updateData;
const ago = ms(Date.now() - at);
console.log(`> ${chalk.white.bgRed('UPDATE NEEDED')} ` +
`Current: ${current}` +
`Latest ${chalk.bold(latest)} (released ${ago} ago)`);
}
});
return update;
}
function check ({ debug = false, debounce = TEN_MINUTES, timeout = 1000 }) {
return new Promise((resolve, reject) => {
const { _last_update_check } = cfg.read();
if (_last_update_check && _last_update_check + debounce > Date.now()) {
if (debug) {
const ago = ms(Date.now() - _last_update_check);
console.log(`> [debug] Skipping update. Last check ${ago} ago.`);
}
return;
}
if (debug) console.log(`> [debug] Checking for updates. Timeout in ${ms(timeout)}.`);
let timer;
let req = request.get('https://registry.npmjs.org/now', (res) => {
if (200 !== res.statusCode) {
if (debug) console.log(`> [debug] Update check error. NPM ${res.statusCode}.`);
resolve(false);
return;
}
res.resume();
const bufs = [];
res.on('data', (buf) => bufs.push(buf));
res.on('error', (err) => {
if (debug) console.log(`> [debug] Update check error: ${err.message}.`);
resolve(false);
});
res.on('end', () => {
clearTimeout(timer);
const buf = Buffer.concat(bufs);
let data;
try {
data = JSON.parse(buf.toString('utf8'));
} catch (err) {
if (debug) console.log(`> [debug] Update check JSON parse error: ${err.message}.`);
resolve(false);
return;
}
const { latest } = data['dist-tags'];
const current = pkg.version;
if (latest !== pkg.version) {
if (debug) console.log(`> [debug] Needs update. Current ${current}, latest ${latest}`);
resolve({
latest,
current,
at: new Date(data.time[latest])
});
} else {
if (debug) console.log(`> [debug] Up to date (${pkg.version}).`);
resolve(false);
}
cfg.merge({ _last_update_check: Date.now() });
});
})
.on('error', (err) => {
if (debug) console.log(`> [debug] Update check error: ${err.message}.`);
resolve(false);
});
timer = setTimeout(() => {
if (debug) console.log(`> [debug] Aborting update check after ${ms(timeout)}.`);
req.abort();
resolve(false);
}, timeout);
});
}