mirror of https://github.com/renovatebot/renovate
259 lines
8.0 KiB
TypeScript
259 lines
8.0 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import { logger } from '../../../logger';
|
|
import { cache } from '../../../util/cache/package/decorator';
|
|
import { queryReleases } from '../../../util/github/graphql';
|
|
import type {
|
|
GithubDigestFile,
|
|
GithubRestAsset,
|
|
GithubRestRelease,
|
|
} from '../../../util/github/types';
|
|
import { getApiBaseUrl, getSourceUrl } from '../../../util/github/url';
|
|
import { hashStream } from '../../../util/hash';
|
|
import { GithubHttp } from '../../../util/http/github';
|
|
import { newlineRegex, regEx } from '../../../util/regex';
|
|
import { Datasource } from '../datasource';
|
|
import type {
|
|
DigestConfig,
|
|
GetReleasesConfig,
|
|
Release,
|
|
ReleaseResult,
|
|
} from '../types';
|
|
|
|
export const cacheNamespace = 'datasource-github-releases';
|
|
|
|
function inferHashAlg(digest: string): string {
|
|
switch (digest.length) {
|
|
case 64:
|
|
return 'sha256';
|
|
default:
|
|
case 96:
|
|
return 'sha512';
|
|
}
|
|
}
|
|
|
|
export class GithubReleaseAttachmentsDatasource extends Datasource {
|
|
static readonly id = 'github-release-attachments';
|
|
|
|
override readonly defaultRegistryUrls = ['https://github.com'];
|
|
|
|
override http: GithubHttp;
|
|
|
|
override readonly releaseTimestampSupport = true;
|
|
// Note: not sure
|
|
override readonly releaseTimestampNote =
|
|
'The release timestamp is determined from the `releaseTimestamp` field in the results.';
|
|
override readonly sourceUrlSupport = 'package';
|
|
override readonly sourceUrlNote =
|
|
'The source URL is determined by using the `packageName` and `registryUrl`.';
|
|
|
|
constructor() {
|
|
super(GithubReleaseAttachmentsDatasource.id);
|
|
this.http = new GithubHttp(GithubReleaseAttachmentsDatasource.id);
|
|
}
|
|
|
|
@cache({
|
|
ttlMinutes: 1440,
|
|
namespace: `datasource-${GithubReleaseAttachmentsDatasource.id}`,
|
|
key: (release: GithubRestRelease, digest: string) =>
|
|
`findDigestFile:${release.html_url}:${digest}`,
|
|
})
|
|
async findDigestFile(
|
|
release: GithubRestRelease,
|
|
digest: string,
|
|
): Promise<GithubDigestFile | null> {
|
|
const smallAssets = release.assets.filter(
|
|
(a: GithubRestAsset) => a.size < 5 * 1024,
|
|
);
|
|
for (const asset of smallAssets) {
|
|
const res = await this.http.get(asset.browser_download_url);
|
|
for (const line of res.body.split(newlineRegex)) {
|
|
const [lineDigest, lineFilename] = line.split(regEx(/\s+/), 2);
|
|
if (lineDigest === digest) {
|
|
return {
|
|
assetName: asset.name,
|
|
digestedFileName: lineFilename,
|
|
currentVersion: release.tag_name,
|
|
currentDigest: lineDigest,
|
|
};
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
@cache({
|
|
ttlMinutes: 1440,
|
|
namespace: `datasource-${GithubReleaseAttachmentsDatasource.id}`,
|
|
key: (asset: GithubRestAsset, algorithm: string) =>
|
|
`downloadAndDigest:${asset.browser_download_url}:${algorithm}`,
|
|
})
|
|
async downloadAndDigest(
|
|
asset: GithubRestAsset,
|
|
algorithm: string,
|
|
): Promise<string> {
|
|
const res = this.http.stream(asset.browser_download_url);
|
|
const digest = await hashStream(res, algorithm);
|
|
return digest;
|
|
}
|
|
|
|
async findAssetWithDigest(
|
|
release: GithubRestRelease,
|
|
digest: string,
|
|
): Promise<GithubDigestFile | null> {
|
|
const algorithm = inferHashAlg(digest);
|
|
const assetsBySize = release.assets.sort(
|
|
(a: GithubRestAsset, b: GithubRestAsset) => {
|
|
if (a.size < b.size) {
|
|
return -1;
|
|
}
|
|
if (a.size > b.size) {
|
|
return 1;
|
|
}
|
|
return 0;
|
|
},
|
|
);
|
|
|
|
for (const asset of assetsBySize) {
|
|
const assetDigest = await this.downloadAndDigest(asset, algorithm);
|
|
if (assetDigest === digest) {
|
|
return {
|
|
assetName: asset.name,
|
|
currentVersion: release.tag_name,
|
|
currentDigest: assetDigest,
|
|
};
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/** Identify the asset associated with a known digest. */
|
|
async findDigestAsset(
|
|
release: GithubRestRelease,
|
|
digest: string,
|
|
): Promise<GithubDigestFile | null> {
|
|
const digestFile = await this.findDigestFile(release, digest);
|
|
if (digestFile) {
|
|
return digestFile;
|
|
}
|
|
|
|
const asset = await this.findAssetWithDigest(release, digest);
|
|
return asset;
|
|
}
|
|
|
|
/** Given a digest asset, find the equivalent digest in a different release. */
|
|
async mapDigestAssetToRelease(
|
|
digestAsset: GithubDigestFile,
|
|
release: GithubRestRelease,
|
|
): Promise<string | null> {
|
|
const current = digestAsset.currentVersion.replace(regEx(/^v/), '');
|
|
const next = release.tag_name.replace(regEx(/^v/), '');
|
|
const releaseChecksumAssetName = digestAsset.assetName.replace(
|
|
current,
|
|
next,
|
|
);
|
|
const releaseAsset = release.assets.find(
|
|
(a: GithubRestAsset) => a.name === releaseChecksumAssetName,
|
|
);
|
|
if (!releaseAsset) {
|
|
return null;
|
|
}
|
|
if (digestAsset.digestedFileName) {
|
|
const releaseFilename = digestAsset.digestedFileName.replace(
|
|
current,
|
|
next,
|
|
);
|
|
const res = await this.http.get(releaseAsset.browser_download_url);
|
|
for (const line of res.body.split(newlineRegex)) {
|
|
const [lineDigest, lineFn] = line.split(regEx(/\s+/), 2);
|
|
if (lineFn === releaseFilename) {
|
|
return lineDigest;
|
|
}
|
|
}
|
|
} else {
|
|
const algorithm = inferHashAlg(digestAsset.currentDigest);
|
|
const newDigest = await this.downloadAndDigest(releaseAsset, algorithm);
|
|
return newDigest;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Attempts to resolve the digest for the specified package.
|
|
*
|
|
* The `newValue` supplied here should be a valid tag for the GitHub release.
|
|
* Requires `currentValue` and `currentDigest`.
|
|
*
|
|
* There may be many assets attached to the release. This function will:
|
|
* - Identify the asset pinned by `currentDigest` in the `currentValue` release
|
|
* - Download small release assets, parse as checksum manifests (e.g. `SHASUMS.txt`).
|
|
* - Download individual assets until `currentDigest` is encountered. This is limited to sha256 and sha512.
|
|
* - Map the hashed asset to `newValue` and return the updated digest as a string
|
|
*/
|
|
override async getDigest(
|
|
{
|
|
packageName: repo,
|
|
currentValue,
|
|
currentDigest,
|
|
registryUrl,
|
|
}: DigestConfig,
|
|
newValue: string,
|
|
): Promise<string | null> {
|
|
logger.debug(
|
|
{ repo, currentValue, currentDigest, registryUrl, newValue },
|
|
'getDigest',
|
|
);
|
|
if (!currentDigest) {
|
|
return null;
|
|
}
|
|
if (!currentValue) {
|
|
return currentDigest;
|
|
}
|
|
|
|
const apiBaseUrl = getApiBaseUrl(registryUrl);
|
|
const { body: currentRelease } = await this.http.getJson<GithubRestRelease>(
|
|
`${apiBaseUrl}repos/${repo}/releases/tags/${currentValue}`,
|
|
);
|
|
const digestAsset = await this.findDigestAsset(
|
|
currentRelease,
|
|
currentDigest,
|
|
);
|
|
let newDigest: string | null;
|
|
if (!digestAsset || newValue === currentValue) {
|
|
newDigest = currentDigest;
|
|
} else {
|
|
const { body: newRelease } = await this.http.getJson<GithubRestRelease>(
|
|
`${apiBaseUrl}repos/${repo}/releases/tags/${newValue}`,
|
|
);
|
|
newDigest = await this.mapDigestAssetToRelease(digestAsset, newRelease);
|
|
}
|
|
return newDigest;
|
|
}
|
|
|
|
/**
|
|
* This function can be used to fetch releases with a customizable versioning
|
|
* (e.g. semver) and with releases.
|
|
*
|
|
* This function will:
|
|
* - Fetch all releases
|
|
* - Sanitize the versions if desired (e.g. strip out leading 'v')
|
|
* - Return a dependency object containing sourceUrl string and releases array
|
|
*/
|
|
async getReleases(config: GetReleasesConfig): Promise<ReleaseResult> {
|
|
const releasesResult = await queryReleases(config, this.http);
|
|
const releases = releasesResult.map((item) => {
|
|
const { version, releaseTimestamp, isStable } = item;
|
|
const result: Release = {
|
|
version,
|
|
gitRef: version,
|
|
releaseTimestamp,
|
|
};
|
|
if (is.boolean(isStable)) {
|
|
result.isStable = isStable;
|
|
}
|
|
return result;
|
|
});
|
|
const sourceUrl = getSourceUrl(config.packageName, config.registryUrl);
|
|
return { sourceUrl, releases };
|
|
}
|
|
}
|