mirror of https://github.com/renovatebot/renovate
172 lines
4.4 KiB
TypeScript
172 lines
4.4 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import type { MergeStrategy } from '../../../config/types';
|
|
import { CONFIG_GIT_URL_UNAVAILABLE } from '../../../constants/error-messages';
|
|
import { logger } from '../../../logger';
|
|
import * as hostRules from '../../../util/host-rules';
|
|
import { regEx } from '../../../util/regex';
|
|
import { parseUrl } from '../../../util/url';
|
|
import { getPrBodyStruct } from '../pr-body';
|
|
import type { GitUrlOption, Pr } from '../types';
|
|
import type { PR, PRMergeMethod, Repo } from './types';
|
|
|
|
export function smartLinks(body: string): string {
|
|
return body?.replace(regEx(/\]\(\.\.\/pull\//g), '](pulls/');
|
|
}
|
|
|
|
export function trimTrailingApiPath(url: string): string {
|
|
return url?.replace(regEx(/api\/v1\/?$/g), '');
|
|
}
|
|
|
|
export function getRepoUrl(
|
|
repo: Repo,
|
|
gitUrl: GitUrlOption | undefined,
|
|
endpoint: string,
|
|
): string {
|
|
if (gitUrl === 'ssh') {
|
|
if (!repo.ssh_url) {
|
|
throw new Error(CONFIG_GIT_URL_UNAVAILABLE);
|
|
}
|
|
logger.debug(`Using SSH URL: ${repo.ssh_url}`);
|
|
return repo.ssh_url;
|
|
}
|
|
|
|
// Find options for current host and determine Git endpoint
|
|
const opts = hostRules.find({
|
|
hostType: 'gitea',
|
|
url: endpoint,
|
|
});
|
|
|
|
if (gitUrl === 'endpoint') {
|
|
const url = parseUrl(endpoint);
|
|
if (!url) {
|
|
throw new Error(CONFIG_GIT_URL_UNAVAILABLE);
|
|
}
|
|
url.username = opts.token ?? '';
|
|
url.pathname = `${url.pathname}${repo.full_name}.git`;
|
|
logger.debug(
|
|
{ url: url.toString() },
|
|
'using URL based on configured endpoint',
|
|
);
|
|
return url.toString();
|
|
}
|
|
|
|
if (!repo.clone_url) {
|
|
throw new Error(CONFIG_GIT_URL_UNAVAILABLE);
|
|
}
|
|
|
|
logger.debug(`Using HTTP URL: ${repo.clone_url}`);
|
|
const repoUrl = parseUrl(repo.clone_url);
|
|
if (!repoUrl) {
|
|
throw new Error(CONFIG_GIT_URL_UNAVAILABLE);
|
|
}
|
|
repoUrl.username = opts.token ?? '';
|
|
return repoUrl.toString();
|
|
}
|
|
|
|
export function getMergeMethod(
|
|
strategy: MergeStrategy | undefined,
|
|
): PRMergeMethod | null {
|
|
switch (strategy) {
|
|
case 'fast-forward':
|
|
return 'rebase';
|
|
case 'merge-commit':
|
|
return 'merge';
|
|
case 'rebase':
|
|
return 'rebase-merge';
|
|
case 'squash':
|
|
return strategy;
|
|
case 'auto':
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
|
|
export const API_PATH = '/api/v1';
|
|
|
|
export const DRAFT_PREFIX = 'WIP: ';
|
|
const reconfigurePrRegex = regEx(/reconfigure$/g);
|
|
|
|
export function toRenovatePR(data: PR, author: string | null): Pr | null {
|
|
if (!data) {
|
|
return null;
|
|
}
|
|
|
|
if (
|
|
!data.base?.ref ||
|
|
!data.head?.label ||
|
|
!data.head?.sha ||
|
|
!data.head?.repo?.full_name
|
|
) {
|
|
logger.trace(
|
|
`Skipping Pull Request #${data.number} due to missing base and/or head branch`,
|
|
);
|
|
return null;
|
|
}
|
|
|
|
const createdBy = data.user?.username;
|
|
if (
|
|
createdBy &&
|
|
author &&
|
|
!reconfigurePrRegex.test(data.head.label) &&
|
|
createdBy !== author
|
|
) {
|
|
return null;
|
|
}
|
|
|
|
let title = data.title;
|
|
let isDraft = false;
|
|
if (title.startsWith(DRAFT_PREFIX)) {
|
|
title = title.substring(DRAFT_PREFIX.length);
|
|
isDraft = true;
|
|
}
|
|
const labels = (data?.labels ?? []).map((l) => l.name);
|
|
|
|
return {
|
|
labels,
|
|
number: data.number,
|
|
state: data.merged ? 'merged' : data.state,
|
|
title,
|
|
isDraft,
|
|
bodyStruct: getPrBodyStruct(data.body),
|
|
sha: data.head.sha,
|
|
sourceBranch: data.head.label,
|
|
targetBranch: data.base.ref,
|
|
sourceRepo: data.head.repo.full_name,
|
|
createdAt: data.created_at,
|
|
cannotMergeReason: data.mergeable
|
|
? undefined
|
|
: `pr.mergeable="${data.mergeable}"`,
|
|
hasAssignees: !!(data.assignee?.login ?? is.nonEmptyArray(data.assignees)),
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Check if a repository is usable.
|
|
* A repo isn't usable if one of the following conditions is met:
|
|
* - The repo is a `mirror`
|
|
* - We don't have pull or push permissions
|
|
* - Pull requests are disabled
|
|
* @param repo Repo to check
|
|
* @returns `true` if the repository is usable, `false` otherwise
|
|
*/
|
|
export function usableRepo(repo: Repo): boolean {
|
|
if (repo.mirror === true) {
|
|
return false;
|
|
}
|
|
|
|
if (repo.permissions.pull === false || repo.permissions.push === false) {
|
|
logger.debug(
|
|
`Skipping repository ${repo.full_name} because of missing pull or push permissions`,
|
|
);
|
|
return false;
|
|
}
|
|
|
|
if (repo.has_pull_requests === false) {
|
|
logger.debug(
|
|
`Skipping repository ${repo.full_name} because pull requests are disabled`,
|
|
);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|