2017-01-14 12:50:39 +00:00
|
|
|
const logger = require('winston');
|
2017-01-07 07:22:48 +00:00
|
|
|
const ghGot = require('gh-got');
|
|
|
|
|
2017-01-11 12:19:59 +00:00
|
|
|
const config = {};
|
2017-01-07 07:22:48 +00:00
|
|
|
|
2017-01-11 13:33:32 +00:00
|
|
|
module.exports = {
|
|
|
|
initRepo,
|
2017-01-17 08:11:42 +00:00
|
|
|
// Search
|
|
|
|
findFilePaths,
|
2017-01-10 12:32:32 +00:00
|
|
|
// Branch
|
2017-01-29 20:25:12 +00:00
|
|
|
branchExists,
|
2017-01-16 17:10:39 +00:00
|
|
|
getBranchPr,
|
2017-01-13 18:18:44 +00:00
|
|
|
// issue
|
2017-01-18 20:17:07 +00:00
|
|
|
addAssignees,
|
2017-01-31 13:54:16 +00:00
|
|
|
addReviewers,
|
2017-01-13 18:18:44 +00:00
|
|
|
addLabels,
|
2017-01-10 12:32:32 +00:00
|
|
|
// PR
|
2017-01-30 06:34:35 +00:00
|
|
|
findPr,
|
2017-01-11 13:33:32 +00:00
|
|
|
checkForClosedPr,
|
|
|
|
createPr,
|
|
|
|
getPr,
|
|
|
|
updatePr,
|
2017-01-15 22:56:09 +00:00
|
|
|
// file
|
2017-02-01 16:43:28 +00:00
|
|
|
commitFileToBranch,
|
2017-02-08 07:43:16 +00:00
|
|
|
commitFilesToBranch,
|
2017-01-15 22:56:09 +00:00
|
|
|
getFile,
|
2017-01-29 20:25:12 +00:00
|
|
|
getFileContent,
|
2017-01-18 18:55:03 +00:00
|
|
|
getFileJson,
|
2017-01-10 12:32:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Initialize GitHub by getting base branch and SHA
|
2017-02-05 08:10:29 +00:00
|
|
|
async function initRepo(repoName, token, endpoint) {
|
2017-01-17 08:11:42 +00:00
|
|
|
logger.debug(`initRepo(${repoName})`);
|
2017-02-01 16:43:28 +00:00
|
|
|
if (token) {
|
|
|
|
process.env.GITHUB_TOKEN = token;
|
|
|
|
} else if (!process.env.GITHUB_TOKEN) {
|
|
|
|
throw new Error(`No token found for GitHub repository ${repoName}`);
|
|
|
|
}
|
2017-02-05 08:10:29 +00:00
|
|
|
if (endpoint) {
|
|
|
|
process.env.GITHUB_ENDPOINT = endpoint;
|
|
|
|
}
|
2017-01-10 12:32:32 +00:00
|
|
|
config.repoName = repoName;
|
2017-01-31 11:19:06 +00:00
|
|
|
try {
|
|
|
|
const res = await ghGot(`repos/${repoName}`);
|
2017-01-29 20:25:12 +00:00
|
|
|
config.owner = res.body.owner.login;
|
|
|
|
logger.debug(`${repoName} owner = ${config.owner}`);
|
|
|
|
config.defaultBranch = res.body.default_branch;
|
|
|
|
logger.debug(`${repoName} default branch = ${config.defaultBranch}`);
|
2017-01-31 11:19:06 +00:00
|
|
|
config.baseCommitSHA = await getBranchCommit(config.defaultBranch);
|
|
|
|
config.baseTreeSHA = await getCommitTree(config.baseCommitSHA);
|
|
|
|
} catch (err) {
|
2017-01-15 22:56:09 +00:00
|
|
|
logger.error(`GitHub init error: ${JSON.stringify(err)}`);
|
2017-01-11 12:19:59 +00:00
|
|
|
throw err;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-01-17 08:11:42 +00:00
|
|
|
// Search
|
|
|
|
|
|
|
|
// Returns an array of file paths in current repo matching the fileName
|
2017-01-31 11:19:06 +00:00
|
|
|
async function findFilePaths(fileName) {
|
|
|
|
const res = await ghGot(`search/code?q=repo:${config.repoName}+filename:${fileName}`);
|
|
|
|
const exactMatches = res.body.items.filter(item => item.name === fileName);
|
2017-02-07 20:45:35 +00:00
|
|
|
|
|
|
|
// GitHub seems to return files in the root with a leading `/`
|
|
|
|
// which then breaks things later on down the line
|
|
|
|
return exactMatches.map(item => item.path.replace(/^\//, ''));
|
2017-01-17 08:11:42 +00:00
|
|
|
}
|
|
|
|
|
2017-01-10 12:32:32 +00:00
|
|
|
// Branch
|
2017-01-29 20:25:12 +00:00
|
|
|
|
|
|
|
// Returns true if branch exists, otherwise false
|
2017-01-31 11:19:06 +00:00
|
|
|
async function branchExists(branchName) {
|
2017-01-29 20:25:12 +00:00
|
|
|
logger.debug(`Checking if branch exists: ${branchName}`);
|
2017-01-31 11:19:06 +00:00
|
|
|
try {
|
|
|
|
const res = await ghGot(`repos/${config.repoName}/git/refs/heads/${branchName}`);
|
2017-01-29 20:25:12 +00:00
|
|
|
if (res.statusCode === 200) {
|
|
|
|
logger.debug('Branch exists');
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// This probably shouldn't happen
|
|
|
|
logger.debug('Branch doesn\'t exist');
|
|
|
|
return false;
|
2017-01-31 11:19:06 +00:00
|
|
|
} catch (error) {
|
2017-01-29 20:25:12 +00:00
|
|
|
if (error.statusCode === 404) {
|
|
|
|
// If file not found, then return false
|
|
|
|
logger.debug('Branch doesn\'t exist');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// Propagate if it's any other error
|
|
|
|
throw error;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
2017-02-01 12:50:28 +00:00
|
|
|
// Returns the Pull Request for a branch. Null if not exists.
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getBranchPr(branchName) {
|
2017-02-01 12:50:28 +00:00
|
|
|
logger.debug(`getBranchPr(${branchName})`);
|
2017-01-16 17:10:39 +00:00
|
|
|
const gotString = `repos/${config.repoName}/pulls?` +
|
|
|
|
`state=open&base=${config.defaultBranch}&head=${config.owner}:${branchName}`;
|
2017-01-31 11:19:06 +00:00
|
|
|
const res = await ghGot(gotString);
|
2017-02-01 16:43:28 +00:00
|
|
|
if (!res.body.length) {
|
|
|
|
return null;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-02-01 16:43:28 +00:00
|
|
|
const prNo = res.body[0].number;
|
|
|
|
return getPr(prNo);
|
2017-01-16 17:10:39 +00:00
|
|
|
}
|
|
|
|
|
2017-01-13 18:18:44 +00:00
|
|
|
// Issue
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function addAssignees(issueNo, assignees) {
|
2017-01-18 20:17:07 +00:00
|
|
|
logger.debug(`Adding assignees ${assignees} to #${issueNo}`);
|
2017-01-31 11:19:06 +00:00
|
|
|
await ghGot.post(`repos/${config.repoName}/issues/${issueNo}/assignees`, {
|
2017-01-18 20:17:07 +00:00
|
|
|
body: {
|
|
|
|
assignees,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-01-31 13:54:16 +00:00
|
|
|
async function addReviewers(issueNo, reviewers) {
|
|
|
|
logger.debug(`Adding reviewers ${reviewers} to #${issueNo}`);
|
|
|
|
await ghGot.post(`repos/${config.repoName}/pulls/${issueNo}/requested_reviewers`, {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.black-cat-preview+json',
|
|
|
|
},
|
|
|
|
body: {
|
|
|
|
reviewers,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function addLabels(issueNo, labels) {
|
2017-01-13 18:18:44 +00:00
|
|
|
logger.debug(`Adding labels ${labels} to #${issueNo}`);
|
2017-01-31 11:19:06 +00:00
|
|
|
await ghGot.post(`repos/${config.repoName}/issues/${issueNo}/labels`, {
|
2017-01-13 18:18:44 +00:00
|
|
|
body: JSON.stringify(labels),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function findPr(branchName, prTitle, state = 'all') {
|
2017-01-30 06:34:35 +00:00
|
|
|
logger.debug(`findPr(${branchName}, ${state})`);
|
2017-01-31 11:19:06 +00:00
|
|
|
const urlString = `repos/${config.repoName}/pulls?head=${config.owner}:${branchName}&state=${state}`;
|
2017-01-30 06:34:35 +00:00
|
|
|
logger.debug(`findPr urlString: ${urlString}`);
|
2017-01-31 11:19:06 +00:00
|
|
|
const res = await ghGot(urlString);
|
|
|
|
let pr = null;
|
|
|
|
res.body.forEach((result) => {
|
2017-02-01 12:50:28 +00:00
|
|
|
if (!prTitle || result.title === prTitle) {
|
2017-01-31 11:19:06 +00:00
|
|
|
pr = result;
|
2017-02-01 12:50:28 +00:00
|
|
|
if (pr.state === 'closed') {
|
|
|
|
pr.isClosed = true;
|
|
|
|
}
|
2017-02-02 17:34:48 +00:00
|
|
|
pr.displayNumber = `Pull Request #${pr.number}`;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return pr;
|
2017-01-30 06:34:35 +00:00
|
|
|
}
|
|
|
|
|
2017-01-10 12:32:32 +00:00
|
|
|
// Pull Request
|
2017-01-31 11:19:06 +00:00
|
|
|
async function checkForClosedPr(branchName, prTitle) {
|
2017-01-30 06:34:35 +00:00
|
|
|
logger.debug(`checkForClosedPr(${branchName}, ${prTitle})`);
|
2017-01-31 11:19:06 +00:00
|
|
|
const url = `repos/${config.repoName}/pulls?state=closed&head=${config.owner}:${branchName}`;
|
|
|
|
const res = await ghGot(url);
|
|
|
|
// Return true if any of the titles match exactly
|
|
|
|
return res.body.some(pr => pr.title === prTitle && pr.head.label === `${config.owner}:${branchName}`);
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-02-01 12:50:28 +00:00
|
|
|
// Creates PR and returns PR number
|
2017-01-31 11:19:06 +00:00
|
|
|
async function createPr(branchName, title, body) {
|
2017-02-08 07:33:54 +00:00
|
|
|
const pr = (await ghGot.post(`repos/${config.repoName}/pulls`, {
|
2017-01-31 11:19:06 +00:00
|
|
|
body: { title, head: branchName, base: config.defaultBranch, body },
|
2017-02-08 07:34:19 +00:00
|
|
|
})).body;
|
2017-02-02 17:34:48 +00:00
|
|
|
pr.displayNumber = `Pull Request #${pr.number}`;
|
2017-02-07 20:45:35 +00:00
|
|
|
return pr;
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-02-01 12:50:28 +00:00
|
|
|
// Gets details for a PR
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getPr(prNo) {
|
2017-02-01 12:50:28 +00:00
|
|
|
if (!prNo) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const pr = (await ghGot(`repos/${config.repoName}/pulls/${prNo}`)).body;
|
|
|
|
if (!pr) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// Harmonise PR values
|
2017-02-02 17:34:48 +00:00
|
|
|
pr.displayNumber = `Pull Request #${pr.number}`;
|
2017-02-01 12:50:28 +00:00
|
|
|
if (pr.state === 'closed') {
|
|
|
|
pr.isClosed = true;
|
|
|
|
}
|
|
|
|
if (pr.mergeable_state === 'dirty') {
|
|
|
|
pr.isUnmergeable = true;
|
|
|
|
}
|
2017-02-09 04:59:50 +00:00
|
|
|
if (pr.additions * pr.deletions === 1 || pr.commits === 1) {
|
2017-02-01 12:50:28 +00:00
|
|
|
pr.canRebase = true;
|
|
|
|
}
|
2017-02-06 06:56:33 +00:00
|
|
|
if (pr.base.sha !== config.baseCommitSHA) {
|
|
|
|
pr.isStale = true;
|
|
|
|
}
|
2017-02-01 12:50:28 +00:00
|
|
|
return pr;
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function updatePr(prNo, title, body) {
|
|
|
|
await ghGot.patch(`repos/${config.repoName}/pulls/${prNo}`, {
|
2017-01-10 22:06:25 +00:00
|
|
|
body: { title, body },
|
2017-01-10 12:32:32 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generic File operations
|
2017-01-29 20:25:12 +00:00
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getFile(filePath, branchName = config.defaultBranch) {
|
|
|
|
const res = await ghGot(`repos/${config.repoName}/contents/${filePath}?ref=${branchName}`);
|
|
|
|
return res.body.content;
|
2017-01-07 21:08:45 +00:00
|
|
|
}
|
2017-01-10 12:32:32 +00:00
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getFileContent(filePath, branchName = config.baseBranch) {
|
|
|
|
try {
|
|
|
|
const file = await getFile(filePath, branchName);
|
|
|
|
return new Buffer(file, 'base64').toString();
|
|
|
|
} catch (error) {
|
2017-01-29 20:25:12 +00:00
|
|
|
if (error.statusCode === 404) {
|
|
|
|
// If file not found, then return null JSON
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// Propagate if it's any other error
|
|
|
|
throw error;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getFileJson(filePath, branchName = config.baseBranch) {
|
|
|
|
try {
|
|
|
|
const file = await getFile(filePath, branchName);
|
|
|
|
return JSON.parse(new Buffer(file, 'base64').toString());
|
|
|
|
} catch (error) {
|
2017-01-18 18:55:03 +00:00
|
|
|
if (error.statusCode === 404) {
|
|
|
|
// If file not found, then return null JSON
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
// Propagate if it's any other error
|
|
|
|
throw error;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-02-01 16:43:28 +00:00
|
|
|
// Add a new commit, create branch if not existing
|
|
|
|
async function commitFileToBranch(
|
|
|
|
branchName,
|
|
|
|
fileName,
|
|
|
|
fileContents,
|
|
|
|
message,
|
|
|
|
parentBranch = config.defaultBranch) {
|
|
|
|
logger.debug(`commitFileToBrach('${branchName}', '${fileName}', fileContents, message, '${parentBranch})'`);
|
2017-02-08 07:43:16 +00:00
|
|
|
return commitFilesToBranch(
|
|
|
|
branchName,
|
|
|
|
[{
|
|
|
|
name: fileName,
|
|
|
|
contents: fileContents,
|
|
|
|
}],
|
|
|
|
message,
|
|
|
|
parentBranch);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a new commit, create branch if not existing
|
|
|
|
async function commitFilesToBranch(
|
|
|
|
branchName,
|
|
|
|
files,
|
|
|
|
message,
|
|
|
|
parentBranch = config.defaultBranch) {
|
|
|
|
logger.debug(`commitFilesToBranch('${branchName}', files, message, '${parentBranch})'`);
|
2017-01-31 11:19:06 +00:00
|
|
|
const parentCommit = await getBranchCommit(parentBranch);
|
|
|
|
const parentTree = await getCommitTree(parentCommit);
|
2017-02-08 07:43:16 +00:00
|
|
|
const fileBlobs = [];
|
|
|
|
// Create blobs
|
|
|
|
for (const file of files) {
|
|
|
|
const blob = await createBlob(file.contents);
|
|
|
|
fileBlobs.push({
|
|
|
|
name: file.name,
|
|
|
|
blob,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// Create tree
|
|
|
|
const tree = await createTree(parentTree, fileBlobs);
|
2017-01-31 11:19:06 +00:00
|
|
|
const commit = await createCommit(parentCommit, tree, message);
|
2017-02-01 16:43:28 +00:00
|
|
|
const isBranchExisting = await branchExists(branchName);
|
|
|
|
if (isBranchExisting) {
|
|
|
|
await updateBranch(branchName, commit);
|
|
|
|
} else {
|
|
|
|
await createBranch(branchName, commit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Internal branch operations
|
|
|
|
|
|
|
|
// Creates a new branch with provided commit
|
|
|
|
async function createBranch(branchName, commit = config.baseCommitSHA) {
|
|
|
|
await ghGot.post(`repos/${config.repoName}/git/refs`, {
|
|
|
|
body: {
|
|
|
|
ref: `refs/heads/${branchName}`,
|
|
|
|
sha: commit,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Internal: Updates an existing branch to new commit sha
|
|
|
|
async function updateBranch(branchName, commit) {
|
|
|
|
logger.debug(`Updating branch ${branchName} with commit ${commit}`);
|
|
|
|
await ghGot.patch(`repos/${config.repoName}/git/refs/heads/${branchName}`, {
|
|
|
|
body: {
|
|
|
|
sha: commit,
|
|
|
|
force: true,
|
|
|
|
},
|
|
|
|
});
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Low-level commit operations
|
|
|
|
|
|
|
|
// Create a blob with fileContents and return sha
|
2017-01-31 11:19:06 +00:00
|
|
|
async function createBlob(fileContents) {
|
2017-01-29 20:25:12 +00:00
|
|
|
logger.debug('Creating blob');
|
2017-01-31 11:19:06 +00:00
|
|
|
return (await ghGot.post(`repos/${config.repoName}/git/blobs`, {
|
2017-01-29 20:25:12 +00:00
|
|
|
body: {
|
|
|
|
encoding: 'base64',
|
|
|
|
content: new Buffer(fileContents).toString('base64'),
|
|
|
|
},
|
2017-01-31 11:19:06 +00:00
|
|
|
})).body.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return the commit SHA for a branch
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getBranchCommit(branchName) {
|
|
|
|
return (await ghGot(`repos/${config.repoName}/git/refs/heads/${branchName}`)).body.object.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Return the tree SHA for a commit
|
2017-01-31 11:19:06 +00:00
|
|
|
async function getCommitTree(commit) {
|
2017-01-29 20:25:12 +00:00
|
|
|
logger.debug(`getCommitTree(${commit})`);
|
2017-01-31 11:19:06 +00:00
|
|
|
return (await ghGot(`repos/${config.repoName}/git/commits/${commit}`)).body.tree.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create a tree and return SHA
|
2017-02-08 07:43:16 +00:00
|
|
|
async function createTree(baseTree, files) {
|
|
|
|
logger.debug(`createTree(${baseTree}, files)`);
|
|
|
|
const body = {
|
|
|
|
base_tree: baseTree,
|
|
|
|
tree: [],
|
|
|
|
};
|
|
|
|
files.forEach((file) => {
|
|
|
|
body.tree.push({
|
|
|
|
path: file.name,
|
|
|
|
mode: '100644',
|
|
|
|
type: 'blob',
|
|
|
|
sha: file.blob,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
logger.debug(body);
|
|
|
|
return (await ghGot.post(`repos/${config.repoName}/git/trees`, { body })).body.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create a commit and return commit SHA
|
2017-01-31 11:19:06 +00:00
|
|
|
async function createCommit(parent, tree, message) {
|
2017-01-29 20:25:12 +00:00
|
|
|
logger.debug(`createCommit(${parent}, ${tree}, ${message})`);
|
2017-01-31 11:19:06 +00:00
|
|
|
return (await ghGot.post(`repos/${config.repoName}/git/commits`, {
|
2017-01-29 20:25:12 +00:00
|
|
|
body: {
|
|
|
|
message,
|
|
|
|
parents: [parent],
|
|
|
|
tree,
|
|
|
|
},
|
2017-01-31 11:19:06 +00:00
|
|
|
})).body.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|