mirror of
https://github.com/astral-sh/uv.git
synced 2025-09-24 19:22:35 +00:00

This enhances the hints generator in the resolver with some heuristic to detect and warn in case of failures due to version mismatches on a local package. Those may be the symptom of name conflict/shadowing with a transitive dependency. Closes: https://github.com/astral-sh/uv/issues/7329 --------- Co-authored-by: Zanie Blue <contact@zanie.dev>
4867 lines
143 KiB
Rust
4867 lines
143 KiB
Rust
#![cfg(all(feature = "python", feature = "pypi"))]
|
||
|
||
use anyhow::Result;
|
||
use assert_cmd::assert::OutputAssertExt;
|
||
use assert_fs::prelude::*;
|
||
use indoc::indoc;
|
||
use insta::assert_snapshot;
|
||
use std::path::Path;
|
||
|
||
use crate::common::{decode_token, packse_index_url};
|
||
use common::{uv_snapshot, TestContext};
|
||
|
||
mod common;
|
||
|
||
/// Add a PyPI requirement.
|
||
#[test]
|
||
fn add_registry() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio==3.7.0"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "3.7.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/c6/b3/fefbf7e78ab3b805dec67d698dc18dd505af7a18a8dd08868c9b4fa736b5/anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce", size = 142737 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/68/fe/7ce1926952c8a403b35029e194555558514b365ad77d75125f521a2bec62/anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0", size = 80873 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "anyio" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "anyio", specifier = "==3.7.0" }]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 4 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
// Adding with an ambiguous Git reference should treat it as a revision.
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage@0.0.1"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 5 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-public-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-public-pypackage@0dacfd662c64cb4ceb16e6cf65a157a8b715b979)
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage").arg("--tag=0.0.1"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 5 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
"uv-public-pypackage",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
uv-public-pypackage = { git = "https://github.com/astral-test/uv-public-pypackage", tag = "0.0.1" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "3.7.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/c6/b3/fefbf7e78ab3b805dec67d698dc18dd505af7a18a8dd08868c9b4fa736b5/anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce", size = 142737 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/68/fe/7ce1926952c8a403b35029e194555558514b365ad77d75125f521a2bec62/anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0", size = 80873 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "anyio" },
|
||
{ name = "uv-public-pypackage" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [
|
||
{ name = "anyio", specifier = "==3.7.0" },
|
||
{ name = "uv-public-pypackage", git = "https://github.com/astral-test/uv-public-pypackage?tag=0.0.1" },
|
||
]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "uv-public-pypackage"
|
||
version = "0.1.0"
|
||
source = { git = "https://github.com/astral-test/uv-public-pypackage?tag=0.0.1#0dacfd662c64cb4ceb16e6cf65a157a8b715b979" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 5 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement from a private repository, with credentials. The resolution should
|
||
/// succeed, but the `pyproject.toml` should omit the credentials.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_private_source() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
let token = decode_token(common::READ_ONLY_GITHUB_TOKEN);
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg(format!("uv-private-pypackage @ git+https://{token}@github.com/astral-test/uv-private-pypackage")), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-private-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-private-pypackage@d780faf0ac91257d4d5a4f0c5a0e4509608c0071)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"uv-private-pypackage",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
uv-private-pypackage = { git = "https://github.com/astral-test/uv-private-pypackage" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "uv-private-pypackage" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "uv-private-pypackage", git = "https://github.com/astral-test/uv-private-pypackage" }]
|
||
|
||
[[package]]
|
||
name = "uv-private-pypackage"
|
||
version = "0.1.0"
|
||
source = { git = "https://github.com/astral-test/uv-private-pypackage#d780faf0ac91257d4d5a4f0c5a0e4509608c0071" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement from a private repository, with credentials. Since `--raw-sources` is
|
||
/// specified, the `pyproject.toml` should retain the credentials.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_private_raw() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
let token = decode_token(common::READ_ONLY_GITHUB_TOKEN);
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg(format!("uv-private-pypackage @ git+https://{token}@github.com/astral-test/uv-private-pypackage")).arg("--raw-sources"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-private-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-private-pypackage@d780faf0ac91257d4d5a4f0c5a0e4509608c0071)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
let filters: Vec<_> = [(token.as_str(), "***")]
|
||
.into_iter()
|
||
.chain(context.filters())
|
||
.collect();
|
||
|
||
insta::with_settings!({
|
||
filters => filters
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"uv-private-pypackage @ git+https://***@github.com/astral-test/uv-private-pypackage",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "uv-private-pypackage" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "uv-private-pypackage", git = "https://github.com/astral-test/uv-private-pypackage" }]
|
||
|
||
[[package]]
|
||
name = "uv-private-pypackage"
|
||
version = "0.1.0"
|
||
source = { git = "https://github.com/astral-test/uv-private-pypackage#d780faf0ac91257d4d5a4f0c5a0e4509608c0071" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_error() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 1 package in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
// Provide a tag without a Git source.
|
||
uv_snapshot!(context.filters(), context.add().arg("flask").arg("--tag").arg("0.0.1"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: `flask` did not resolve to a Git repository, but a Git reference (`--tag 0.0.1`) was provided.
|
||
"###);
|
||
|
||
// Provide a tag with a non-Git source.
|
||
uv_snapshot!(context.filters(), context.add().arg("flask @ https://files.pythonhosted.org/packages/61/80/ffe1da13ad9300f87c93af113edd0638c75138c42a0994becfacac078c06/flask-3.0.3-py3-none-any.whl").arg("--branch").arg("0.0.1"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: `flask` did not resolve to a Git repository, but a Git reference (`--branch 0.0.1`) was provided.
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_branch() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage").arg("--branch").arg("test-branch"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-public-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-public-pypackage@0dacfd662c64cb4ceb16e6cf65a157a8b715b979)
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement using the `--raw-sources` API.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_raw() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
// Use an ambiguous tag reference, which would otherwise not resolve.
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage@0.0.1").arg("--raw-sources"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 5 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-public-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-public-pypackage@0dacfd662c64cb4ceb16e6cf65a157a8b715b979)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
"uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage@0.0.1",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "3.7.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/c6/b3/fefbf7e78ab3b805dec67d698dc18dd505af7a18a8dd08868c9b4fa736b5/anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce", size = 142737 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/68/fe/7ce1926952c8a403b35029e194555558514b365ad77d75125f521a2bec62/anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0", size = 80873 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "anyio" },
|
||
{ name = "uv-public-pypackage" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [
|
||
{ name = "anyio", specifier = "==3.7.0" },
|
||
{ name = "uv-public-pypackage", git = "https://github.com/astral-test/uv-public-pypackage?rev=0.0.1" },
|
||
]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "uv-public-pypackage"
|
||
version = "0.1.0"
|
||
source = { git = "https://github.com/astral-test/uv-public-pypackage?rev=0.0.1#0dacfd662c64cb4ceb16e6cf65a157a8b715b979" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 5 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement without the `git+` prefix.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_implicit() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
// Omit the `git+` prefix.
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ https://github.com/astral-test/uv-public-pypackage.git"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 5 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-public-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-public-pypackage.git@b270df1a2fb5d012294e9aaf05e7e0bab1e6a389)
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// `--raw-sources` should be considered conflicting with sources-specific arguments, like `--tag`.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_raw_error() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Provide a tag without a Git source.
|
||
uv_snapshot!(context.filters(), context.add().arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage").arg("--tag").arg("0.0.1").arg("--raw-sources"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: the argument '--tag <TAG>' cannot be used with '--raw-sources'
|
||
|
||
Usage: uv add --cache-dir [CACHE_DIR] --tag <TAG> --exclude-newer <EXCLUDE_NEWER> <PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
For more information, try '--help'.
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add an unnamed requirement.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_unnamed() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("git+https://github.com/astral-test/uv-public-pypackage").arg("--tag=0.0.1"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ uv-public-pypackage==0.1.0 (from git+https://github.com/astral-test/uv-public-pypackage@0dacfd662c64cb4ceb16e6cf65a157a8b715b979)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"uv-public-pypackage",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
uv-public-pypackage = { git = "https://github.com/astral-test/uv-public-pypackage", tag = "0.0.1" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "uv-public-pypackage" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "uv-public-pypackage", git = "https://github.com/astral-test/uv-public-pypackage?tag=0.0.1" }]
|
||
|
||
[[package]]
|
||
name = "uv-public-pypackage"
|
||
version = "0.1.0"
|
||
source = { git = "https://github.com/astral-test/uv-public-pypackage?tag=0.0.1#0dacfd662c64cb4ceb16e6cf65a157a8b715b979" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add and remove a development dependency.
|
||
#[test]
|
||
fn add_remove_dev() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio==3.7.0").arg("--dev"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv]
|
||
dev-dependencies = [
|
||
"anyio==3.7.0",
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "3.7.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/c6/b3/fefbf7e78ab3b805dec67d698dc18dd505af7a18a8dd08868c9b4fa736b5/anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce", size = 142737 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/68/fe/7ce1926952c8a403b35029e194555558514b365ad77d75125f521a2bec62/anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0", size = 80873 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
|
||
[package.dev-dependencies]
|
||
dev = [
|
||
{ name = "anyio" },
|
||
]
|
||
|
||
[package.metadata]
|
||
|
||
[package.metadata.requires-dev]
|
||
dev = [{ name = "anyio", specifier = "==3.7.0" }]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 4 packages in [TIME]
|
||
"###);
|
||
|
||
// This should fail without --dev.
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
warning: `anyio` is a development dependency; try calling `uv remove --dev`
|
||
error: The dependency `anyio` could not be found in `dependencies`
|
||
"###);
|
||
|
||
// Remove the dependency.
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio").arg("--dev"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 1 package in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 4 packages in [TIME]
|
||
Installed 1 package in [TIME]
|
||
- anyio==3.7.0
|
||
- idna==3.6
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
- sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv]
|
||
dev-dependencies = []
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 1 package in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add and remove an optional dependency.
|
||
#[test]
|
||
fn add_remove_optional() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio==3.7.0").arg("--optional=io"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[project.optional-dependencies]
|
||
io = [
|
||
"anyio==3.7.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "3.7.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/c6/b3/fefbf7e78ab3b805dec67d698dc18dd505af7a18a8dd08868c9b4fa736b5/anyio-3.7.0.tar.gz", hash = "sha256:275d9973793619a5374e1c89a4f4ad3f4b0a5510a2b5b939444bee8f4c4d37ce", size = 142737 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/68/fe/7ce1926952c8a403b35029e194555558514b365ad77d75125f521a2bec62/anyio-3.7.0-py3-none-any.whl", hash = "sha256:eddca883c4175f14df8aedce21054bfca3adb70ffe76a9f607aef9d7fa2ea7f0", size = 80873 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
|
||
[package.optional-dependencies]
|
||
io = [
|
||
{ name = "anyio" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "anyio", marker = "extra == 'io'", specifier = "==3.7.0" }]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile. At present, this will _uninstall_ the packages since `sync` does
|
||
// not include extras by default.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Uninstalled 3 packages in [TIME]
|
||
- anyio==3.7.0
|
||
- idna==3.6
|
||
- sniffio==1.3.1
|
||
"###);
|
||
|
||
// This should fail without --optional.
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
warning: `anyio` is an optional dependency; try calling `uv remove --optional io`
|
||
error: The dependency `anyio` could not be found in `dependencies`
|
||
"###);
|
||
|
||
// Remove the dependency.
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio").arg("--optional=io"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 1 package in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[project.optional-dependencies]
|
||
io = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 1 package in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
fn add_remove_inline_optional() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
optional-dependencies = { io = [
|
||
"anyio==3.7.0",
|
||
] }
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("typing-extensions").arg("--optional=types"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 5 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ typing-extensions==4.10.0
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
optional-dependencies = { io = [
|
||
"anyio==3.7.0",
|
||
], types = [
|
||
"typing-extensions>=4.10.0",
|
||
] }
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
uv_snapshot!(context.filters(), context.remove().arg("typing-extensions").arg("--optional=types"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Uninstalled 2 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
- typing-extensions==4.10.0
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
optional-dependencies = { io = [
|
||
"anyio==3.7.0",
|
||
], types = [] }
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add and remove a workspace dependency.
|
||
#[test]
|
||
fn add_remove_workspace() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let workspace = context.temp_dir.child("pyproject.toml");
|
||
workspace.write_str(indoc! {r#"
|
||
[tool.uv.workspace]
|
||
members = ["child1", "child2"]
|
||
"#})?;
|
||
|
||
let pyproject_toml = context.temp_dir.child("child1/pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let pyproject_toml = context.temp_dir.child("child2/pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "child2"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding a workspace package with a mismatched source should error.
|
||
let mut add_cmd = context.add();
|
||
add_cmd
|
||
.arg("child2 @ git+https://github.com/astral-test/uv-public-pypackage")
|
||
.arg("--package")
|
||
.arg("child1")
|
||
.current_dir(&context.temp_dir);
|
||
|
||
uv_snapshot!(context.filters(), add_cmd, @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: Workspace dependency `child2` must refer to local directory, not a Git repository
|
||
"###);
|
||
|
||
// Workspace packages should be detected automatically.
|
||
let child1 = context.temp_dir.join("child1");
|
||
let mut add_cmd = context.add();
|
||
add_cmd
|
||
.arg("child2")
|
||
.arg("--package")
|
||
.arg("child1")
|
||
.current_dir(&context.temp_dir);
|
||
|
||
uv_snapshot!(context.filters(), add_cmd, @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ child1==0.1.0 (from file://[TEMP_DIR]/child1)
|
||
+ child2==0.1.0 (from file://[TEMP_DIR]/child2)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(child1.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"child2",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
child2 = { workspace = true }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[manifest]
|
||
members = [
|
||
"child1",
|
||
"child2",
|
||
]
|
||
|
||
[[package]]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
source = { editable = "child1" }
|
||
dependencies = [
|
||
{ name = "child2" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "child2", editable = "child2" }]
|
||
|
||
[[package]]
|
||
name = "child2"
|
||
version = "0.1.0"
|
||
source = { editable = "child2" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen").current_dir(&child1), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
// Remove the dependency.
|
||
uv_snapshot!(context.filters(), context.remove().arg("child2").current_dir(&child1), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 2 packages in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ child1==0.1.0 (from file://[TEMP_DIR]/child1)
|
||
- child2==0.1.0 (from file://[TEMP_DIR]/child2)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(child1.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[manifest]
|
||
members = [
|
||
"child1",
|
||
"child2",
|
||
]
|
||
|
||
[[package]]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
source = { editable = "child1" }
|
||
|
||
[[package]]
|
||
name = "child2"
|
||
version = "0.1.0"
|
||
source = { editable = "child2" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen").current_dir(&child1), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 1 package in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a workspace dependency as an editable.
|
||
#[test]
|
||
fn add_workspace_editable() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let workspace = context.temp_dir.child("pyproject.toml");
|
||
workspace.write_str(indoc! {r#"
|
||
[project]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[tool.uv.workspace]
|
||
members = ["child1", "child2"]
|
||
"#})?;
|
||
|
||
let pyproject_toml = context.temp_dir.child("child1/pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let pyproject_toml = context.temp_dir.child("child2/pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "child2"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let child1 = context.temp_dir.join("child1");
|
||
let mut add_cmd = context.add();
|
||
add_cmd.arg("child2").arg("--editable").current_dir(&child1);
|
||
|
||
uv_snapshot!(context.filters(), add_cmd, @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 3 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ child1==0.1.0 (from file://[TEMP_DIR]/child1)
|
||
+ child2==0.1.0 (from file://[TEMP_DIR]/child2)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(child1.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"child2",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
child2 = { workspace = true }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[manifest]
|
||
members = [
|
||
"child1",
|
||
"child2",
|
||
"parent",
|
||
]
|
||
|
||
[[package]]
|
||
name = "child1"
|
||
version = "0.1.0"
|
||
source = { editable = "child1" }
|
||
dependencies = [
|
||
{ name = "child2" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "child2", editable = "child2" }]
|
||
|
||
[[package]]
|
||
name = "child2"
|
||
version = "0.1.0"
|
||
source = { editable = "child2" }
|
||
|
||
[[package]]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
source = { virtual = "." }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen").current_dir(&child1), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a workspace dependency via its path.
|
||
#[test]
|
||
fn add_workspace_path() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let workspace = context.temp_dir.child("pyproject.toml");
|
||
workspace.write_str(indoc! {r#"
|
||
[project]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[tool.uv.workspace]
|
||
members = ["child"]
|
||
"#})?;
|
||
|
||
let pyproject_toml = context.temp_dir.child("child/pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "child"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("./child"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
+ child==0.1.0 (from file://[TEMP_DIR]/child)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.child("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"child",
|
||
]
|
||
|
||
[tool.uv.workspace]
|
||
members = ["child"]
|
||
|
||
[tool.uv.sources]
|
||
child = { workspace = true }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[manifest]
|
||
members = [
|
||
"child",
|
||
"parent",
|
||
]
|
||
|
||
[[package]]
|
||
name = "child"
|
||
version = "0.1.0"
|
||
source = { editable = "child" }
|
||
|
||
[[package]]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
source = { virtual = "." }
|
||
dependencies = [
|
||
{ name = "child" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "child", editable = "child" }]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 1 package in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a path dependency.
|
||
#[test]
|
||
fn add_path() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let workspace = context.temp_dir.child("workspace");
|
||
workspace.child("pyproject.toml").write_str(indoc! {r#"
|
||
[project]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let child = workspace.child("packages").child("child");
|
||
child.child("pyproject.toml").write_str(indoc! {r#"
|
||
[project]
|
||
name = "child"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg(Path::new("packages").join("child")).current_dir(workspace.path()), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Using Python 3.12.[X] interpreter at: [PYTHON-3.12]
|
||
Creating virtual environment at: .venv
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ child==0.1.0 (from file://[TEMP_DIR]/workspace/packages/child)
|
||
+ parent==0.1.0 (from file://[TEMP_DIR]/workspace)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(workspace.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"child",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
child = { path = "packages/child" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// `uv add` implies a full lock and sync, including development dependencies.
|
||
let lock = fs_err::read_to_string(workspace.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "child"
|
||
version = "0.1.0"
|
||
source = { directory = "packages/child" }
|
||
|
||
[[package]]
|
||
name = "parent"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "child" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "child", directory = "packages/child" }]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen").current_dir(workspace.path()), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Update a requirement, modifying the source and extras.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn update() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["requests==2.31.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 6 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 6 packages in [TIME]
|
||
Installed 6 packages in [TIME]
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ requests==2.31.0
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
// Enable an extra (note the version specifier should be preserved).
|
||
uv_snapshot!(context.filters(), context.add().arg("requests[security]"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 6 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"requests[security]==2.31.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Enable extras using the CLI flag and add a marker.
|
||
uv_snapshot!(context.filters(), context.add().arg("requests; python_version > '3.7'").args(["--extra=use_chardet_on_py3", "--extra=socks"]), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 3 packages in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 3 packages in [TIME]
|
||
+ chardet==5.2.0
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ pysocks==1.7.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"requests[security]==2.31.0",
|
||
"requests[socks,use-chardet-on-py3]>=2.31.0 ; python_full_version >= '3.8'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Change the source by specifying a version (note the extras, markers, and version should be
|
||
// preserved).
|
||
uv_snapshot!(context.filters(), context.add().arg("requests @ git+https://github.com/psf/requests").arg("--tag=v2.32.3"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Uninstalled 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
- requests==2.31.0
|
||
+ requests==2.32.3 (from git+https://github.com/psf/requests@0e322af87745eff34caffe4df68456ebc20d9068)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"requests[security]==2.31.0",
|
||
"requests[socks,use-chardet-on-py3]>=2.31.0 ; python_full_version >= '3.8'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
requests = { git = "https://github.com/psf/requests", tag = "v2.32.3" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "certifi"
|
||
version = "2024.2.2"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/71/da/e94e26401b62acd6d91df2b52954aceb7f561743aa5ccc32152886c76c96/certifi-2024.2.2.tar.gz", hash = "sha256:0569859f95fc761b18b45ef421b1290a0f65f147e92a1e5eb3e635f9a5e4e66f", size = 164886 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/ba/06/a07f096c664aeb9f01624f858c3add0a4e913d6c96257acb4fce61e7de14/certifi-2024.2.2-py3-none-any.whl", hash = "sha256:dc383c07b76109f368f6106eee2b593b04a011ea4d55f652c6ca24a754d1cdd1", size = 163774 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "chardet"
|
||
version = "5.2.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/f3/0d/f7b6ab21ec75897ed80c17d79b15951a719226b9fababf1e40ea74d69079/chardet-5.2.0.tar.gz", hash = "sha256:1b3b6ff479a8c414bc3fa2c0852995695c4a026dcd6d0633b2dd092ca39c1cf7", size = 2069618 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/38/6f/f5fbc992a329ee4e0f288c1fe0e2ad9485ed064cac731ed2fe47dcc38cbf/chardet-5.2.0-py3-none-any.whl", hash = "sha256:e1cf59446890a00105fe7b7912492ea04b6e6f06d4b742b2c788469e34c82970", size = 199385 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "charset-normalizer"
|
||
version = "3.3.2"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/63/09/c1bc53dab74b1816a00d8d030de5bf98f724c52c1635e07681d312f20be8/charset-normalizer-3.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5", size = 104809 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/d1/b2/fcedc8255ec42afee97f9e6f0145c734bbe104aac28300214593eb326f1d/charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8", size = 192892 },
|
||
{ url = "https://files.pythonhosted.org/packages/2e/7d/2259318c202f3d17f3fe6438149b3b9e706d1070fe3fcbb28049730bb25c/charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b", size = 122213 },
|
||
{ url = "https://files.pythonhosted.org/packages/3a/52/9f9d17c3b54dc238de384c4cb5a2ef0e27985b42a0e5cc8e8a31d918d48d/charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6", size = 119404 },
|
||
{ url = "https://files.pythonhosted.org/packages/99/b0/9c365f6d79a9f0f3c379ddb40a256a67aa69c59609608fe7feb6235896e1/charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a", size = 137275 },
|
||
{ url = "https://files.pythonhosted.org/packages/91/33/749df346e93d7a30cdcb90cbfdd41a06026317bfbfb62cd68307c1a3c543/charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389", size = 147518 },
|
||
{ url = "https://files.pythonhosted.org/packages/72/1a/641d5c9f59e6af4c7b53da463d07600a695b9824e20849cb6eea8a627761/charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa", size = 140182 },
|
||
{ url = "https://files.pythonhosted.org/packages/ee/fb/14d30eb4956408ee3ae09ad34299131fb383c47df355ddb428a7331cfa1e/charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b", size = 141869 },
|
||
{ url = "https://files.pythonhosted.org/packages/df/3e/a06b18788ca2eb6695c9b22325b6fde7dde0f1d1838b1792a0076f58fe9d/charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed", size = 144042 },
|
||
{ url = "https://files.pythonhosted.org/packages/45/59/3d27019d3b447a88fe7e7d004a1e04be220227760264cc41b405e863891b/charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26", size = 138275 },
|
||
{ url = "https://files.pythonhosted.org/packages/7b/ef/5eb105530b4da8ae37d506ccfa25057961b7b63d581def6f99165ea89c7e/charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d", size = 144819 },
|
||
{ url = "https://files.pythonhosted.org/packages/a2/51/e5023f937d7f307c948ed3e5c29c4b7a3e42ed2ee0b8cdf8f3a706089bf0/charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068", size = 149415 },
|
||
{ url = "https://files.pythonhosted.org/packages/24/9d/2e3ef673dfd5be0154b20363c5cdcc5606f35666544381bee15af3778239/charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143", size = 141212 },
|
||
{ url = "https://files.pythonhosted.org/packages/5b/ae/ce2c12fcac59cb3860b2e2d76dc405253a4475436b1861d95fe75bdea520/charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4", size = 142167 },
|
||
{ url = "https://files.pythonhosted.org/packages/ed/3a/a448bf035dce5da359daf9ae8a16b8a39623cc395a2ffb1620aa1bce62b0/charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7", size = 93041 },
|
||
{ url = "https://files.pythonhosted.org/packages/b6/7c/8debebb4f90174074b827c63242c23851bdf00a532489fba57fef3416e40/charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001", size = 100397 },
|
||
{ url = "https://files.pythonhosted.org/packages/28/76/e6222113b83e3622caa4bb41032d0b1bf785250607392e1b778aca0b8a7d/charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc", size = 48543 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "requests", extra = ["socks", "use-chardet-on-py3"] },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [
|
||
{ name = "requests", extras = ["security"], git = "https://github.com/psf/requests?tag=v2.32.3" },
|
||
{ name = "requests", extras = ["socks", "use-chardet-on-py3"], marker = "python_full_version >= '3.8'", git = "https://github.com/psf/requests?tag=v2.32.3" },
|
||
]
|
||
|
||
[[package]]
|
||
name = "pysocks"
|
||
version = "1.7.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bd/11/293dd436aea955d45fc4e8a35b6ae7270f5b8e00b53cf6c024c83b657a11/PySocks-1.7.1.tar.gz", hash = "sha256:3f8804571ebe159c380ac6de37643bb4685970655d3bba243530d6558b799aa0", size = 284429 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/8d/59/b4572118e098ac8e46e399a1dd0f2d85403ce8bbaad9ec79373ed6badaf9/PySocks-1.7.1-py3-none-any.whl", hash = "sha256:2725bd0a9925919b9b51739eea5f9e2bae91e83288108a9ad338b2e3a4435ee5", size = 16725 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "requests"
|
||
version = "2.32.3"
|
||
source = { git = "https://github.com/psf/requests?tag=v2.32.3#0e322af87745eff34caffe4df68456ebc20d9068" }
|
||
dependencies = [
|
||
{ name = "certifi" },
|
||
{ name = "charset-normalizer" },
|
||
{ name = "idna" },
|
||
{ name = "urllib3" },
|
||
]
|
||
|
||
[package.optional-dependencies]
|
||
socks = [
|
||
{ name = "pysocks" },
|
||
]
|
||
use-chardet-on-py3 = [
|
||
{ name = "chardet" },
|
||
]
|
||
|
||
[[package]]
|
||
name = "urllib3"
|
||
version = "2.2.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/7a/50/7fd50a27caa0652cd4caf224aa87741ea41d3265ad13f010886167cfcc79/urllib3-2.2.1.tar.gz", hash = "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19", size = 291020 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/a2/73/a68704750a7679d0b6d3ad7aa8d4da8e14e151ae82e6fee774e6e0d05ec8/urllib3-2.2.1-py3-none-any.whl", hash = "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d", size = 121067 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 8 packages in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add and update a requirement, with different markers
|
||
#[test]
|
||
fn add_update_marker() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = ["requests>=2.30; python_version >= '3.11'"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 6 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 6 packages in [TIME]
|
||
Installed 6 packages in [TIME]
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ requests==2.31.0
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
// Restrict the `requests` version for Python <3.11
|
||
uv_snapshot!(context.filters(), context.add().arg("requests>=2.0,<2.29; python_version < '3.11'"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
// Should add a new line for the dependency since the marker does not match an existing one
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = [
|
||
"requests>=2.0,<2.29 ; python_full_version < '3.11'",
|
||
"requests>=2.30; python_version >= '3.11'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Change the restricted `requests` version for Python <3.11
|
||
uv_snapshot!(context.filters(), context.add().arg("requests>=2.0,<2.20; python_version < '3.11'"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 10 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
// Should mutate the existing dependency since the marker matches
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = [
|
||
"requests>=2.0,<2.20 ; python_full_version < '3.11'",
|
||
"requests>=2.30; python_version >= '3.11'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Restrict the `requests` version on Windows and Python >3.11
|
||
uv_snapshot!(context.filters(), context.add().arg("requests>=2.31 ; sys_platform == 'win32' and python_version > '3.11'"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 3 packages in [TIME]
|
||
Uninstalled 3 packages in [TIME]
|
||
Installed 3 packages in [TIME]
|
||
- idna==3.6
|
||
+ idna==2.7
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
- urllib3==2.2.1
|
||
+ urllib3==1.23
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
// Should add a new line for the dependency since the marker does not match an existing one
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = [
|
||
"requests>=2.0,<2.20 ; python_full_version < '3.11'",
|
||
"requests>=2.30; python_version >= '3.11'",
|
||
"requests>=2.31 ; python_full_version >= '3.12' and sys_platform == 'win32'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Restrict the `requests` version on Windows
|
||
uv_snapshot!(context.filters(), context.add().arg("requests>=2.10 ; sys_platform == 'win32'"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
// Should add a new line for the dependency since the marker does not exactly match an existing
|
||
// one — although it is a subset of the existing marker.
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = [
|
||
"requests>=2.0,<2.20 ; python_full_version < '3.11'",
|
||
"requests>=2.10 ; sys_platform == 'win32'",
|
||
"requests>=2.30; python_version >= '3.11'",
|
||
"requests>=2.31 ; python_full_version >= '3.12' and sys_platform == 'win32'",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Remove `requests`
|
||
uv_snapshot!(context.filters(), context.remove().arg("requests"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 1 package in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 6 packages in [TIME]
|
||
Installed 1 package in [TIME]
|
||
- certifi==2024.2.2
|
||
- charset-normalizer==3.3.2
|
||
- idna==2.7
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
- requests==2.31.0
|
||
- urllib3==1.23
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
// Should remove all variants of `requests`
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.8"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn update_source_replace_url() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"requests[security] @ https://files.pythonhosted.org/packages/f9/9b/335f9764261e915ed497fcdeb11df5dfd6f7bf257d4a6a2a686d80da4d54/requests-2.32.3-py3-none-any.whl"
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Change the source. The existing URL should be removed.
|
||
uv_snapshot!(context.filters(), context.add().arg("requests @ git+https://github.com/psf/requests").arg("--tag=v2.32.3"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 6 packages in [TIME]
|
||
Prepared 6 packages in [TIME]
|
||
Installed 6 packages in [TIME]
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ requests==2.32.3 (from git+https://github.com/psf/requests@0e322af87745eff34caffe4df68456ebc20d9068)
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"requests[security]",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
requests = { git = "https://github.com/psf/requests", tag = "v2.32.3" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Adding a dependency does not remove untracked dependencies from the environment.
|
||
#[test]
|
||
fn add_inexact() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio == 3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
// Manually remove a dependency.
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("iniconfig==2.0.0"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Uninstalled 1 package in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ iniconfig==2.0.0
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"iniconfig==2.0.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "iniconfig"
|
||
version = "2.0.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/d7/4b/cbd8e699e64a6f16ca3a8220661b5f83792b3017d0f79807cb8708d33913/iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3", size = 4646 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/ef/a6/62565a6e1cf69e10f5727360368e451d4b7f58beeac6173dc9db836a5b46/iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374", size = 5892 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "iniconfig" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "iniconfig", specifier = "==2.0.0" }]
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile without removing extraneous packages from the environment.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen").arg("--inexact"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 2 packages in [TIME]
|
||
"###);
|
||
|
||
// Install from the lockfile, performing an exact sync.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Uninstalled 3 packages in [TIME]
|
||
- anyio==3.7.0
|
||
- idna==3.6
|
||
- sniffio==1.3.1
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Remove a PyPI requirement.
|
||
#[test]
|
||
fn remove_registry() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.lock(), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 1 package in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Uninstalled 4 packages in [TIME]
|
||
Installed 1 package in [TIME]
|
||
- anyio==3.7.0
|
||
- idna==3.6
|
||
~ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
- sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Install from the lockfile.
|
||
uv_snapshot!(context.filters(), context.sync().arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Audited 1 package in [TIME]
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
fn add_preserves_indentation_in_pyproject_toml() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("requests==2.31.0"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 8 packages in [TIME]
|
||
Installed 8 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ requests==2.31.0
|
||
+ sniffio==1.3.1
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
"requests==2.31.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
fn add_puts_default_indentation_in_pyproject_toml_if_not_observed() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio==3.7.0"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("requests==2.31.0"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 8 packages in [TIME]
|
||
Prepared 8 packages in [TIME]
|
||
Installed 8 packages in [TIME]
|
||
+ anyio==3.7.0
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ requests==2.31.0
|
||
+ sniffio==1.3.1
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
"requests==2.31.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a requirement without updating the lockfile.
|
||
#[test]
|
||
fn add_frozen() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio==3.7.0").arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
assert!(!context.temp_dir.join("uv.lock").exists());
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a requirement without updating the environment.
|
||
#[test]
|
||
fn add_no_sync() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio==3.7.0").arg("--no-sync"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio==3.7.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
assert!(context.temp_dir.join("uv.lock").exists());
|
||
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
fn add_reject_multiple_git_ref_flags() {
|
||
let context = TestContext::new("3.12");
|
||
|
||
// --tag and --branch
|
||
uv_snapshot!(context
|
||
.add()
|
||
.arg("foo")
|
||
.arg("--tag")
|
||
.arg("0.0.1")
|
||
.arg("--branch")
|
||
.arg("test"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: the argument '--tag <TAG>' cannot be used with '--branch <BRANCH>'
|
||
|
||
Usage: uv add --cache-dir [CACHE_DIR] --tag <TAG> --exclude-newer <EXCLUDE_NEWER> <PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
For more information, try '--help'.
|
||
"###
|
||
);
|
||
|
||
// --tag and --rev
|
||
uv_snapshot!(context
|
||
.add()
|
||
.arg("foo")
|
||
.arg("--tag")
|
||
.arg("0.0.1")
|
||
.arg("--rev")
|
||
.arg("326b943"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: the argument '--tag <TAG>' cannot be used with '--rev <REV>'
|
||
|
||
Usage: uv add --cache-dir [CACHE_DIR] --tag <TAG> --exclude-newer <EXCLUDE_NEWER> <PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
For more information, try '--help'.
|
||
"###
|
||
);
|
||
|
||
// --tag and --tag
|
||
uv_snapshot!(context
|
||
.add()
|
||
.arg("foo")
|
||
.arg("--tag")
|
||
.arg("0.0.1")
|
||
.arg("--tag")
|
||
.arg("0.0.2"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: the argument '--tag <TAG>' cannot be used multiple times
|
||
|
||
Usage: uv add [OPTIONS] <PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
For more information, try '--help'.
|
||
"###
|
||
);
|
||
}
|
||
|
||
/// Avoiding persisting `add` calls when resolution fails.
|
||
#[test]
|
||
fn add_error() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("xyz"), @r###"
|
||
success: false
|
||
exit_code: 1
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
× No solution found when resolving dependencies:
|
||
╰─▶ Because there are no versions of xyz and your project depends on xyz, we can conclude that your project's requirements are unsatisfiable.
|
||
help: If this is intentional, run `uv add --frozen` to skip the lock and sync steps.
|
||
"###);
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("xyz").arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
"###);
|
||
|
||
let lock = context.temp_dir.join("uv.lock");
|
||
assert!(!lock.exists());
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Set a lower bound when adding unconstrained dependencies.
|
||
#[test]
|
||
fn add_lower_bound() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `anyio` should include a lower-bound.
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio>=4.3.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Avoid setting a lower bound when updating existing dependencies.
|
||
#[test]
|
||
fn add_lower_bound_existing() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `anyio` should _not_ set a lower-bound, since it's already present (even if
|
||
// unconstrained).
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Avoid setting a lower bound with `--raw-sources`.
|
||
#[test]
|
||
fn add_lower_bound_raw() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = ["anyio"]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `anyio` should _not_ set a lower-bound when using `--raw-sources`.
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio").arg("--raw-sources"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Set a lower bound when adding unconstrained dev dependencies.
|
||
#[test]
|
||
fn add_lower_bound_dev() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `anyio` should include a lower-bound.
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio").arg("--dev"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv]
|
||
dev-dependencies = [
|
||
"anyio>=4.3.0",
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Set a lower bound when adding unconstrained optional dependencies.
|
||
#[test]
|
||
fn add_lower_bound_optional() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `anyio` should include a lower-bound.
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio").arg("--optional=io"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[project.optional-dependencies]
|
||
io = [
|
||
"anyio>=4.3.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[[package]]
|
||
name = "anyio"
|
||
version = "4.3.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
dependencies = [
|
||
{ name = "idna" },
|
||
{ name = "sniffio" },
|
||
]
|
||
sdist = { url = "https://files.pythonhosted.org/packages/db/4d/3970183622f0330d3c23d9b8a5f52e365e50381fd484d08e3285104333d3/anyio-4.3.0.tar.gz", hash = "sha256:f75253795a87df48568485fd18cdd2a3fa5c4f7c5be8e5e36637733fce06fed6", size = 159642 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/14/fd/2f20c40b45e4fb4324834aea24bd4afdf1143390242c0b33774da0e2e34f/anyio-4.3.0-py3-none-any.whl", hash = "sha256:048e05d0f6caeed70d731f3db756d35dcc1f35747c8c403364a8332c630441b8", size = 85584 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "idna"
|
||
version = "3.6"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/bf/3f/ea4b9117521a1e9c50344b909be7886dd00a519552724809bb1f486986c2/idna-3.6.tar.gz", hash = "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", size = 175426 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/c2/e7/a82b05cf63a603df6e68d59ae6a68bf5064484a0718ea5033660af4b54a9/idna-3.6-py3-none-any.whl", hash = "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f", size = 61567 },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
|
||
[package.optional-dependencies]
|
||
io = [
|
||
{ name = "anyio" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "anyio", marker = "extra == 'io'", specifier = ">=4.3.0" }]
|
||
|
||
[[package]]
|
||
name = "sniffio"
|
||
version = "1.3.1"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/a2/87/a6771e1546d97e7e041b6ae58d80074f81b7d5121207425c964ddf5cfdbd/sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc", size = 20372 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/e9/44/75a9c9421471a6c4805dbf2356f7c181a29c1879239abab1ea2cc8f38b40/sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2", size = 10235 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Omit the local segment when adding dependencies (since `>=1.2.3+local` is invalid).
|
||
#[test]
|
||
fn add_lower_bound_local() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `torch` should include a lower-bound, but no local segment.
|
||
uv_snapshot!(context.filters(), context.add().arg("local-simple-a").arg("--extra-index-url").arg(packse_index_url()).env_remove("UV_EXCLUDE_NEWER"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ local-simple-a==1.2.3+foo
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"local-simple-a>=1.2.3",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[[package]]
|
||
name = "local-simple-a"
|
||
version = "1.2.3+foo"
|
||
source = { registry = "https://astral-sh.github.io/packse/PACKSE_VERSION/simple-html/" }
|
||
sdist = { url = "https://astral-sh.github.io/packse/PACKSE_VERSION/files/local_simple_a-1.2.3+foo.tar.gz", hash = "sha256:ebd55c4a79d0a5759126657cb289ff97558902abcfb142e036b993781497edac" }
|
||
wheels = [
|
||
{ url = "https://astral-sh.github.io/packse/PACKSE_VERSION/files/local_simple_a-1.2.3+foo-py3-none-any.whl", hash = "sha256:6f30e2e709b3e171cd734bb58705229a582587c29e0a7041227435583c7224cc" },
|
||
]
|
||
|
||
[[package]]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
source = { editable = "." }
|
||
dependencies = [
|
||
{ name = "local-simple-a" },
|
||
]
|
||
|
||
[package.metadata]
|
||
requires-dist = [{ name = "local-simple-a", specifier = ">=1.2.3" }]
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add dependencies to a (legacy) non-project workspace root.
|
||
#[test]
|
||
fn add_non_project() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r"
|
||
[tool.uv.workspace]
|
||
members = []
|
||
"})?;
|
||
|
||
// Adding `iniconfig` should fail, since virtual workspace roots don't support production
|
||
// dependencies.
|
||
uv_snapshot!(context.filters(), context.add().arg("iniconfig"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: Project is missing a `[project]` table; add a `[project]` table to use production dependencies, or run `uv add --dev` instead
|
||
"###);
|
||
|
||
// Adding `iniconfig` as optional should fail, since virtual workspace roots don't support
|
||
// optional dependencies.
|
||
uv_snapshot!(context.filters(), context.add().arg("iniconfig").arg("--optional").arg("async"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: Project is missing a `[project]` table; add a `[project]` table to use optional dependencies, or run `uv add --dev` instead
|
||
"###);
|
||
|
||
// Adding `iniconfig` as a dev dependency should succeed.
|
||
uv_snapshot!(context.filters(), context.add().arg("iniconfig").arg("--dev"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
warning: No `requires-python` value found in the workspace. Defaulting to `>=3.12`.
|
||
Resolved 1 package in [TIME]
|
||
Prepared 1 package in [TIME]
|
||
Installed 1 package in [TIME]
|
||
+ iniconfig==2.0.0
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[tool.uv]
|
||
dev-dependencies = [
|
||
"iniconfig>=2.0.0",
|
||
]
|
||
[tool.uv.workspace]
|
||
members = []
|
||
"###
|
||
);
|
||
});
|
||
|
||
let lock = fs_err::read_to_string(context.temp_dir.join("uv.lock"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
lock, @r###"
|
||
version = 1
|
||
requires-python = ">=3.12"
|
||
|
||
[options]
|
||
exclude-newer = "2024-03-25T00:00:00Z"
|
||
|
||
[manifest]
|
||
requirements = [{ name = "iniconfig", specifier = ">=2.0.0" }]
|
||
|
||
[[package]]
|
||
name = "iniconfig"
|
||
version = "2.0.0"
|
||
source = { registry = "https://pypi.org/simple" }
|
||
sdist = { url = "https://files.pythonhosted.org/packages/d7/4b/cbd8e699e64a6f16ca3a8220661b5f83792b3017d0f79807cb8708d33913/iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3", size = 4646 }
|
||
wheels = [
|
||
{ url = "https://files.pythonhosted.org/packages/ef/a6/62565a6e1cf69e10f5727360368e451d4b7f58beeac6173dc9db836a5b46/iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374", size = 5892 },
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add the same requirement multiple times.
|
||
#[test]
|
||
fn add_repeat() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Prepared 4 packages in [TIME]
|
||
Installed 4 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ idna==3.6
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio>=4.3.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 4 packages in [TIME]
|
||
Audited 4 packages in [TIME]
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio>=4.3.0",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add from requirement file.
|
||
#[test]
|
||
fn add_requirements_file() -> Result<()> {
|
||
let context = TestContext::new("3.12").with_filtered_counts();
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let requirements_txt = context.temp_dir.child("requirements.txt");
|
||
requirements_txt
|
||
.write_str("Flask==2.3.2\nanyio @ git+https://github.com/agronholm/anyio.git@4.4.0")?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("-r").arg("requirements.txt"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved [N] packages in [TIME]
|
||
Prepared [N] packages in [TIME]
|
||
Installed [N] packages in [TIME]
|
||
+ anyio==4.4.0 (from git+https://github.com/agronholm/anyio.git@053e8f0a0f7b0f4a47a012eb5c6b1d9d84344e6a)
|
||
+ blinker==1.7.0
|
||
+ click==8.1.7
|
||
+ flask==2.3.2
|
||
+ idna==3.6
|
||
+ itsdangerous==2.1.2
|
||
+ jinja2==3.1.3
|
||
+ markupsafe==2.1.5
|
||
+ project==0.1.0 (from file://[TEMP_DIR]/)
|
||
+ sniffio==1.3.1
|
||
+ werkzeug==3.0.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"anyio",
|
||
"flask==2.3.2",
|
||
]
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[tool.uv.sources]
|
||
anyio = { git = "https://github.com/agronholm/anyio.git", rev = "4.4.0" }
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Passing a `setup.py` should fail.
|
||
uv_snapshot!(context.filters(), context.add().arg("-r").arg("setup.py"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: Adding requirements from a `setup.py` is not supported in `uv add`
|
||
"###);
|
||
|
||
// Passing nothing should fail.
|
||
uv_snapshot!(context.filters(), context.add(), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
error: the following required arguments were not provided:
|
||
<PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
Usage: uv add --cache-dir [CACHE_DIR] --exclude-newer <EXCLUDE_NEWER> <PACKAGES|--requirements <REQUIREMENTS>>
|
||
|
||
For more information, try '--help'.
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Add to a PEP 732 script.
|
||
#[test]
|
||
fn add_script() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("anyio").arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "anyio",
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
#[test]
|
||
fn add_script_relative_path() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let project = context.temp_dir.child("project");
|
||
project.child("pyproject.toml").write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
print("Hello, world!")
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("./project").arg("--editable").arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = [
|
||
# "project",
|
||
# ]
|
||
#
|
||
# [tool.uv.sources]
|
||
# project = { path = "project", editable = true }
|
||
# ///
|
||
print("Hello, world!")
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add to a script without an existing metadata table.
|
||
#[test]
|
||
fn add_script_without_metadata_table() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["rich", "requests<3"]).arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add to a script without an existing metadata table, but with a shebang.
|
||
#[test]
|
||
fn add_script_without_metadata_table_with_shebang() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
#!/usr/bin/env python3
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["rich", "requests<3"]).arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
#!/usr/bin/env python3
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add to a script with a metadata table and a shebang.
|
||
#[test]
|
||
fn add_script_with_metadata_table_and_shebang() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
#!/usr/bin/env python3
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = []
|
||
# ///
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["rich", "requests<3"]).arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
#!/usr/bin/env python3
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add to a script without a metadata table, but with a docstring.
|
||
#[test]
|
||
fn add_script_without_metadata_table_with_docstring() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
"""This is a script."""
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["rich", "requests<3"]).arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.12"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
"""This is a script."""
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Remove from a PEP732 script,
|
||
#[test]
|
||
fn remove_script() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# "anyio",
|
||
# ]
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.remove().arg("anyio").arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "requests<3",
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Remove last dependency PEP732 script
|
||
#[test]
|
||
fn remove_last_dep_script() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "rich",
|
||
# ]
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.remove().arg("rich").arg("--script").arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = []
|
||
# ///
|
||
|
||
import requests
|
||
from rich.pretty import pprint
|
||
|
||
resp = requests.get("https://peps.python.org/api/peps.json")
|
||
data = resp.json()
|
||
pprint([(k, v["title"]) for k, v in data.items()][:10])
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Add a Git requirement to PEP732 script.
|
||
#[test]
|
||
#[cfg(feature = "git")]
|
||
fn add_git_to_script() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let script = context.temp_dir.child("script.py");
|
||
script.write_str(indoc! {r#"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "anyio",
|
||
# ]
|
||
# ///
|
||
|
||
import anyio
|
||
import uv_public_pypackage
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context
|
||
.add()
|
||
.arg("uv-public-pypackage @ git+https://github.com/astral-test/uv-public-pypackage")
|
||
.arg("--tag=0.0.1")
|
||
.arg("--script")
|
||
.arg("script.py"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Updated `script.py`
|
||
"###);
|
||
|
||
let script_content = fs_err::read_to_string(context.temp_dir.join("script.py"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
script_content, @r###"
|
||
# /// script
|
||
# requires-python = ">=3.11"
|
||
# dependencies = [
|
||
# "anyio",
|
||
# "uv-public-pypackage",
|
||
# ]
|
||
#
|
||
# [tool.uv.sources]
|
||
# uv-public-pypackage = { git = "https://github.com/astral-test/uv-public-pypackage", tag = "0.0.1" }
|
||
# ///
|
||
|
||
import anyio
|
||
import uv_public_pypackage
|
||
"###
|
||
);
|
||
});
|
||
|
||
// Ensure that the script runs without error.
|
||
context.run().arg("script.py").assert().success();
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Revert changes to a `pyproject.toml` the `add` fails.
|
||
#[test]
|
||
fn fail_to_add_revert_project() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"#})?;
|
||
|
||
// Adding `pytorch==1.0.2` should produce an error
|
||
let filters = std::iter::once((r"exit code: 1", "exit status: 1"))
|
||
.chain(context.filters())
|
||
.collect::<Vec<_>>();
|
||
uv_snapshot!(filters, context.add().arg("pytorch==1.0.2"), @r###"
|
||
success: false
|
||
exit_code: 2
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 2 packages in [TIME]
|
||
error: Failed to prepare distributions
|
||
Caused by: Failed to fetch wheel: pytorch==1.0.2
|
||
Caused by: Build backend failed to build wheel through `build_wheel()` with exit status: 1
|
||
--- stdout:
|
||
|
||
--- stderr:
|
||
Traceback (most recent call last):
|
||
File "<string>", line 11, in <module>
|
||
File "[CACHE_DIR]/builds-v0/[TMP]/build_meta.py", line 410, in build_wheel
|
||
return self._build_with_temp_dir(
|
||
^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||
File "[CACHE_DIR]/builds-v0/[TMP]/build_meta.py", line 395, in _build_with_temp_dir
|
||
self.run_setup()
|
||
File "[CACHE_DIR]/builds-v0/[TMP]/build_meta.py", line 487, in run_setup
|
||
super().run_setup(setup_script=setup_script)
|
||
File "[CACHE_DIR]/builds-v0/[TMP]/build_meta.py", line 311, in run_setup
|
||
exec(code, locals())
|
||
File "<string>", line 15, in <module>
|
||
Exception: You tried to install "pytorch". The package named for PyTorch is "torch"
|
||
---
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
|
||
[build-system]
|
||
requires = ["setuptools>=42"]
|
||
build-backend = "setuptools.build_meta"
|
||
"###
|
||
);
|
||
});
|
||
|
||
Ok(())
|
||
}
|
||
|
||
/// Ensure that the added dependencies are sorted if the dependency list was already sorted prior
|
||
/// to the operation.
|
||
#[test]
|
||
fn sorted_dependencies() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
description = "Add your description here"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"CacheControl[filecache]>=0.14,<0.15",
|
||
"iniconfig",
|
||
]
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["typing-extensions", "anyio"]), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 13 packages in [TIME]
|
||
Prepared 12 packages in [TIME]
|
||
Installed 12 packages in [TIME]
|
||
+ anyio==4.3.0
|
||
+ cachecontrol==0.14.0
|
||
+ certifi==2024.2.2
|
||
+ charset-normalizer==3.3.2
|
||
+ filelock==3.13.1
|
||
+ idna==3.6
|
||
+ iniconfig==2.0.0
|
||
+ msgpack==1.0.8
|
||
+ requests==2.31.0
|
||
+ sniffio==1.3.1
|
||
+ typing-extensions==4.10.0
|
||
+ urllib3==2.2.1
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
description = "Add your description here"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"CacheControl[filecache]>=0.14,<0.15",
|
||
"anyio>=4.3.0",
|
||
"iniconfig",
|
||
"typing-extensions>=4.10.0",
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Ensure that the custom ordering of the dependencies is preserved
|
||
/// after adding a package.
|
||
#[test]
|
||
fn custom_dependencies() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
description = "Add your description here"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"yarl",
|
||
"CacheControl[filecache]>=0.14,<0.15",
|
||
"mwparserfromhell",
|
||
"pywikibot",
|
||
"sentry-sdk",
|
||
]
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().arg("pydantic").arg("--frozen"), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
description = "Add your description here"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"yarl",
|
||
"CacheControl[filecache]>=0.14,<0.15",
|
||
"mwparserfromhell",
|
||
"pywikibot",
|
||
"sentry-sdk",
|
||
"pydantic",
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Regression test for: <https://github.com/astral-sh/uv/issues/7259>
|
||
#[test]
|
||
fn update_offset() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"iniconfig",
|
||
]
|
||
"#})?;
|
||
|
||
uv_snapshot!(context.filters(), context.add().args(["typing-extensions", "iniconfig"]), @r###"
|
||
success: true
|
||
exit_code: 0
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
Resolved 3 packages in [TIME]
|
||
Prepared 2 packages in [TIME]
|
||
Installed 2 packages in [TIME]
|
||
+ iniconfig==2.0.0
|
||
+ typing-extensions==4.10.0
|
||
"###);
|
||
|
||
let pyproject_toml = fs_err::read_to_string(context.temp_dir.join("pyproject.toml"))?;
|
||
|
||
insta::with_settings!({
|
||
filters => context.filters(),
|
||
}, {
|
||
assert_snapshot!(
|
||
pyproject_toml, @r###"
|
||
[project]
|
||
name = "project"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = [
|
||
"iniconfig",
|
||
"typing-extensions>=4.10.0",
|
||
]
|
||
"###
|
||
);
|
||
});
|
||
Ok(())
|
||
}
|
||
|
||
/// Check hint for <https://github.com/astral-sh/uv/issues/7329>
|
||
/// if there is a broken cyclic dependency on a local package.
|
||
#[test]
|
||
fn add_shadowed_name() -> Result<()> {
|
||
let context = TestContext::new("3.12");
|
||
|
||
let pyproject_toml = context.temp_dir.child("pyproject.toml");
|
||
pyproject_toml.write_str(indoc! {r#"
|
||
[project]
|
||
name = "dagster"
|
||
version = "0.1.0"
|
||
requires-python = ">=3.12"
|
||
dependencies = []
|
||
"#})?;
|
||
|
||
// Pinned constrained, check for a direct dependency loop.
|
||
uv_snapshot!(context.filters(), context.add().arg("dagster-webserver==1.6.13"), @r###"
|
||
success: false
|
||
exit_code: 1
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
× No solution found when resolving dependencies:
|
||
╰─▶ Because dagster-webserver==1.6.13 depends on your project and your project depends on dagster-webserver==1.6.13, we can conclude that your project's requirements are unsatisfiable.
|
||
|
||
hint: The package `dagster-webserver` depends on the package `dagster` but the name is shadowed by your project. Consider changing the name of the project.
|
||
help: If this is intentional, run `uv add --frozen` to skip the lock and sync steps.
|
||
"###);
|
||
|
||
// Constraint with several available versions, check for an indirect dependency loop.
|
||
uv_snapshot!(context.filters(), context.add().arg("dagster-webserver>=1.6.11,<1.7.0"), @r###"
|
||
success: false
|
||
exit_code: 1
|
||
----- stdout -----
|
||
|
||
----- stderr -----
|
||
× No solution found when resolving dependencies:
|
||
╰─▶ Because only the following versions of dagster-webserver are available:
|
||
dagster-webserver<=1.6.13
|
||
dagster-webserver>1.7.0
|
||
and dagster-webserver==1.6.11 depends on your project, we can conclude that all of:
|
||
dagster-webserver>=1.6.11,<1.6.12
|
||
dagster-webserver>1.6.13,<1.7.0
|
||
depend on your project.
|
||
And because dagster-webserver==1.6.12 depends on your project, we can conclude that all of:
|
||
dagster-webserver>=1.6.11,<1.6.13
|
||
dagster-webserver>1.6.13,<1.7.0
|
||
depend on your project.
|
||
And because dagster-webserver==1.6.13 depends on your project and your project depends on dagster-webserver>=1.6.11,<1.7.0, we can conclude that your project's requirements are unsatisfiable.
|
||
|
||
hint: The package `dagster-webserver` depends on the package `dagster` but the name is shadowed by your project. Consider changing the name of the project.
|
||
help: If this is intentional, run `uv add --frozen` to skip the lock and sync steps.
|
||
"###);
|
||
|
||
Ok(())
|
||
}
|