renovate/lib/api/gitlab.js

456 lines
12 KiB
JavaScript
Raw Normal View History

let logger = require('../helpers/logger');
2017-02-11 07:14:19 +00:00
const glGot = require('gl-got');
const config = {};
module.exports = {
getRepos,
2017-02-11 07:14:19 +00:00
initRepo,
// Search
findFilePaths,
// Branch
branchExists,
createBranch,
getBranch,
2017-02-11 07:14:19 +00:00
getBranchPr,
getBranchStatus,
deleteBranch,
2017-02-11 07:14:19 +00:00
// issue
addAssignees,
addReviewers,
addLabels,
// PR
findPr,
checkForClosedPr,
createPr,
getPr,
updatePr,
mergePr,
2017-02-11 07:14:19 +00:00
// file
commitFilesToBranch,
getFile,
getFileContent,
getFileJson,
createFile,
updateFile,
2017-02-11 07:14:19 +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.GITLAB_TOKEN = token;
} else if (!process.env.GITLAB_TOKEN) {
throw new Error('No token found for getRepos');
}
if (endpoint) {
process.env.GITLAB_ENDPOINT = endpoint;
}
try {
const res = await glGot('projects');
return res.body.map(repo => repo.path_with_namespace);
} catch (err) {
logger.error(`GitLab getRepos error: ${JSON.stringify(err)}`);
throw err;
}
}
2017-02-11 07:14:19 +00:00
// Initialize GitLab by getting base branch
async function initRepo(repoName, token, endpoint, repoLogger) {
if (repoLogger) {
logger = repoLogger;
}
2017-02-11 07:14:19 +00:00
logger.debug(`initRepo(${repoName})`);
if (token) {
process.env.GITLAB_TOKEN = token;
} else if (!process.env.GITLAB_TOKEN) {
throw new Error(`No token found for GitLab repository ${repoName}`);
}
if (token) {
process.env.GITLAB_TOKEN = token;
}
if (endpoint) {
process.env.GITLAB_ENDPOINT = endpoint;
}
try {
logger.debug(`Determining Gitlab API version`);
// projects/owned route deprecated in v4
await glGot(`projects/owned`);
config.apiVersion = 'v3';
} catch (err) {
config.apiVersion = 'v4';
}
logger.debug(`Detected Gitlab API ${config.apiVersion}`);
2017-02-11 07:14:19 +00:00
config.repoName = repoName.replace('/', '%2F');
try {
const res = await glGot(`projects/${config.repoName}`);
config.defaultBranch = res.body.default_branch;
logger.debug(`${repoName} default branch = ${config.defaultBranch}`);
// Discover our user email
config.email = (await glGot(`user`)).body.email;
2017-02-11 07:14:19 +00:00
} catch (err) {
logger.error(`GitLab init error: ${JSON.stringify(err)}`);
throw err;
}
return config;
2017-02-11 07:14:19 +00:00
}
// Search
// Returns an array of file paths in current repo matching the fileName
async function findFilePaths(fileName) {
logger.debug("Can't find multiple package.json files in GitLab");
2017-02-11 07:14:19 +00:00
return [fileName];
}
// Branch
// Returns true if branch exists, otherwise false
async function branchExists(branchName) {
logger.debug(`Checking if branch exists: ${branchName}`);
try {
const url = `projects/${config.repoName}/repository/branches/${branchName}`;
const res = await glGot(url);
if (res.statusCode === 200) {
logger.debug('Branch exists');
return true;
}
// This probably shouldn't happen
2017-04-21 08:12:41 +00:00
logger.debug("Branch doesn't exist");
2017-02-11 07:14:19 +00:00
return false;
} catch (error) {
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-02-11 07:14:19 +00:00
return false;
}
// Propagate if it's any other error
throw error;
}
}
// Returns branch object
async function getBranch(branchName) {
logger.debug(`getBranch(${branchName})`);
const url = `projects/${config.repoName}/repository/branches/${branchName}`;
try {
return (await glGot(url)).body;
} catch (err) {
logger.warn(`Failed to getBranch ${branchName}`);
logger.debug(JSON.stringify(err));
return null;
}
}
2017-02-11 07:14:19 +00:00
// Returns the Pull Request for a branch. Null if not exists.
async function getBranchPr(branchName) {
logger.debug(`getBranchPr(${branchName})`);
const urlString = `projects/${config.repoName}/merge_requests?state=opened`;
const res = await glGot(urlString);
logger.debug(`Got res with ${res.body.length} results`);
let pr = null;
2017-04-21 08:12:41 +00:00
res.body.forEach(result => {
2017-02-11 07:14:19 +00:00
if (result.source_branch === branchName) {
pr = result;
}
});
if (!pr) {
return null;
}
return getPr(config.apiVersion === 'v3' ? pr.id : pr.iid);
2017-02-11 07:14:19 +00:00
}
// Returns the combined status for a branch.
async function getBranchStatus(branchName) {
logger.debug(`getBranchStatus(${branchName})`);
// First, get the branch to find the commit SHA
let url = `projects/${config.repoName}/repository/branches/${branchName}`;
let res = await glGot(url);
const branchSha = res.body.commit.id;
// Now, check the statuses for that commit
url = `projects/${config.repoName}/repository/commits/${branchSha}/statuses`;
res = await glGot(url);
logger.debug(`Got res with ${res.body.length} results`);
if (res.body.length === 0) {
// Return 'pending' if we have no status checks
return 'pending';
}
let status = 'success';
// Return 'success' if all are success
2017-04-21 08:12:41 +00:00
res.body.forEach(check => {
// If one is failed then don't overwrite that
if (status !== 'failure') {
if (check.status === 'failed') {
status = 'failure';
} else if (check.status !== 'success') {
status = check.status;
}
}
});
return status;
}
async function deleteBranch(branchName) {
await glGot.delete(
`projects/${config.repoName}/repository/branches/${branchName}`
);
}
2017-02-11 07:14:19 +00:00
// Issue
async function addAssignees(prNo, assignees) {
logger.debug(`Adding assignees ${assignees} to #${prNo}`);
if (assignees.length > 1) {
logger.error('Cannot assign more than one assignee to Merge Requests');
}
let url = `projects/${config.repoName}/merge_requests/${prNo}`;
url = `${url}?assignee_id=${assignees[0]}`;
await glGot.put(url);
}
async function addReviewers(prNo, reviewers) {
logger.debug(`addReviewers('${prNo}, '${reviewers})`);
logger.error('No reviewer functionality in GitLab');
}
async function addLabels(prNo, labels) {
logger.debug(`Adding labels ${labels} to #${prNo}`);
let url = `projects/${config.repoName}/merge_requests/${prNo}`;
url = `${url}?labels=${labels.join(',')}`;
await glGot.put(url);
}
async function findPr(branchName, prTitle, state = 'all') {
logger.debug(`findPr(${branchName}, ${prTitle}, ${state})`);
const urlString = `projects/${config.repoName}/merge_requests?state=${state}`;
const res = await glGot(urlString);
let pr = null;
2017-04-21 08:12:41 +00:00
res.body.forEach(result => {
if (
(!prTitle || result.title === prTitle) &&
result.source_branch === branchName
) {
2017-02-11 07:14:19 +00:00
pr = result;
// GitHub uses number, GitLab uses iid
pr.number = pr.id;
pr.body = pr.description;
pr.displayNumber = `Merge Request #${pr.iid}`;
if (pr.state !== 'opened') {
pr.isClosed = true;
}
}
});
return pr;
}
// Pull Request
async function checkForClosedPr(branchName, prTitle) {
const pr = await findPr(branchName, prTitle, 'closed');
if (pr) {
return true;
}
return false;
}
async function createPr(branchName, title, body) {
logger.debug(`Creating Merge Request: ${title}`);
const description = body
.replace(/Pull Request/g, 'Merge Request')
.replace(/PR/g, 'MR');
2017-02-11 07:14:19 +00:00
const res = await glGot.post(`projects/${config.repoName}/merge_requests`, {
body: {
source_branch: branchName,
target_branch: config.defaultBranch,
remove_source_branch: true,
2017-02-11 07:14:19 +00:00
title,
description,
2017-02-11 07:14:19 +00:00
},
});
const pr = res.body;
pr.number = pr.id;
pr.displayNumber = `Merge Request #${pr.iid}`;
return pr;
}
async function getPr(prNo) {
logger.debug(`getPr(${prNo})`);
const url = `projects/${config.repoName}/merge_requests/${prNo}`;
const pr = (await glGot(url)).body;
// Harmonize fields with GitHub
pr.number = config.apiVersion === 'v3' ? pr.id : pr.iid;
2017-02-11 07:14:19 +00:00
pr.displayNumber = `Merge Request #${pr.iid}`;
pr.body = pr.description;
if (pr.state === 'closed' || pr.state === 'merged') {
logger.debug('pr is closed');
pr.isClosed = true;
}
if (pr.merge_status === 'cannot_be_merged') {
logger.debug('pr cannot be merged');
pr.isUnmergeable = true;
}
// Check if the most recent branch commit is by us
// If not then we don't allow it to be rebased, in case someone's changes would be lost
const branch = await getBranch(pr.source_branch);
if (branch && branch.commit.author_email === config.email) {
pr.canRebase = true;
}
2017-02-11 07:14:19 +00:00
return pr;
}
async function updatePr(prNo, title, body) {
const description = body
.replace(/Pull Request/g, 'Merge Request')
.replace(/PR/g, 'MR');
2017-02-11 07:14:19 +00:00
await glGot.put(`projects/${config.repoName}/merge_requests/${prNo}`, {
body: {
title,
description,
2017-02-11 07:14:19 +00:00
},
});
}
async function mergePr(pr) {
2017-04-21 08:12:41 +00:00
await glGot.put(
`projects/${config.repoName}/merge_requests/${pr.number}/merge`,
{
body: {
should_remove_source_branch: true,
},
2017-04-21 08:25:49 +00:00
}
2017-04-21 08:12:41 +00:00
);
}
2017-02-11 07:14:19 +00:00
// Generic File operations
async function getFile(filePath, branchName = config.defaultBranch) {
// Gitlab API v3 support
let url;
if (config.apiVersion === 'v3') {
url = `projects/${config.repoName}/repository/files?file_path=${filePath}&ref=${branchName}`;
} else {
url = `projects/${config.repoName}/repository/files/${filePath}?ref=${branchName}`;
}
const res = await glGot(url);
2017-02-11 07:14:19 +00:00
return res.body.content;
}
async function getFileContent(filePath, branchName) {
try {
const file = await getFile(filePath, branchName);
return new Buffer(file, 'base64').toString();
} catch (error) {
if (error.statusCode === 404) {
// If file not found, then return null JSON
return null;
}
// Propagate if it's any other error
throw error;
}
}
async function getFileJson(filePath, branchName) {
const fileContent = await getFileContent(filePath, branchName);
return JSON.parse(fileContent);
2017-02-11 07:14:19 +00:00
}
async function createFile(branchName, filePath, fileContents, message) {
// Gitlab API v3 support
let url;
const opts = {};
if (config.apiVersion === 'v3') {
url = `projects/${config.repoName}/repository/files`;
opts.body = {
2017-02-11 07:14:19 +00:00
file_path: filePath,
branch_name: branchName,
commit_message: message,
encoding: 'base64',
content: new Buffer(fileContents).toString('base64'),
};
} else {
url = `projects/${config.repoName}/repository/files/${filePath}`;
opts.body = {
branch: branchName,
commit_message: message,
encoding: 'base64',
content: new Buffer(fileContents).toString('base64'),
};
}
await glGot.post(url, opts);
2017-02-11 07:14:19 +00:00
}
async function updateFile(branchName, filePath, fileContents, message) {
// Gitlab API v3 support
let url;
const opts = {};
if (config.apiVersion === 'v3') {
url = `projects/${config.repoName}/repository/files`;
opts.body = {
2017-02-11 07:14:19 +00:00
file_path: filePath,
branch_name: branchName,
commit_message: message,
encoding: 'base64',
content: new Buffer(fileContents).toString('base64'),
};
} else {
url = `projects/${config.repoName}/repository/files/${filePath}`;
opts.body = {
branch: branchName,
commit_message: message,
encoding: 'base64',
content: new Buffer(fileContents).toString('base64'),
};
}
await glGot.put(url, opts);
2017-02-11 07:14:19 +00:00
}
// Add a new commit, create branch if not existing
async function commitFilesToBranch(
branchName,
files,
message,
2017-04-21 08:25:49 +00:00
parentBranch = config.defaultBranch
2017-04-21 08:12:41 +00:00
) {
logger.debug(
`commitFilesToBranch('${branchName}', files, message, '${parentBranch})'`
);
2017-02-11 07:14:19 +00:00
if (branchName !== parentBranch) {
const isBranchExisting = await branchExists(branchName);
if (isBranchExisting) {
logger.debug(`Branch ${branchName} already exists`);
} else {
logger.debug(`Creating branch ${branchName}`);
await createBranch(branchName);
}
}
for (const file of files) {
const existingFile = await getFileContent(file.name, branchName);
if (existingFile) {
logger.debug(`${file.name} exists - updating it`);
await updateFile(branchName, file.name, file.contents, message);
} else {
logger.debug(`Creating file ${file.name}`);
await createFile(branchName, file.name, file.contents, message);
}
}
}
// Internal branch operations
// Creates a new branch with provided commit
async function createBranch(branchName, ref = config.defaultBranch) {
// Gitlab API v3 support
const opts = {};
if (config.apiVersion === 'v3') {
opts.body = {
2017-02-11 07:14:19 +00:00
branch_name: branchName,
ref,
};
} else {
opts.body = {
branch: branchName,
ref,
};
}
await glGot.post(`projects/${config.repoName}/repository/branches`, opts);
2017-02-11 07:14:19 +00:00
}