updates/updates.js

234 lines
5.6 KiB
JavaScript
Raw Normal View History

2017-12-09 07:45:06 +00:00
#!/usr/bin/env node
2017-12-03 11:15:02 +00:00
"use strict";
const args = require("minimist")(process.argv.slice(2), {
boolean: [
"update", "u", "json", "j", "color", "no-color", "version", "v",
"help", "h", "prerelease", "p"
],
alias: {
u: "update",
p: "prerelease",
j: "json",
v: "version",
h: "help"
}
});
if (args.help) {
process.stdout.write(`usage: updates [options]
2017-12-03 11:15:02 +00:00
Options:
-u, --update Update package.json
-p, --prerelease Update to prerelease versions
-j, --json Output a JSON object
-c, --color Force-enable color output
-n, --no-color Disable color output
-v, --version Print the version
-h, --help Print this help
2017-12-03 11:15:02 +00:00
Exit Codes:
0 Success
1 Error
255 Dependencies are up to date
2017-12-03 11:15:02 +00:00
Examples:
$ updates
2017-12-03 11:35:25 +00:00
$ updates -u
$ updates -j
`);
process.exit(0);
}
2017-12-03 11:15:02 +00:00
const path = require("path");
2017-12-06 18:11:40 +00:00
if (args.version) {
2018-03-03 19:47:48 +00:00
console.info(require(path.join(__dirname, "package.json")).version);
2017-12-06 18:11:40 +00:00
process.exit(0);
}
2017-12-09 05:49:35 +00:00
if (process.argv.includes("-n")) process.argv.push("--no-color");
if (process.argv.includes("-c")) process.argv.push("--color");
2017-12-06 18:11:40 +00:00
const fs = require("fs");
2018-01-16 20:58:58 +00:00
const rp = require("request-promise-native");
2017-12-03 11:15:02 +00:00
const semver = require("semver");
const columnify = require("columnify");
const chalk = require("chalk");
const esc = require("escape-string-regexp");
const url = "https://registry.npmjs.org/";
const packageFile = path.join(process.cwd(), "package.json");
2017-12-03 11:15:02 +00:00
const dependencyTypes = [
2017-12-03 11:15:02 +00:00
"dependencies",
"devDependencies",
"peerDependencies",
"bundledDependencies",
"optionalDependencies"
];
let pkg, pkgStr;
const deps = {};
try {
pkgStr = fs.readFileSync(packageFile, "utf8");
} catch (err) {
finish(new Error("Unable to open package.json"));
}
try {
pkg = JSON.parse(pkgStr);
} catch (err) {
finish(new Error("Error parsing package.json:" + err.message));
}
dependencyTypes.forEach(function(key) {
2017-12-03 11:15:02 +00:00
if (pkg[key]) {
Object.keys(pkg[key]).forEach(function(name) {
2017-12-09 05:47:15 +00:00
const old = pkg[key][name];
if (isValidSemverRange(old)) {
deps[name] = {old};
2017-12-03 11:15:02 +00:00
}
});
}
});
2018-01-16 20:58:58 +00:00
Promise.all(Object.keys(deps).map(dep => rp(url + dep))).then(function(responses) {
responses.forEach(function(res) {
2018-01-16 20:58:58 +00:00
const registryData = JSON.parse(res);
const dep = registryData.name;
2017-12-09 05:47:15 +00:00
const oldRange = deps[dep].old;
const highestVersion = findHighestVersion(Object.keys(registryData["versions"]));
const newRange = updateRange(oldRange, highestVersion);
if (!highestVersion || oldRange === newRange) {
delete deps[dep];
2017-12-03 12:17:31 +00:00
} else {
2017-12-09 05:47:15 +00:00
deps[dep].new = newRange;
2017-12-03 12:17:31 +00:00
}
});
// log results
if (!Object.keys(deps).length) {
finish("All packages are up to date.", {exitCode: 255});
2017-12-03 15:30:59 +00:00
}
// exit if -u is not given
if (!args.update) {
finish(0);
2017-12-03 11:15:02 +00:00
}
fs.writeFile(packageFile, updatePkg(), "utf8", function(err) {
2017-12-03 11:15:02 +00:00
if (err) {
finish(new Error("Error writing package.json:" + err.message));
2017-12-03 11:15:02 +00:00
} else {
finish("package.json updated!");
2017-12-03 11:15:02 +00:00
}
});
});
function finish(obj, opts) {
opts = opts || {};
const output = {};
if (typeof obj === "string") {
output.message = obj;
} else if (obj instanceof Error) {
output.error = obj.message;
}
if (args.json) {
output.results = deps;
2018-03-03 19:47:48 +00:00
console.info(JSON.stringify(output, null, 2));
2017-12-03 12:17:31 +00:00
} else {
if (Object.keys(deps).length) {
2018-03-03 19:47:48 +00:00
console.info(formatDeps(deps));
}
if (output.message || output.error) {
2018-03-03 19:47:48 +00:00
console.info(output.message || output.error);
2017-12-03 12:17:31 +00:00
}
}
let exitCode;
if (opts.exitCode) {
exitCode = opts.exitCode;
} else {
opts.exitCode = output.error ? 1 : 0;
}
process.exit(exitCode);
}
2017-12-03 13:12:30 +00:00
function highlightDiff(a, b, added) {
const aParts = a.split(/\./);
const bParts = b.split(/\./);
let res = "";
for (let i = 0; i < aParts.length; i++) {
if (aParts[i] !== bParts[i]) {
if (/^[0-9]+$/.test(aParts[i])) {
res += chalk[added ? "green" : "red"](aParts.slice(i).join("."));
} else {
res += aParts[i].split("").map(function(char) {
if (/^[0-9]+$/.test(char)) {
return chalk[added ? "green" : "red"](char + ".");
} else {
return char;
}
2017-12-03 13:25:59 +00:00
}).join("") + chalk[added ? "green" : "red"](aParts.slice(i + 1).join("."));
2017-12-03 13:12:30 +00:00
}
break;
} else res += aParts[i] + ".";
}
return res;
}
function formatDeps() {
return columnify(Object.keys(deps).map(function(dep) {
return {
"name": dep,
"old": highlightDiff(deps[dep].old, deps[dep].new, false),
"new": highlightDiff(deps[dep].new, deps[dep].old, true),
};
2017-12-03 12:17:31 +00:00
}), {
2017-12-03 11:15:02 +00:00
columnSplitter: " ",
});
}
function updatePkg() {
2017-12-03 11:15:02 +00:00
let newPkgStr = pkgStr;
Object.keys(deps).forEach(function(dep) {
const re = new RegExp(`"${esc(dep)}": +"${esc(deps[dep].old)}"`, "g");
newPkgStr = newPkgStr.replace(re, `"${dep}": "${deps[dep].new}"`);
2017-12-03 11:15:02 +00:00
});
return newPkgStr;
}
// naive regex replace
function updateRange(range, version) {
return range.replace(/[0-9]+\.[0-9]+\.[0-9]+(-.+)?/g, version);
2017-12-03 11:15:02 +00:00
}
function isValidSemverRange(range) {
let valid = false;
try {
semver.Range(range);
valid = true;
} catch (err) {}
return valid;
}
// find the newest version, ignoring prerelease version unless they are requested
function findHighestVersion(versions) {
let highest;
while (versions.length) {
const parsed = semver.parse(versions.pop());
if (!args.prerelease && parsed.prerelease.length) continue;
if (semver.gt(parsed.version, highest || "0.0.0")) {
highest = parsed.version;
}
}
return highest;
}