Guillermo Rauch
9 years ago
3 changed files with 140 additions and 19 deletions
@ -0,0 +1,115 @@ |
|||||
|
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); |
||||
|
}); |
||||
|
} |
Loading…
Reference in new issue