Skip to content

Commit 66739f1

Browse files
committed
Command alias or Alias command #1091
Dearest Reviewer, This pull request closes #1091 which is a request to support aliases. This is not as powerful as something like git's alias, however, I think it sticks true to the original request. I high jack the processing of the args. After a few flags are checked and the args are parsed I check the config file for alias.COMMAND. If it is there I split it like args does and replace args[1] (the original command) with the alias command and its 'flags'. I have also included default short hand commands b, t and r. Thanks! Becker
1 parent 5716f32 commit 66739f1

File tree

3 files changed

+161
-3
lines changed

3 files changed

+161
-3
lines changed

src/bin/cargo.rs

+49-3
Original file line numberDiff line numberDiff line change
@@ -7,13 +7,14 @@ extern crate toml;
77
#[macro_use] extern crate log;
88

99
use std::collections::BTreeSet;
10+
use std::collections::HashMap;
1011
use std::env;
1112
use std::fs;
1213
use std::path::{Path,PathBuf};
1314

1415
use cargo::core::shell::Verbosity;
1516
use cargo::execute_main_without_stdin;
16-
use cargo::util::{self, CliResult, lev_distance, Config, human};
17+
use cargo::util::{self, CliResult, lev_distance, Config, human, CargoResult};
1718
use cargo::util::CliError;
1819
use cargo::util::process_builder::process;
1920

@@ -138,7 +139,7 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
138139
return Ok(None)
139140
}
140141

141-
let args = match &flags.arg_command[..] {
142+
let mut args = match &flags.arg_command[..] {
142143
// For the commands `cargo` and `cargo help`, re-execute ourselves as
143144
// `cargo -h` so we can go through the normal process of printing the
144145
// help message.
@@ -166,9 +167,30 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
166167
// For all other invocations, we're of the form `cargo foo args...`. We
167168
// use the exact environment arguments to preserve tokens like `--` for
168169
// example.
169-
_ => env::args().collect(),
170+
_ => {
171+
let mut default_alias = HashMap::new();
172+
default_alias.insert("b", "build".to_string());
173+
default_alias.insert("t", "test".to_string());
174+
default_alias.insert("r", "run".to_string());
175+
let mut args: Vec<String> = env::args().collect();
176+
if let Some(new_command) = default_alias.get(&args[1][..]){
177+
args[1] = new_command.clone();
178+
}
179+
args
180+
}
170181
};
171182

183+
let alias_list = try!(aliased_command(&config, &args[1]));
184+
if let Some(alias_command) = alias_list {
185+
// Replace old command with new command and flags
186+
let chain = args.iter().take(1)
187+
.chain(alias_command.iter())
188+
.chain(args.iter().skip(2))
189+
.map(|s| s.to_string())
190+
.collect();
191+
args = chain;
192+
}
193+
172194
macro_rules! cmd{
173195
($name:ident) => (if args[1] == stringify!($name).replace("_", "-") {
174196
config.shell().set_verbosity(Verbosity::Verbose);
@@ -186,6 +208,30 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
186208
Ok(None)
187209
}
188210

211+
fn aliased_command(config: &Config, command: &String) -> CargoResult<Option<Vec<String>>> {
212+
let alias_name = format!("alias.{}", command);
213+
let mut result = Ok(None);
214+
match config.get_string(&alias_name) {
215+
Ok(value) => {
216+
if let Some(record) = value {
217+
let alias_commands = record.val.split_whitespace()
218+
.map(|s| s.to_string())
219+
.collect();
220+
result = Ok(Some(alias_commands));
221+
}
222+
},
223+
Err(_) => {
224+
let value = try!(config.get_list(&alias_name));
225+
if let Some(record) = value {
226+
let alias_commands: Vec<String> = record.val.iter()
227+
.map(|s| s.0.to_string()).collect();
228+
result = Ok(Some(alias_commands));
229+
}
230+
}
231+
}
232+
result
233+
}
234+
189235
fn find_closest(config: &Config, cmd: &str) -> Option<String> {
190236
let cmds = list_commands(config);
191237
// Only consider candidates with a lev_distance of 3 or less so we don't

src/doc/config.md

+9
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,15 @@ color = 'auto' # whether cargo colorizes output
9393
# Network configuration
9494
[net]
9595
retry = 2 # number of times a network call will automatically retried
96+
97+
# Alias cargo commands. The first 3 aliases are built in. If your
98+
command requires grouped whitespace use the list format.
99+
[alias]
100+
b = "build"
101+
t = "test"
102+
r = "run"
103+
rr = "run --release"
104+
space_example = ["run", "--release", "--", "\"command list\""]
96105
```
97106

98107
# Environment Variables

tests/cargo_alias_config.rs

+103
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,103 @@
1+
extern crate cargotest;
2+
extern crate hamcrest;
3+
use cargotest::support::{project, execs, basic_bin_manifest};
4+
use hamcrest::{assert_that};
5+
6+
#[test]
7+
fn alias_incorrect_config_type() {
8+
let p = project("foo")
9+
.file("Cargo.toml", &basic_bin_manifest("foo"))
10+
.file("src/main.rs", r#"
11+
fn main() {
12+
}"#)
13+
.file(".cargo/config",r#"
14+
[alias]
15+
b-cargo-test = 5
16+
"#);;
17+
18+
assert_that(p.cargo_process("b-cargo-test").arg("-v"),
19+
execs().with_status(101).
20+
with_stderr_contains("[ERROR] invalid configuration \
21+
for key `alias.b-cargo-test`
22+
expected a list, but found a integer in [..]"));
23+
}
24+
25+
26+
#[test]
27+
fn alias_default_config_overrides_config() {
28+
let p = project("foo")
29+
.file("Cargo.toml", &basic_bin_manifest("foo"))
30+
.file("src/main.rs", r#"
31+
fn main() {
32+
}"#)
33+
.file(".cargo/config",r#"
34+
[alias]
35+
b = "not_build"
36+
"#);;
37+
38+
assert_that(p.cargo_process("b").arg("-v"),
39+
execs().with_status(0).
40+
with_stderr_contains("[COMPILING] foo v0.5.0 [..]"));
41+
}
42+
43+
#[test]
44+
fn alias_config() {
45+
let p = project("foo")
46+
.file("Cargo.toml", &basic_bin_manifest("foo"))
47+
.file("src/main.rs", r#"
48+
fn main() {
49+
}"#)
50+
.file(".cargo/config",r#"
51+
[alias]
52+
b-cargo-test = "build"
53+
"#);;
54+
55+
assert_that(p.cargo_process("b-cargo-test").arg("-v"),
56+
execs().with_status(0).
57+
with_stderr_contains("[COMPILING] foo v0.5.0 [..]
58+
[RUNNING] `rustc [..] --crate-name foo --crate-type \
59+
bin -g --out-dir [..] --emit=dep-info,link -L dependency=[..]\
60+
-L dependency=[..]"));
61+
}
62+
63+
#[test]
64+
fn alias_list_test() {
65+
let p = project("foo")
66+
.file("Cargo.toml", &basic_bin_manifest("foo"))
67+
.file("src/main.rs", r#"
68+
fn main() {
69+
}"#)
70+
.file(".cargo/config",r#"
71+
[alias]
72+
b-cargo-test = ["build", "--release"]
73+
"#);;
74+
75+
assert_that(p.cargo_process("b-cargo-test").arg("-v"),
76+
execs().with_status(0).
77+
with_stderr_contains("[COMPILING] foo v0.5.0 [..]").
78+
with_stderr_contains("[RUNNING] `rustc [..] --crate-name foo \
79+
--crate-type bin -C opt-level=3 --out-dir [..]\
80+
--emit=dep-info,link -L dependency=[..]")
81+
);
82+
}
83+
84+
#[test]
85+
fn alias_with_flags_config() {
86+
let p = project("foo")
87+
.file("Cargo.toml", &basic_bin_manifest("foo"))
88+
.file("src/main.rs", r#"
89+
fn main() {
90+
}"#)
91+
.file(".cargo/config",r#"
92+
[alias]
93+
b-cargo-test = "build --release"
94+
"#);;
95+
96+
assert_that(p.cargo_process("b-cargo-test").arg("-v"),
97+
execs().with_status(0).
98+
with_stderr_contains("[COMPILING] foo v0.5.0 [..]").
99+
with_stderr_contains("[RUNNING] `rustc [..] --crate-name foo \
100+
--crate-type bin -C opt-level=3 --out-dir [..]\
101+
--emit=dep-info,link -L dependency=[..]")
102+
);
103+
}

0 commit comments

Comments
 (0)