mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-23 13:06:27 +00:00
01be64987b
Co-authored-by: Rhys Arkins <rhys@arkins.net>
888 lines
23 KiB
TypeScript
888 lines
23 KiB
TypeScript
import URL from 'url';
|
|
import addrs from 'email-addresses';
|
|
import parseDiff from 'parse-diff';
|
|
import { RenovateConfig } from '../../config/common';
|
|
import {
|
|
REPOSITORY_DISABLED,
|
|
REPOSITORY_NOT_FOUND,
|
|
} from '../../constants/error-messages';
|
|
import { PLATFORM_TYPE_BITBUCKET } from '../../constants/platforms';
|
|
import { PR_STATE_ALL, PR_STATE_OPEN } from '../../constants/pull-requests';
|
|
import { logger } from '../../logger';
|
|
import { BranchStatus } from '../../types';
|
|
import * as hostRules from '../../util/host-rules';
|
|
import { BitbucketHttp, setBaseUrl } from '../../util/http/bitbucket';
|
|
import { sanitize } from '../../util/sanitize';
|
|
import {
|
|
BranchStatusConfig,
|
|
CommitFilesConfig,
|
|
CreatePRConfig,
|
|
EnsureCommentConfig,
|
|
EnsureCommentRemovalConfig,
|
|
EnsureIssueConfig,
|
|
EnsureIssueResult,
|
|
FindPRConfig,
|
|
Issue,
|
|
PlatformConfig,
|
|
Pr,
|
|
RepoConfig,
|
|
RepoParams,
|
|
VulnerabilityAlert,
|
|
} from '../common';
|
|
import GitStorage, { StatusResult } from '../git/storage';
|
|
import { smartTruncate } from '../utils/pr-body';
|
|
import { readOnlyIssueBody } from '../utils/read-only-issue-body';
|
|
import * as comments from './comments';
|
|
import * as utils from './utils';
|
|
|
|
const bitbucketHttp = new BitbucketHttp();
|
|
|
|
const BITBUCKET_PROD_ENDPOINT = 'https://api.bitbucket.org/';
|
|
|
|
let config: utils.Config = {} as any;
|
|
|
|
export function initPlatform({
|
|
endpoint,
|
|
username,
|
|
password,
|
|
}: RenovateConfig): Promise<PlatformConfig> {
|
|
if (!(username && password)) {
|
|
throw new Error(
|
|
'Init: You must configure a Bitbucket username and password'
|
|
);
|
|
}
|
|
if (endpoint && endpoint !== BITBUCKET_PROD_ENDPOINT) {
|
|
logger.warn(
|
|
`Init: Bitbucket Cloud endpoint should generally be ${BITBUCKET_PROD_ENDPOINT} but is being configured to a different value. Did you mean to use Bitbucket Server?`
|
|
);
|
|
}
|
|
// TODO: Add a connection check that endpoint/username/password combination are valid
|
|
const platformConfig: PlatformConfig = {
|
|
endpoint: endpoint || BITBUCKET_PROD_ENDPOINT,
|
|
};
|
|
return Promise.resolve(platformConfig);
|
|
}
|
|
|
|
// Get all repositories that the user has access to
|
|
export async function getRepos(): Promise<string[]> {
|
|
logger.debug('Autodiscovering Bitbucket Cloud repositories');
|
|
try {
|
|
const repos = await utils.accumulateValues<{ full_name: string }>(
|
|
`/2.0/repositories/?role=contributor`
|
|
);
|
|
return repos.map((repo) => repo.full_name);
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.error({ err }, `bitbucket getRepos error`);
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
// Initialize bitbucket by getting base branch and SHA
|
|
export async function initRepo({
|
|
repository,
|
|
localDir,
|
|
optimizeForDisabled,
|
|
bbUseDefaultReviewers,
|
|
endpoint = BITBUCKET_PROD_ENDPOINT,
|
|
}: RepoParams): Promise<RepoConfig> {
|
|
logger.debug(`initRepo("${repository}")`);
|
|
const opts = hostRules.find({
|
|
hostType: PLATFORM_TYPE_BITBUCKET,
|
|
url: endpoint,
|
|
});
|
|
setBaseUrl(endpoint);
|
|
config = {
|
|
repository,
|
|
username: opts.username,
|
|
bbUseDefaultReviewers: bbUseDefaultReviewers !== false,
|
|
} as any;
|
|
let info: utils.RepoInfo;
|
|
try {
|
|
info = utils.repoInfoTransformer(
|
|
(await bitbucketHttp.getJson(`/2.0/repositories/${repository}`)).body
|
|
);
|
|
|
|
if (optimizeForDisabled) {
|
|
interface RenovateConfig {
|
|
enabled: boolean;
|
|
}
|
|
|
|
let renovateConfig: RenovateConfig;
|
|
try {
|
|
renovateConfig = (
|
|
await bitbucketHttp.getJson<RenovateConfig>(
|
|
`/2.0/repositories/${repository}/src/${info.mainbranch}/renovate.json`
|
|
)
|
|
).body;
|
|
} catch {
|
|
// Do nothing
|
|
}
|
|
if (renovateConfig && renovateConfig.enabled === false) {
|
|
throw new Error(REPOSITORY_DISABLED);
|
|
}
|
|
}
|
|
|
|
Object.assign(config, {
|
|
owner: info.owner,
|
|
defaultBranch: info.mainbranch,
|
|
baseBranch: info.mainbranch,
|
|
mergeMethod: info.mergeMethod,
|
|
has_issues: info.has_issues,
|
|
});
|
|
|
|
logger.debug(`${repository} owner = ${config.owner}`);
|
|
} catch (err) /* istanbul ignore next */ {
|
|
if (err.statusCode === 404) {
|
|
throw new Error(REPOSITORY_NOT_FOUND);
|
|
}
|
|
logger.debug({ err }, 'Unknown Bitbucket initRepo error');
|
|
throw err;
|
|
}
|
|
|
|
const { hostname } = URL.parse(endpoint);
|
|
|
|
// Converts API hostnames to their respective HTTP git hosts:
|
|
// `api.bitbucket.org` to `bitbucket.org`
|
|
// `api-staging.<host>` to `staging.<host>`
|
|
const hostnameWithoutApiPrefix = /api[.|-](.+)/.exec(hostname)[1];
|
|
|
|
const url = GitStorage.getUrl({
|
|
protocol: 'https',
|
|
auth: `${opts.username}:${opts.password}`,
|
|
hostname: hostnameWithoutApiPrefix,
|
|
repository,
|
|
});
|
|
|
|
config.storage = new GitStorage();
|
|
await config.storage.initRepo({
|
|
...config,
|
|
localDir,
|
|
url,
|
|
});
|
|
const repoConfig: RepoConfig = {
|
|
baseBranch: config.baseBranch,
|
|
isFork: info.isFork,
|
|
};
|
|
return repoConfig;
|
|
}
|
|
|
|
// Returns true if repository has rule enforcing PRs are up-to-date with base branch before merging
|
|
export function getRepoForceRebase(): Promise<boolean> {
|
|
// BB doesnt have an option to flag staled branches
|
|
return Promise.resolve(false);
|
|
}
|
|
|
|
// Search
|
|
|
|
// Get full file list
|
|
export function getFileList(): Promise<string[]> {
|
|
return config.storage.getFileList();
|
|
}
|
|
|
|
export async function setBaseBranch(
|
|
branchName = config.baseBranch
|
|
): Promise<string> {
|
|
logger.debug(`Setting baseBranch to ${branchName}`);
|
|
config.baseBranch = branchName;
|
|
delete config.baseCommitSHA;
|
|
const baseBranchSha = await config.storage.setBaseBranch(branchName);
|
|
return baseBranchSha;
|
|
}
|
|
|
|
export /* istanbul ignore next */ function setBranchPrefix(
|
|
branchPrefix: string
|
|
): Promise<void> {
|
|
return config.storage.setBranchPrefix(branchPrefix);
|
|
}
|
|
|
|
// Branch
|
|
|
|
// Returns true if branch exists, otherwise false
|
|
export function branchExists(branchName: string): Promise<boolean> {
|
|
return config.storage.branchExists(branchName);
|
|
}
|
|
|
|
export function getAllRenovateBranches(
|
|
branchPrefix: string
|
|
): Promise<string[]> {
|
|
return config.storage.getAllRenovateBranches(branchPrefix);
|
|
}
|
|
|
|
export function isBranchStale(branchName: string): Promise<boolean> {
|
|
return config.storage.isBranchStale(branchName);
|
|
}
|
|
|
|
export function getFile(
|
|
filePath: string,
|
|
branchName?: string
|
|
): Promise<string> {
|
|
return config.storage.getFile(filePath, branchName);
|
|
}
|
|
|
|
// istanbul ignore next
|
|
function matchesState(state: string, desiredState: string): boolean {
|
|
if (desiredState === PR_STATE_ALL) {
|
|
return true;
|
|
}
|
|
if (desiredState.startsWith('!')) {
|
|
return state !== desiredState.substring(1);
|
|
}
|
|
return state === desiredState;
|
|
}
|
|
|
|
export async function getPrList(): Promise<Pr[]> {
|
|
logger.debug('getPrList()');
|
|
if (!config.prList) {
|
|
logger.debug('Retrieving PR list');
|
|
let url = `/2.0/repositories/${config.repository}/pullrequests?`;
|
|
url += utils.prStates.all.map((state) => 'state=' + state).join('&');
|
|
const prs = await utils.accumulateValues(url, undefined, undefined, 50);
|
|
config.prList = prs.map(utils.prInfo);
|
|
logger.debug({ length: config.prList.length }, 'Retrieved Pull Requests');
|
|
}
|
|
return config.prList;
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
export async function getPrFiles(pr: Pr): Promise<string[]> {
|
|
return config.storage.getBranchFiles(pr.branchName, pr.targetBranch);
|
|
}
|
|
|
|
export async function findPr({
|
|
branchName,
|
|
prTitle,
|
|
state = PR_STATE_ALL,
|
|
}: FindPRConfig): Promise<Pr | null> {
|
|
logger.debug(`findPr(${branchName}, ${prTitle}, ${state})`);
|
|
const prList = await getPrList();
|
|
const pr = prList.find(
|
|
(p) =>
|
|
p.branchName === branchName &&
|
|
(!prTitle || p.title === prTitle) &&
|
|
matchesState(p.state, state)
|
|
);
|
|
if (pr) {
|
|
logger.debug(`Found PR #${pr.number}`);
|
|
}
|
|
return pr;
|
|
}
|
|
|
|
export async function deleteBranch(
|
|
branchName: string,
|
|
closePr?: boolean
|
|
): Promise<void> {
|
|
if (closePr) {
|
|
const pr = await findPr({ branchName, state: PR_STATE_OPEN });
|
|
if (pr) {
|
|
await bitbucketHttp.postJson(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${pr.number}/decline`
|
|
);
|
|
}
|
|
}
|
|
return config.storage.deleteBranch(branchName);
|
|
}
|
|
|
|
export function getBranchLastCommitTime(branchName: string): Promise<Date> {
|
|
return config.storage.getBranchLastCommitTime(branchName);
|
|
}
|
|
|
|
// istanbul ignore next
|
|
export function getRepoStatus(): Promise<StatusResult> {
|
|
return config.storage.getRepoStatus();
|
|
}
|
|
|
|
export function mergeBranch(branchName: string): Promise<void> {
|
|
return config.storage.mergeBranch(branchName);
|
|
}
|
|
|
|
export function commitFiles({
|
|
branchName,
|
|
files,
|
|
message,
|
|
}: CommitFilesConfig): Promise<string | null> {
|
|
return config.storage.commitFiles({
|
|
branchName,
|
|
files,
|
|
message,
|
|
});
|
|
}
|
|
|
|
export function getCommitMessages(): Promise<string[]> {
|
|
return config.storage.getCommitMessages();
|
|
}
|
|
|
|
async function isPrConflicted(prNo: number): Promise<boolean> {
|
|
const diff = (
|
|
await bitbucketHttp.get(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${prNo}/diff`,
|
|
{ json: false } as any
|
|
)
|
|
).body;
|
|
|
|
return utils.isConflicted(parseDiff(diff));
|
|
}
|
|
|
|
interface PrResponse {
|
|
id: string;
|
|
state: string;
|
|
links: {
|
|
commits: {
|
|
href: string;
|
|
};
|
|
};
|
|
source: {
|
|
branch: {
|
|
name: string;
|
|
};
|
|
};
|
|
}
|
|
|
|
// Gets details for a PR
|
|
export async function getPr(prNo: number): Promise<Pr | null> {
|
|
const pr = (
|
|
await bitbucketHttp.getJson<PrResponse>(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${prNo}`
|
|
)
|
|
).body;
|
|
|
|
// istanbul ignore if
|
|
if (!pr) {
|
|
return null;
|
|
}
|
|
|
|
const res: any = {
|
|
displayNumber: `Pull Request #${pr.id}`,
|
|
...utils.prInfo(pr),
|
|
isModified: false,
|
|
};
|
|
|
|
if (utils.prStates.open.includes(pr.state)) {
|
|
res.isConflicted = await isPrConflicted(prNo);
|
|
|
|
// TODO: Is that correct? Should we check getBranchStatus like gitlab?
|
|
res.canMerge = !res.isConflicted;
|
|
|
|
// we only want the first two commits, because size tells us the overall number
|
|
const url = pr.links.commits.href + '?pagelen=2';
|
|
const { body } = await bitbucketHttp.getJson<utils.PagedResult<Commit>>(
|
|
url
|
|
);
|
|
const size = body.size || body.values.length;
|
|
|
|
// istanbul ignore if
|
|
if (size === undefined) {
|
|
logger.warn({ prNo, url, body }, 'invalid response so can rebase');
|
|
} else if (size === 1) {
|
|
if (global.gitAuthor) {
|
|
const author = addrs.parseOneAddress(
|
|
body.values[0].author.raw
|
|
) as addrs.ParsedMailbox;
|
|
if (author.address !== global.gitAuthor.email) {
|
|
logger.debug(
|
|
{ prNo },
|
|
'PR is modified: 1 commit but not by configured gitAuthor'
|
|
);
|
|
res.isModified = true;
|
|
}
|
|
}
|
|
} else {
|
|
logger.debug({ prNo }, `PR is modified: Found ${size} commits`);
|
|
res.isModified = true;
|
|
}
|
|
}
|
|
if (await branchExists(pr.source.branch.name)) {
|
|
res.isStale = await isBranchStale(pr.source.branch.name);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
const escapeHash = (input: string): string =>
|
|
input ? input.replace(/#/g, '%23') : input;
|
|
|
|
interface BranchResponse {
|
|
target: {
|
|
hash: string;
|
|
};
|
|
}
|
|
|
|
// Return the commit SHA for a branch
|
|
async function getBranchCommit(branchName: string): Promise<string | null> {
|
|
try {
|
|
const branch = (
|
|
await bitbucketHttp.getJson<BranchResponse>(
|
|
`/2.0/repositories/${config.repository}/refs/branches/${escapeHash(
|
|
branchName
|
|
)}`
|
|
)
|
|
).body;
|
|
return branch.target.hash;
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.debug({ err }, `getBranchCommit('${branchName}') failed'`);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// Returns the Pull Request for a branch. Null if not exists.
|
|
export async function getBranchPr(branchName: string): Promise<Pr | null> {
|
|
logger.debug(`getBranchPr(${branchName})`);
|
|
const existingPr = await findPr({
|
|
branchName,
|
|
state: PR_STATE_OPEN,
|
|
});
|
|
return existingPr ? getPr(existingPr.number) : null;
|
|
}
|
|
|
|
async function getStatus(
|
|
branchName: string,
|
|
useCache = true
|
|
): Promise<utils.BitbucketStatus[]> {
|
|
const sha = await getBranchCommit(branchName);
|
|
return utils.accumulateValues<utils.BitbucketStatus>(
|
|
`/2.0/repositories/${config.repository}/commit/${sha}/statuses`,
|
|
'get',
|
|
{ useCache }
|
|
);
|
|
}
|
|
// Returns the combined status for a branch.
|
|
export async function getBranchStatus(
|
|
branchName: string,
|
|
requiredStatusChecks?: string[]
|
|
): Promise<BranchStatus> {
|
|
logger.debug(`getBranchStatus(${branchName})`);
|
|
if (!requiredStatusChecks) {
|
|
// null means disable status checks, so it always succeeds
|
|
logger.debug('Status checks disabled = returning "success"');
|
|
return BranchStatus.green;
|
|
}
|
|
if (requiredStatusChecks.length) {
|
|
// This is Unsupported
|
|
logger.warn({ requiredStatusChecks }, `Unsupported requiredStatusChecks`);
|
|
return BranchStatus.red;
|
|
}
|
|
const statuses = await getStatus(branchName);
|
|
logger.debug({ branch: branchName, statuses }, 'branch status check result');
|
|
if (!statuses.length) {
|
|
logger.debug('empty branch status check result = returning "pending"');
|
|
return BranchStatus.yellow;
|
|
}
|
|
const noOfFailures = statuses.filter(
|
|
(status: { state: string }) =>
|
|
status.state === 'FAILED' || status.state === 'STOPPED'
|
|
).length;
|
|
if (noOfFailures) {
|
|
return BranchStatus.red;
|
|
}
|
|
const noOfPending = statuses.filter(
|
|
(status: { state: string }) => status.state === 'INPROGRESS'
|
|
).length;
|
|
if (noOfPending) {
|
|
return BranchStatus.yellow;
|
|
}
|
|
return BranchStatus.green;
|
|
}
|
|
|
|
const bbToRenovateStatusMapping: Record<string, BranchStatus> = {
|
|
SUCCESSFUL: BranchStatus.green,
|
|
INPROGRESS: BranchStatus.yellow,
|
|
FAILED: BranchStatus.red,
|
|
};
|
|
|
|
export async function getBranchStatusCheck(
|
|
branchName: string,
|
|
context: string
|
|
): Promise<BranchStatus | null> {
|
|
const statuses = await getStatus(branchName);
|
|
const bbState = (statuses.find((status) => status.key === context) || {})
|
|
.state;
|
|
return bbToRenovateStatusMapping[bbState] || null;
|
|
}
|
|
|
|
export async function setBranchStatus({
|
|
branchName,
|
|
context,
|
|
description,
|
|
state,
|
|
url: targetUrl,
|
|
}: BranchStatusConfig): Promise<void> {
|
|
const sha = await getBranchCommit(branchName);
|
|
|
|
// TargetUrl can not be empty so default to bitbucket
|
|
const url = targetUrl || /* istanbul ignore next */ 'http://bitbucket.org';
|
|
|
|
const body = {
|
|
name: context,
|
|
state: utils.buildStates[state],
|
|
key: context,
|
|
description,
|
|
url,
|
|
};
|
|
|
|
await bitbucketHttp.postJson(
|
|
`/2.0/repositories/${config.repository}/commit/${sha}/statuses/build`,
|
|
{ body }
|
|
);
|
|
// update status cache
|
|
await getStatus(branchName, false);
|
|
}
|
|
|
|
type BbIssue = { id: number; content?: { raw: string } };
|
|
|
|
async function findOpenIssues(title: string): Promise<BbIssue[]> {
|
|
try {
|
|
const filter = encodeURIComponent(
|
|
[
|
|
`title=${JSON.stringify(title)}`,
|
|
'(state = "new" OR state = "open")',
|
|
`reporter.username="${config.username}"`,
|
|
].join(' AND ')
|
|
);
|
|
return (
|
|
(
|
|
await bitbucketHttp.getJson<{ values: BbIssue[] }>(
|
|
`/2.0/repositories/${config.repository}/issues?q=${filter}`
|
|
)
|
|
).body.values || /* istanbul ignore next */ []
|
|
);
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.warn({ err }, 'Error finding issues');
|
|
return [];
|
|
}
|
|
}
|
|
|
|
export async function findIssue(title: string): Promise<Issue> {
|
|
logger.debug(`findIssue(${title})`);
|
|
|
|
/* istanbul ignore if */
|
|
if (!config.has_issues) {
|
|
logger.debug('Issues are disabled - cannot findIssue');
|
|
return null;
|
|
}
|
|
const issues = await findOpenIssues(title);
|
|
if (!issues.length) {
|
|
return null;
|
|
}
|
|
const [issue] = issues;
|
|
return {
|
|
number: issue.id,
|
|
body: issue.content && issue.content.raw,
|
|
};
|
|
}
|
|
|
|
async function closeIssue(issueNumber: number): Promise<void> {
|
|
await bitbucketHttp.putJson(
|
|
`/2.0/repositories/${config.repository}/issues/${issueNumber}`,
|
|
{
|
|
body: { state: 'closed' },
|
|
}
|
|
);
|
|
}
|
|
|
|
export function getPrBody(input: string): string {
|
|
// Remove any HTML we use
|
|
return smartTruncate(input, 50000)
|
|
.replace(
|
|
'you tick the rebase/retry checkbox',
|
|
'rename PR to start with "rebase!"'
|
|
)
|
|
.replace(/<\/?summary>/g, '**')
|
|
.replace(/<\/?details>/g, '')
|
|
.replace(new RegExp(`\n---\n\n.*?<!-- rebase-check -->.*?\n`), '')
|
|
.replace(/\]\(\.\.\/pull\//g, '](../../pull-requests/');
|
|
}
|
|
|
|
export async function ensureIssue({
|
|
title,
|
|
body,
|
|
}: EnsureIssueConfig): Promise<EnsureIssueResult | null> {
|
|
logger.debug(`ensureIssue()`);
|
|
const description = getPrBody(sanitize(body));
|
|
|
|
/* istanbul ignore if */
|
|
if (!config.has_issues) {
|
|
logger.warn('Issues are disabled - cannot ensureIssue');
|
|
logger.debug({ title }, 'Failed to ensure Issue');
|
|
return null;
|
|
}
|
|
try {
|
|
const issues = await findOpenIssues(title);
|
|
if (issues.length) {
|
|
// Close any duplicates
|
|
for (const issue of issues.slice(1)) {
|
|
await closeIssue(issue.id);
|
|
}
|
|
const [issue] = issues;
|
|
if (String(issue.content.raw).trim() !== description.trim()) {
|
|
logger.debug('Issue updated');
|
|
await bitbucketHttp.putJson(
|
|
`/2.0/repositories/${config.repository}/issues/${issue.id}`,
|
|
{
|
|
body: {
|
|
content: {
|
|
raw: readOnlyIssueBody(description),
|
|
markup: 'markdown',
|
|
},
|
|
},
|
|
}
|
|
);
|
|
return 'updated';
|
|
}
|
|
} else {
|
|
logger.info('Issue created');
|
|
await bitbucketHttp.postJson(
|
|
`/2.0/repositories/${config.repository}/issues`,
|
|
{
|
|
body: {
|
|
title,
|
|
content: {
|
|
raw: readOnlyIssueBody(description),
|
|
markup: 'markdown',
|
|
},
|
|
},
|
|
}
|
|
);
|
|
return 'created';
|
|
}
|
|
} catch (err) /* istanbul ignore next */ {
|
|
if (err.message.startsWith('Repository has no issue tracker.')) {
|
|
logger.debug(
|
|
`Issues are disabled, so could not create issue: ${err.message}`
|
|
);
|
|
} else {
|
|
logger.warn({ err }, 'Could not ensure issue');
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
export /* istanbul ignore next */ async function getIssueList(): Promise<
|
|
Issue[]
|
|
> {
|
|
logger.debug(`getIssueList()`);
|
|
|
|
/* istanbul ignore if */
|
|
if (!config.has_issues) {
|
|
logger.debug('Issues are disabled - cannot getIssueList');
|
|
return [];
|
|
}
|
|
try {
|
|
const filter = encodeURIComponent(
|
|
[
|
|
'(state = "new" OR state = "open")',
|
|
`reporter.username="${config.username}"`,
|
|
].join(' AND ')
|
|
);
|
|
return (
|
|
(
|
|
await bitbucketHttp.getJson<{ values: Issue[] }>(
|
|
`/2.0/repositories/${config.repository}/issues?q=${filter}`
|
|
)
|
|
).body.values || /* istanbul ignore next */ []
|
|
);
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.warn({ err }, 'Error finding issues');
|
|
return [];
|
|
}
|
|
}
|
|
|
|
export async function ensureIssueClosing(title: string): Promise<void> {
|
|
/* istanbul ignore if */
|
|
if (!config.has_issues) {
|
|
logger.debug('Issues are disabled - cannot ensureIssueClosing');
|
|
return;
|
|
}
|
|
const issues = await findOpenIssues(title);
|
|
for (const issue of issues) {
|
|
await closeIssue(issue.id);
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
export function addAssignees(
|
|
_prNr: number,
|
|
_assignees: string[]
|
|
): Promise<void> {
|
|
// Bitbucket supports "participants" and "reviewers" so does not seem to have the concept of "assignee"
|
|
logger.warn('Cannot add assignees');
|
|
return Promise.resolve();
|
|
}
|
|
|
|
export async function addReviewers(
|
|
prId: number,
|
|
reviewers: string[]
|
|
): Promise<void> {
|
|
logger.debug(`Adding reviewers ${reviewers} to #${prId}`);
|
|
|
|
const { title } = await getPr(prId);
|
|
|
|
const body = {
|
|
title,
|
|
reviewers: reviewers.map((username: string) => ({ username })),
|
|
};
|
|
|
|
await bitbucketHttp.putJson(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${prId}`,
|
|
{
|
|
body,
|
|
}
|
|
);
|
|
}
|
|
|
|
export /* istanbul ignore next */ function deleteLabel(): never {
|
|
throw new Error('deleteLabel not implemented');
|
|
}
|
|
|
|
export function ensureComment({
|
|
number,
|
|
topic,
|
|
content,
|
|
}: EnsureCommentConfig): Promise<boolean> {
|
|
// https://developer.atlassian.com/bitbucket/api/2/reference/search?q=pullrequest+comment
|
|
return comments.ensureComment({
|
|
config,
|
|
number,
|
|
topic,
|
|
content: sanitize(content),
|
|
});
|
|
}
|
|
|
|
export function ensureCommentRemoval({
|
|
number: prNo,
|
|
topic,
|
|
content,
|
|
}: EnsureCommentRemovalConfig): Promise<void> {
|
|
return comments.ensureCommentRemoval(config, prNo, topic, content);
|
|
}
|
|
|
|
// Creates PR and returns PR number
|
|
export async function createPr({
|
|
branchName,
|
|
prTitle: title,
|
|
prBody: description,
|
|
useDefaultBranch = true,
|
|
}: CreatePRConfig): Promise<Pr> {
|
|
// labels is not supported in Bitbucket: https://bitbucket.org/site/master/issues/11976/ability-to-add-labels-to-pull-requests-bb
|
|
|
|
const base = useDefaultBranch
|
|
? config.defaultBranch
|
|
: /* istanbul ignore next */ config.baseBranch;
|
|
|
|
logger.debug({ repository: config.repository, title, base }, 'Creating PR');
|
|
|
|
let reviewers: { uuid: { raw: string } }[] = [];
|
|
|
|
if (config.bbUseDefaultReviewers) {
|
|
const reviewersResponse = (
|
|
await bitbucketHttp.getJson<utils.PagedResult<Reviewer>>(
|
|
`/2.0/repositories/${config.repository}/default-reviewers`
|
|
)
|
|
).body;
|
|
reviewers = reviewersResponse.values.map((reviewer: Reviewer) => ({
|
|
uuid: reviewer.uuid,
|
|
}));
|
|
}
|
|
|
|
const body = {
|
|
title,
|
|
description: sanitize(description),
|
|
source: {
|
|
branch: {
|
|
name: branchName,
|
|
},
|
|
},
|
|
destination: {
|
|
branch: {
|
|
name: base,
|
|
},
|
|
},
|
|
close_source_branch: true,
|
|
reviewers,
|
|
};
|
|
|
|
try {
|
|
const prInfo = (
|
|
await bitbucketHttp.postJson<PrResponse>(
|
|
`/2.0/repositories/${config.repository}/pullrequests`,
|
|
{
|
|
body,
|
|
}
|
|
)
|
|
).body;
|
|
// TODO: fix types
|
|
const pr: Pr = {
|
|
number: prInfo.id,
|
|
displayNumber: `Pull Request #${prInfo.id}`,
|
|
isModified: false,
|
|
} as any;
|
|
// istanbul ignore if
|
|
if (config.prList) {
|
|
config.prList.push(pr);
|
|
}
|
|
return pr;
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.warn({ err }, 'Error creating pull request');
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
interface Reviewer {
|
|
uuid: { raw: string };
|
|
}
|
|
|
|
interface Commit {
|
|
author: { raw: string };
|
|
}
|
|
|
|
export async function updatePr(
|
|
prNo: number,
|
|
title: string,
|
|
description: string
|
|
): Promise<void> {
|
|
logger.debug(`updatePr(${prNo}, ${title}, body)`);
|
|
await bitbucketHttp.putJson(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${prNo}`,
|
|
{
|
|
body: { title, description: sanitize(description) },
|
|
}
|
|
);
|
|
}
|
|
|
|
export async function mergePr(
|
|
prNo: number,
|
|
branchName: string
|
|
): Promise<boolean> {
|
|
logger.debug(`mergePr(${prNo}, ${branchName})`);
|
|
|
|
try {
|
|
await bitbucketHttp.postJson(
|
|
`/2.0/repositories/${config.repository}/pullrequests/${prNo}/merge`,
|
|
{
|
|
body: {
|
|
close_source_branch: true,
|
|
merge_strategy: 'merge_commit',
|
|
message: 'auto merged',
|
|
},
|
|
}
|
|
);
|
|
delete config.baseCommitSHA;
|
|
logger.debug('Automerging succeeded');
|
|
} catch (err) /* istanbul ignore next */ {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// Pull Request
|
|
|
|
export function cleanRepo(): Promise<void> {
|
|
// istanbul ignore if
|
|
if (config.storage && config.storage.cleanRepo) {
|
|
config.storage.cleanRepo();
|
|
}
|
|
config = {} as any;
|
|
return Promise.resolve();
|
|
}
|
|
|
|
export function getVulnerabilityAlerts(): Promise<VulnerabilityAlert[]> {
|
|
return Promise.resolve([]);
|
|
}
|