0
0
Fork 0
mirror of https://github.com/renovatebot/renovate.git synced 2025-05-12 23:51:55 +00:00
renovatebot_renovate/lib/modules/platform/gitea/pr-cache.ts
Michael Kriese c7737befb7
feat(gitea/forgejo): set poster server filter for pr cache (#34374)
Co-authored-by: Sebastian Poxhofer <secustor@users.noreply.github.com>
2025-02-21 13:47:43 +00:00

177 lines
5.1 KiB
TypeScript

import { dequal } from 'dequal';
import { DateTime } from 'luxon';
import { TEMPORARY_ERROR } from '../../../constants/error-messages';
import { logger } from '../../../logger';
import * as memCache from '../../../util/cache/memory';
import { getCache } from '../../../util/cache/repository';
import type { GiteaHttp } from '../../../util/http/gitea';
import type { HttpResponse } from '../../../util/http/types';
import { getQueryString, parseLinkHeader, parseUrl } from '../../../util/url';
import type { Pr } from '../types';
import type { GiteaPrCacheData, PR } from './types';
import { API_PATH, toRenovatePR } from './utils';
export class GiteaPrCache {
private cache: GiteaPrCacheData;
private items: Pr[] = [];
private constructor(
private repo: string,
private readonly ignorePrAuthor: boolean,
private author: string | null,
) {
const repoCache = getCache();
repoCache.platform ??= {};
repoCache.platform.gitea ??= {};
let pullRequestCache = repoCache.platform.gitea.pullRequestsCache as
| GiteaPrCacheData
| undefined;
if (!pullRequestCache || pullRequestCache.author !== author) {
pullRequestCache = {
items: {},
updated_at: null,
author,
};
}
repoCache.platform.gitea.pullRequestsCache = pullRequestCache;
this.cache = pullRequestCache;
this.updateItems();
}
static forceSync(): void {
memCache.set('gitea-pr-cache-synced', false);
}
private static async init(
http: GiteaHttp,
repo: string,
ignorePrAuthor: boolean,
author: string | null,
): Promise<GiteaPrCache> {
const res = new GiteaPrCache(repo, ignorePrAuthor, author);
const isSynced = memCache.get<true | undefined>('gitea-pr-cache-synced');
if (!isSynced) {
await res.sync(http);
memCache.set('gitea-pr-cache-synced', true);
}
return res;
}
private getPrs(): Pr[] {
return this.items;
}
static async getPrs(
http: GiteaHttp,
repo: string,
ignorePrAuthor: boolean,
author: string,
): Promise<Pr[]> {
const prCache = await GiteaPrCache.init(http, repo, ignorePrAuthor, author);
return prCache.getPrs();
}
private setPr(item: Pr): void {
this.cache.items[item.number] = item;
this.updateItems();
}
static async setPr(
http: GiteaHttp,
repo: string,
ignorePrAuthor: boolean,
author: string,
item: Pr,
): Promise<void> {
const prCache = await GiteaPrCache.init(http, repo, ignorePrAuthor, author);
prCache.setPr(item);
}
private reconcile(rawItems: (PR | null)[]): boolean {
const { items } = this.cache;
let { updated_at } = this.cache;
const cacheTime = updated_at ? DateTime.fromISO(updated_at) : null;
let needNextPage = true;
for (const rawItem of rawItems) {
if (!rawItem) {
logger.warn('Gitea PR is empty, throwing temporary error');
// Gitea API sometimes returns empty PRs, so we throw a temporary error
// https://github.com/go-gitea/gitea/blob/fcd096231ac2deaefbca10a7db1b9b01f1da93d7/services/convert/pull.go#L34-L52
throw new Error(TEMPORARY_ERROR);
}
const id = rawItem.number;
const newItem = toRenovatePR(rawItem, this.author);
if (!newItem) {
continue;
}
const oldItem = items[id];
if (dequal(oldItem, newItem)) {
needNextPage = false;
continue;
}
items[id] = newItem;
const itemTime = DateTime.fromISO(rawItem.updated_at);
if (!cacheTime || itemTime > cacheTime) {
updated_at = rawItem.updated_at;
}
}
this.cache.updated_at = updated_at;
return needNextPage;
}
private async sync(http: GiteaHttp): Promise<GiteaPrCache> {
let query: string | null = getQueryString({
state: 'all',
sort: 'recentupdate',
// Fetch 100 PRs on the first run to ensure we have the most recent PRs.
// Gitea / Forgejo will cap appropriate (50 by default, see `MAX_RESPONSE_ITEMS`).
// https://docs.gitea.com/administration/config-cheat-sheet#api-api
// https://forgejo.org/docs/latest/admin/config-cheat-sheet/#api-api
limit: this.items.length ? 20 : 100,
// Supported since Gitea 1.23.0 and Forgejo v10.0.0.
// Will be ignoded by older instances.
...(this.ignorePrAuthor ? {} : { poster: this.author }),
});
while (query) {
// TODO: use zod, typescript can't infer the type of the response #22198
const res: HttpResponse<(PR | null)[]> = await http.getJsonUnchecked(
`${API_PATH}/repos/${this.repo}/pulls?${query}`,
{
memCache: false,
paginate: false,
},
);
const needNextPage = this.reconcile(res.body);
if (!needNextPage) {
break;
}
const uri = parseUrl(parseLinkHeader(res.headers.link)?.next?.url);
query = uri ? uri.search : null;
}
this.updateItems();
return this;
}
/**
* Ensure the pr cache starts with the most recent PRs.
* JavaScript ensures that the cache is sorted by PR number.
*/
private updateItems(): void {
this.items = Object.values(this.cache.items).reverse();
}
}