mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-27 05:09:37 +00:00
ef1cf5906c
Co-authored-by: Rhys Arkins <rhys@arkins.net>
901 lines
25 KiB
TypeScript
901 lines
25 KiB
TypeScript
import { HeadObjectCommand, S3Client } from '@aws-sdk/client-s3';
|
|
import { mockClient } from 'aws-sdk-client-mock';
|
|
import { codeBlock } from 'common-tags';
|
|
import { GoogleAuth as _googleAuth } from 'google-auth-library';
|
|
import { DateTime } from 'luxon';
|
|
import type { Release, ReleaseResult } from '..';
|
|
import { getPkgReleases } from '..';
|
|
import { Fixtures } from '../../../../test/fixtures';
|
|
import * as httpMock from '../../../../test/http-mock';
|
|
import { mocked } from '../../../../test/util';
|
|
import { EXTERNAL_HOST_ERROR } from '../../../constants/error-messages';
|
|
import * as hostRules from '../../../util/host-rules';
|
|
import { id as versioning } from '../../versioning/maven';
|
|
import { postprocessRelease } from '../postprocess-release';
|
|
import { MAVEN_REPO } from './common';
|
|
import { MavenDatasource } from '.';
|
|
|
|
const googleAuth = mocked(_googleAuth);
|
|
jest.mock('google-auth-library');
|
|
|
|
const datasource = MavenDatasource.id;
|
|
|
|
const baseUrl = 'https://repo.maven.apache.org/maven2';
|
|
const baseUrlCustom = 'https://custom.registry.renovatebot.com';
|
|
|
|
const arRegistry = 'maven.pkg.dev/some-project/some-repository';
|
|
const baseUrlAR = `artifactregistry://${arRegistry}`;
|
|
const baseUrlARHttps = `https://${arRegistry}`;
|
|
|
|
interface SnapshotOpts {
|
|
version: string;
|
|
meta?: string;
|
|
}
|
|
|
|
interface MockOpts {
|
|
dep?: string;
|
|
base?: string;
|
|
meta?: string | null;
|
|
pom?: string | null;
|
|
latest?: string;
|
|
snapshots?: SnapshotOpts[] | null;
|
|
}
|
|
|
|
function mockGenericPackage(opts: MockOpts = {}) {
|
|
const {
|
|
dep = 'org.example:package',
|
|
base = baseUrl,
|
|
latest = '2.0.0',
|
|
snapshots,
|
|
} = opts;
|
|
const meta =
|
|
opts.meta === undefined ? Fixtures.get('metadata.xml') : opts.meta;
|
|
const pom = opts.pom === undefined ? Fixtures.get('pom.xml') : opts.pom;
|
|
|
|
const scope = httpMock.scope(base);
|
|
|
|
const [group, artifact] = dep.split(':');
|
|
const packagePath = `${group.replace(/\./g, '/')}/${artifact}`;
|
|
|
|
if (meta) {
|
|
scope.get(`/${packagePath}/maven-metadata.xml`).reply(200, meta);
|
|
}
|
|
|
|
if (pom) {
|
|
if (latest.endsWith('-SNAPSHOT')) {
|
|
const [major, minor, patch] = latest
|
|
.replace('-SNAPSHOT', '')
|
|
.split('.')
|
|
.map((x) => parseInt(x, 10))
|
|
.map((x) => (x < 10 ? `0${x}` : `${x}`));
|
|
scope
|
|
.get(
|
|
`/${packagePath}/${latest}/${artifact}-${latest.replace(
|
|
'-SNAPSHOT',
|
|
'',
|
|
)}-20200101.${major}${minor}${patch}-${parseInt(patch, 10)}.pom`,
|
|
)
|
|
.reply(200, pom);
|
|
} else {
|
|
scope
|
|
.get(`/${packagePath}/${latest}/${artifact}-${latest}.pom`)
|
|
.reply(200, pom);
|
|
}
|
|
}
|
|
|
|
if (snapshots) {
|
|
snapshots.forEach((snapshot) => {
|
|
if (snapshot.meta) {
|
|
scope
|
|
.get(`/${packagePath}/${snapshot.version}/maven-metadata.xml`)
|
|
.reply(200, snapshot.meta);
|
|
} else {
|
|
scope
|
|
.get(`/${packagePath}/${snapshot.version}/maven-metadata.xml`)
|
|
.reply(404, '');
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
function get(
|
|
packageName = 'org.example:package',
|
|
...registryUrls: string[]
|
|
): Promise<ReleaseResult | null> {
|
|
const conf = { versioning, datasource, packageName };
|
|
return getPkgReleases(registryUrls ? { ...conf, registryUrls } : conf);
|
|
}
|
|
|
|
describe('modules/datasource/maven/index', () => {
|
|
beforeEach(() => {
|
|
hostRules.add({
|
|
hostType: datasource,
|
|
matchHost: 'custom.registry.renovatebot.com',
|
|
token: '123test',
|
|
});
|
|
});
|
|
|
|
afterEach(() => {
|
|
hostRules.clear();
|
|
});
|
|
|
|
it('returns null when metadata is not found', async () => {
|
|
httpMock
|
|
.scope(baseUrl)
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(404);
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns releases', async () => {
|
|
mockGenericPackage();
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('returns releases when only snapshot', async () => {
|
|
const meta = Fixtures.get('metadata-snapshot-version.xml');
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('metadata-snapshot-only.xml'),
|
|
latest: '1.0.3-SNAPSHOT',
|
|
snapshots: [
|
|
{
|
|
version: '1.0.3-SNAPSHOT',
|
|
meta,
|
|
},
|
|
],
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toEqual({
|
|
display: 'org.example:package',
|
|
group: 'org.example',
|
|
homepage: 'https://package.example.org/about',
|
|
name: 'package',
|
|
packageScope: 'org.example',
|
|
registryUrl: 'https://repo.maven.apache.org/maven2',
|
|
releases: [{ version: '1.0.3-SNAPSHOT' }],
|
|
});
|
|
});
|
|
|
|
it('handles invalid snapshot', async () => {
|
|
const meta = Fixtures.get('metadata-snapshot-version-invalid.xml');
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.get('/org/example/package/1.0.3-SNAPSHOT/package-1.0.3-SNAPSHOT.pom')
|
|
.reply(200, meta);
|
|
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('metadata-snapshot-only.xml'),
|
|
pom: null,
|
|
latest: '1.0.3-SNAPSHOT',
|
|
snapshots: [
|
|
{
|
|
version: '1.0.3-SNAPSHOT',
|
|
meta,
|
|
},
|
|
],
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toEqual({
|
|
display: 'org.example:package',
|
|
group: 'org.example',
|
|
name: 'package',
|
|
packageScope: 'org.example',
|
|
registryUrl: 'https://repo.maven.apache.org/maven2',
|
|
releases: [{ version: '1.0.3-SNAPSHOT' }],
|
|
});
|
|
});
|
|
|
|
it('returns releases from custom repository', async () => {
|
|
mockGenericPackage({ base: baseUrlCustom });
|
|
|
|
const res = await get('org.example:package', baseUrlCustom);
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('falls back to next registry url', async () => {
|
|
mockGenericPackage();
|
|
httpMock
|
|
.scope('https://failed_repo')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(404);
|
|
httpMock
|
|
.scope('https://unauthorized_repo')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(403);
|
|
httpMock
|
|
.scope('https://empty_repo')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(200, 'non-sense');
|
|
httpMock
|
|
.scope('https://unknown_error')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.replyWithError('unknown');
|
|
|
|
const res = await get(
|
|
'org.example:package',
|
|
'https://failed_repo/',
|
|
'https://unauthorized_repo/',
|
|
'https://empty_repo',
|
|
'https://unknown_error',
|
|
baseUrl,
|
|
);
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('throws EXTERNAL_HOST_ERROR for 50x', async () => {
|
|
httpMock
|
|
.scope(baseUrl)
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(503);
|
|
|
|
await expect(get()).rejects.toThrow(EXTERNAL_HOST_ERROR);
|
|
});
|
|
|
|
it('ignores unsupported protocols', async () => {
|
|
const base = baseUrl.replace('https', 'http');
|
|
mockGenericPackage({ base });
|
|
|
|
const res = await get(
|
|
'org.example:package',
|
|
'ftp://protocol_error_repo',
|
|
base,
|
|
);
|
|
|
|
expect(res?.releases).toMatchSnapshot();
|
|
});
|
|
|
|
it('skips registry with invalid metadata structure', async () => {
|
|
mockGenericPackage();
|
|
httpMock
|
|
.scope('https://invalid_metadata_repo')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(200, Fixtures.get('metadata-invalid.xml'));
|
|
|
|
const res = await get(
|
|
'org.example:package',
|
|
'https://invalid_metadata_repo',
|
|
baseUrl,
|
|
);
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('skips registry with invalid XML', async () => {
|
|
mockGenericPackage();
|
|
httpMock
|
|
.scope('https://invalid_metadata_repo')
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(200, '###');
|
|
|
|
const res = await get(
|
|
'org.example:package',
|
|
'https://invalid_metadata_repo',
|
|
baseUrl,
|
|
);
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('handles optional slash at the end of registry url', async () => {
|
|
mockGenericPackage();
|
|
const resA = await get('org.example:package', baseUrl.replace(/\/+$/, ''));
|
|
mockGenericPackage();
|
|
const resB = await get('org.example:package', baseUrl.replace(/\/*$/, '/'));
|
|
expect(resA).not.toBeNull();
|
|
expect(resB).not.toBeNull();
|
|
expect(resA?.releases).toEqual(resB?.releases);
|
|
});
|
|
|
|
it('returns null for invalid registryUrls', async () => {
|
|
const res = await get(
|
|
'org.example:package',
|
|
|
|
'${project.baseUri}../../repository/',
|
|
);
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('supports scm.url values prefixed with "scm:"', async () => {
|
|
const pom = Fixtures.get('pom.scm-prefix.xml');
|
|
mockGenericPackage({ pom });
|
|
|
|
const res = await get();
|
|
|
|
expect(res?.sourceUrl).toBe('https://github.com/example/test');
|
|
});
|
|
|
|
describe('supports relocation', () => {
|
|
it('with only groupId present', async () => {
|
|
const pom = codeBlock`
|
|
<project>
|
|
<distributionManagement>
|
|
<relocation>
|
|
<groupId>io.example</groupId>
|
|
</relocation>
|
|
</distributionManagement>
|
|
</project>
|
|
`;
|
|
mockGenericPackage({ pom });
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
replacementName: 'io.example:package',
|
|
replacementVersion: '2.0.0',
|
|
});
|
|
});
|
|
|
|
it('with only artifactId present', async () => {
|
|
const pom = codeBlock`
|
|
<project>
|
|
<distributionManagement>
|
|
<relocation>
|
|
<artifactId>foo</artifactId>
|
|
</relocation>
|
|
</distributionManagement>
|
|
</project>
|
|
`;
|
|
mockGenericPackage({ pom });
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
replacementName: 'org.example:foo',
|
|
replacementVersion: '2.0.0',
|
|
});
|
|
});
|
|
|
|
it('with all elments present', async () => {
|
|
const pom = codeBlock`
|
|
<project>
|
|
<distributionManagement>
|
|
<relocation>
|
|
<groupId>io.example</groupId>
|
|
<artifactId>foo</artifactId>
|
|
<version>1.2.3</version>
|
|
<message>test relocation</message>
|
|
</relocation>
|
|
</distributionManagement>
|
|
</project>
|
|
`;
|
|
mockGenericPackage({ pom });
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
replacementName: 'io.example:foo',
|
|
replacementVersion: '1.2.3',
|
|
deprecationMessage: 'test relocation',
|
|
});
|
|
});
|
|
});
|
|
|
|
it('removes authentication header after redirect', async () => {
|
|
const frontendHost = 'frontend_for_private_s3_repository';
|
|
const frontendUrl = `https://${frontendHost}/maven2`;
|
|
const backendUrl = 'https://private_s3_repository/maven2';
|
|
const metadataPath = '/org/example/package/maven-metadata.xml';
|
|
const pomfilePath = '/org/example/package/2.0.0/package-2.0.0.pom';
|
|
const queryStr = '?X-Amz-Algorithm=AWS4-HMAC-SHA256';
|
|
|
|
hostRules.add({
|
|
hostType: datasource,
|
|
matchHost: frontendHost,
|
|
username: 'username',
|
|
password: 'password',
|
|
timeout: 20000,
|
|
});
|
|
|
|
httpMock
|
|
.scope(frontendUrl)
|
|
.get(metadataPath)
|
|
.basicAuth({ user: 'username', pass: 'password' })
|
|
.reply(302, '', {
|
|
Location: `${backendUrl}${metadataPath}${queryStr}`,
|
|
})
|
|
.get(pomfilePath)
|
|
.basicAuth({ user: 'username', pass: 'password' })
|
|
.reply(302, '', {
|
|
Location: `${backendUrl}${pomfilePath}${queryStr}`,
|
|
});
|
|
httpMock
|
|
.scope(backendUrl, { badheaders: ['authorization'] })
|
|
.get(`${metadataPath}${queryStr}`)
|
|
.reply(200, Fixtures.get('metadata.xml'))
|
|
.get(`${pomfilePath}${queryStr}`)
|
|
.reply(200, Fixtures.get('pom.xml'));
|
|
|
|
const res = await get('org.example:package', frontendUrl);
|
|
|
|
expect(res).toMatchSnapshot();
|
|
});
|
|
|
|
it('supports artifactregistry urls with auth', async () => {
|
|
const pomfilePath = '/org/example/package/2.0.0/package-2.0.0.pom';
|
|
hostRules.clear();
|
|
|
|
httpMock
|
|
.scope(baseUrlARHttps)
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.matchHeader(
|
|
'authorization',
|
|
'Basic b2F1dGgyYWNjZXNzdG9rZW46c29tZS10b2tlbg==',
|
|
)
|
|
.reply(200, Fixtures.get('metadata.xml'));
|
|
|
|
httpMock
|
|
.scope(baseUrlARHttps)
|
|
.get(pomfilePath)
|
|
.matchHeader(
|
|
'authorization',
|
|
'Basic b2F1dGgyYWNjZXNzdG9rZW46c29tZS10b2tlbg==',
|
|
)
|
|
.reply(200, Fixtures.get('pom.xml'));
|
|
|
|
googleAuth.mockImplementation(
|
|
jest.fn().mockImplementation(() => ({
|
|
getAccessToken: jest.fn().mockResolvedValue('some-token'),
|
|
})),
|
|
);
|
|
|
|
const res = await get('org.example:package', baseUrlAR);
|
|
|
|
expect(res).toEqual({
|
|
display: 'org.example:package',
|
|
group: 'org.example',
|
|
homepage: 'https://package.example.org/about',
|
|
name: 'package',
|
|
packageScope: 'org.example',
|
|
registryUrl:
|
|
'artifactregistry://maven.pkg.dev/some-project/some-repository',
|
|
releases: [
|
|
{ version: '0.0.1' },
|
|
{ version: '1.0.0' },
|
|
{ version: '1.0.1' },
|
|
{ version: '1.0.2' },
|
|
{ version: '1.0.3-SNAPSHOT' },
|
|
{ version: '1.0.4-SNAPSHOT' },
|
|
{ version: '1.0.5-SNAPSHOT' },
|
|
{ version: '2.0.0' },
|
|
],
|
|
isPrivate: true,
|
|
});
|
|
expect(googleAuth).toHaveBeenCalledTimes(2);
|
|
});
|
|
|
|
it('supports artifactregistry urls without auth', async () => {
|
|
const pomfilePath = '/org/example/package/2.0.0/package-2.0.0.pom';
|
|
hostRules.clear();
|
|
|
|
httpMock
|
|
.scope(baseUrlARHttps)
|
|
.get('/org/example/package/maven-metadata.xml')
|
|
.reply(200, Fixtures.get('metadata.xml'));
|
|
|
|
httpMock
|
|
.scope(baseUrlARHttps)
|
|
.get(pomfilePath)
|
|
.reply(200, Fixtures.get('pom.xml'));
|
|
|
|
googleAuth.mockImplementation(
|
|
jest.fn().mockImplementation(() => ({
|
|
getAccessToken: jest.fn().mockResolvedValue(undefined),
|
|
})),
|
|
);
|
|
|
|
const res = await get('org.example:package', baseUrlAR);
|
|
|
|
expect(res).toEqual({
|
|
display: 'org.example:package',
|
|
group: 'org.example',
|
|
homepage: 'https://package.example.org/about',
|
|
name: 'package',
|
|
packageScope: 'org.example',
|
|
registryUrl:
|
|
'artifactregistry://maven.pkg.dev/some-project/some-repository',
|
|
releases: [
|
|
{ version: '0.0.1' },
|
|
{ version: '1.0.0' },
|
|
{ version: '1.0.1' },
|
|
{ version: '1.0.2' },
|
|
{ version: '1.0.3-SNAPSHOT' },
|
|
{ version: '1.0.4-SNAPSHOT' },
|
|
{ version: '1.0.5-SNAPSHOT' },
|
|
{ version: '2.0.0' },
|
|
],
|
|
isPrivate: true,
|
|
});
|
|
expect(googleAuth).toHaveBeenCalledTimes(2);
|
|
});
|
|
|
|
describe('fetching parent info', () => {
|
|
const parentPackage = {
|
|
dep: 'org.example:parent',
|
|
meta: null,
|
|
pom: Fixtures.get('parent-scm-homepage/pom.xml'),
|
|
latest: '1.0.0',
|
|
};
|
|
|
|
it('should get source and homepage from parent', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-no-info/meta.xml'),
|
|
pom: Fixtures.get('child-no-info/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
mockGenericPackage(parentPackage);
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/parent-scm/parent',
|
|
homepage: 'https://parent-home.example.com',
|
|
});
|
|
});
|
|
|
|
it('should deal with missing parent fields', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-empty/meta.xml'),
|
|
pom: Fixtures.get('child-empty/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
display: 'org.example:package',
|
|
group: 'org.example',
|
|
name: 'package',
|
|
});
|
|
expect(res).not.toHaveProperty('homepage');
|
|
expect(res).not.toHaveProperty('sourceUrl');
|
|
});
|
|
|
|
it('should deal with circular hierarchy', async () => {
|
|
const parentPom = Fixtures.get('child-parent-cycle/parent.pom.xml');
|
|
const parentPomMock = {
|
|
dep: 'org.example:parent',
|
|
meta: null,
|
|
pom: parentPom,
|
|
latest: '2.0.0',
|
|
};
|
|
|
|
const childMeta = Fixtures.get('child-parent-cycle/child.meta.xml');
|
|
const childPom = Fixtures.get('child-parent-cycle/child.pom.xml');
|
|
const childPomMock = {
|
|
dep: 'org.example:child',
|
|
meta: null,
|
|
pom: childPom,
|
|
latest: '2.0.0',
|
|
};
|
|
|
|
mockGenericPackage({
|
|
...childPomMock,
|
|
meta: childMeta,
|
|
});
|
|
mockGenericPackage(parentPomMock);
|
|
mockGenericPackage(childPomMock);
|
|
mockGenericPackage(parentPomMock);
|
|
mockGenericPackage(childPomMock);
|
|
mockGenericPackage(parentPomMock);
|
|
|
|
const res = await get('org.example:child');
|
|
|
|
expect(res).toMatchObject({
|
|
homepage: 'https://parent-home.example.com',
|
|
});
|
|
});
|
|
|
|
it('should get source from own pom and homepage from parent', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-scm/meta.xml'),
|
|
pom: Fixtures.get('child-scm/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
mockGenericPackage(parentPackage);
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/child-scm/child',
|
|
homepage: 'https://parent-home.example.com',
|
|
});
|
|
});
|
|
|
|
it('should get homepage from own pom and source from parent', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-url/meta.xml'),
|
|
pom: Fixtures.get('child-url/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
mockGenericPackage(parentPackage);
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/parent-scm/parent',
|
|
homepage: 'https://child-home.example.com',
|
|
});
|
|
});
|
|
|
|
it('should get homepage and source from own pom', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-all-info/meta.xml'),
|
|
pom: Fixtures.get('child-all-info/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/child-scm/child',
|
|
homepage: 'https://child-home.example.com',
|
|
});
|
|
});
|
|
|
|
it('should be able to detect git@github.com:child-scm as valid sourceUrl', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-scm-gitatcolon/meta.xml'),
|
|
pom: Fixtures.get('child-scm-gitatcolon/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/child-scm/child',
|
|
});
|
|
});
|
|
|
|
it('should be able to detect git@github.com/child-scm as valid sourceUrl', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-scm-gitatslash/meta.xml'),
|
|
pom: Fixtures.get('child-scm-gitatslash/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/child-scm/child',
|
|
});
|
|
});
|
|
|
|
it('should be able to detect git://@github.com/child-scm as valid sourceUrl', async () => {
|
|
mockGenericPackage({
|
|
meta: Fixtures.get('child-scm-gitprotocol/meta.xml'),
|
|
pom: Fixtures.get('child-scm-gitprotocol/pom.xml'),
|
|
latest: '2.0.0',
|
|
});
|
|
|
|
const res = await get();
|
|
|
|
expect(res).toMatchObject({
|
|
sourceUrl: 'https://github.com/child-scm/child',
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('post-fetch release validation', () => {
|
|
it('returns null for 404', async () => {
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.head('/foo/bar/1.2.3/bar-1.2.3.pom')
|
|
.reply(404);
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: MAVEN_REPO },
|
|
{ version: '1.2.3' },
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns null for unknown error', async () => {
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.head('/foo/bar/1.2.3/bar-1.2.3.pom')
|
|
.replyWithError('unknown error');
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: MAVEN_REPO },
|
|
{ version: '1.2.3' },
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns original value for 200 response', async () => {
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.head('/foo/bar/1.2.3/bar-1.2.3.pom')
|
|
.reply(200);
|
|
const releaseOrig: Release = { version: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: MAVEN_REPO },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBe(releaseOrig);
|
|
});
|
|
|
|
it('returns original value for 200 response with versionOrig', async () => {
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.head('/foo/bar/1.2.3/bar-1.2.3.pom')
|
|
.reply(200);
|
|
const releaseOrig: Release = { version: '1.2', versionOrig: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: MAVEN_REPO },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBe(releaseOrig);
|
|
});
|
|
|
|
it('returns original value for invalid configs', async () => {
|
|
const releaseOrig: Release = { version: '1.2.3' };
|
|
expect(
|
|
await postprocessRelease(
|
|
{ datasource, registryUrl: MAVEN_REPO },
|
|
releaseOrig,
|
|
),
|
|
).toBe(releaseOrig);
|
|
expect(
|
|
await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar' },
|
|
releaseOrig,
|
|
),
|
|
).toBe(releaseOrig);
|
|
});
|
|
|
|
it('adds releaseTimestamp', async () => {
|
|
httpMock
|
|
.scope(MAVEN_REPO)
|
|
.head('/foo/bar/1.2.3/bar-1.2.3.pom')
|
|
.reply(200, '', { 'Last-Modified': '2024-01-01T00:00:00.000Z' });
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: MAVEN_REPO },
|
|
{ version: '1.2.3' },
|
|
);
|
|
|
|
expect(res).toEqual({
|
|
version: '1.2.3',
|
|
releaseTimestamp: '2024-01-01T00:00:00.000Z',
|
|
});
|
|
});
|
|
|
|
describe('S3', () => {
|
|
const s3mock = mockClient(S3Client);
|
|
|
|
afterEach(() => {
|
|
s3mock.reset();
|
|
});
|
|
|
|
it('checks package', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.resolvesOnce({});
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
{ version: '1.2.3' },
|
|
);
|
|
|
|
expect(res).toEqual({ version: '1.2.3' });
|
|
});
|
|
|
|
it('supports timestamp', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.resolvesOnce({
|
|
LastModified: DateTime.fromISO(
|
|
'2024-01-01T00:00:00.000Z',
|
|
).toJSDate(),
|
|
});
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
{ version: '1.2.3' },
|
|
);
|
|
|
|
expect(res).toEqual({
|
|
version: '1.2.3',
|
|
releaseTimestamp: '2024-01-01T00:00:00.000Z',
|
|
});
|
|
});
|
|
|
|
it('returns null for deleted object', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.resolvesOnce({ DeleteMarker: true });
|
|
|
|
const releaseOrig = { version: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns null for NotFound response', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.rejectsOnce('NotFound');
|
|
|
|
const releaseOrig = { version: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns null for NoSuchKey response', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.rejectsOnce('NoSuchKey');
|
|
|
|
const releaseOrig = { version: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
|
|
it('returns null for unknown error', async () => {
|
|
s3mock
|
|
.on(HeadObjectCommand, {
|
|
Bucket: 'bucket',
|
|
Key: 'foo/bar/1.2.3/bar-1.2.3.pom',
|
|
})
|
|
.rejectsOnce('Unknown');
|
|
|
|
const releaseOrig = { version: '1.2.3' };
|
|
|
|
const res = await postprocessRelease(
|
|
{ datasource, packageName: 'foo:bar', registryUrl: 's3://bucket' },
|
|
releaseOrig,
|
|
);
|
|
|
|
expect(res).toBeNull();
|
|
});
|
|
});
|
|
});
|
|
});
|