mirror of
https://github.com/astral-sh/uv.git
synced 2025-07-07 21:35:00 +00:00

## Summary This PR enables the proof-of-concept resolver to backtrack by way of using the `pubgrub-rs` crate. Rather than using PubGrub as a _framework_ (implementing the `DependencyProvider` trait, letting PubGrub call us), I've instead copied over PubGrub's primary solver hook (which is only ~100 lines or so) and modified it for our purposes (e.g., made it async). There's a lot to improve here, but it's a start that will let us understand PubGrub's appropriateness for this problem space. A few observations: - In simple cases, the resolver is slower than our current (naive) resolver. I think it's just that the pipelining isn't as efficient as in the naive case, where we can just stream package and version fetches concurrently without any bottlenecks. - A lot of the code here relates to bridging PubGrub with our own abstractions -- so we need a `PubGrubPackage`, a `PubGrubVersion`, etc.
81 lines
3.1 KiB
Rust
81 lines
3.1 KiB
Rust
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
use pubgrub::error::PubGrubError;
|
|
use pubgrub::range::Range;
|
|
use pubgrub::report::{DefaultStringReporter, Reporter};
|
|
use pubgrub::solver::{resolve, OfflineDependencyProvider};
|
|
use pubgrub::version::SemanticVersion;
|
|
|
|
// `root` depends on `menu`, `icons 1.0.0` and `intl 5.0.0`
|
|
// `menu 1.0.0` depends on `dropdown < 2.0.0`
|
|
// `menu >= 1.1.0` depends on `dropdown >= 2.0.0`
|
|
// `dropdown 1.8.0` depends on `intl 3.0.0`
|
|
// `dropdown >= 2.0.0` depends on `icons 2.0.0`
|
|
// `icons` has no dependency
|
|
// `intl` has no dependency
|
|
#[rustfmt::skip]
|
|
fn main() {
|
|
let mut dependency_provider = OfflineDependencyProvider::<&str, SemanticVersion>::new();
|
|
// Direct dependencies: menu and icons.
|
|
dependency_provider.add_dependencies("root", (1, 0, 0), vec![
|
|
("menu", Range::any()),
|
|
("icons", Range::exact((1, 0, 0))),
|
|
("intl", Range::exact((5, 0, 0))),
|
|
]);
|
|
|
|
// Dependencies of the menu lib.
|
|
dependency_provider.add_dependencies("menu", (1, 0, 0), vec![
|
|
("dropdown", Range::strictly_lower_than((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("menu", (1, 1, 0), vec![
|
|
("dropdown", Range::higher_than((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("menu", (1, 2, 0), vec![
|
|
("dropdown", Range::higher_than((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("menu", (1, 3, 0), vec![
|
|
("dropdown", Range::higher_than((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("menu", (1, 4, 0), vec![
|
|
("dropdown", Range::higher_than((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("menu", (1, 5, 0), vec![
|
|
("dropdown", Range::higher_than((2, 0, 0))),
|
|
]);
|
|
|
|
// Dependencies of the dropdown lib.
|
|
dependency_provider.add_dependencies("dropdown", (1, 8, 0), vec![
|
|
("intl", Range::exact((3, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("dropdown", (2, 0, 0), vec![
|
|
("icons", Range::exact((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("dropdown", (2, 1, 0), vec![
|
|
("icons", Range::exact((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("dropdown", (2, 2, 0), vec![
|
|
("icons", Range::exact((2, 0, 0))),
|
|
]);
|
|
dependency_provider.add_dependencies("dropdown", (2, 3, 0), vec![
|
|
("icons", Range::exact((2, 0, 0))),
|
|
]);
|
|
|
|
// Icons have no dependencies.
|
|
dependency_provider.add_dependencies("icons", (1, 0, 0), vec![]);
|
|
dependency_provider.add_dependencies("icons", (2, 0, 0), vec![]);
|
|
|
|
// Intl have no dependencies.
|
|
dependency_provider.add_dependencies("intl", (3, 0, 0), vec![]);
|
|
dependency_provider.add_dependencies("intl", (4, 0, 0), vec![]);
|
|
dependency_provider.add_dependencies("intl", (5, 0, 0), vec![]);
|
|
|
|
// Run the algorithm.
|
|
match resolve(&dependency_provider, "root", (1, 0, 0)) {
|
|
Ok(sol) => println!("{:?}", sol),
|
|
Err(PubGrubError::NoSolution(mut derivation_tree)) => {
|
|
derivation_tree.collapse_no_versions();
|
|
eprintln!("{}", DefaultStringReporter::report(&derivation_tree));
|
|
}
|
|
Err(err) => panic!("{:?}", err),
|
|
};
|
|
}
|