2017-06-29 05:29:41 +00:00
|
|
|
let logger = require('../logger');
|
2017-01-07 07:22:48 +00:00
|
|
|
const ghGot = require('gh-got');
|
|
|
|
|
2017-08-04 16:13:49 +00:00
|
|
|
// istanbul ignore next
|
|
|
|
function sleep(ms) {
|
|
|
|
// eslint-disable-next-line promise/avoid-new
|
|
|
|
return new Promise(resolve => setTimeout(resolve, ms));
|
|
|
|
}
|
|
|
|
|
|
|
|
async function ghGotRetry(path, opts, retries = 5) {
|
2017-07-30 09:06:15 +00:00
|
|
|
try {
|
|
|
|
const res = await ghGot(path, opts);
|
|
|
|
return res;
|
|
|
|
} catch (err) {
|
2017-08-04 16:13:49 +00:00
|
|
|
if (err.statusCode >= 500 && err.statusCode < 600 && retries > 0) {
|
|
|
|
logger.debug(`Retrying statusCode ${err.statusCode}`);
|
|
|
|
// istanbul ignore if
|
|
|
|
if (process.env.NODE_ENV !== 'test') {
|
|
|
|
await sleep(5000 / retries);
|
|
|
|
}
|
2017-07-30 09:06:15 +00:00
|
|
|
return ghGotRetry(path, opts, retries - 1);
|
|
|
|
}
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const helpers = ['get', 'post', 'put', 'patch', 'head', 'delete'];
|
|
|
|
|
|
|
|
for (const x of helpers) {
|
|
|
|
ghGotRetry[x] = async (path, opts, retries = 3) => {
|
|
|
|
try {
|
|
|
|
const res = await ghGot[x](path, opts);
|
|
|
|
return res;
|
|
|
|
} catch (err) {
|
|
|
|
if (err.statusCode === 502 && retries > 0) {
|
|
|
|
return ghGotRetry[x](path, opts, retries - 1);
|
|
|
|
}
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
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 = {
|
2017-06-02 20:06:15 +00:00
|
|
|
// GitHub App
|
|
|
|
getInstallations,
|
|
|
|
getInstallationToken,
|
|
|
|
getInstallationRepositories,
|
|
|
|
// Initialization
|
2017-04-21 05:00:26 +00:00
|
|
|
getRepos,
|
2017-01-11 13:33:32 +00:00
|
|
|
initRepo,
|
2017-07-06 08:26:18 +00:00
|
|
|
setBaseBranch,
|
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-07-05 09:57:22 +00:00
|
|
|
getAllRenovateBranches,
|
2017-06-08 04:18:21 +00:00
|
|
|
isBranchStale,
|
2017-01-16 17:10:39 +00:00
|
|
|
getBranchPr,
|
2017-04-17 04:46:24 +00:00
|
|
|
getBranchStatus,
|
2017-08-06 13:38:10 +00:00
|
|
|
setBranchStatus,
|
2017-06-03 13:27:11 +00:00
|
|
|
deleteBranch,
|
2017-06-08 04:18:21 +00:00
|
|
|
mergeBranch,
|
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,
|
2017-07-05 09:57:22 +00:00
|
|
|
getAllPrs,
|
2017-01-11 13:33:32 +00:00
|
|
|
updatePr,
|
2017-04-20 11:01:23 +00:00
|
|
|
mergePr,
|
2017-01-15 22:56:09 +00:00
|
|
|
// file
|
2017-07-27 07:36:36 +00:00
|
|
|
getSubDirectories,
|
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-07-07 05:54:09 +00:00
|
|
|
// Commits
|
|
|
|
getCommitMessages,
|
2017-01-10 12:32:32 +00:00
|
|
|
};
|
|
|
|
|
2017-06-02 20:06:15 +00:00
|
|
|
// Get all installations for a GitHub app
|
|
|
|
async function getInstallations(appToken) {
|
|
|
|
logger.debug('getInstallations(appToken)');
|
|
|
|
try {
|
|
|
|
const url = 'app/installations';
|
|
|
|
const options = {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.machine-man-preview+json',
|
|
|
|
authorization: `Bearer ${appToken}`,
|
|
|
|
},
|
|
|
|
};
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(url, options);
|
2017-06-02 20:06:15 +00:00
|
|
|
logger.debug(`Returning ${res.body.length} results`);
|
|
|
|
return res.body;
|
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `GitHub getInstallations error`);
|
2017-06-02 20:06:15 +00:00
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the user's installation token
|
|
|
|
async function getInstallationToken(appToken, installationId) {
|
|
|
|
logger.debug(`getInstallationToken(appToken, ${installationId})`);
|
|
|
|
try {
|
|
|
|
const url = `installations/${installationId}/access_tokens`;
|
|
|
|
const options = {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.machine-man-preview+json',
|
|
|
|
authorization: `Bearer ${appToken}`,
|
|
|
|
},
|
|
|
|
};
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry.post(url, options);
|
2017-06-02 20:06:15 +00:00
|
|
|
return res.body.token;
|
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `GitHub getInstallationToken error`);
|
2017-06-02 20:06:15 +00:00
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get all repositories for a user's installation
|
|
|
|
async function getInstallationRepositories(userToken) {
|
|
|
|
logger.debug('getInstallationRepositories(userToken)');
|
|
|
|
try {
|
|
|
|
const url = 'installation/repositories';
|
|
|
|
const options = {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.machine-man-preview+json',
|
|
|
|
authorization: `token ${userToken}`,
|
|
|
|
},
|
|
|
|
};
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(url, options);
|
2017-06-02 20:06:15 +00:00
|
|
|
logger.debug(
|
2017-06-02 20:40:00 +00:00
|
|
|
`Returning ${res.body.repositories.length} results from a total of ${res
|
|
|
|
.body.total_count}`
|
2017-06-02 20:06:15 +00:00
|
|
|
);
|
|
|
|
return res.body;
|
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `GitHub getInstallationRepositories error`);
|
2017-06-02 20:06:15 +00:00
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 05:00:26 +00:00
|
|
|
// Get all repositories that the user has access to
|
|
|
|
async function getRepos(token, endpoint) {
|
|
|
|
logger.debug('getRepos(token, endpoint)');
|
|
|
|
if (token) {
|
|
|
|
process.env.GITHUB_TOKEN = token;
|
|
|
|
} else if (!process.env.GITHUB_TOKEN) {
|
|
|
|
throw new Error('No token found for getRepos');
|
|
|
|
}
|
|
|
|
if (endpoint) {
|
|
|
|
process.env.GITHUB_ENDPOINT = endpoint;
|
|
|
|
}
|
|
|
|
try {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry('user/repos');
|
2017-04-21 05:00:26 +00:00
|
|
|
return res.body.map(repo => repo.full_name);
|
|
|
|
} catch (err) /* istanbul ignore next */ {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `GitHub getRepos error`);
|
2017-04-21 05:00:26 +00:00
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-10 12:32:32 +00:00
|
|
|
// Initialize GitHub by getting base branch and SHA
|
2017-06-22 07:03:36 +00:00
|
|
|
async function initRepo(repoName, token, endpoint, repoLogger) {
|
2017-06-25 05:36:13 +00:00
|
|
|
logger = repoLogger || logger;
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug(`initRepo("${repoName}")`);
|
2017-06-22 07:03:36 +00:00
|
|
|
if (repoLogger) {
|
|
|
|
logger = repoLogger;
|
|
|
|
}
|
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-07-26 08:56:11 +00:00
|
|
|
const platformConfig = {};
|
2017-01-31 11:19:06 +00:00
|
|
|
try {
|
2017-08-04 12:51:21 +00:00
|
|
|
const res = await ghGotRetry(`repos/${repoName}`, {
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.loki-preview+json',
|
|
|
|
},
|
|
|
|
});
|
2017-08-01 11:31:27 +00:00
|
|
|
logger.trace({ repositoryDetails: res.body }, 'Repository details');
|
2017-07-30 20:59:53 +00:00
|
|
|
platformConfig.privateRepo = res.body.private === true;
|
|
|
|
platformConfig.isFork = res.body.fork === true;
|
2017-01-29 20:25:12 +00:00
|
|
|
config.owner = res.body.owner.login;
|
|
|
|
logger.debug(`${repoName} owner = ${config.owner}`);
|
2017-07-06 08:26:18 +00:00
|
|
|
// Use default branch as PR target unless later overridden
|
2017-07-06 12:12:52 +00:00
|
|
|
config.defaultBranch = res.body.default_branch;
|
|
|
|
config.baseBranch = config.defaultBranch;
|
2017-07-06 08:26:18 +00:00
|
|
|
logger.debug(`${repoName} default branch = ${config.baseBranch}`);
|
|
|
|
config.baseCommitSHA = await getBranchCommit(config.baseBranch);
|
2017-04-20 11:01:23 +00:00
|
|
|
if (res.body.allow_rebase_merge) {
|
|
|
|
config.mergeMethod = 'rebase';
|
|
|
|
} else if (res.body.allow_squash_merge) {
|
|
|
|
config.mergeMethod = 'squash';
|
2017-06-03 07:40:13 +00:00
|
|
|
} else if (res.body.allow_merge_commit) {
|
2017-04-20 11:01:23 +00:00
|
|
|
config.mergeMethod = 'merge';
|
2017-06-03 07:40:13 +00:00
|
|
|
} else {
|
|
|
|
logger.debug('Could not find allowed merge methods for repo');
|
2017-04-20 11:01:23 +00:00
|
|
|
}
|
2017-07-26 08:56:11 +00:00
|
|
|
platformConfig.repoForceRebase = false;
|
2017-07-25 16:18:19 +00:00
|
|
|
try {
|
|
|
|
const branchProtection = await getBranchProtection(config.baseBranch);
|
2017-07-26 06:19:27 +00:00
|
|
|
if (branchProtection.strict) {
|
2017-07-25 16:18:19 +00:00
|
|
|
logger.debug('Repo has branch protection and needs PRs up-to-date');
|
2017-07-26 08:56:11 +00:00
|
|
|
platformConfig.repoForceRebase = true;
|
2017-07-25 16:18:19 +00:00
|
|
|
} else {
|
|
|
|
logger.debug(
|
|
|
|
'Repo has branch protection but does not require up-to-date'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
if (err.statusCode === 404) {
|
|
|
|
logger.debug('Repo has no branch protection');
|
2017-07-26 04:55:25 +00:00
|
|
|
} else if (err.statusCode === 403) {
|
|
|
|
logger.debug('Do not have permissions to detect branch protection');
|
2017-07-25 16:18:19 +00:00
|
|
|
} else {
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|
2017-04-20 11:01:23 +00:00
|
|
|
} catch (err) /* istanbul ignore next */ {
|
2017-07-18 12:31:48 +00:00
|
|
|
if (err.statusCode === 409) {
|
|
|
|
logger.debug('Repository is not initiated');
|
|
|
|
throw new Error('uninitiated');
|
|
|
|
}
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, 'Unknown GitHub initRepo error');
|
2017-01-11 12:19:59 +00:00
|
|
|
throw err;
|
2017-01-31 11:19:06 +00:00
|
|
|
}
|
2017-07-26 08:56:11 +00:00
|
|
|
return platformConfig;
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-07-25 16:18:19 +00:00
|
|
|
async function getBranchProtection(branchName) {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(
|
2017-07-26 06:19:27 +00:00
|
|
|
`repos/${config.repoName}/branches/${branchName}/protection/required_status_checks`,
|
2017-07-25 16:18:19 +00:00
|
|
|
{
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.loki-preview+json',
|
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|
|
|
|
return res.body;
|
|
|
|
}
|
|
|
|
|
2017-07-06 08:26:18 +00:00
|
|
|
async function setBaseBranch(branchName) {
|
|
|
|
if (branchName) {
|
2017-07-06 12:12:52 +00:00
|
|
|
logger.debug(`Setting baseBranch to ${branchName}`);
|
2017-07-06 08:26:18 +00:00
|
|
|
config.baseBranch = branchName;
|
|
|
|
config.baseCommitSHA = await getBranchCommit(config.baseBranch);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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) {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(
|
2017-05-10 07:26:09 +00:00
|
|
|
`search/code?q=repo:${config.repoName}+filename:${fileName}`
|
|
|
|
);
|
2017-01-31 11:19:06 +00:00
|
|
|
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 {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(
|
2017-05-10 07:26:09 +00:00
|
|
|
`repos/${config.repoName}/git/refs/heads/${branchName}`
|
|
|
|
);
|
2017-07-30 06:20:19 +00:00
|
|
|
if (Array.isArray(res.body)) {
|
|
|
|
// This seems to happen if GitHub has partial matches, so we check ref
|
|
|
|
const matchedBranch = res.body.some(
|
|
|
|
branch => branch.ref === `refs/heads/${branchName}`
|
|
|
|
);
|
|
|
|
if (matchedBranch) {
|
|
|
|
logger.debug('Branch exists');
|
|
|
|
} else {
|
|
|
|
logger.debug('No matching branches');
|
2017-04-17 06:39:11 +00:00
|
|
|
}
|
2017-07-30 06:20:19 +00:00
|
|
|
return matchedBranch;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
2017-07-30 06:20:19 +00:00
|
|
|
// This should happen if there's an exact match
|
|
|
|
return res.body.ref === `refs/heads/${branchName}`;
|
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
|
2017-04-21 08:12:41 +00:00
|
|
|
logger.debug("Branch doesn't exist");
|
2017-01-29 20:25:12 +00:00
|
|
|
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-08-04 06:32:22 +00:00
|
|
|
async function getAllRenovateBranches(branchPrefix) {
|
2017-07-05 09:57:22 +00:00
|
|
|
logger.trace('getAllRenovateBranches');
|
2017-07-30 09:06:15 +00:00
|
|
|
const allBranches = (await ghGotRetry(
|
|
|
|
`repos/${config.repoName}/git/refs/heads`
|
|
|
|
)).body;
|
2017-07-05 09:57:22 +00:00
|
|
|
return allBranches.reduce((arr, branch) => {
|
2017-08-04 06:32:22 +00:00
|
|
|
if (branch.ref.indexOf(`refs/heads/${branchPrefix}`) === 0) {
|
2017-07-05 09:57:22 +00:00
|
|
|
arr.push(branch.ref.substring('refs/heads/'.length));
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
}, []);
|
|
|
|
}
|
|
|
|
|
2017-06-08 04:18:21 +00:00
|
|
|
async function isBranchStale(branchName) {
|
|
|
|
// Check if branch's parent SHA = master SHA
|
|
|
|
logger.debug(`isBranchStale(${branchName})`);
|
|
|
|
const branchCommit = await getBranchCommit(branchName);
|
|
|
|
logger.debug(`branchCommit=${branchCommit}`);
|
|
|
|
const commitDetails = await getCommitDetails(branchCommit);
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug({ commitDetails }, `commitDetails`);
|
2017-06-08 04:18:21 +00:00
|
|
|
const parentSha = commitDetails.parents[0].sha;
|
|
|
|
logger.debug(`parentSha=${parentSha}`);
|
|
|
|
// Return true if the SHAs don't match
|
|
|
|
return parentSha !== config.baseCommitSHA;
|
|
|
|
}
|
|
|
|
|
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-04-21 08:12:41 +00:00
|
|
|
const gotString =
|
|
|
|
`repos/${config.repoName}/pulls?` +
|
2017-07-06 08:26:18 +00:00
|
|
|
`state=open&base=${config.baseBranch}&head=${config.owner}:${branchName}`;
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(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-04-17 04:46:24 +00:00
|
|
|
// Returns the combined status for a branch.
|
2017-07-05 05:02:25 +00:00
|
|
|
async function getBranchStatus(branchName, requiredStatusChecks) {
|
2017-04-17 04:46:24 +00:00
|
|
|
logger.debug(`getBranchStatus(${branchName})`);
|
2017-07-05 05:02:25 +00:00
|
|
|
if (!requiredStatusChecks) {
|
|
|
|
// null means disable status checks, so it always succeeds
|
|
|
|
return 'success';
|
|
|
|
}
|
|
|
|
if (requiredStatusChecks.length) {
|
|
|
|
// This is Unsupported
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.warn({ requiredStatusChecks }, `Unsupported requiredStatusChecks`);
|
2017-07-05 05:02:25 +00:00
|
|
|
return 'failed';
|
|
|
|
}
|
2017-04-17 04:46:24 +00:00
|
|
|
const gotString = `repos/${config.repoName}/commits/${branchName}/status`;
|
|
|
|
logger.debug(gotString);
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(gotString);
|
2017-04-17 04:46:24 +00:00
|
|
|
return res.body.state;
|
|
|
|
}
|
|
|
|
|
2017-08-06 13:38:10 +00:00
|
|
|
async function setBranchStatus(
|
|
|
|
branchName,
|
|
|
|
context,
|
|
|
|
description,
|
|
|
|
state,
|
|
|
|
targetUrl
|
|
|
|
) {
|
|
|
|
const branchCommit = await getBranchCommit(branchName);
|
|
|
|
const url = `repos/${config.repoName}/statuses/${branchCommit}`;
|
|
|
|
const options = {
|
|
|
|
state,
|
|
|
|
description,
|
|
|
|
context,
|
|
|
|
};
|
|
|
|
if (targetUrl) {
|
|
|
|
options.target_url = targetUrl;
|
|
|
|
}
|
|
|
|
await ghGotRetry.post(url, { body: options });
|
|
|
|
}
|
|
|
|
|
2017-06-03 13:27:11 +00:00
|
|
|
async function deleteBranch(branchName) {
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.delete(
|
|
|
|
`repos/${config.repoName}/git/refs/heads/${branchName}`
|
|
|
|
);
|
2017-06-03 13:27:11 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 04:18:21 +00:00
|
|
|
async function mergeBranch(branchName, mergeType) {
|
|
|
|
logger.debug(`mergeBranch(${branchName}, ${mergeType})`);
|
|
|
|
if (mergeType === 'branch-push') {
|
2017-07-06 08:26:18 +00:00
|
|
|
const url = `repos/${config.repoName}/git/refs/heads/${config.baseBranch}`;
|
2017-06-08 04:18:21 +00:00
|
|
|
const options = {
|
|
|
|
body: {
|
|
|
|
sha: await getBranchCommit(branchName),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
try {
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.patch(url, options);
|
2017-06-08 04:18:21 +00:00
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `Error pushing branch merge for ${branchName}`);
|
2017-06-08 04:18:21 +00:00
|
|
|
throw new Error('branch-push failed');
|
|
|
|
}
|
|
|
|
} else if (mergeType === 'branch-merge-commit') {
|
|
|
|
const url = `repos/${config.repoName}/merges`;
|
|
|
|
const options = {
|
|
|
|
body: {
|
2017-07-06 08:26:18 +00:00
|
|
|
base: config.baseBranch,
|
2017-06-08 04:18:21 +00:00
|
|
|
head: branchName,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
try {
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.post(url, options);
|
2017-06-08 04:18:21 +00:00
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `Error pushing branch merge for ${branchName}`);
|
2017-06-08 04:18:21 +00:00
|
|
|
throw new Error('branch-push failed');
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw new Error(`Unsupported branch merge type: ${mergeType}`);
|
|
|
|
}
|
|
|
|
// Update base commit
|
2017-07-06 08:26:18 +00:00
|
|
|
config.baseCommitSHA = await getBranchCommit(config.baseBranch);
|
2017-06-08 04:18:21 +00:00
|
|
|
// Delete branch
|
|
|
|
await deleteBranch(branchName);
|
|
|
|
}
|
|
|
|
|
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-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.post(
|
|
|
|
`repos/${config.repoName}/issues/${issueNo}/assignees`,
|
|
|
|
{
|
|
|
|
body: {
|
|
|
|
assignees,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|
2017-01-18 20:17:07 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 13:54:16 +00:00
|
|
|
async function addReviewers(issueNo, reviewers) {
|
|
|
|
logger.debug(`Adding reviewers ${reviewers} to #${issueNo}`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.post(
|
2017-04-21 08:12:41 +00:00
|
|
|
`repos/${config.repoName}/pulls/${issueNo}/requested_reviewers`,
|
|
|
|
{
|
|
|
|
headers: {
|
|
|
|
accept: 'application/vnd.github.black-cat-preview+json',
|
|
|
|
},
|
|
|
|
body: {
|
|
|
|
reviewers,
|
|
|
|
},
|
2017-04-21 08:25:49 +00:00
|
|
|
}
|
2017-04-21 08:12:41 +00:00
|
|
|
);
|
2017-01-31 13:54:16 +00:00
|
|
|
}
|
|
|
|
|
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-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.post(`repos/${config.repoName}/issues/${issueNo}/labels`, {
|
2017-07-06 17:24:09 +00:00
|
|
|
body: labels,
|
2017-01-13 18:18:44 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
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-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(urlString);
|
2017-01-31 11:19:06 +00:00
|
|
|
let pr = null;
|
2017-04-21 08:12:41 +00:00
|
|
|
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}`;
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(url);
|
2017-01-31 11:19:06 +00:00
|
|
|
// Return true if any of the titles match exactly
|
2017-04-21 08:12:41 +00:00
|
|
|
return res.body.some(
|
|
|
|
pr =>
|
2017-04-21 08:25:49 +00:00
|
|
|
pr.title === prTitle && pr.head.label === `${config.owner}:${branchName}`
|
2017-04-21 08:12:41 +00:00
|
|
|
);
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-02-01 12:50:28 +00:00
|
|
|
// Creates PR and returns PR number
|
2017-07-06 12:12:52 +00:00
|
|
|
async function createPr(branchName, title, body, useDefaultBranch) {
|
|
|
|
const base = useDefaultBranch ? config.defaultBranch : config.baseBranch;
|
2017-07-30 09:06:15 +00:00
|
|
|
const pr = (await ghGotRetry.post(`repos/${config.repoName}/pulls`, {
|
2017-07-06 08:26:18 +00:00
|
|
|
body: {
|
|
|
|
title,
|
|
|
|
head: branchName,
|
2017-07-06 12:12:52 +00:00
|
|
|
base,
|
2017-07-06 08:26:18 +00:00
|
|
|
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;
|
|
|
|
}
|
2017-07-30 09:06:15 +00:00
|
|
|
const pr = (await ghGotRetry(`repos/${config.repoName}/pulls/${prNo}`)).body;
|
2017-02-01 12:50:28 +00:00
|
|
|
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;
|
|
|
|
}
|
2017-03-13 09:05:19 +00:00
|
|
|
if (!pr.isClosed) {
|
|
|
|
if (pr.mergeable_state === 'dirty') {
|
2017-06-28 11:20:31 +00:00
|
|
|
logger.debug(`PR mergeable state is dirty`);
|
2017-03-13 09:05:19 +00:00
|
|
|
pr.isUnmergeable = true;
|
|
|
|
}
|
|
|
|
if (pr.commits === 1) {
|
|
|
|
// Only one commit was made - must have been renovate
|
2017-06-28 11:20:31 +00:00
|
|
|
logger.debug('Only 1 commit in PR so rebase is possible');
|
2017-03-13 09:05:19 +00:00
|
|
|
pr.canRebase = true;
|
|
|
|
} else {
|
|
|
|
// Check if only one author of all commits
|
|
|
|
logger.debug('Checking all commits');
|
2017-07-30 09:06:15 +00:00
|
|
|
const prCommits = (await ghGotRetry(
|
2017-05-10 07:26:09 +00:00
|
|
|
`repos/${config.repoName}/pulls/${prNo}/commits`
|
|
|
|
)).body;
|
2017-03-13 09:05:19 +00:00
|
|
|
const authors = prCommits.reduce((arr, commit) => {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.trace({ commit }, `Checking commit`);
|
2017-07-04 17:22:09 +00:00
|
|
|
let author = 'unknown';
|
|
|
|
if (commit.author) {
|
|
|
|
author = commit.author.login;
|
|
|
|
} else if (commit.commit && commit.commit.author) {
|
|
|
|
author = commit.commit.author.email;
|
|
|
|
} else {
|
|
|
|
logger.debug('Could not determine commit author');
|
|
|
|
}
|
|
|
|
logger.debug(`Commit author is: ${author}`);
|
2017-03-13 09:05:19 +00:00
|
|
|
if (arr.indexOf(author) === -1) {
|
|
|
|
arr.push(author);
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
}, []);
|
|
|
|
logger.debug(`Author list: ${authors}`);
|
|
|
|
if (authors.length === 1) {
|
|
|
|
pr.canRebase = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (pr.base.sha !== config.baseCommitSHA) {
|
|
|
|
pr.isStale = true;
|
|
|
|
}
|
2017-02-06 06:56:33 +00:00
|
|
|
}
|
2017-02-01 12:50:28 +00:00
|
|
|
return pr;
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-07-05 09:57:22 +00:00
|
|
|
async function getAllPrs() {
|
2017-07-30 09:06:15 +00:00
|
|
|
const all = (await ghGotRetry(`repos/${config.repoName}/pulls?state=open`))
|
|
|
|
.body;
|
2017-07-05 09:57:22 +00:00
|
|
|
return all.map(pr => ({
|
|
|
|
number: pr.number,
|
|
|
|
branchName: pr.head.ref,
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2017-01-31 11:19:06 +00:00
|
|
|
async function updatePr(prNo, title, body) {
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.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
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-04-20 11:01:23 +00:00
|
|
|
async function mergePr(pr) {
|
2017-06-03 07:40:13 +00:00
|
|
|
const url = `repos/${config.repoName}/pulls/${pr.number}/merge`;
|
|
|
|
const options = {
|
|
|
|
body: {},
|
|
|
|
};
|
|
|
|
if (config.mergeMethod) {
|
|
|
|
// This path is taken if we have auto-detected the allowed merge types from the repo
|
|
|
|
options.body.merge_method = config.mergeMethod;
|
|
|
|
try {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug({ options, url }, `mergePr`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.put(url, options);
|
2017-06-03 07:40:13 +00:00
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `Failed to ${options.body.merge_method} PR`);
|
2017-06-03 07:40:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// We need to guess the merge method and try squash -> rebase -> merge
|
|
|
|
options.body.merge_method = 'rebase';
|
|
|
|
try {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug({ options, url }, `mergePr`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.put(url, options);
|
2017-06-03 07:40:13 +00:00
|
|
|
} catch (err1) {
|
2017-08-01 11:31:27 +00:00
|
|
|
logger.debug({ err: err1 }, `Failed to ${options.body.merge_method} PR`);
|
2017-06-03 07:40:13 +00:00
|
|
|
try {
|
|
|
|
options.body.merge_method = 'squash';
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug({ options, url }, `mergePr`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.put(url, options);
|
2017-06-03 07:40:13 +00:00
|
|
|
} catch (err2) {
|
|
|
|
logger.debug(
|
2017-07-19 06:05:26 +00:00
|
|
|
{ err: err2 },
|
|
|
|
`Failed to ${options.body.merge_method} PR`
|
2017-06-03 07:40:13 +00:00
|
|
|
);
|
|
|
|
try {
|
|
|
|
options.body.merge_method = 'merge';
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.debug({ options, url }, `mergePr`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.put(url, options);
|
2017-06-03 07:40:13 +00:00
|
|
|
} catch (err3) {
|
|
|
|
logger.debug(
|
2017-07-19 06:05:26 +00:00
|
|
|
{ err: err3 },
|
|
|
|
`Failed to ${options.body.merge_method} PR`
|
2017-06-03 07:40:13 +00:00
|
|
|
);
|
2017-08-02 05:16:31 +00:00
|
|
|
logger.warn('All merge attempts failed');
|
2017-06-03 07:40:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-06-07 13:42:20 +00:00
|
|
|
// Update base branch SHA
|
2017-07-06 08:26:18 +00:00
|
|
|
config.baseCommitSHA = await getBranchCommit(config.baseBranch);
|
2017-04-20 11:01:23 +00:00
|
|
|
// Delete branch
|
2017-06-03 13:27:11 +00:00
|
|
|
await deleteBranch(pr.head.ref);
|
2017-04-20 11:01:23 +00:00
|
|
|
}
|
|
|
|
|
2017-01-10 12:32:32 +00:00
|
|
|
// Generic File operations
|
2017-01-29 20:25:12 +00:00
|
|
|
|
2017-07-06 08:26:18 +00:00
|
|
|
async function getFile(filePath, branchName = config.baseBranch) {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(
|
2017-05-10 07:26:09 +00:00
|
|
|
`repos/${config.repoName}/contents/${filePath}?ref=${branchName}`
|
|
|
|
);
|
2017-01-31 11:19:06 +00:00
|
|
|
return res.body.content;
|
2017-01-07 21:08:45 +00:00
|
|
|
}
|
2017-01-10 12:32:32 +00:00
|
|
|
|
2017-07-06 08:26:18 +00:00
|
|
|
async function getFileContent(filePath, branchName = config.baseBranch) {
|
2017-07-06 12:12:52 +00:00
|
|
|
logger.trace(
|
|
|
|
`getFileContent(filePath=${filePath}, branchName=${branchName})`
|
|
|
|
);
|
2017-01-31 11:19:06 +00:00
|
|
|
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-07-06 12:12:52 +00:00
|
|
|
async function getFileJson(filePath, branchName) {
|
|
|
|
logger.trace(`getFileJson(filePath=${filePath}, branchName=${branchName})`);
|
2017-06-04 06:06:40 +00:00
|
|
|
let fileJson = null;
|
|
|
|
try {
|
|
|
|
fileJson = JSON.parse(await getFileContent(filePath, branchName));
|
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `Failed to parse JSON for ${filePath}`);
|
2017-06-04 06:06:40 +00:00
|
|
|
}
|
|
|
|
return fileJson;
|
2017-01-10 12:32:32 +00:00
|
|
|
}
|
|
|
|
|
2017-07-27 07:36:36 +00:00
|
|
|
async function getSubDirectories(path) {
|
|
|
|
logger.trace(`getSubDirectories(path=${path})`);
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(`repos/${config.repoName}/contents/${path}`);
|
2017-07-27 07:36:36 +00:00
|
|
|
const directoryList = [];
|
|
|
|
res.body.forEach(item => {
|
|
|
|
if (item.type === 'dir') {
|
|
|
|
directoryList.push(item.name);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return directoryList;
|
|
|
|
}
|
|
|
|
|
2017-02-08 07:43:16 +00:00
|
|
|
// Add a new commit, create branch if not existing
|
|
|
|
async function commitFilesToBranch(
|
|
|
|
branchName,
|
|
|
|
files,
|
|
|
|
message,
|
2017-07-06 08:26:18 +00:00
|
|
|
parentBranch = config.baseBranch
|
2017-04-21 08:12:41 +00:00
|
|
|
) {
|
2017-05-10 07:26:09 +00:00
|
|
|
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) {
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.post(`repos/${config.repoName}/git/refs`, {
|
2017-02-01 16:43:28 +00:00
|
|
|
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}`);
|
2017-07-30 09:06:15 +00:00
|
|
|
await ghGotRetry.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-07-30 09:06:15 +00:00
|
|
|
return (await ghGotRetry.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) {
|
2017-07-30 09:06:15 +00:00
|
|
|
return (await ghGotRetry(
|
|
|
|
`repos/${config.repoName}/git/refs/heads/${branchName}`
|
|
|
|
)).body.object.sha;
|
2017-01-29 20:25:12 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 04:18:21 +00:00
|
|
|
async function getCommitDetails(commit) {
|
|
|
|
logger.debug(`getCommitDetails(${commit})`);
|
2017-07-30 09:06:15 +00:00
|
|
|
const results = await ghGotRetry(
|
|
|
|
`repos/${config.repoName}/git/commits/${commit}`
|
|
|
|
);
|
2017-06-08 04:18:21 +00:00
|
|
|
return results.body;
|
|
|
|
}
|
|
|
|
|
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-07-30 09:06:15 +00:00
|
|
|
return (await ghGotRetry(`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: [],
|
|
|
|
};
|
2017-04-21 08:12:41 +00:00
|
|
|
files.forEach(file => {
|
2017-02-08 07:43:16 +00:00
|
|
|
body.tree.push({
|
|
|
|
path: file.name,
|
|
|
|
mode: '100644',
|
|
|
|
type: 'blob',
|
|
|
|
sha: file.blob,
|
|
|
|
});
|
|
|
|
});
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.trace({ body }, 'createTree body');
|
2017-07-30 09:06:15 +00:00
|
|
|
return (await ghGotRetry.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-07-30 09:06:15 +00:00
|
|
|
return (await ghGotRetry.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
|
|
|
}
|
2017-07-07 05:54:09 +00:00
|
|
|
|
|
|
|
async function getCommitMessages() {
|
|
|
|
logger.debug('getCommitMessages');
|
|
|
|
try {
|
2017-07-30 09:06:15 +00:00
|
|
|
const res = await ghGotRetry(`repos/${config.repoName}/commits`);
|
2017-07-07 05:54:09 +00:00
|
|
|
return res.body.map(commit => commit.commit.message);
|
|
|
|
} catch (err) {
|
2017-07-19 06:05:26 +00:00
|
|
|
logger.error({ err }, `getCommitMessages error`);
|
2017-07-07 05:54:09 +00:00
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|