mirror of https://github.com/renovatebot/renovate
133 lines
4.1 KiB
TypeScript
133 lines
4.1 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import pMap from 'p-map';
|
|
import { logger } from '../../../logger';
|
|
import * as packageCache from '../../../util/cache/package';
|
|
import { cache } from '../../../util/cache/package/decorator';
|
|
import { regEx } from '../../../util/regex';
|
|
import { joinUrlParts } from '../../../util/url';
|
|
import * as semanticVersioning from '../../versioning/semver';
|
|
import { Datasource } from '../datasource';
|
|
import type { Release } from '../index';
|
|
import type { GetReleasesConfig, ReleaseResult } from '../types';
|
|
import { DenoAPIModuleResponse, DenoAPIModuleVersionResponse } from './schema';
|
|
|
|
export class DenoDatasource extends Datasource {
|
|
static readonly id = 'deno';
|
|
|
|
override readonly customRegistrySupport = true;
|
|
|
|
override readonly registryStrategy = 'first';
|
|
|
|
override readonly defaultVersioning = semanticVersioning.id;
|
|
|
|
override readonly defaultRegistryUrls = ['https://apiland.deno.dev'];
|
|
|
|
override readonly releaseTimestampSupport = true;
|
|
override readonly releaseTimestampNote =
|
|
'The release timestamp is determined from the `uploaded_at` field in the results.';
|
|
override readonly sourceUrlSupport = 'release';
|
|
override readonly sourceUrlNote =
|
|
'The source URL is determined from the `repository` field in the results.';
|
|
|
|
constructor() {
|
|
super(DenoDatasource.id);
|
|
}
|
|
|
|
@cache({
|
|
namespace: `datasource-${DenoDatasource.id}`,
|
|
key: ({ packageName, registryUrl }: GetReleasesConfig) =>
|
|
// TODO: types (#22198)
|
|
`getReleases:${registryUrl}:${packageName}`,
|
|
})
|
|
async getReleases({
|
|
packageName,
|
|
registryUrl,
|
|
}: GetReleasesConfig): Promise<ReleaseResult | null> {
|
|
const massagedRegistryUrl = registryUrl!;
|
|
|
|
const extractResult = regEx(
|
|
/^(https:\/\/deno.land\/)(?<rawPackageName>[^@\s]+)/,
|
|
).exec(packageName);
|
|
const rawPackageName = extractResult?.groups?.rawPackageName;
|
|
if (is.nullOrUndefined(rawPackageName)) {
|
|
logger.debug(
|
|
`Could not extract rawPackageName from packageName: "${packageName}"`,
|
|
);
|
|
return null;
|
|
}
|
|
|
|
// remove third-party prefix if defined. The only internal library is `std` and is available under the same API
|
|
const massagedPackageName = rawPackageName.replace('x/', '');
|
|
|
|
// https://apiland.deno.dev/v2/modules/postgres
|
|
const moduleAPIURL = joinUrlParts(
|
|
massagedRegistryUrl,
|
|
'v2/modules',
|
|
massagedPackageName,
|
|
);
|
|
|
|
return await this.getReleaseResult(moduleAPIURL);
|
|
}
|
|
|
|
@cache({
|
|
namespace: `datasource-${DenoDatasource.id}`,
|
|
key: (moduleAPIURL) => `getReleaseResult:${moduleAPIURL}`,
|
|
})
|
|
async getReleaseResult(moduleAPIURL: string): Promise<ReleaseResult> {
|
|
const detailsCacheKey = `details:${moduleAPIURL}`;
|
|
const releasesCache: Record<string, Release> =
|
|
(await packageCache.get(
|
|
`datasource-${DenoDatasource.id}`,
|
|
detailsCacheKey,
|
|
)) ?? {};
|
|
let cacheModified = false;
|
|
|
|
const {
|
|
body: { versions, tags },
|
|
} = await this.http.getJson(moduleAPIURL, DenoAPIModuleResponse);
|
|
|
|
// get details for the versions
|
|
const releases = await pMap(
|
|
versions,
|
|
async (version) => {
|
|
const cacheRelease = releasesCache[version];
|
|
// istanbul ignore if
|
|
if (cacheRelease) {
|
|
return cacheRelease;
|
|
}
|
|
|
|
// https://apiland.deno.dev/v2/modules/postgres/v0.17.0
|
|
const url = joinUrlParts(moduleAPIURL, version);
|
|
const { body: release } = await this.http.getJson(
|
|
url,
|
|
DenoAPIModuleVersionResponse.catch(({ error: err }) => {
|
|
logger.warn(
|
|
{ err },
|
|
`Deno: failed to get version details for ${version}`,
|
|
);
|
|
return { version };
|
|
}),
|
|
);
|
|
|
|
releasesCache[release.version] = release;
|
|
cacheModified = true;
|
|
|
|
return release;
|
|
},
|
|
{ concurrency: 5 },
|
|
);
|
|
|
|
if (cacheModified) {
|
|
// 1 week. Releases at Deno are immutable, therefore we can use a long term cache here.
|
|
await packageCache.set(
|
|
`datasource-${DenoDatasource.id}`,
|
|
detailsCacheKey,
|
|
releasesCache,
|
|
10080,
|
|
);
|
|
}
|
|
|
|
return { releases, tags };
|
|
}
|
|
}
|