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