renovate/lib/workers/repository/process/fetch.spec.ts

260 lines
8.0 KiB
TypeScript

import type { RenovateConfig } from '../../../../test/util';
import { mocked } from '../../../../test/util';
import { getConfig } from '../../../config/defaults';
import { MavenDatasource } from '../../../modules/datasource/maven';
import type { PackageFile } from '../../../modules/manager/types';
import { ExternalHostError } from '../../../types/errors/external-host-error';
import { Result } from '../../../util/result';
import { fetchUpdates } from './fetch';
import * as lookup from './lookup';
const lookupUpdates = mocked(lookup).lookupUpdates;
jest.mock('./lookup');
describe('workers/repository/process/fetch', () => {
describe('fetchUpdates()', () => {
let config: RenovateConfig;
beforeEach(() => {
config = getConfig();
});
it('handles empty deps', async () => {
const packageFiles: Record<string, PackageFile[]> = {
npm: [{ packageFile: 'package.json', deps: [] }],
};
await fetchUpdates(config, packageFiles);
expect(packageFiles).toEqual({
npm: [{ deps: [], packageFile: 'package.json' }],
});
});
it('handles ignored, skipped and disabled', async () => {
config.ignoreDeps = ['abcd'];
config.packageRules = [
{
matchPackageNames: ['foo'],
enabled: false,
},
];
const packageFiles: Record<string, PackageFile[]> = {
npm: [
{
packageFile: 'package.json',
deps: [
{ depName: 'abcd' },
{ depName: 'foo' },
{ depName: 'skipped', skipReason: 'some-reason' as never },
],
},
],
};
await fetchUpdates(config, packageFiles);
expect(packageFiles).toEqual({
npm: [
{
deps: [
{
depName: 'abcd',
packageName: 'abcd',
skipReason: 'ignored',
skipStage: 'lookup',
updates: [],
},
{
depName: 'foo',
packageName: 'foo',
skipReason: 'disabled',
skipStage: 'lookup',
updates: [],
},
{
depName: 'skipped',
packageName: 'skipped',
skipReason: 'some-reason',
skipStage: 'lookup',
updates: [],
},
],
packageFile: 'package.json',
},
],
});
expect(packageFiles.npm[0].deps[0].skipReason).toBe('ignored');
expect(packageFiles.npm[0].deps[0].updates).toHaveLength(0);
expect(packageFiles.npm[0].deps[1].skipReason).toBe('disabled');
expect(packageFiles.npm[0].deps[1].updates).toHaveLength(0);
});
it('fetches updates', async () => {
config.rangeStrategy = 'auto';
config.constraints = { some: 'different' };
const packageFiles: any = {
maven: [
{
packageFile: 'pom.xml',
extractedConstraints: { some: 'constraint', other: 'constraint' },
deps: [{ datasource: MavenDatasource.id, depName: 'bbb' }],
},
],
};
lookupUpdates.mockResolvedValue({ updates: ['a', 'b'] } as never);
await fetchUpdates(config, packageFiles);
expect(packageFiles).toEqual({
maven: [
{
deps: [
{
datasource: 'maven',
depName: 'bbb',
packageName: 'bbb',
updates: ['a', 'b'],
},
],
extractedConstraints: { other: 'constraint', some: 'constraint' },
packageFile: 'pom.xml',
},
],
});
});
it('skips deps with empty names', async () => {
const packageFiles: Record<string, PackageFile[]> = {
docker: [
{
packageFile: 'values.yaml',
deps: [
{ depName: '', currentValue: '2.8.11', datasource: 'docker' },
{ depName: 'abcd' },
{ currentValue: '2.8.11', datasource: 'docker' },
{ depName: ' ' },
{},
{ depName: undefined },
{ depName: { oh: 'no' } as unknown as string },
],
},
],
};
await fetchUpdates(config, packageFiles);
expect(packageFiles.docker[0].deps[0].skipReason).toBe('invalid-name');
expect(packageFiles.docker[0].deps[1].skipReason).toBeUndefined();
expect(packageFiles.docker[0].deps[2].skipReason).toBe('invalid-name');
expect(packageFiles.docker[0].deps[3].skipReason).toBe('invalid-name');
expect(packageFiles.docker[0].deps[4].skipReason).toBe('invalid-name');
expect(packageFiles.docker[0].deps[5].skipReason).toBe('invalid-name');
expect(packageFiles.docker[0].deps[6].skipReason).toBe('invalid-name');
});
it('skips internal deps by default', async () => {
const packageFiles: Record<string, PackageFile[]> = {
docker: [
{
packageFile: 'values.yaml',
deps: [
{
depName: 'dep-name',
currentValue: '2.8.11',
datasource: 'docker',
isInternal: true,
},
],
},
],
};
await fetchUpdates(config, packageFiles);
expect(packageFiles.docker[0].deps[0].skipReason).toBe(
'internal-package',
);
expect(packageFiles.docker[0].deps[0].updates).toHaveLength(0);
});
it('fetch updates for internal deps if updateInternalDeps is true', async () => {
config.updateInternalDeps = true;
config.rangeStrategy = 'auto';
const packageFiles: any = {
maven: [
{
packageFile: 'pom.xml',
deps: [
{
datasource: MavenDatasource.id,
depName: 'bbb',
isInternal: true,
},
],
},
],
};
lookupUpdates.mockResolvedValue({ updates: ['a', 'b'] } as never);
await fetchUpdates(config, packageFiles);
expect(packageFiles.maven[0].deps[0].updates).toHaveLength(2);
});
it('throws lookup errors for onboarded repos', async () => {
config.rangeStrategy = 'auto';
const packageFiles: any = {
maven: [
{
packageFile: 'pom.xml',
deps: [{ datasource: MavenDatasource.id, depName: 'bbb' }],
},
],
};
lookupUpdates.mockResolvedValueOnce(Result.err(new Error('some error')));
await expect(
fetchUpdates({ ...config, repoIsOnboarded: true }, packageFiles),
).rejects.toThrow();
});
it('throws lookup errors for not onboarded repos', async () => {
config.rangeStrategy = 'auto';
const packageFiles: any = {
maven: [
{
packageFile: 'pom.xml',
deps: [{ datasource: MavenDatasource.id, depName: 'bbb' }],
},
],
};
lookupUpdates.mockResolvedValueOnce(Result.err(new Error('some error')));
await expect(
fetchUpdates({ ...config, repoIsOnboarded: true }, packageFiles),
).rejects.toThrow();
});
it('produces external host warnings for not onboarded repos', async () => {
config.rangeStrategy = 'auto';
const packageFiles: any = {
maven: [
{
packageFile: 'pom.xml',
deps: [{ datasource: MavenDatasource.id, depName: 'bbb' }],
},
],
};
const err = new ExternalHostError(new Error('some error'));
lookupUpdates.mockRejectedValueOnce(err);
await fetchUpdates({ ...config, repoIsOnboarded: false }, packageFiles);
expect(packageFiles).toMatchObject({
maven: [
{
deps: [
{
depName: 'bbb',
warnings: [
{ topic: 'Lookup Error', message: 'bbb: some error' },
],
},
],
},
],
});
});
});
});