Skip to content

Commit e2d1488

Browse files
committed
Auto merge of #12083 - Eh2406:names, r=weihanglo
do not try an exponential number of package names re #11934, and as discussed in the cargo team meeting, this changes the strategy to "the original, all underscore, and all dashes". I was excessively proud of the `hyphen_combination_num` based implementation when I came up with it. But it's always been a hack. I'm glad to be the one to remove it.
2 parents 569b648 + d6021c9 commit e2d1488

File tree

3 files changed

+11
-91
lines changed

3 files changed

+11
-91
lines changed

src/cargo/sources/registry/index.rs

-85
Original file line numberDiff line numberDiff line change
@@ -81,91 +81,6 @@ use std::path::Path;
8181
use std::str;
8282
use std::task::{ready, Poll};
8383

84-
/// Crates.io treats hyphen and underscores as interchangeable, but the index and old Cargo do not.
85-
/// Therefore, the index must store uncanonicalized version of the name so old Cargo's can find it.
86-
/// This loop tries all possible combinations of switching hyphen and underscores to find the
87-
/// uncanonicalized one. As all stored inputs have the correct spelling, we start with the spelling
88-
/// as-provided.
89-
pub struct UncanonicalizedIter<'s> {
90-
input: &'s str,
91-
num_hyphen_underscore: u32,
92-
hyphen_combination_num: u16,
93-
}
94-
95-
impl<'s> UncanonicalizedIter<'s> {
96-
pub fn new(input: &'s str) -> Self {
97-
let num_hyphen_underscore = input.chars().filter(|&c| c == '_' || c == '-').count() as u32;
98-
UncanonicalizedIter {
99-
input,
100-
num_hyphen_underscore,
101-
hyphen_combination_num: 0,
102-
}
103-
}
104-
}
105-
106-
impl<'s> Iterator for UncanonicalizedIter<'s> {
107-
type Item = String;
108-
109-
fn next(&mut self) -> Option<Self::Item> {
110-
if self.hyphen_combination_num > 0
111-
&& self.hyphen_combination_num.trailing_zeros() >= self.num_hyphen_underscore
112-
{
113-
return None;
114-
}
115-
116-
let ret = Some(
117-
self.input
118-
.chars()
119-
.scan(0u16, |s, c| {
120-
// the check against 15 here's to prevent
121-
// shift overflow on inputs with more than 15 hyphens
122-
if (c == '_' || c == '-') && *s <= 15 {
123-
let switch = (self.hyphen_combination_num & (1u16 << *s)) > 0;
124-
let out = if (c == '_') ^ switch { '_' } else { '-' };
125-
*s += 1;
126-
Some(out)
127-
} else {
128-
Some(c)
129-
}
130-
})
131-
.collect(),
132-
);
133-
self.hyphen_combination_num += 1;
134-
ret
135-
}
136-
}
137-
138-
#[test]
139-
fn no_hyphen() {
140-
assert_eq!(
141-
UncanonicalizedIter::new("test").collect::<Vec<_>>(),
142-
vec!["test".to_string()]
143-
)
144-
}
145-
146-
#[test]
147-
fn two_hyphen() {
148-
assert_eq!(
149-
UncanonicalizedIter::new("te-_st").collect::<Vec<_>>(),
150-
vec![
151-
"te-_st".to_string(),
152-
"te__st".to_string(),
153-
"te--st".to_string(),
154-
"te_-st".to_string()
155-
]
156-
)
157-
}
158-
159-
#[test]
160-
fn overflow_hyphen() {
161-
assert_eq!(
162-
UncanonicalizedIter::new("te-_-_-_-_-_-_-_-_-st")
163-
.take(100)
164-
.count(),
165-
100
166-
)
167-
}
168-
16984
/// Manager for handling the on-disk index.
17085
///
17186
/// Note that local and remote registries store the index differently. Local

src/cargo/sources/registry/mod.rs

+9-3
Original file line numberDiff line numberDiff line change
@@ -850,9 +850,15 @@ impl<'cfg> Source for RegistrySource<'cfg> {
850850
// names to the original name. The resolver will later
851851
// reject any candidates that have the wrong name, and with this it'll
852852
// along the way produce helpful "did you mean?" suggestions.
853-
for name_permutation in
854-
index::UncanonicalizedIter::new(&dep.package_name()).take(1024)
855-
{
853+
// For now we only try the canonical lysing `-` to `_` and vice versa.
854+
// More advanced fuzzy searching become in the future.
855+
for name_permutation in [
856+
dep.package_name().replace('-', "_"),
857+
dep.package_name().replace('_', "-"),
858+
] {
859+
if name_permutation.as_str() == dep.package_name().as_str() {
860+
continue;
861+
}
856862
any_pending |= self
857863
.index
858864
.query_inner(

tests/testsuite/registry.rs

+2-3
Original file line numberDiff line numberDiff line change
@@ -3166,7 +3166,7 @@ fn not_found_permutations() {
31663166
authors = []
31673167
31683168
[dependencies]
3169-
a-b-c = "1.0"
3169+
a-b_c = "1.0"
31703170
"#,
31713171
)
31723172
.file("src/lib.rs", "")
@@ -3177,7 +3177,7 @@ fn not_found_permutations() {
31773177
.with_stderr(
31783178
"\
31793179
[UPDATING] `dummy-registry` index
3180-
error: no matching package named `a-b-c` found
3180+
error: no matching package named `a-b_c` found
31813181
location searched: registry `crates-io`
31823182
required by package `foo v0.0.1 ([ROOT]/foo)`
31833183
",
@@ -3190,7 +3190,6 @@ required by package `foo v0.0.1 ([ROOT]/foo)`
31903190
&[
31913191
"/index/a-/b-/a-b-c",
31923192
"/index/a-/b_/a-b_c",
3193-
"/index/a_/b-/a_b-c",
31943193
"/index/a_/b_/a_b_c"
31953194
]
31963195
);

0 commit comments

Comments
 (0)