Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(managers/custom): generic manager for json files #32784

Open
wants to merge 45 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
cb0dd89
implement jsonata manager
RahulGautamSingh Nov 26, 2024
6caecbd
validation
RahulGautamSingh Nov 28, 2024
739b28f
Merge branch 'main' into feat/generic-manager
RahulGautamSingh Nov 28, 2024
0432ece
fix tests
RahulGautamSingh Nov 28, 2024
8f88d8d
Merge branch 'feat/generic-manager' of https://github.com/RahulGautam…
RahulGautamSingh Nov 28, 2024
754324d
update docs
RahulGautamSingh Nov 28, 2024
c497553
fix tests
RahulGautamSingh Nov 28, 2024
f8a32ea
fix lint issue
RahulGautamSingh Nov 28, 2024
a375c51
refactor
RahulGautamSingh Nov 28, 2024
0b54c6d
fix ci issues
RahulGautamSingh Nov 28, 2024
4c14ac6
remove duplicate code
RahulGautamSingh Nov 28, 2024
caf4513
add jsonata to customType allowed values
RahulGautamSingh Nov 28, 2024
55fc985
Apply Suggestions
RahulGautamSingh Nov 29, 2024
69f6745
refactor: remove unused types
RahulGautamSingh Nov 29, 2024
dabb627
docs: refactor
RahulGautamSingh Nov 29, 2024
7f60365
Update docs/usage/configuration-options.md
RahulGautamSingh Nov 29, 2024
bfd00e1
apply suggestions
RahulGautamSingh Nov 30, 2024
292c91f
docs: redo structure
RahulGautamSingh Nov 30, 2024
1324d83
apply suggestions
RahulGautamSingh Dec 5, 2024
ff13a21
Apply Suggestions
RahulGautamSingh Dec 7, 2024
c8db815
docs: remove redundant codeblock
RahulGautamSingh Dec 7, 2024
c92ad2b
refactor: tests
RahulGautamSingh Dec 7, 2024
28e64e5
fix: docs
RahulGautamSingh Dec 11, 2024
cb25557
fix: types
RahulGautamSingh Dec 11, 2024
d4cfb2d
feat: add new field fileFormat
RahulGautamSingh Dec 11, 2024
96c0024
refactor: simplify logic for handleMatching()
RahulGautamSingh Dec 11, 2024
1ef8764
refactor: apply DRY concept
RahulGautamSingh Dec 11, 2024
ffd6228
fix(types): indentation
RahulGautamSingh Dec 11, 2024
c746d87
Merge branch 'main' into feat/generic-manager
RahulGautamSingh Dec 17, 2024
81d9121
Merge branch 'main' into feat/generic-manager
RahulGautamSingh Dec 18, 2024
4e4739c
validation for JSONata manager
RahulGautamSingh Dec 18, 2024
e66b916
docs(customManagers): fileFormat
RahulGautamSingh Dec 18, 2024
e28ba81
Apply Suggestion
RahulGautamSingh Dec 18, 2024
76b0cc4
fix issues
RahulGautamSingh Dec 18, 2024
0e48b18
apply suggestions
RahulGautamSingh Dec 18, 2024
f4ec390
Apply Suggestions
RahulGautamSingh Dec 19, 2024
66f2577
fix test
RahulGautamSingh Dec 19, 2024
a632dc6
Merge branch 'main' into feat/generic-manager
RahulGautamSingh Dec 19, 2024
d1da658
rebase
RahulGautamSingh Dec 19, 2024
e9ab58a
matchStrings: update description
RahulGautamSingh Dec 19, 2024
a36550e
update docs
RahulGautamSingh Dec 19, 2024
3788661
fix test
RahulGautamSingh Dec 21, 2024
cd68238
Apply Suggestions
RahulGautamSingh Jan 19, 2025
211927d
fix tests
RahulGautamSingh Jan 19, 2025
6f4fa17
fix lint isue
RahulGautamSingh Jan 19, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 26 additions & 2 deletions docs/usage/configuration-options.md
Original file line number Diff line number Diff line change
Expand Up @@ -777,6 +777,19 @@ Example:
}
```

```json
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
{
"customManagers": [
{
"customType": "jsonata",
"matchStrings": [
"packages.{ \"depName\": package, \"currentValue\": version }"
]
}
]
}
```
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved

### datasourceTemplate

If the `datasource` for a dependency is not captured with a named group then it can be defined in config using this field.
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
Expand All @@ -799,18 +812,29 @@ It will be compiled using Handlebars and the regex `groups` result.

### matchStrings

Each `matchStrings` must be a valid regular expression, optionally with named capture groups.
Each `matchStrings` must be one of the two:
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved

1. a valid regular expression, optionally with named capture groups (if using `customType=regex`)
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
2. a valid, escaped [JSONata](https://docs.jsonata.org/overview.html) query (if using `customType=json`)
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved

Example:

```json
```json title="matchStrings with a valid regular expression"
{
"matchStrings": [
"ENV .*?_VERSION=(?<currentValue>.*) # (?<datasource>.*?)/(?<depName>.*?)\\s"
]
}
```

```json title="matchStrings with a valid JSONata query"
{
"matchStrings": [
"packages.{ \"depName\": package, \"currentValue\": version }"
]
}
```

### matchStringsStrategy

`matchStringsStrategy` controls behavior when multiple `matchStrings` values are provided.
Expand Down
5 changes: 2 additions & 3 deletions lib/config/options/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2733,18 +2733,17 @@ const options: RenovateOptions[] = [
description:
'Custom manager to use. Valid only within a `customManagers` object.',
type: 'string',
allowedValues: ['regex'],
allowedValues: ['jsonata', 'regex'],
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
parents: ['customManagers'],
cli: false,
env: false,
},
{
name: 'matchStrings',
description:
'Regex capture rule to use. Valid only within a `customManagers` object.',
'Regex pattern or JSONata query to use. Valid only within a `customManagers` object.',
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
type: 'array',
subType: 'string',
format: 'regex',
parents: ['customManagers'],
cli: false,
env: false,
Expand Down
2 changes: 2 additions & 0 deletions lib/modules/manager/custom/api.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
import type { ManagerApi } from '../types';
import * as jsonata from './jsonata';
import * as regex from './regex';

const api = new Map<string, ManagerApi>();
export default api;

api.set('regex', regex);
api.set('jsonata', jsonata);
2 changes: 2 additions & 0 deletions lib/modules/manager/custom/index.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ describe('modules/manager/custom/index', () => {
expect(customManager.isCustomManager('npm')).toBe(false);
expect(customManager.isCustomManager('regex')).toBe(true);
expect(customManager.isCustomManager('custom.regex')).toBe(false);
expect(customManager.isCustomManager('jsonata')).toBe(true);
expect(customManager.isCustomManager('custom.jsonata')).toBe(false);
});
});
});
270 changes: 270 additions & 0 deletions lib/modules/manager/custom/jsonata/index.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,270 @@
import { codeBlock } from 'common-tags';
import { logger } from '../../../../../test/util';
import type { JsonataExtractConfig } from './types';
import { defaultConfig, extractPackageFile } from '.';

describe('modules/manager/custom/jsonata/index', () => {
it('has default config', () => {
expect(defaultConfig).toEqual({
pinDigests: false,
});
});

it('extracts data when no templates are used', async () => {
const json = codeBlock`
{
"packages": [
{
"dep_name": "foo",
"package_name": "fii",
"current_value": "1.2.3",
"current_digest": "1234",
"data_source": "nuget",
"versioning": "maven",
"extract_version": "custom-extract-version",
"registry_url": "https://registry.npmjs.org",
"dep_type": "dev"
}
]
}`;
const config = {
matchStrings: [
`packages.{
"depName": dep_name,
"packageName": package_name,
"currentValue": current_value,
"currentDigest": current_digest,
"datasource": data_source,
"versioning": versioning,
"extractVersion": extract_version,
"registryUrl": registry_url,
"depType": dep_type
}`,
],
};
const res = await extractPackageFile(json, 'unused', config);

expect(res?.deps).toHaveLength(1);
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
expect(res).toMatchObject({
deps: [
{
depName: 'foo',
packageName: 'fii',
currentValue: '1.2.3',
currentDigest: '1234',
datasource: 'nuget',
versioning: 'maven',
extractVersion: 'custom-extract-version',
registryUrls: ['https://registry.npmjs.org/'],
depType: 'dev',
},
],
matchStrings: config.matchStrings,
});
});

it('applies templates', async () => {
const json = codeBlock`
{
"packages": [
{
"dep_name": "foo",
"package_name": "fii",
"current_value": "1.2.3",
"current_digest": "1234",
"data_source": "nuget",
"versioning": "maven",
"extract_version": "custom-extract-version",
"registry_url": "https://registry.npmjs.org",
"dep_type": "dev"
},
{
}]
}`;
const config = {
matchStrings: [
`packages.{
"depName": dep_name,
"packageName": package_name,
"currentValue": current_value,
"currentDigest": current_digest,
"datasource": data_source,
"versioning": versioning,
"extractVersion": extract_version,
"registryUrl": registry_url,
"depType": dep_type
}`,
],
depNameTemplate:
'{{#if depName}}{{depName}}{{else}}default-dep-name{{/if}}',
packageNameTemplate:
'{{#if packageName}}{{packageName}}{{else}}default-package-name{{/if}}',
currentValueTemplate:
'{{#if currentValue}}{{currentValue}}{{else}}default-current-value{{/if}}',
currentDigestTemplate:
'{{#if currentDigest}}{{currentDigest}}{{else}}default-current-digest{{/if}}',
datasourceTemplate:
'{{#if datasource}}{{datasource}}{{else}}default-datasource{{/if}}',
versioningTemplate:
'{{#if versioning}}{{versioning}}{{else}}default-versioning{{/if}}',
extractVersionTemplate:
'{{#if extractVersion}}{{extractVersion}}{{else}}default-extract-version{{/if}}',
registryUrlTemplate:
'{{#if registryUrl}}{{registryUrl}}{{else}}https://default.registry.url{{/if}}',
depTypeTemplate:
'{{#if depType}}{{depType}}{{else}}default-dep-type{{/if}}',
};
const res = await extractPackageFile(json, 'unused', config);

expect(res?.deps).toHaveLength(2);
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
expect(res).toMatchObject({
deps: [
{
depName: 'foo',
packageName: 'fii',
currentValue: '1.2.3',
currentDigest: '1234',
datasource: 'nuget',
versioning: 'maven',
extractVersion: 'custom-extract-version',
registryUrls: ['https://registry.npmjs.org/'],
depType: 'dev',
},
{
depName: 'default-dep-name',
packageName: 'default-package-name',
currentValue: 'default-current-value',
currentDigest: 'default-current-digest',
datasource: 'default-datasource',
versioning: 'default-versioning',
extractVersion: 'default-extract-version',
registryUrls: ['https://default.registry.url/'],
depType: 'default-dep-type',
},
],
matchStrings: config.matchStrings,
});
});

it('logs warning if query result does not match schema', async () => {
const json = codeBlock`
{
"packages": [
{
"dep_name": "foo",
"package_name": "fii",
"current_value": 1,
"current_digest": "1234",
"data_source": "nuget",
"versioning": "maven",
"extract_version": "custom-extract-version",
"registry_url": "https://registry.npmjs.org",
"dep_type": "dev"
}
]
}`;
const config = {
matchStrings: [
`packages.{
"depName": dep_name,
"currentValue": current_value,
"datasource": data_source
}`,
],
};
const res = await extractPackageFile(json, 'unused', config);

expect(res).toBeNull();
expect(logger.logger.warn).toHaveBeenCalledWith(
expect.anything(),
'Error while parsing dep info',
);
});

it('returns null when content is not json', async () => {
const res = await extractPackageFile(
'not-json',
'foo-file',
{} as JsonataExtractConfig,
);
expect(res).toBeNull();
expect(logger.logger.warn).toHaveBeenCalledWith(
expect.anything(),
'Invalid JSON file(parsing failed)',
);
});

it('returns null when no content', async () => {
const res = await extractPackageFile(
'',
'foo-file',
{} as JsonataExtractConfig,
);
expect(res).toBeNull();
});

it('returns null if no dependencies found', async () => {
const config = {
matchStrings: [
'packages.{ "depName": package, "currentValue": version, "versioning ": versioning }',
],
};
const res = await extractPackageFile('{}', 'unused', config);
expect(res).toBeNull();
});

it('returns null if invalid template', async () => {
const config = {
matchStrings: [`{"depName": "foo"}`],
versioningTemplate: '{{#if versioning}}{{versioning}}{{else}}semver', // invalid template
};
const res = await extractPackageFile('{}', 'unused', config);
expect(res).toBeNull();
expect(logger.logger.warn).toHaveBeenCalledWith(
expect.anything(),
'Error compiling template for JSONata manager',
);
});

it('extracts and does not apply a registryUrlTemplate if the result is an invalid url', async () => {
const config = {
matchStrings: [`{"depName": "foo"}`],
registryUrlTemplate: 'this-is-not-a-valid-url-{{depName}}',
};
const res = await extractPackageFile('{}', 'unused', config);
expect(res).not.toBeNull();
expect(logger.logger.warn).toHaveBeenCalledWith(
{ url: 'this-is-not-a-valid-url-foo' },
'Invalid JSONata manager registryUrl',
);
});

it('extracts multiple dependencies with multiple matchStrings', async () => {
const config = {
matchStrings: [`{"depName": "foo"}`, `{"depName": "bar"}`],
};
const res = await extractPackageFile('{}', 'unused', config);
expect(res?.deps).toHaveLength(2);
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
});

it('excludes and warns if invalid jsonata query found', async () => {
const config = {
matchStrings: ['{', `{"depName": "foo"}`, `{"depName": "bar"}`],
};
const res = await extractPackageFile('{}', 'unused', config);
expect(res?.deps).toHaveLength(2);
RahulGautamSingh marked this conversation as resolved.
Show resolved Hide resolved
expect(logger.logger.warn).toHaveBeenCalledWith(
{ err: expect.any(Object), query: '{' },
'Failed to compile JSONata query',
);
});

it('extracts dependency with autoReplaceStringTemplate', async () => {
const config = {
matchStrings: [`{"depName": "foo"}`],
autoReplaceStringTemplate: 'auto-replace-string-template',
};
const res = await extractPackageFile('{}', 'values.yaml', config);
expect(res?.autoReplaceStringTemplate).toBe('auto-replace-string-template');
});
});
Loading
Loading