feat(maven): Support 'releaseTimestamp' for Maven Central (#7061)

Co-authored-by: Michael Kriese <michael.kriese@visualon.de>
This commit is contained in:
Sergio Zharinov 2020-08-27 11:23:21 +04:00 committed by GitHub
parent f411dde51a
commit 6c085243d6
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 52 additions and 13 deletions

View file

@ -1,7 +1,7 @@
import fs from 'fs'; import fs from 'fs';
import { resolve } from 'path'; import { resolve } from 'path';
import nock from 'nock'; import nock from 'nock';
import { getPkgReleases } from '..'; import { Release, getPkgReleases } from '..';
import { EXTERNAL_HOST_ERROR } from '../../constants/error-messages'; import { EXTERNAL_HOST_ERROR } from '../../constants/error-messages';
import * as hostRules from '../../util/host-rules'; import * as hostRules from '../../util/host-rules';
import * as mavenVersioning from '../../versioning/maven'; import * as mavenVersioning from '../../versioning/maven';
@ -93,10 +93,12 @@ describe('datasource/maven', () => {
'8.0.12': 500, '8.0.12': 500,
}).forEach(([v, status]) => { }).forEach(([v, status]) => {
const path = `/maven2/mysql/mysql-connector-java/${v}/mysql-connector-java-${v}.pom`; const path = `/maven2/mysql/mysql-connector-java/${v}/mysql-connector-java-${v}.pom`;
nock('https://repo.maven.apache.org').head(path).reply(status, '', {}); nock('https://repo.maven.apache.org')
.head(path)
.reply(status, '', { 'Last-Modified': `good timestamp for ${v}` });
nock('http://frontend_for_private_s3_repository') nock('http://frontend_for_private_s3_repository')
.head(path) .head(path)
.reply(status, '', {}); .reply(status, '', { 'Last-Modified': `bad timestamp for ${v}` });
}); });
}); });
@ -104,7 +106,7 @@ describe('datasource/maven', () => {
nock.enableNetConnect(); nock.enableNetConnect();
}); });
function generateReleases(versions) { function generateReleases(versions): Release[] {
return versions.map((v) => ({ version: v })); return versions.map((v) => ({ version: v }));
} }
@ -164,7 +166,12 @@ describe('datasource/maven', () => {
depName: 'mysql:mysql-connector-java', depName: 'mysql:mysql-connector-java',
registryUrls: ['https://repo.maven.apache.org/maven2/'], registryUrls: ['https://repo.maven.apache.org/maven2/'],
}); });
expect(releases.releases).toEqual(generateReleases(MYSQL_VERSIONS)); expect(releases.releases).toEqual(
generateReleases(MYSQL_VERSIONS).map(({ version }) => ({
version,
releaseTimestamp: `good timestamp for ${version}`,
}))
);
}); });
it('should return all versions of a specific library if a repository fails', async () => { it('should return all versions of a specific library if a repository fails', async () => {
@ -179,7 +186,12 @@ describe('datasource/maven', () => {
'http://empty_repo', 'http://empty_repo',
], ],
}); });
expect(releases.releases).toEqual(generateReleases(MYSQL_VERSIONS)); expect(releases.releases).toEqual(
generateReleases(MYSQL_VERSIONS).map(({ version }) => ({
version,
releaseTimestamp: `good timestamp for ${version}`,
}))
);
}); });
it('should throw external-host-error if default maven repo fails', async () => { it('should throw external-host-error if default maven repo fails', async () => {
@ -208,7 +220,12 @@ describe('datasource/maven', () => {
'ftp://protocol_error_repo', 'ftp://protocol_error_repo',
], ],
}); });
expect(releases.releases).toEqual(generateReleases(MYSQL_VERSIONS)); expect(releases.releases).toEqual(
generateReleases(MYSQL_VERSIONS).map(({ version }) => ({
version,
releaseTimestamp: `good timestamp for ${version}`,
}))
);
}); });
it('should return all versions of a specific library if a repository fails because invalid metadata file is found in another repository', async () => { it('should return all versions of a specific library if a repository fails because invalid metadata file is found in another repository', async () => {
@ -233,7 +250,12 @@ describe('datasource/maven', () => {
'http://invalid_metadata_repo/maven2/', 'http://invalid_metadata_repo/maven2/',
], ],
}); });
expect(releases.releases).toEqual(generateReleases(MYSQL_VERSIONS)); expect(releases.releases).toEqual(
generateReleases(MYSQL_VERSIONS).map(({ version }) => ({
version,
releaseTimestamp: `good timestamp for ${version}`,
}))
);
}); });
it('should return all versions of a specific library if a repository fails because a metadata file is not xml', async () => { it('should return all versions of a specific library if a repository fails because a metadata file is not xml', async () => {
@ -251,7 +273,12 @@ describe('datasource/maven', () => {
'http://invalid_metadata_repo/maven2/', 'http://invalid_metadata_repo/maven2/',
], ],
}); });
expect(releases.releases).toEqual(generateReleases(MYSQL_VERSIONS)); expect(releases.releases).toEqual(
generateReleases(MYSQL_VERSIONS).map(({ version }) => ({
version,
releaseTimestamp: `good timestamp for ${version}`,
}))
);
}); });
it('should return all versions of a specific library if a repository does not end with /', async () => { it('should return all versions of a specific library if a repository does not end with /', async () => {

View file

@ -181,7 +181,7 @@ function isValidArtifactsInfo(
return versions.every((v) => info[v] !== undefined); return versions.every((v) => info[v] !== undefined);
} }
type ArtifactInfoResult = [string, boolean | null]; type ArtifactInfoResult = [string, boolean | string | null];
async function getArtifactInfo( async function getArtifactInfo(
version: string, version: string,
@ -239,7 +239,14 @@ async function filterMissingArtifacts(
return versions return versions
.filter((v) => artifactsInfo[v]) .filter((v) => artifactsInfo[v])
.map((version) => ({ version })); .map((version) => {
const release: Release = { version };
const releaseTimestamp = artifactsInfo[version];
if (releaseTimestamp && typeof releaseTimestamp === 'string') {
release.releaseTimestamp = releaseTimestamp;
}
return release;
});
} }
export async function getReleases({ export async function getReleases({

View file

@ -99,10 +99,15 @@ export async function downloadHttpProtocol(
export async function isHttpResourceExists( export async function isHttpResourceExists(
pkgUrl: url.URL | string, pkgUrl: url.URL | string,
hostType = id hostType = id
): Promise<boolean | null> { ): Promise<boolean | string | null> {
try { try {
const httpClient = httpByHostType(hostType); const httpClient = httpByHostType(hostType);
await httpClient.head(pkgUrl.toString()); const res = await httpClient.head(pkgUrl.toString());
const pkgUrlHost = url.parse(pkgUrl.toString()).host;
if (pkgUrlHost === 'repo.maven.apache.org') {
const timestamp = res?.headers?.['last-modified'];
return timestamp || true;
}
return true; return true;
} catch (err) { } catch (err) {
if (isNotFoundError(err)) { if (isNotFoundError(err)) {