mirror of
https://github.com/renovatebot/renovate.git
synced 2025-05-12 23:51:55 +00:00
199 lines
5.2 KiB
TypeScript
199 lines
5.2 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import { logger } from '../../logger';
|
|
import type { BranchConfig, BranchUpgradeConfig } from '../types';
|
|
|
|
export type Limit = 'Commits';
|
|
interface LimitValue {
|
|
max: number | null;
|
|
current: number;
|
|
}
|
|
|
|
const limits = new Map<Limit, LimitValue>();
|
|
|
|
export function resetAllLimits(): void {
|
|
limits.clear();
|
|
}
|
|
|
|
export function setMaxLimit(key: Limit, val: unknown): void {
|
|
const max = typeof val === 'number' ? Math.max(0, val) : null;
|
|
limits.set(key, { current: 0, max });
|
|
logger.debug(`${key} limit = ${max!}`);
|
|
}
|
|
|
|
export function incLimitedValue(key: Limit, incBy = 1): void {
|
|
const limit = limits.get(key) ?? { max: null, current: 0 };
|
|
limits.set(key, {
|
|
...limit,
|
|
current: limit.current + incBy,
|
|
});
|
|
}
|
|
|
|
function handleCommitsLimit(): boolean {
|
|
const limit = limits.get('Commits');
|
|
// TODO: fix me?
|
|
// eslint-disable-next-line @typescript-eslint/prefer-optional-chain
|
|
if (!limit || limit.max === null) {
|
|
return false;
|
|
}
|
|
const { max, current } = limit;
|
|
return max - current <= 0;
|
|
}
|
|
|
|
export type CountName = 'ConcurrentPRs' | 'HourlyPRs' | 'Branches';
|
|
|
|
type BranchLimitName =
|
|
| 'branchConcurrentLimit'
|
|
| 'prConcurrentLimit'
|
|
| 'prHourlyLimit';
|
|
|
|
export const counts = new Map<CountName, number>();
|
|
|
|
export function getCount(key: CountName): number {
|
|
const count = counts.get(key);
|
|
// istanbul ignore if: should not happen
|
|
if (!is.integer(count)) {
|
|
logger.debug(`Could not compute the count of ${key}, returning zero.`);
|
|
return 0;
|
|
}
|
|
return count;
|
|
}
|
|
|
|
export function setCount(key: CountName, val: number): void {
|
|
counts.set(key, val);
|
|
logger.debug(`${key} count = ${val}`);
|
|
}
|
|
|
|
export function incCountValue(key: CountName, incBy = 1): void {
|
|
const count = getCount(key);
|
|
counts.set(key, count + incBy);
|
|
}
|
|
|
|
function handleConcurrentLimits(
|
|
key: Exclude<CountName, 'HourlyPRs'>,
|
|
config: BranchConfig,
|
|
): boolean {
|
|
const limitKey =
|
|
key === 'Branches' ? 'branchConcurrentLimit' : 'prConcurrentLimit';
|
|
|
|
// calculate the limits for this branch
|
|
const hourlyLimit = calcLimit(config.upgrades, 'prHourlyLimit');
|
|
const hourlyPrCount = getCount('HourlyPRs');
|
|
|
|
// if a limit is defined ( >0 ) and limit reached return true ie. limit has been reached
|
|
if (hourlyLimit && hourlyPrCount >= hourlyLimit) {
|
|
return true;
|
|
}
|
|
|
|
const limitValue = calcLimit(config.upgrades, limitKey);
|
|
const currentCount = getCount(key);
|
|
|
|
if (limitValue && currentCount >= limitValue) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
export function calcLimit(
|
|
upgrades: BranchUpgradeConfig[],
|
|
limitName: BranchLimitName,
|
|
): number {
|
|
logger.debug(
|
|
{
|
|
limits: upgrades.map((upg) => {
|
|
return { depName: upg.depName, [limitName]: upg[limitName] };
|
|
}),
|
|
},
|
|
`${limitName} of the upgrades present in this branch`,
|
|
);
|
|
|
|
if (hasMultipleLimits(upgrades, limitName)) {
|
|
logger.once.debug(
|
|
`Branch has multiple ${limitName} limits. The lowest among these will be selected.`,
|
|
);
|
|
}
|
|
|
|
let lowestLimit = Number.MAX_SAFE_INTEGER;
|
|
for (const upgrade of upgrades) {
|
|
let limit = upgrade[limitName];
|
|
|
|
// inherit prConcurrentLimit value incase branchConcurrentLimit is null
|
|
if (!is.number(limit) && limitName === 'branchConcurrentLimit') {
|
|
limit = upgrade.prConcurrentLimit;
|
|
}
|
|
|
|
// istanbul ignore if: should never happen as all limits get a default value
|
|
if (is.undefined(limit)) {
|
|
limit = Number.MAX_SAFE_INTEGER;
|
|
}
|
|
|
|
// no limit
|
|
if (limit === 0 || limit === null) {
|
|
logger.debug(
|
|
`${limitName} of this branch is unlimited, because atleast one of the upgrade has it's ${limitName} set to "No limit" ie. 0 or null`,
|
|
);
|
|
return 0;
|
|
}
|
|
|
|
// limit is set
|
|
lowestLimit = limit < lowestLimit ? limit : lowestLimit;
|
|
}
|
|
|
|
logger.debug(
|
|
`Calculated lowest ${limitName} among the upgrades present in this branch is ${lowestLimit}.`,
|
|
);
|
|
return lowestLimit;
|
|
}
|
|
|
|
export function hasMultipleLimits(
|
|
upgrades: BranchUpgradeConfig[],
|
|
limitName: BranchLimitName,
|
|
): boolean {
|
|
if (upgrades.length === 1) {
|
|
return false;
|
|
}
|
|
|
|
const distinctLimits = new Set<number>();
|
|
for (const upgrade of upgrades) {
|
|
let limitValue = upgrade[limitName];
|
|
|
|
// inherit prConcurrentLimit value incase branchConcurrentLimit is null
|
|
if (limitName === 'branchConcurrentLimit' && !is.number(limitValue)) {
|
|
limitValue = upgrade.prConcurrentLimit;
|
|
}
|
|
|
|
// istanbul ignore if: should not happen as the limits are of type number
|
|
if (limitValue === null) {
|
|
limitValue = 0;
|
|
}
|
|
|
|
if (!is.undefined(limitValue) && !distinctLimits.has(limitValue)) {
|
|
distinctLimits.add(limitValue);
|
|
}
|
|
}
|
|
|
|
return distinctLimits.size > 1;
|
|
}
|
|
|
|
export function isLimitReached(limit: 'Commits'): boolean;
|
|
export function isLimitReached(
|
|
limit: 'Branches' | 'ConcurrentPRs',
|
|
config: BranchConfig,
|
|
): boolean;
|
|
export function isLimitReached(
|
|
limit: 'Commits' | 'Branches' | 'ConcurrentPRs',
|
|
config?: BranchConfig,
|
|
): boolean {
|
|
if (limit === 'Commits') {
|
|
return handleCommitsLimit();
|
|
}
|
|
|
|
if (config) {
|
|
return handleConcurrentLimits(limit, config);
|
|
}
|
|
|
|
// istanbul ignore next: should not happen
|
|
throw new Error(
|
|
'Config is required for computing limits for Branches and PullRequests',
|
|
);
|
|
}
|