git-upstream-template
Version:
Synchronize git templates generated by GitHub as if they were forks
274 lines • 13 kB
JavaScript
;
var __makeTemplateObject = (this && this.__makeTemplateObject) || function (cooked, raw) {
if (Object.defineProperty) { Object.defineProperty(cooked, "raw", { value: raw }); } else { cooked.raw = raw; }
return cooked;
};
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __generator = (this && this.__generator) || function (thisArg, body) {
var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
function verb(n) { return function (v) { return step([n, v]); }; }
function step(op) {
if (f) throw new TypeError("Generator is already executing.");
while (_) try {
if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
if (y = 0, t) op = [op[0] & 2, t.value];
switch (op[0]) {
case 0: case 1: t = op; break;
case 4: _.label++; return { value: op[1], done: false };
case 5: _.label++; y = op[1]; op = [0]; continue;
case 7: op = _.ops.pop(); _.trys.pop(); continue;
default:
if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
if (t[2]) _.ops.pop();
_.trys.pop(); continue;
}
op = body.call(thisArg, _);
} catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
}
};
var __read = (this && this.__read) || function (o, n) {
var m = typeof Symbol === "function" && o[Symbol.iterator];
if (!m) return o;
var i = m.call(o), r, ar = [], e;
try {
while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);
}
catch (error) { e = { error: error }; }
finally {
try {
if (r && !r.done && (m = i["return"])) m.call(i);
}
finally { if (e) throw e.error; }
}
return ar;
};
Object.defineProperty(exports, "__esModule", { value: true });
var chalk = require("chalk");
var child_process_1 = require("child_process");
var inquirer = require("inquirer");
var DEBUG = false;
function git(cmd, _a) {
var _b = _a === void 0 ? { noPager: false, verbose: false } : _a, noPager = _b.noPager, verbose = _b.verbose;
var gitCmd = "git" + ((noPager && " --no-pager") || "");
return run(gitCmd + " " + cmd, { verbose: verbose });
}
exports.git = git;
function run(inputCommand, _a) {
var verbose = (_a === void 0 ? { verbose: false } : _a).verbose;
var output = "";
var _b = __read((inputCommand.match(/[A-z0-9\-\_\:\/\\\.\@\!\#\$\%\^\&\*\(\)\{\}\[\]\;\<\>\=\+\~]+|"(?:[^\"]|(?<=\\)")+"|'(?:[^\']|(?<=\\)')+'/g) || []).map(function (arg) { return arg.replace(/^"|"$|\\(?="|')/g, ""); })), cmd = _b[0], args = _b.slice(1);
if (DEBUG)
console.log("\u00BB " + inputCommand, cmd, args);
var child = child_process_1.spawn(cmd, args, { stdio: ["inherit", "pipe", "pipe"] });
if (verbose) {
child.stdout.pipe(process.stdout);
child.stderr.pipe(process.stderr);
}
child.stdout.on("data", function (chunk) {
output += chunk;
});
child.stderr.on("data", function (chunk) {
output += chunk;
});
return new Promise(function (res, rej) {
if (DEBUG)
console.log("\u00AB " + output);
child.on("exit", function (code) {
if (code) {
rej(output);
}
else {
res(output);
}
});
});
}
function addRemote(remoteName, remoteUrl) {
return __awaiter(this, void 0, void 0, function () {
return __generator(this, function (_a) {
return [2 /*return*/, successful(function () { return git("remote add -f " + remoteName + " " + remoteUrl); })];
});
});
}
exports.addRemote = addRemote;
function removeRemote(remoteName) {
return __awaiter(this, void 0, void 0, function () {
return __generator(this, function (_a) {
return [2 /*return*/, successful(function () { return git("remote remove " + remoteName); })];
});
});
}
exports.removeRemote = removeRemote;
function getCurrentBranchName() {
return __awaiter(this, void 0, void 0, function () {
var branchOutput;
return __generator(this, function (_a) {
switch (_a.label) {
case 0: return [4 /*yield*/, git("branch")];
case 1:
branchOutput = _a.sent();
return [2 /*return*/, branchOutput.match(/\*\s(\S+)/)[1]];
}
});
});
}
exports.getCurrentBranchName = getCurrentBranchName;
function getUpdates(updateBranch) {
return __awaiter(this, void 0, void 0, function () {
var currentBranch, currentHashes, currentMessages, currentDates, templateHashes, templateMessages, templateDates, forkDate, afterFork, notApplied, updates;
return __generator(this, function (_a) {
switch (_a.label) {
case 0: return [4 /*yield*/, getCurrentBranchName()];
case 1:
currentBranch = _a.sent();
return [4 /*yield*/, git("log " + currentBranch + " --format=%h")];
case 2:
currentHashes = (_a.sent()).trim().split("\n");
return [4 /*yield*/, git("log " + currentBranch + " --format=%s")];
case 3:
currentMessages = (_a.sent()).trim().split("\n");
return [4 /*yield*/, git("log " + currentBranch + " --format=%at")];
case 4:
currentDates = (_a.sent()).trim().split("\n");
return [4 /*yield*/, git("log " + updateBranch + " --format=%h")];
case 5:
templateHashes = (_a.sent()).trim().split("\n");
return [4 /*yield*/, git("log " + updateBranch + " --format=%s")];
case 6:
templateMessages = (_a.sent()).trim().split("\n");
return [4 /*yield*/, git("log " + updateBranch + " --format=%at")];
case 7:
templateDates = (_a.sent()).trim().split("\n");
forkDate = +currentDates[currentDates.length - 1];
afterFork = function (commit) { return commit.timestamp >= forkDate; };
notApplied = function (commit) {
return currentMessages.findIndex(function (msg) { return msg.includes("🔄") && msg.includes(commit.hash); }) === -1 &&
currentHashes.findIndex(function (hash) { return hash === commit.hash; }) === -1;
};
updates = templateHashes
.map(function (hash, idx) { return ({ hash: hash, message: templateMessages[idx], timestamp: +templateDates[idx] }); })
.filter(notApplied)
.filter(afterFork);
return [2 /*return*/, updates];
}
});
});
}
exports.getUpdates = getUpdates;
function successful(fn) {
return __awaiter(this, void 0, void 0, function () {
var _a;
return __generator(this, function (_b) {
switch (_b.label) {
case 0:
_b.trys.push([0, 2, , 3]);
return [4 /*yield*/, fn()];
case 1:
_b.sent();
return [2 /*return*/, true];
case 2:
_a = _b.sent();
return [2 /*return*/, false];
case 3: return [2 /*return*/];
}
});
});
}
exports.successful = successful;
function applyUpdate(commit) {
return __awaiter(this, void 0, void 0, function () {
function successfullyCommits() {
return __awaiter(this, void 0, void 0, function () {
var stderr_1;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
_a.trys.push([0, 2, , 3]);
return [4 /*yield*/, git("commit -m \"" + commitMessage + "\"", { verbose: true })];
case 1:
_a.sent();
return [2 /*return*/, true];
case 2:
stderr_1 = _a.sent();
return [2 /*return*/, stderr_1.includes("working tree clean")];
case 3: return [2 /*return*/];
}
});
});
}
var commitMessage, stashed, update;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
commitMessage = generateUpdateCommitMessage(commit);
console.log(chalk.default.yellow(templateObject_1 || (templateObject_1 = __makeTemplateObject(["Stashing your current working directory before applying updates..."], ["Stashing your current working directory before applying updates..."]))));
return [4 /*yield*/, git("stash save Before applying upstream-template update " + commit.hash, {
verbose: true
})];
case 1:
stashed = !(_a.sent()).includes("No local changes");
update = extractUpdateCommit(commit);
if (!update) return [3 /*break*/, 4];
return [4 /*yield*/, successful(function () { return run("yarn upgrade " + update.package + "@" + update.version, { verbose: true }); })];
case 2:
_a.sent();
return [4 /*yield*/, successful(function () { return git("add -u", { verbose: true }); })];
case 3:
_a.sent();
return [3 /*break*/, 6];
case 4: return [4 /*yield*/, successful(function () { return git("cherry-pick -X ignore-all-space " + commit.hash + " --no-commit"); })];
case 5:
_a.sent();
_a.label = 6;
case 6: return [4 /*yield*/, successfullyCommits()];
case 7:
if (!!(_a.sent())) return [3 /*break*/, 9];
return [4 /*yield*/, inquirer.prompt({
message: chalk.default.yellow(templateObject_2 || (templateObject_2 = __makeTemplateObject(["Resolve/stage conflicts and press any key to continue..."], ["Resolve/stage conflicts and press any key to continue..."]))),
name: "value"
})];
case 8:
_a.sent();
return [3 /*break*/, 6];
case 9:
if (!stashed) return [3 /*break*/, 11];
return [4 /*yield*/, git("stash pop", { verbose: true })];
case 10:
_a.sent();
_a.label = 11;
case 11: return [2 /*return*/];
}
});
});
}
exports.applyUpdate = applyUpdate;
function generateUpdateCommitMessage(commit) {
return "\uD83D\uDD04 " + commit.hash + ": " + commit.message.replace(/(["'$`\\])/g, '\\$1');
}
function extractUpdateCommit(commit) {
var update = /Bump (\S+) from \S+ to (\S+)/g.exec(commit.message);
if (update) {
return {
package: update[1],
version: update[2]
};
}
else {
return false;
}
}
var templateObject_1, templateObject_2;
//# sourceMappingURL=git.js.map