mirror of https://github.com/renovatebot/renovate
168 lines
4.7 KiB
TypeScript
168 lines
4.7 KiB
TypeScript
import { logger } from '../../../logger';
|
|
import { cache } from '../../../util/cache/package/decorator';
|
|
import type { GithubRestRelease } from '../../../util/github/types';
|
|
import { getApiBaseUrl } from '../../../util/github/url';
|
|
import { GithubHttp } from '../../../util/http/github';
|
|
import { regEx } from '../../../util/regex';
|
|
import { streamToString } from '../../../util/streams';
|
|
import { coerceString } from '../../../util/string';
|
|
import { parseUrl } from '../../../util/url';
|
|
import { id } from '../../versioning/hermit';
|
|
import { Datasource } from '../datasource';
|
|
import type { GetReleasesConfig, ReleaseResult } from '../types';
|
|
import type { HermitSearchResult } from './types';
|
|
|
|
/**
|
|
* Hermit Datasource searches a given package from the specified `hermit-packages`
|
|
* repository. It expects the search manifest to come from an asset `index.json` from
|
|
* a release named index.
|
|
*/
|
|
export class HermitDatasource extends Datasource {
|
|
static readonly id = 'hermit';
|
|
|
|
override readonly customRegistrySupport = true;
|
|
|
|
override readonly registryStrategy = 'first';
|
|
|
|
override readonly defaultVersioning = id;
|
|
|
|
override readonly defaultRegistryUrls = [
|
|
'https://github.com/cashapp/hermit-packages',
|
|
];
|
|
|
|
override readonly sourceUrlSupport = 'release';
|
|
override readonly sourceUrlNote =
|
|
'The source URL is determined from the `Repository` field in the results.';
|
|
|
|
pathRegex: RegExp;
|
|
|
|
constructor() {
|
|
super(HermitDatasource.id);
|
|
this.http = new GithubHttp(id);
|
|
this.pathRegex = regEx('^/(?<owner>[^/]+)/(?<repo>[^/]+)$');
|
|
}
|
|
|
|
@cache({
|
|
namespace: `datasource-${HermitDatasource.id}`,
|
|
key: ({ registryUrl, packageName }: GetReleasesConfig) =>
|
|
`getReleases:${registryUrl ?? ''}-${packageName}`,
|
|
})
|
|
async getReleases({
|
|
packageName,
|
|
registryUrl,
|
|
}: GetReleasesConfig): Promise<ReleaseResult | null> {
|
|
logger.trace(`HermitDataSource.getReleases()`);
|
|
|
|
if (!registryUrl) {
|
|
logger.error('registryUrl must be supplied');
|
|
return null;
|
|
}
|
|
|
|
const parsedUrl = parseUrl(registryUrl);
|
|
if (parsedUrl === null) {
|
|
logger.warn({ registryUrl }, 'invalid registryUrl given');
|
|
return null;
|
|
}
|
|
|
|
if (!registryUrl.startsWith('https://github.com/')) {
|
|
logger.warn({ registryUrl }, 'Only Github registryUrl is supported');
|
|
return null;
|
|
}
|
|
|
|
const items = await this.getHermitSearchManifest(parsedUrl);
|
|
|
|
if (items === null) {
|
|
return null;
|
|
}
|
|
|
|
const res = items.find((i) => i.Name === packageName);
|
|
|
|
if (!res) {
|
|
logger.debug(
|
|
`Could not find hermit package ${packageName} at URL ${registryUrl}`,
|
|
);
|
|
return null;
|
|
}
|
|
|
|
const sourceUrl = res.Repository;
|
|
|
|
return {
|
|
sourceUrl,
|
|
releases: [
|
|
...res.Versions.map((v) => ({
|
|
version: v,
|
|
sourceUrl,
|
|
})),
|
|
...res.Channels.map((v) => ({
|
|
version: v,
|
|
sourceUrl,
|
|
})),
|
|
],
|
|
};
|
|
}
|
|
|
|
/**
|
|
* getHermitSearchManifest fetch the index.json from release
|
|
* named index, parses it and returned the parsed JSON result
|
|
*/
|
|
@cache({
|
|
namespace: `datasource-${HermitDatasource.id}`,
|
|
key: (u) => `getHermitSearchManifest:${u.toString()}`,
|
|
})
|
|
async getHermitSearchManifest(u: URL): Promise<HermitSearchResult[] | null> {
|
|
const registryUrl = u.toString();
|
|
const host = coerceString(u.host);
|
|
const groups = this.pathRegex.exec(coerceString(u.pathname))?.groups;
|
|
if (!groups) {
|
|
logger.warn(
|
|
{ registryUrl },
|
|
'failed to get owner and repo from given url',
|
|
);
|
|
return null;
|
|
}
|
|
|
|
const { owner, repo } = groups;
|
|
|
|
const apiBaseUrl = getApiBaseUrl(`https://${host}`);
|
|
|
|
const indexRelease = await this.http.getJson<GithubRestRelease>(
|
|
`${apiBaseUrl}repos/${owner}/${repo}/releases/tags/index`,
|
|
);
|
|
|
|
// finds asset with name index.json
|
|
const asset = indexRelease.body.assets.find(
|
|
(asset) => asset.name === 'index.json',
|
|
);
|
|
|
|
if (!asset) {
|
|
logger.warn(
|
|
{ registryUrl },
|
|
`can't find asset index.json in the given registryUrl`,
|
|
);
|
|
return null;
|
|
}
|
|
|
|
// stream down the content of index.json
|
|
// Note: need to use stream here with
|
|
// the accept header as octet-stream to
|
|
// download asset from private github repository
|
|
// see GithubDoc:
|
|
// https://docs.github.com/en/rest/releases/assets#get-a-release-asset
|
|
const indexContent = await streamToString(
|
|
this.http.stream(asset.url, {
|
|
headers: {
|
|
accept: 'application/octet-stream',
|
|
},
|
|
}),
|
|
);
|
|
|
|
try {
|
|
return JSON.parse(indexContent) as HermitSearchResult[];
|
|
} catch {
|
|
logger.warn('error parsing hermit search manifest from remote respond');
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|