import * as configValidation from './validation'; import { RenovateConfig } from '.'; describe('config/validation', () => { describe('validateConfig(config)', () => { it('returns deprecation warnings', async () => { const config = { prTitle: 'something', }; const { warnings } = await configValidation.validateConfig(config); expect(warnings).toHaveLength(1); expect(warnings).toMatchSnapshot(); }); it('catches invalid templates', async () => { const config = { commitMessage: '{{{something}}', }; const { errors } = await configValidation.validateConfig(config); expect(errors).toHaveLength(1); expect(errors).toMatchSnapshot(); }); it('catches invalid allowedVersions regex', async () => { const config = { packageRules: [ { packageNames: ['foo'], allowedVersions: '/^2/', }, { packageNames: ['bar'], allowedVersions: '/***$}{]][/', }, { packageNames: ['baz'], allowedVersions: '!/^2/', }, { packageNames: ['quack'], allowedVersions: '!/***$}{]][/', }, ], }; const { errors } = await configValidation.validateConfig(config); expect(errors).toHaveLength(2); expect(errors).toMatchSnapshot(); }); it('returns nested errors', async () => { const config: RenovateConfig = { foo: 1, schedule: ['after 5pm'], timezone: 'Asia/Singapore', packageRules: [ { packagePatterns: ['*'], excludePackagePatterns: ['abc ([a-z]+) ([a-z]+))'], }, ], lockFileMaintenance: { bar: 2, }, major: null, }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(3); expect(errors).toMatchSnapshot(); }); it('included unsupported manager', async () => { const config = { packageRules: [ { managers: ['foo'], }, ], }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(1); expect(errors[0].message.includes('ansible')).toBe(true); }); it('included managers of the wrong type', async () => { const config = { packageRules: [ { managers: 'string not an array', }, ], }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(2); expect(errors).toMatchSnapshot(); }); it('errors for all types', async () => { const config: RenovateConfig = { allowedVersions: 'foo', enabled: 1 as any, schedule: ['every 15 mins every weekday'], timezone: 'Asia', labels: 5 as any, semanticCommitType: 7 as any, lockFileMaintenance: false as any, extends: [':timezone(Europe/Brussel)'], packageRules: [ { excludePackageNames: ['foo'], enabled: true, }, { foo: 1, }, 'what?' as any, { packagePatterns: 'abc ([a-z]+) ([a-z]+))', excludePackagePatterns: ['abc ([a-z]+) ([a-z]+))'], enabled: false, }, ], major: null, }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toMatchSnapshot(); expect(errors).toHaveLength(12); }); it('selectors outside packageRules array trigger errors', async () => { const config = { packageNames: ['angular'], meteor: { packageRules: [ { packageNames: ['meteor'], }, ], }, docker: { minor: { packageNames: ['testPackage'], }, }, }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toMatchSnapshot(); expect(errors).toHaveLength(2); }); it('ignore packageRule nesting validation for presets', async () => { const config = { description: ['All angular.js packages'], packageNames: [ 'angular', 'angular-animate', 'angular-scroll', 'angular-sanitize', ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toMatchSnapshot(); expect(errors).toHaveLength(0); }); it('errors for unsafe fileMatches', async () => { const config = { npm: { fileMatch: ['abc ([a-z]+) ([a-z]+))'], }, docker: { fileMatch: ['x?+'], }, }; const { warnings, errors } = await configValidation.validateConfig( config ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(2); expect(errors).toMatchSnapshot(); }); it('validates regEx for each fileMatch', async () => { const config = { fileMatch: ['js', '***$}{]]['], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(1); }); it('errors if no regexManager matchStrings', async () => { const config = { regexManagers: [ { matchStrings: [], }, ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(1); }); it('validates regEx for each matchStrings', async () => { const config = { regexManagers: [ { matchStrings: ['***$}{]]['], }, ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(1); }); it('passes if regexManager fields are present', async () => { const config = { regexManagers: [ { matchStrings: ['ENV (?.*?)\\s'], depNameTemplate: 'foo', datasourceTemplate: 'bar', }, ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(0); }); it('errors if extra regexManager fields are present', async () => { const config = { regexManagers: [ { matchStrings: ['ENV (?.*?)\\s'], depNameTemplate: 'foo', datasourceTemplate: 'bar', automerge: true, }, ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(1); }); it('errors if regexManager fields are missing', async () => { const config = { regexManagers: [ { matchStrings: ['ENV (.*?)\\s'], depNameTemplate: 'foo', datasourceTemplate: 'bar', }, ], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toMatchSnapshot(); expect(errors).toHaveLength(1); }); it('ignore keys', async () => { const config = { $schema: 'renovate.json', }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(0); }); it('validates timezone preset', async () => { const config = { extends: [':timezone', ':timezone(Europe/Berlin)'], }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(0); }); it('does not validate compatibility children', async () => { const config = { compatibility: { packageRules: [{}] }, }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(0); }); it('validates object with ignored children', async () => { const config = { prBodyDefinitions: {}, }; const { warnings, errors } = await configValidation.validateConfig( config, true ); expect(warnings).toHaveLength(0); expect(errors).toHaveLength(0); }); }); });