mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-15 09:06:25 +00:00
cedc8e2f6b
* feat: add conan revisions * Adding getDigest and moved function * Apply suggestions from code review Co-authored-by: Michael Kriese <michael.kriese@visualon.de> * Changing undefined condition * Minimize tests * Extract isCommon to new common.ts * Removing isComment * Update lib/modules/datasource/conan/types.ts Co-authored-by: Michael Kriese <michael.kriese@visualon.de> * Removing digest from getReleases * Removing digest from packageName * cleaning up and fixing caching * Apply suggestions from code review Co-authored-by: Michael Kriese <michael.kriese@visualon.de> * Adding back eslint-disable-next-line * forcing registryUrl * move to joinurlparts * Refactor depName and userAndChannel Co-authored-by: Michael Kriese <michael.kriese@visualon.de>
147 lines
4.4 KiB
TypeScript
147 lines
4.4 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(
|
|
depName: string,
|
|
userAndChannel: string
|
|
): Promise<ReleaseResult | null> {
|
|
if (userAndChannel && userAndChannel !== '@_/_') {
|
|
logger.debug(
|
|
{ depName, 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/${depName}/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.depName,
|
|
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 depName = conanPackage.depName;
|
|
const userAndChannel = '@' + conanPackage.userAndChannel;
|
|
if (
|
|
is.string(registryUrl) &&
|
|
ensureTrailingSlash(registryUrl) === defaultRegistryUrl
|
|
) {
|
|
return this.getConanCenterReleases(depName, userAndChannel);
|
|
}
|
|
|
|
logger.trace({ depName, registryUrl }, 'Looking up conan api dependency');
|
|
if (registryUrl) {
|
|
const url = ensureTrailingSlash(registryUrl);
|
|
const lookupUrl = joinUrlParts(url, `v2/conans/search?q=${depName}`);
|
|
|
|
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;
|
|
}
|
|
}
|