202 lines
4.3 KiB
JavaScript
202 lines
4.3 KiB
JavaScript
|
// Native
|
||
|
const {URL} = require('url');
|
||
|
const {join} = require('path');
|
||
|
const fs = require('fs');
|
||
|
const {promisify} = require('util');
|
||
|
const {tmpdir} = require('os');
|
||
|
|
||
|
// Packages
|
||
|
const registryUrl = require('registry-url');
|
||
|
|
||
|
const writeFile = promisify(fs.writeFile);
|
||
|
const mkdir = promisify(fs.mkdir);
|
||
|
const readFile = promisify(fs.readFile);
|
||
|
const compareVersions = (a, b) => a.localeCompare(b, 'en-US', {numeric: true});
|
||
|
const encode = value => encodeURIComponent(value).replace(/^%40/, '@');
|
||
|
|
||
|
const getFile = async (details, distTag) => {
|
||
|
const rootDir = tmpdir();
|
||
|
const subDir = join(rootDir, 'update-check');
|
||
|
|
||
|
if (!fs.existsSync(subDir)) {
|
||
|
mkdir(subDir);
|
||
|
}
|
||
|
|
||
|
let name = `${details.name}-${distTag}.json`;
|
||
|
|
||
|
if (details.scope) {
|
||
|
name = `${details.scope}-${name}`;
|
||
|
}
|
||
|
|
||
|
return join(subDir, name);
|
||
|
};
|
||
|
|
||
|
const evaluateCache = async (file, time, interval) => {
|
||
|
if (fs.existsSync(file)) {
|
||
|
const content = await readFile(file, 'utf8');
|
||
|
const {lastUpdate, latest} = JSON.parse(content);
|
||
|
const nextCheck = lastUpdate + interval;
|
||
|
|
||
|
// As long as the time of the next check is in
|
||
|
// the future, we don't need to run it yet
|
||
|
if (nextCheck > time) {
|
||
|
return {
|
||
|
shouldCheck: false,
|
||
|
latest
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
shouldCheck: true,
|
||
|
latest: null
|
||
|
};
|
||
|
};
|
||
|
|
||
|
const updateCache = async (file, latest, lastUpdate) => {
|
||
|
const content = JSON.stringify({
|
||
|
latest,
|
||
|
lastUpdate
|
||
|
});
|
||
|
|
||
|
await writeFile(file, content, 'utf8');
|
||
|
};
|
||
|
|
||
|
const loadPackage = (url, authInfo) => new Promise((resolve, reject) => {
|
||
|
const options = {
|
||
|
host: url.hostname,
|
||
|
path: url.pathname,
|
||
|
port: url.port,
|
||
|
headers: {
|
||
|
accept: 'application/vnd.npm.install-v1+json; q=1.0, application/json; q=0.8, */*'
|
||
|
},
|
||
|
timeout: 2000
|
||
|
};
|
||
|
|
||
|
if (authInfo) {
|
||
|
options.headers.authorization = `${authInfo.type} ${authInfo.token}`;
|
||
|
}
|
||
|
|
||
|
const {get} = require(url.protocol === 'https:' ? 'https' : 'http');
|
||
|
get(options, response => {
|
||
|
const {statusCode} = response;
|
||
|
|
||
|
if (statusCode !== 200) {
|
||
|
const error = new Error(`Request failed with code ${statusCode}`);
|
||
|
error.code = statusCode;
|
||
|
|
||
|
reject(error);
|
||
|
|
||
|
// Consume response data to free up RAM
|
||
|
response.resume();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
let rawData = '';
|
||
|
response.setEncoding('utf8');
|
||
|
|
||
|
response.on('data', chunk => {
|
||
|
rawData += chunk;
|
||
|
});
|
||
|
|
||
|
response.on('end', () => {
|
||
|
try {
|
||
|
const parsedData = JSON.parse(rawData);
|
||
|
resolve(parsedData);
|
||
|
} catch (e) {
|
||
|
reject(e);
|
||
|
}
|
||
|
});
|
||
|
}).on('error', reject).on('timeout', reject);
|
||
|
});
|
||
|
|
||
|
const getMostRecent = async ({full, scope}, distTag) => {
|
||
|
const regURL = registryUrl(scope);
|
||
|
const url = new URL(full, regURL);
|
||
|
|
||
|
let spec = null;
|
||
|
|
||
|
try {
|
||
|
spec = await loadPackage(url);
|
||
|
} catch (err) {
|
||
|
// We need to cover:
|
||
|
// 401 or 403 for when we don't have access
|
||
|
// 404 when the package is hidden
|
||
|
if (err.code && String(err.code).startsWith(4)) {
|
||
|
// We only want to load this package for when we
|
||
|
// really need to use the token
|
||
|
const registryAuthToken = require('registry-auth-token');
|
||
|
const authInfo = registryAuthToken(regURL, {recursive: true});
|
||
|
|
||
|
spec = await loadPackage(url, authInfo);
|
||
|
} else {
|
||
|
throw err;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const version = spec['dist-tags'][distTag];
|
||
|
|
||
|
if (!version) {
|
||
|
throw new Error(`Distribution tag ${distTag} is not available`);
|
||
|
}
|
||
|
|
||
|
return version;
|
||
|
};
|
||
|
|
||
|
const defaultConfig = {
|
||
|
interval: 3600000,
|
||
|
distTag: 'latest'
|
||
|
};
|
||
|
|
||
|
const getDetails = name => {
|
||
|
const spec = {
|
||
|
full: encode(name)
|
||
|
};
|
||
|
|
||
|
if (name.includes('/')) {
|
||
|
const parts = name.split('/');
|
||
|
|
||
|
spec.scope = parts[0];
|
||
|
spec.name = parts[1];
|
||
|
} else {
|
||
|
spec.scope = null;
|
||
|
spec.name = name;
|
||
|
}
|
||
|
|
||
|
return spec;
|
||
|
};
|
||
|
|
||
|
module.exports = async (pkg, config) => {
|
||
|
if (typeof pkg !== 'object') {
|
||
|
throw new Error('The first parameter should be your package.json file content');
|
||
|
}
|
||
|
|
||
|
const details = getDetails(pkg.name);
|
||
|
const time = Date.now();
|
||
|
const {distTag, interval} = Object.assign({}, defaultConfig, config);
|
||
|
const file = await getFile(details, distTag);
|
||
|
|
||
|
let latest = null;
|
||
|
let shouldCheck = true;
|
||
|
|
||
|
({shouldCheck, latest} = await evaluateCache(file, time, interval));
|
||
|
|
||
|
if (shouldCheck) {
|
||
|
latest = await getMostRecent(details, distTag);
|
||
|
|
||
|
// If we pulled an update, we need to update the cache
|
||
|
await updateCache(file, latest, time);
|
||
|
}
|
||
|
|
||
|
const comparision = compareVersions(pkg.version, latest);
|
||
|
|
||
|
if (comparision === -1) {
|
||
|
return {
|
||
|
latest,
|
||
|
fromCache: !shouldCheck
|
||
|
};
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
};
|