Skip to content

Commit 64c3217

Browse files
committed
Auto merge of #4229 - azerupi:install-version-range, r=matklad
Implement semver ranges for install --vers This implements the design discussed in #4207 It allows to specify semver ranges on `cargo install ... --vers` 1. The first character of the `--vers` value is checked, if there is none we return an error. 2. If it is one of `<`, `>`, `=`, `^`, `~` we parse the value as a `VersionReq`, otherwise we parse it as a `Version`. 3. If the parsing as a `Version` fails but parsing as `VersionReq` succeeds, we add a note to the warning mentioning that a qualifier should be used to specify a semver range. This catches versions with less than tree digits. Otherwise, the previous behaviour is preserved with the warning of backwards compatibility. This means that - `cargo install ... --vers "^1.2.3"` will be parsed as a range - `cargo install ... --vers 1.2.3` will be parsed as a version - `cargo install ... --vers 1.2` will be parsed as a version for backwards compatibility reasons, fail and be passed through as is,**but** we will add a note `if you want to specify semver range, add an explicit qualifier, like ^1.2` - `cargo install ... --vers blah` will be parsed as a version for backwards compatibility reasons (which is weird because it is not even a valid semver range) and produce an `unknown error` down the line. I have left this behaviour untouched because it worked like that before, but I can easily make it error sooner with a better message.
2 parents 854bc16 + d9afb2d commit 64c3217

File tree

1 file changed

+40
-13
lines changed

1 file changed

+40
-13
lines changed

src/cargo/ops/cargo_install.rs

+40-13
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use std::io::SeekFrom;
88
use std::path::{Path, PathBuf};
99
use std::sync::Arc;
1010

11-
use semver::Version;
11+
use semver::{Version, VersionReq};
1212
use tempdir::TempDir;
1313
use toml;
1414

@@ -276,18 +276,45 @@ fn select_pkg<'a, T>(mut source: T,
276276
Some(name) => {
277277
let vers = match vers {
278278
Some(v) => {
279-
match v.parse::<Version>() {
280-
Ok(v) => Some(format!("={}", v)),
281-
Err(_) => {
282-
let msg = format!("the `--vers` provided, `{}`, is \
283-
not a valid semver version\n\n\
284-
historically Cargo treated this \
285-
as a semver version requirement \
286-
accidentally\nand will continue \
287-
to do so, but this behavior \
288-
will be removed eventually", v);
289-
config.shell().warn(&msg)?;
290-
Some(v.to_string())
279+
280+
// If the version begins with character <, >, =, ^, ~ parse it as a
281+
// version range, otherwise parse it as a specific version
282+
let first = v.chars()
283+
.nth(0)
284+
.ok_or("no version provided for the `--vers` flag")?;
285+
286+
match first {
287+
'<' | '>' | '=' | '^' | '~' => match v.parse::<VersionReq>() {
288+
Ok(v) => Some(v.to_string()),
289+
Err(_) => {
290+
let msg = format!("the `--vers` provided, `{}`, is \
291+
not a valid semver version requirement\n\n
292+
Please have a look at \
293+
http://doc.crates.io/specifying-dependencies.html \
294+
for the correct format", v);
295+
return Err(msg.into());
296+
}
297+
},
298+
_ => match v.parse::<Version>() {
299+
Ok(v) => Some(format!("={}", v)),
300+
Err(_) => {
301+
let mut msg = format!("the `--vers` provided, `{}`, is \
302+
not a valid semver version\n\n\
303+
historically Cargo treated this \
304+
as a semver version requirement \
305+
accidentally\nand will continue \
306+
to do so, but this behavior \
307+
will be removed eventually", v);
308+
309+
// If it is not a valid version but it is a valid version
310+
// requirement, add a note to the warning
311+
if v.parse::<VersionReq>().is_ok() {
312+
msg.push_str(&format!("\nif you want to specify semver range, \
313+
add an explicit qualifier, like ^{}", v));
314+
}
315+
config.shell().warn(&msg)?;
316+
Some(v.to_string())
317+
}
291318
}
292319
}
293320
}

0 commit comments

Comments
 (0)