renovate/lib/modules/datasource/conan/index.ts

155 lines
4.5 KiB
TypeScript

import is from '@sindresorhus/is';
import { load } from 'js-yaml';
import { logger } from '../../../logger';
import { cache } from '../../../util/cache/package/decorator';
import { GithubHttp } from '../../../util/http/github';
import { ensureTrailingSlash, joinUrlParts } from '../../../util/url';
import { Datasource } from '../datasource';
import type {
DigestConfig,
GetReleasesConfig,
Release,
ReleaseResult,
} from '../types';
import {
conanDatasourceRegex,
datasource,
defaultRegistryUrl,
getConanPackage,
} from './common';
import type { ConanJSON, ConanRevisionsJSON, ConanYAML } from './types';
export class ConanDatasource extends Datasource {
static readonly id = datasource;
override readonly defaultRegistryUrls = [defaultRegistryUrl];
override readonly caching = true;
override readonly registryStrategy = 'merge';
githubHttp: GithubHttp;
constructor(id = ConanDatasource.id) {
super(id);
this.githubHttp = new GithubHttp(id);
}
async getConanCenterReleases(
conanName: string,
userAndChannel: string
): Promise<ReleaseResult | null> {
if (userAndChannel && userAndChannel !== '@_/_') {
logger.debug(
{ conanName, userAndChannel },
'User/channel not supported for Conan Center lookups'
);
return null;
}
const url = `https://api.github.com/repos/conan-io/conan-center-index/contents/recipes/${conanName}/config.yml`;
const res = await this.githubHttp.get(url, {
headers: { accept: 'application/vnd.github.v3.raw' },
});
const doc = load(res.body, {
json: true,
}) as ConanYAML;
return {
releases: Object.keys(doc?.versions ?? {}).map((version) => ({
version,
})),
};
}
@cache({
namespace: `datasource-${datasource}-revisions`,
key: ({ registryUrl, packageName }: DigestConfig, newValue?: string) =>
// TODO: types (#7154)
`${registryUrl!}:${packageName}:${newValue!}`,
})
override async getDigest(
{ registryUrl, packageName }: DigestConfig,
newValue?: string
): Promise<string | null> {
if (is.undefined(newValue) || is.undefined(registryUrl)) {
return null;
}
const url = ensureTrailingSlash(registryUrl);
const conanPackage = getConanPackage(packageName);
const revisionLookUp = joinUrlParts(
url,
'v2/conans/',
conanPackage.conanName,
newValue,
conanPackage.userAndChannel,
'/revisions'
);
const revisionRep = await this.http.getJson<ConanRevisionsJSON>(
revisionLookUp
);
const revisions = revisionRep?.body.revisions;
return revisions?.[0].revision ?? null;
}
@cache({
namespace: `datasource-${datasource}`,
key: ({ registryUrl, packageName }: GetReleasesConfig) =>
// TODO: types (#7154)
// eslint-disable-next-line @typescript-eslint/restrict-template-expressions
`${registryUrl}:${packageName}`,
})
async getReleases({
registryUrl,
packageName,
}: GetReleasesConfig): Promise<ReleaseResult | null> {
const conanPackage = getConanPackage(packageName);
const userAndChannel = '@' + conanPackage.userAndChannel;
if (
is.string(registryUrl) &&
ensureTrailingSlash(registryUrl) === defaultRegistryUrl
) {
return this.getConanCenterReleases(
conanPackage.conanName,
userAndChannel
);
}
logger.trace(
{ packageName, registryUrl },
'Looking up conan api dependency'
);
if (registryUrl) {
const url = ensureTrailingSlash(registryUrl);
const lookupUrl = joinUrlParts(
url,
`v2/conans/search?q=${conanPackage.conanName}`
);
try {
const rep = await this.http.getJson<ConanJSON>(lookupUrl);
const versions = rep?.body;
if (versions) {
logger.trace({ lookupUrl }, 'Got conan api result');
const dep: ReleaseResult = { releases: [] };
for (const resultString of Object.values(versions.results ?? {})) {
const fromMatch = conanDatasourceRegex.exec(resultString);
if (fromMatch?.groups?.version && fromMatch?.groups?.userChannel) {
const version = fromMatch.groups.version;
if (fromMatch.groups.userChannel === userAndChannel) {
const result: Release = {
version,
};
dep.releases.push(result);
}
}
}
return dep;
}
} catch (err) {
this.handleGenericErrors(err);
}
}
return null;
}
}