mirror of https://github.com/renovatebot/renovate
630 lines
18 KiB
TypeScript
630 lines
18 KiB
TypeScript
import { codeBlock } from 'common-tags';
|
|
import { Fixtures } from '../../../../test/fixtures';
|
|
import { fs } from '../../../../test/util';
|
|
import type { ExtractConfig } from '../types';
|
|
import { extractPackageFile } from '.';
|
|
|
|
jest.mock('../../../util/fs');
|
|
|
|
function mockReadLocalFile(files: Record<string, string | null>) {
|
|
fs.readLocalFile.mockImplementation((file): Promise<any> => {
|
|
let content: string | null = null;
|
|
if (file in files) {
|
|
content = files[file];
|
|
}
|
|
return Promise.resolve(content);
|
|
});
|
|
}
|
|
|
|
const cargo1toml = Fixtures.get('Cargo.1.toml');
|
|
const cargo2toml = Fixtures.get('Cargo.2.toml');
|
|
const cargo3toml = Fixtures.get('Cargo.3.toml');
|
|
const cargo4toml = Fixtures.get('Cargo.4.toml');
|
|
const cargo5toml = Fixtures.get('Cargo.5.toml');
|
|
const cargo6configtoml = Fixtures.get('cargo.6.config.toml');
|
|
const cargo6toml = Fixtures.get('Cargo.6.toml');
|
|
const cargo7toml = Fixtures.get('Cargo.7.toml');
|
|
|
|
const lockfileUpdateCargotoml = Fixtures.get('lockfile-update/Cargo.toml');
|
|
|
|
describe('modules/manager/cargo/extract', () => {
|
|
describe('extractPackageFile()', () => {
|
|
const config: ExtractConfig = {};
|
|
|
|
beforeEach(() => {
|
|
delete process.env.CARGO_REGISTRIES_PRIVATE_CRATES_INDEX;
|
|
delete process.env.CARGO_REGISTRIES_MCORBIN_INDEX;
|
|
});
|
|
|
|
it('returns null for invalid toml', async () => {
|
|
expect(
|
|
await extractPackageFile('invalid toml', 'Cargo.toml', config),
|
|
).toBeNull();
|
|
});
|
|
|
|
it('returns null for empty dependencies', async () => {
|
|
const cargotoml = '[dependencies]\n';
|
|
expect(
|
|
await extractPackageFile(cargotoml, 'Cargo.toml', config),
|
|
).toBeNull();
|
|
});
|
|
|
|
it('returns null for empty dev-dependencies', async () => {
|
|
const cargotoml = '[dev-dependencies]\n';
|
|
expect(
|
|
await extractPackageFile(cargotoml, 'Cargo.toml', config),
|
|
).toBeNull();
|
|
});
|
|
|
|
it('returns null for empty custom target', async () => {
|
|
const cargotoml = '[target."foo".dependencies]\n';
|
|
expect(
|
|
await extractPackageFile(cargotoml, 'Cargo.toml', config),
|
|
).toBeNull();
|
|
});
|
|
|
|
it('extracts multiple dependencies simple', async () => {
|
|
const res = await extractPackageFile(cargo1toml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(15);
|
|
});
|
|
|
|
it('extracts multiple dependencies advanced', async () => {
|
|
const res = await extractPackageFile(cargo2toml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(18 + 6 + 1);
|
|
});
|
|
|
|
it('handles inline tables', async () => {
|
|
const res = await extractPackageFile(cargo3toml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(8);
|
|
});
|
|
|
|
it('handles standard tables', async () => {
|
|
const res = await extractPackageFile(cargo4toml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(6);
|
|
});
|
|
|
|
it('extracts platform specific dependencies', async () => {
|
|
const res = await extractPackageFile(cargo5toml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(4);
|
|
});
|
|
|
|
it('extracts registry urls from .cargo/config.toml', async () => {
|
|
mockReadLocalFile({ '.cargo/config.toml': cargo6configtoml });
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(3);
|
|
});
|
|
|
|
it('extracts registry urls from .cargo/config (legacy path)', async () => {
|
|
mockReadLocalFile({ '.cargo/config.toml': cargo6configtoml });
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(3);
|
|
});
|
|
|
|
it('extracts overridden registry indexes from .cargo/config.toml', async () => {
|
|
mockReadLocalFile({
|
|
'.cargo/config.toml': codeBlock`[registries]
|
|
private-crates = { index = "https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git" }
|
|
|
|
[registries.mcorbin]
|
|
index = "https://github.com/mcorbin/testregistry"
|
|
|
|
[source.crates-io]
|
|
replace-with = "mcorbin"
|
|
|
|
[source.mcorbin]
|
|
replace-with = "private-crates"`,
|
|
});
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.1.0',
|
|
datasource: 'crate',
|
|
depName: 'proprietary-crate',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
registryUrls: [
|
|
'https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git',
|
|
],
|
|
},
|
|
{
|
|
currentValue: '3.0.0',
|
|
datasource: 'crate',
|
|
depName: 'mcorbin-test',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
registryUrls: [
|
|
'https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git',
|
|
],
|
|
},
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
registryUrls: [
|
|
'https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git',
|
|
],
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts overridden source registry indexes from .cargo/config.toml', async () => {
|
|
mockReadLocalFile({
|
|
'.cargo/config.toml': codeBlock`[source.crates-io-replacement]
|
|
registry = "https://github.com/replacement/testregistry"
|
|
|
|
[source.crates-io]
|
|
replace-with = "crates-io-replacement"`,
|
|
});
|
|
const res = await extractPackageFile(cargo7toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
registryUrls: ['https://github.com/replacement/testregistry'],
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts registries overridden to the default', async () => {
|
|
mockReadLocalFile({
|
|
'.cargo/config.toml': codeBlock`[source.mcorbin]
|
|
replace-with = "crates-io"
|
|
|
|
[source.private-crates]
|
|
replace-with = "mcorbin"`,
|
|
});
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.1.0',
|
|
datasource: 'crate',
|
|
depName: 'proprietary-crate',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
},
|
|
{
|
|
currentValue: '3.0.0',
|
|
datasource: 'crate',
|
|
depName: 'mcorbin-test',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
},
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts registries with an empty config.toml', async () => {
|
|
mockReadLocalFile({ '.cargo/config.toml': '' });
|
|
const res = await extractPackageFile(cargo5toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.2.37',
|
|
datasource: 'crate',
|
|
depName: 'wasm-bindgen',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
target: 'cfg(target_arch = "wasm32")',
|
|
},
|
|
{
|
|
currentValue: '0.3.14',
|
|
datasource: 'crate',
|
|
depName: 'js-sys',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
target: 'cfg(target_arch = "wasm32")',
|
|
},
|
|
{
|
|
currentValue: '',
|
|
datasource: 'crate',
|
|
depName: 'js_relative_import',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
skipReason: 'path-dependency',
|
|
target: 'cfg(target_arch = "wasm32")',
|
|
},
|
|
{
|
|
currentValue: '0.3.14',
|
|
datasource: 'crate',
|
|
depName: 'web-sys',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
target: 'cfg(target_arch = "wasm32")',
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts registry urls from environment', async () => {
|
|
process.env.CARGO_REGISTRIES_PRIVATE_CRATES_INDEX =
|
|
'https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git';
|
|
process.env.CARGO_REGISTRIES_MCORBIN_INDEX =
|
|
'https://github.com/mcorbin/testregistry';
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.1.0',
|
|
datasource: 'crate',
|
|
depName: 'proprietary-crate',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
registryUrls: [
|
|
'https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git',
|
|
],
|
|
},
|
|
{
|
|
currentValue: '3.0.0',
|
|
datasource: 'crate',
|
|
depName: 'mcorbin-test',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
registryUrls: ['https://github.com/mcorbin/testregistry'],
|
|
},
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts workspace dependencies', async () => {
|
|
const cargoToml = codeBlock`
|
|
[package]
|
|
name = "renovate-test"
|
|
version = "0.1.0"
|
|
authors = ["John Doe <john.doe@example.org>"]
|
|
edition = "2018"
|
|
|
|
[dependencies]
|
|
git2 = "0.14.0"
|
|
|
|
[workspace]
|
|
members = ["pcap-sys"]
|
|
|
|
[workspace.dependencies]
|
|
serde = "1.0.146"
|
|
tokio = { version = "1.21.1" }`;
|
|
const res = await extractPackageFile(cargoToml, 'Cargo.toml', config);
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.14.0',
|
|
datasource: 'crate',
|
|
depName: 'git2',
|
|
depType: 'dependencies',
|
|
managerData: { nestedVersion: false },
|
|
},
|
|
{
|
|
currentValue: '1.0.146',
|
|
datasource: 'crate',
|
|
depName: 'serde',
|
|
depType: 'workspace.dependencies',
|
|
managerData: { nestedVersion: false },
|
|
},
|
|
{
|
|
currentValue: '1.21.1',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'workspace.dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('skips workspace dependency', async () => {
|
|
const cargotoml = '[dependencies]\nfoobar = { workspace = true }';
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '',
|
|
datasource: 'crate',
|
|
depName: 'foobar',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
skipReason: 'inherited-dependency',
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('skips unknown registries', async () => {
|
|
const cargotoml =
|
|
'[dependencies]\nfoobar = { version = "0.1.0", registry = "not-listed" }';
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(1);
|
|
});
|
|
|
|
it('fails to parse cargo config with invalid TOML', async () => {
|
|
mockReadLocalFile({ '.cargo/config': '[registries' });
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(3);
|
|
});
|
|
|
|
it('ignore cargo config registries with missing index', async () => {
|
|
mockReadLocalFile({ '.cargo/config': '[registries.mine]\nfoo = "bar"' });
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(3);
|
|
});
|
|
|
|
it('ignore cargo config source replaced registries with missing index', async () => {
|
|
mockReadLocalFile({
|
|
'.cargo/config': codeBlock`[registries.mine]
|
|
foo = "bar"
|
|
|
|
[source.crates-io]
|
|
replace-with = "mine"`,
|
|
});
|
|
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.1.0',
|
|
datasource: 'crate',
|
|
depName: 'proprietary-crate',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
{
|
|
currentValue: '3.0.0',
|
|
datasource: 'crate',
|
|
depName: 'mcorbin-test',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('ignore cargo config with circular registry source replacements', async () => {
|
|
mockReadLocalFile({
|
|
'.cargo/config': codeBlock`[registries]
|
|
private-crates = { index = "https://dl.cloudsmith.io/basic/my-org/my-repo/cargo/index.git" }
|
|
|
|
[registries.mcorbin]
|
|
index = "https://github.com/mcorbin/testregistry"
|
|
|
|
[source.crates-io]
|
|
replace-with = "mcorbin"
|
|
|
|
[source.mcorbin]
|
|
replace-with = "private-crates"
|
|
|
|
[source.private-crates]
|
|
replace-with = "mcorbin"
|
|
`,
|
|
});
|
|
|
|
const res = await extractPackageFile(cargo6toml, 'Cargo.toml', {
|
|
...config,
|
|
});
|
|
expect(res?.deps).toEqual([
|
|
{
|
|
currentValue: '0.1.0',
|
|
datasource: 'crate',
|
|
depName: 'proprietary-crate',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
{
|
|
currentValue: '3.0.0',
|
|
datasource: 'crate',
|
|
depName: 'mcorbin-test',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: true,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
{
|
|
currentValue: '0.2',
|
|
datasource: 'crate',
|
|
depName: 'tokio',
|
|
depType: 'dependencies',
|
|
managerData: {
|
|
nestedVersion: false,
|
|
},
|
|
skipReason: 'unknown-registry',
|
|
},
|
|
]);
|
|
});
|
|
|
|
it('extracts original package name of renamed dependencies', async () => {
|
|
const cargotoml =
|
|
'[dependencies]\nboolector-solver = { package = "boolector", version = "0.4.0" }';
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
|
|
expect(res?.deps).toMatchSnapshot();
|
|
expect(res?.deps).toHaveLength(1);
|
|
expect(res?.deps[0].packageName).toBe('boolector');
|
|
});
|
|
|
|
it('extracts locked versions', async () => {
|
|
const cargolock = Fixtures.get('lockfile-update/Cargo.3.lock');
|
|
mockReadLocalFile({ 'Cargo.lock': cargolock });
|
|
fs.findLocalSiblingOrParent.mockResolvedValueOnce('Cargo.lock');
|
|
|
|
const cargotoml = codeBlock`
|
|
[package]
|
|
name = "test"
|
|
version = "0.1.0"
|
|
edition = "2021"
|
|
[dependencies]
|
|
syn = "2.0"
|
|
`;
|
|
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
expect(res?.deps).toMatchObject([{ lockedVersion: '2.0.1' }]);
|
|
});
|
|
|
|
it('extracts locked versions for renamed packages', async () => {
|
|
const cargolock = Fixtures.get('lockfile-update/Cargo.1.lock');
|
|
mockReadLocalFile({ 'Cargo.lock': cargolock });
|
|
fs.findLocalSiblingOrParent.mockResolvedValueOnce('Cargo.lock');
|
|
|
|
const res = await extractPackageFile(
|
|
lockfileUpdateCargotoml,
|
|
'Cargo.toml',
|
|
config,
|
|
);
|
|
expect(res?.deps).toMatchObject([
|
|
{ lockedVersion: '2.0.1' },
|
|
{ lockedVersion: '1.0.1' },
|
|
]);
|
|
});
|
|
|
|
it('handles missing locked versions', async () => {
|
|
const cargolock = Fixtures.get('lockfile-update/Cargo.2.lock');
|
|
mockReadLocalFile({ 'Cargo.lock': cargolock });
|
|
fs.findLocalSiblingOrParent.mockResolvedValueOnce('Cargo.lock');
|
|
|
|
const res = await extractPackageFile(
|
|
lockfileUpdateCargotoml,
|
|
'Cargo.toml',
|
|
config,
|
|
);
|
|
expect(res?.deps).toMatchObject([
|
|
{ lockedVersion: '2.0.1' },
|
|
expect.not.objectContaining({ lockedVersion: expect.anything() }),
|
|
]);
|
|
});
|
|
|
|
it('handles invalid versions in the toml file', async () => {
|
|
const cargolock = Fixtures.get('lockfile-update/Cargo.3.lock');
|
|
mockReadLocalFile({ 'Cargo.lock': cargolock });
|
|
fs.findLocalSiblingOrParent.mockResolvedValueOnce('Cargo.lock');
|
|
|
|
const cargotoml = codeBlock`
|
|
[package]
|
|
name = "test"
|
|
version = "0.1.0"
|
|
edition = "2021"
|
|
[dependencies]
|
|
syn = "2.foo.1"
|
|
`;
|
|
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
expect(res?.deps).not.toHaveProperty('lockedVersion');
|
|
});
|
|
|
|
it('handles invalid lock file', async () => {
|
|
mockReadLocalFile({ 'Cargo.lock': 'foo' });
|
|
fs.findLocalSiblingOrParent.mockResolvedValueOnce('Cargo.lock');
|
|
|
|
const res = await extractPackageFile(
|
|
lockfileUpdateCargotoml,
|
|
'Cargo.toml',
|
|
config,
|
|
);
|
|
expect(res?.deps).toMatchObject([
|
|
expect.not.objectContaining({ lockedVersion: expect.anything() }),
|
|
expect.not.objectContaining({ lockedVersion: expect.anything() }),
|
|
]);
|
|
});
|
|
|
|
it('should extract project version', async () => {
|
|
const cargotoml = codeBlock`
|
|
[package]
|
|
name = "test"
|
|
version = "0.1.0"
|
|
edition = "2021"
|
|
[dependencies]
|
|
syn = "2.0"
|
|
`;
|
|
|
|
const res = await extractPackageFile(cargotoml, 'Cargo.toml', config);
|
|
expect(res?.packageFileVersion).toBe('0.1.0');
|
|
});
|
|
});
|
|
});
|