aube_resolver/builder.rs
1use crate::FxHashMap;
2use crate::{
3 DependencyPolicy, MinimumReleaseAge, ReadPackageHook, ResolutionMode, ResolvedPackage,
4 Resolver, SupportedArchitectures, override_rule,
5};
6use aube_registry::client::RegistryClient;
7use std::collections::{BTreeMap, BTreeSet};
8use std::path::PathBuf;
9use std::sync::Arc;
10use tokio::sync::mpsc;
11
12impl Resolver {
13 /// Stream-channel capacity between resolver and fetch coordinator.
14 ///
15 /// Was 64 (matched fetch concurrency). Bumped to 1024 because the
16 /// channel is just a backpressure-bounded mpsc — its job is to
17 /// absorb resolver bursts so the BFS loop never blocks on
18 /// `send().await` while the fetch coordinator is mid-tarball. Each
19 /// `ResolvedPackage` is ~200 bytes, so 1024 = ~200 KiB worst-case.
20 /// Real install graphs sustain 5–10 in-flight packages per fetch
21 /// permit, so 1024 covers ~20 000-pkg installs without backpressure
22 /// while still bounding heap on a runaway producer.
23 const DEFAULT_STREAM_CAPACITY: usize = 1024;
24
25 pub fn new(client: Arc<RegistryClient>) -> Self {
26 Self {
27 client,
28 // 1024 covers typical monorepo without rehash. 5000-pkg tail pays one grow.
29 cache: FxHashMap::with_capacity_and_hasher(1024, Default::default()),
30 resolved_tx: None,
31 packument_cache_dir: None,
32 packument_full_cache_dir: None,
33 auto_install_peers: true,
34 exclude_links_from_lockfile: false,
35 supported_architectures: SupportedArchitectures::default(),
36 overrides: BTreeMap::new(),
37 override_rules: Vec::new(),
38 ignored_optional_dependencies: BTreeSet::new(),
39 resolution_mode: ResolutionMode::Highest,
40 project_root: PathBuf::from("."),
41 ignore_scripts: false,
42 minimum_release_age: None,
43 catalogs: BTreeMap::new(),
44 read_package_hook: None,
45 dependency_policy: DependencyPolicy::default(),
46 vulnerable_ranges: BTreeMap::new(),
47 git_shallow_hosts: Vec::new(),
48 peers_suffix_max_length: 1000,
49 dedupe_peer_dependents: true,
50 dedupe_peers: false,
51 resolve_peers_from_workspace_root: true,
52 registry_supports_time_field: false,
53 force_metadata_primer: false,
54 packument_network_concurrency: None,
55 }
56 }
57
58 /// Create a resolver that streams resolved packages through a channel.
59 /// Returns `(resolver, receiver)`. The receiver yields packages as they're
60 /// discovered, allowing tarball fetches to start during resolution.
61 pub fn with_stream(client: Arc<RegistryClient>) -> (Self, mpsc::Receiver<ResolvedPackage>) {
62 Self::with_stream_capacity(client, Self::DEFAULT_STREAM_CAPACITY)
63 }
64
65 /// Create a streaming resolver with a bounded resolved-package buffer.
66 pub fn with_stream_capacity(
67 client: Arc<RegistryClient>,
68 capacity: usize,
69 ) -> (Self, mpsc::Receiver<ResolvedPackage>) {
70 let (tx, rx) = mpsc::channel(capacity.max(1));
71 (
72 Self {
73 client,
74 // 1024 covers typical monorepo without rehash. 5000-pkg tail pays one grow.
75 cache: FxHashMap::with_capacity_and_hasher(1024, Default::default()),
76 resolved_tx: Some(tx),
77 packument_cache_dir: None,
78 packument_full_cache_dir: None,
79 auto_install_peers: true,
80 exclude_links_from_lockfile: false,
81 supported_architectures: SupportedArchitectures::default(),
82 overrides: BTreeMap::new(),
83 override_rules: Vec::new(),
84 ignored_optional_dependencies: BTreeSet::new(),
85 resolution_mode: ResolutionMode::Highest,
86 project_root: PathBuf::from("."),
87 ignore_scripts: false,
88 minimum_release_age: None,
89 catalogs: BTreeMap::new(),
90 read_package_hook: None,
91 dependency_policy: DependencyPolicy::default(),
92 vulnerable_ranges: BTreeMap::new(),
93 git_shallow_hosts: Vec::new(),
94 peers_suffix_max_length: 1000,
95 dedupe_peer_dependents: true,
96 dedupe_peers: false,
97 resolve_peers_from_workspace_root: true,
98 registry_supports_time_field: false,
99 force_metadata_primer: false,
100 packument_network_concurrency: None,
101 },
102 rx,
103 )
104 }
105
106 pub fn with_packument_network_concurrency(mut self, n: Option<usize>) -> Self {
107 self.packument_network_concurrency = n.filter(|&n| n > 0);
108 self
109 }
110
111 /// Enable disk-backed packument caching with ETag/Last-Modified revalidation.
112 pub fn with_packument_cache(mut self, cache_dir: std::path::PathBuf) -> Self {
113 self.packument_cache_dir = Some(cache_dir);
114 self
115 }
116
117 /// Disk cache for full (non-corgi) packuments, used in
118 /// `ResolutionMode::TimeBased` so we can read the `time:` map.
119 pub fn with_packument_full_cache(mut self, cache_dir: std::path::PathBuf) -> Self {
120 self.packument_full_cache_dir = Some(cache_dir);
121 self
122 }
123
124 /// Set the resolution mode. Defaults to `Highest` (pnpm's classic
125 /// behavior). `TimeBased` switches direct deps to lowest-satisfying
126 /// and constrains transitives by a publish-date cutoff.
127 pub fn with_resolution_mode(mut self, mode: ResolutionMode) -> Self {
128 self.resolution_mode = mode;
129 self
130 }
131
132 /// Configure pnpm v11's `minimumReleaseAge` family of settings.
133 /// Pass `None` (or a config with `minutes == 0`) to disable.
134 pub fn with_minimum_release_age(mut self, mra: Option<MinimumReleaseAge>) -> Self {
135 self.minimum_release_age = mra.filter(|m| m.minutes > 0);
136 self
137 }
138
139 /// Whether the resolver should round-trip registry `time:` entries
140 /// into the output graph. pnpm only writes `time:` to its lockfile
141 /// when one of `resolution-mode=time-based` / `minimumReleaseAge`
142 /// is active — otherwise the field is dead weight and, worse, shows
143 /// up as churn in a pnpm ↔ aube diff. Gate the insertion at the
144 /// two `resolved_times.insert` call sites on this predicate so
145 /// Highest-mode installs never populate the map.
146 pub(crate) fn should_record_times(&self) -> bool {
147 self.resolution_mode == ResolutionMode::TimeBased
148 || self.minimum_release_age.is_some()
149 || self.dependency_policy.trust_policy == crate::TrustPolicy::NoDowngrade
150 }
151
152 /// Override the default `auto-install-peers=true` behavior. pnpm reads
153 /// this from `.npmrc` or `pnpm-workspace.yaml`; aube's install command
154 /// plumbs the resolved value through here before running resolution.
155 pub fn with_auto_install_peers(mut self, auto_install_peers: bool) -> Self {
156 self.auto_install_peers = auto_install_peers;
157 self
158 }
159
160 /// Configure pnpm's `peersSuffixMaxLength`. When the peer suffix on a
161 /// `dep_path` would exceed this many bytes, the post-pass replaces it
162 /// with `_<10-char-sha256-hex>`. Default 1000 (pnpm's default).
163 pub fn with_peers_suffix_max_length(mut self, max_length: usize) -> Self {
164 self.peers_suffix_max_length = max_length;
165 self
166 }
167
168 /// Override the default `dedupe-peer-dependents=true` behavior. When
169 /// false, the peer-context pass keeps every distinct ancestor-scope
170 /// variant of a package instead of collapsing peer-equivalent ones
171 /// into a single dep_path. Plumbed from `.npmrc` /
172 /// `pnpm-workspace.yaml` via the install command.
173 pub fn with_dedupe_peer_dependents(mut self, value: bool) -> Self {
174 self.dedupe_peer_dependents = value;
175 self
176 }
177
178 /// Override the default `dedupe-peers=false` behavior. When true,
179 /// peer suffixes in the lockfile drop the peer name and emit only
180 /// the resolved version — `(18.2.0)` instead of `(react@18.2.0)`.
181 /// Plumbed from `.npmrc` / `pnpm-workspace.yaml` via the install
182 /// command.
183 pub fn with_dedupe_peers(mut self, value: bool) -> Self {
184 self.dedupe_peers = value;
185 self
186 }
187
188 /// Override the default `resolve-peers-from-workspace-root=true`
189 /// behavior. When false, peer resolution stops at the importer's
190 /// own scope + BFS-auto-installed transitives instead of consulting
191 /// the workspace root's direct deps as a fallback tier. Plumbed
192 /// from `.npmrc` / `pnpm-workspace.yaml` via the install command.
193 pub fn with_resolve_peers_from_workspace_root(mut self, value: bool) -> Self {
194 self.resolve_peers_from_workspace_root = value;
195 self
196 }
197
198 /// Configure pnpm's `registry-supports-time-field`. When true,
199 /// the resolver keeps using the abbreviated (corgi) packument
200 /// path even when `time:` is needed, saving one full-packument
201 /// fetch per distinct package. Safe for registries that embed
202 /// `time` in their abbreviated responses (Verdaccio 5.15.1+, JSR,
203 /// most in-house mirrors); leave at the default `false` for
204 /// npmjs.org.
205 pub fn with_registry_supports_time_field(mut self, value: bool) -> Self {
206 self.registry_supports_time_field = value;
207 self
208 }
209
210 /// Force the bundled metadata primer on for npm-compatible
211 /// mirrors. Normally the primer only seeds npmjs.org cache entries
212 /// because it was generated from npmjs metadata.
213 pub fn with_force_metadata_primer(mut self, value: bool) -> Self {
214 self.force_metadata_primer = value;
215 self
216 }
217
218 /// Configure pnpm's `exclude-links-from-lockfile` setting. Only
219 /// affects lockfile serialization — the resolver still builds the
220 /// same graph either way, but the value is stamped into
221 /// `LockfileGraph::settings` so the pnpm writer can filter `link:`
222 /// importer entries on write.
223 pub fn with_exclude_links_from_lockfile(mut self, value: bool) -> Self {
224 self.exclude_links_from_lockfile = value;
225 self
226 }
227
228 /// Override the host platform triple used when filtering optional
229 /// dependencies. See [`platform::SupportedArchitectures`].
230 pub fn with_supported_architectures(mut self, value: SupportedArchitectures) -> Self {
231 self.supported_architectures = value;
232 self
233 }
234
235 /// Provide dependency overrides. The map's keys are selector
236 /// strings — bare name, `parent>child`, `foo@<2`, `**/foo`, or any
237 /// combination thereof — and values are version specifiers (or
238 /// `npm:` aliases). Keys are compiled into `override_rule`
239 /// structures; unparseable keys are dropped. Whenever the resolver
240 /// encounters a task matching a rule (by name + ancestor chain +
241 /// optional version constraints), the requested range is replaced
242 /// with the rule's replacement before any packument fetch or
243 /// version pick. Workspace + manifest sources are merged by the
244 /// caller.
245 pub fn with_overrides(mut self, overrides: BTreeMap<String, String>) -> Self {
246 self.override_rules = override_rule::compile(&overrides);
247 self.overrides = overrides;
248 self
249 }
250
251 /// Provide workspace catalog ranges. Outer key is the catalog name
252 /// (`default` for the unnamed `catalog:` field in
253 /// `pnpm-workspace.yaml`); inner key is the package name. The
254 /// resolver rewrites `catalog:` and `catalog:<name>` task ranges
255 /// against this map before the override / npm-alias passes, and
256 /// records the picks in the output graph's `catalogs` field.
257 pub fn with_catalogs(mut self, catalogs: BTreeMap<String, BTreeMap<String, String>>) -> Self {
258 self.catalogs = catalogs;
259 self
260 }
261
262 /// Set the project root used to resolve `file:` / `link:` paths.
263 /// `file:./vendor/foo` resolves against this directory, and a
264 /// matching directory / tarball is read to drive resolution of the
265 /// local package's transitive deps.
266 pub fn with_project_root(mut self, project_root: PathBuf) -> Self {
267 self.project_root = project_root;
268 self
269 }
270
271 pub fn with_ignore_scripts(mut self, ignore_scripts: bool) -> Self {
272 self.ignore_scripts = ignore_scripts;
273 self
274 }
275
276 /// Names to strip from every `optionalDependencies` map before
277 /// enqueueing (pnpm's `pnpm.ignoredOptionalDependencies`). Applied
278 /// to both root and transitive optional deps. Empty by default.
279 pub fn with_ignored_optional_dependencies(mut self, ignored: BTreeSet<String>) -> Self {
280 self.ignored_optional_dependencies = ignored;
281 self
282 }
283
284 /// Install a `readPackage` hook. The resolver calls it once per
285 /// version-picked packument before enqueueing transitives; see
286 /// [`ReadPackageHook`] for what mutations are honored.
287 pub fn with_read_package_hook(mut self, hook: Box<dyn ReadPackageHook>) -> Self {
288 self.read_package_hook = Some(hook);
289 self
290 }
291
292 /// Configure dependency resolution policy settings such as
293 /// `packageExtensions`, `allowedDeprecatedVersions`, `trustPolicy*`,
294 /// and `blockExoticSubdeps`.
295 pub fn with_dependency_policy(mut self, policy: DependencyPolicy) -> Self {
296 self.dependency_policy = policy;
297 self
298 }
299
300 /// Prefer non-vulnerable versions for the supplied audit ranges.
301 /// Used by `audit --fix=update` to reuse the normal resolver while
302 /// steering only vulnerable packages away from affected versions.
303 pub fn with_vulnerable_ranges(mut self, ranges: BTreeMap<String, Vec<String>>) -> Self {
304 self.vulnerable_ranges = ranges;
305 self
306 }
307
308 /// Set the `git-shallow-hosts` list used when cloning git deps.
309 /// When a git URL's host matches an entry here (exact match,
310 /// same as pnpm), aube attempts a shallow fetch by SHA; other
311 /// hosts get a plain `git fetch origin`. An empty list forces
312 /// every git dep through the full-fetch path.
313 pub fn with_git_shallow_hosts(mut self, hosts: Vec<String>) -> Self {
314 self.git_shallow_hosts = hosts;
315 self
316 }
317}