blob: 0196863c5ece61b30f22d335bd6380c76d6979c3 [file] [log] [blame]
//! Tests for the `cargo init` command.
use std::env;
use std::fs::{self, File};
use std::io::prelude::*;
use std::process::Command;
use cargo_test_support::{command_is_available, paths, Execs};
fn cargo_process(s: &str) -> Execs {
let mut execs = cargo_test_support::cargo_process(s);
execs.cwd(&paths::root()).env("HOME", &paths::home());
execs
}
fn mercurial_available() -> bool {
let result = Command::new("hg")
.arg("--version")
.output()
.map(|o| o.status.success())
.unwrap_or(false);
if !result {
println!("`hg` not available, skipping test");
}
result
}
#[cargo_test]
fn simple_lib() {
cargo_process("init --lib --vcs none --edition 2015")
.env("USER", "foo")
.with_stderr("[CREATED] library package")
.run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(!paths::root().join(".gitignore").is_file());
cargo_process("build").run();
}
#[cargo_test]
fn simple_bin() {
let path = paths::root().join("foo");
fs::create_dir(&path).unwrap();
cargo_process("init --bin --vcs none --edition 2015")
.env("USER", "foo")
.cwd(&path)
.with_stderr("[CREATED] binary (application) package")
.run();
assert!(paths::root().join("foo/Cargo.toml").is_file());
assert!(paths::root().join("foo/src/main.rs").is_file());
cargo_process("build").cwd(&path).run();
assert!(paths::root()
.join(&format!("foo/target/debug/foo{}", env::consts::EXE_SUFFIX))
.is_file());
}
#[cargo_test]
fn simple_git_ignore_exists() {
// write a .gitignore file with two entries
fs::create_dir_all(paths::root().join("foo")).unwrap();
fs::write(
paths::root().join("foo/.gitignore"),
"/target\n**/some.file",
)
.unwrap();
cargo_process("init --lib foo --edition 2015")
.env("USER", "foo")
.run();
assert!(paths::root().is_dir());
assert!(paths::root().join("foo/Cargo.toml").is_file());
assert!(paths::root().join("foo/src/lib.rs").is_file());
assert!(paths::root().join("foo/.git").is_dir());
assert!(paths::root().join("foo/.gitignore").is_file());
let fp = paths::root().join("foo/.gitignore");
let mut contents = String::new();
File::open(&fp)
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert_eq!(
contents,
"/target\n\
**/some.file\n\n\
#Added by cargo\n\
#\n\
#already existing elements were commented out\n\
\n\
#/target\n\
Cargo.lock\n",
);
cargo_process("build").cwd(&paths::root().join("foo")).run();
}
#[cargo_test]
fn git_ignore_exists_no_conflicting_entries() {
// write a .gitignore file with one entry
fs::create_dir_all(paths::root().join("foo")).unwrap();
fs::write(paths::root().join("foo/.gitignore"), "**/some.file").unwrap();
cargo_process("init --lib foo --edition 2015")
.env("USER", "foo")
.run();
let fp = paths::root().join("foo/.gitignore");
let mut contents = String::new();
File::open(&fp)
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert_eq!(
contents,
"**/some.file\n\n\
#Added by cargo\n\
\n\
/target\n\
Cargo.lock\n",
);
}
#[cargo_test]
fn both_lib_and_bin() {
cargo_process("init --lib --bin")
.env("USER", "foo")
.with_status(101)
.with_stderr("[ERROR] can't specify both lib and binary outputs")
.run();
}
fn bin_already_exists(explicit: bool, rellocation: &str) {
let path = paths::root().join("foo");
fs::create_dir_all(&path.join("src")).unwrap();
let sourcefile_path = path.join(rellocation);
let content = br#"
fn main() {
println!("Hello, world 2!");
}
"#;
File::create(&sourcefile_path)
.unwrap()
.write_all(content)
.unwrap();
if explicit {
cargo_process("init --bin --vcs none")
.env("USER", "foo")
.cwd(&path)
.run();
} else {
cargo_process("init --vcs none")
.env("USER", "foo")
.cwd(&path)
.run();
}
assert!(paths::root().join("foo/Cargo.toml").is_file());
assert!(!paths::root().join("foo/src/lib.rs").is_file());
// Check that our file is not overwritten
let mut new_content = Vec::new();
File::open(&sourcefile_path)
.unwrap()
.read_to_end(&mut new_content)
.unwrap();
assert_eq!(Vec::from(content as &[u8]), new_content);
}
#[cargo_test]
fn bin_already_exists_explicit() {
bin_already_exists(true, "src/main.rs")
}
#[cargo_test]
fn bin_already_exists_implicit() {
bin_already_exists(false, "src/main.rs")
}
#[cargo_test]
fn bin_already_exists_explicit_nosrc() {
bin_already_exists(true, "main.rs")
}
#[cargo_test]
fn bin_already_exists_implicit_nosrc() {
bin_already_exists(false, "main.rs")
}
#[cargo_test]
fn bin_already_exists_implicit_namenosrc() {
bin_already_exists(false, "foo.rs")
}
#[cargo_test]
fn bin_already_exists_implicit_namesrc() {
bin_already_exists(false, "src/foo.rs")
}
#[cargo_test]
fn confused_by_multiple_lib_files() {
let path = paths::root().join("foo");
fs::create_dir_all(&path.join("src")).unwrap();
let sourcefile_path1 = path.join("src/lib.rs");
File::create(&sourcefile_path1)
.unwrap()
.write_all(br#"fn qqq () { println!("Hello, world 2!"); }"#)
.unwrap();
let sourcefile_path2 = path.join("lib.rs");
File::create(&sourcefile_path2)
.unwrap()
.write_all(br#" fn qqq () { println!("Hello, world 3!"); }"#)
.unwrap();
cargo_process("init --vcs none").env("USER", "foo").cwd(&path).with_status(101).with_stderr(
"[ERROR] cannot have a package with multiple libraries, found both `src/lib.rs` and `lib.rs`",
)
.run();
assert!(!paths::root().join("foo/Cargo.toml").is_file());
}
#[cargo_test]
fn multibin_project_name_clash() {
let path = paths::root().join("foo");
fs::create_dir(&path).unwrap();
let sourcefile_path1 = path.join("foo.rs");
File::create(&sourcefile_path1)
.unwrap()
.write_all(br#"fn main () { println!("Hello, world 2!"); }"#)
.unwrap();
let sourcefile_path2 = path.join("main.rs");
File::create(&sourcefile_path2)
.unwrap()
.write_all(br#"fn main () { println!("Hello, world 3!"); }"#)
.unwrap();
cargo_process("init --lib --vcs none")
.env("USER", "foo")
.cwd(&path)
.with_status(101)
.with_stderr(
"\
[ERROR] multiple possible binary sources found:
main.rs
foo.rs
cannot automatically generate Cargo.toml as the main target would be ambiguous
",
)
.run();
assert!(!paths::root().join("foo/Cargo.toml").is_file());
}
fn lib_already_exists(rellocation: &str) {
let path = paths::root().join("foo");
fs::create_dir_all(&path.join("src")).unwrap();
let sourcefile_path = path.join(rellocation);
let content = br#"
pub fn qqq() {}
"#;
File::create(&sourcefile_path)
.unwrap()
.write_all(content)
.unwrap();
cargo_process("init --vcs none")
.env("USER", "foo")
.cwd(&path)
.run();
assert!(paths::root().join("foo/Cargo.toml").is_file());
assert!(!paths::root().join("foo/src/main.rs").is_file());
// Check that our file is not overwritten
let mut new_content = Vec::new();
File::open(&sourcefile_path)
.unwrap()
.read_to_end(&mut new_content)
.unwrap();
assert_eq!(Vec::from(content as &[u8]), new_content);
}
#[cargo_test]
fn lib_already_exists_src() {
lib_already_exists("src/lib.rs");
}
#[cargo_test]
fn lib_already_exists_nosrc() {
lib_already_exists("lib.rs");
}
#[cargo_test]
fn simple_git() {
cargo_process("init --lib --vcs git")
.env("USER", "foo")
.run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(paths::root().join(".git").is_dir());
assert!(paths::root().join(".gitignore").is_file());
}
#[cargo_test]
fn auto_git() {
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(paths::root().join(".git").is_dir());
assert!(paths::root().join(".gitignore").is_file());
}
#[cargo_test]
fn invalid_dir_name() {
let foo = &paths::root().join("foo.bar");
fs::create_dir_all(&foo).unwrap();
cargo_process("init")
.cwd(foo.clone())
.env("USER", "foo")
.with_status(101)
.with_stderr(
"\
[ERROR] Invalid character `.` in crate name: `foo.bar`
use --name to override crate name
",
)
.run();
assert!(!foo.join("Cargo.toml").is_file());
}
#[cargo_test]
fn reserved_name() {
let test = &paths::root().join("test");
fs::create_dir_all(&test).unwrap();
cargo_process("init")
.cwd(test.clone())
.env("USER", "foo")
.with_status(101)
.with_stderr(
"\
[ERROR] The name `test` cannot be used as a crate name\n\
use --name to override crate name
",
)
.run();
assert!(!test.join("Cargo.toml").is_file());
}
#[cargo_test]
fn git_autodetect() {
fs::create_dir(&paths::root().join(".git")).unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(paths::root().join(".git").is_dir());
assert!(paths::root().join(".gitignore").is_file());
}
#[cargo_test]
fn mercurial_autodetect() {
fs::create_dir(&paths::root().join(".hg")).unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(!paths::root().join(".git").is_dir());
assert!(paths::root().join(".hgignore").is_file());
}
#[cargo_test]
fn gitignore_appended_not_replaced() {
fs::create_dir(&paths::root().join(".git")).unwrap();
File::create(&paths::root().join(".gitignore"))
.unwrap()
.write_all(b"qqqqqq\n")
.unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join("Cargo.toml").is_file());
assert!(paths::root().join("src/lib.rs").is_file());
assert!(paths::root().join(".git").is_dir());
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(contents.contains(r#"qqqqqq"#));
}
#[cargo_test]
fn gitignore_added_newline_in_existing() {
fs::create_dir(&paths::root().join(".git")).unwrap();
File::create(&paths::root().join(".gitignore"))
.unwrap()
.write_all(b"first")
.unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(contents.starts_with("first\n"));
}
#[cargo_test]
fn gitignore_no_newline_in_new() {
fs::create_dir(&paths::root().join(".git")).unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(!contents.starts_with('\n'));
}
#[cargo_test]
fn mercurial_added_newline_in_existing() {
fs::create_dir(&paths::root().join(".hg")).unwrap();
File::create(&paths::root().join(".hgignore"))
.unwrap()
.write_all(b"first")
.unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join(".hgignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".hgignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(contents.starts_with("first\n"));
}
#[cargo_test]
fn mercurial_no_newline_in_new() {
fs::create_dir(&paths::root().join(".hg")).unwrap();
cargo_process("init --lib").env("USER", "foo").run();
assert!(paths::root().join(".hgignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".hgignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(!contents.starts_with('\n'));
}
#[cargo_test]
fn terminating_newline_in_new_git_ignore() {
cargo_process("init --vcs git --lib")
.env("USER", "foo")
.run();
let content = fs::read_to_string(&paths::root().join(".gitignore")).unwrap();
let mut last_chars = content.chars().rev();
assert_eq!(last_chars.next(), Some('\n'));
assert_ne!(last_chars.next(), Some('\n'));
}
#[cargo_test]
fn terminating_newline_in_new_mercurial_ignore() {
if !mercurial_available() {
return;
}
cargo_process("init --vcs hg --lib")
.env("USER", "foo")
.run();
let content = fs::read_to_string(&paths::root().join(".hgignore")).unwrap();
let mut last_chars = content.chars().rev();
assert_eq!(last_chars.next(), Some('\n'));
assert_ne!(last_chars.next(), Some('\n'));
}
#[cargo_test]
fn terminating_newline_in_existing_git_ignore() {
fs::create_dir(&paths::root().join(".git")).unwrap();
fs::write(&paths::root().join(".gitignore"), b"first").unwrap();
cargo_process("init --lib").env("USER", "foo").run();
let content = fs::read_to_string(&paths::root().join(".gitignore")).unwrap();
let mut last_chars = content.chars().rev();
assert_eq!(last_chars.next(), Some('\n'));
assert_ne!(last_chars.next(), Some('\n'));
}
#[cargo_test]
fn terminating_newline_in_existing_mercurial_ignore() {
fs::create_dir(&paths::root().join(".hg")).unwrap();
fs::write(&paths::root().join(".hgignore"), b"first").unwrap();
cargo_process("init --lib").env("USER", "foo").run();
let content = fs::read_to_string(&paths::root().join(".hgignore")).unwrap();
let mut last_chars = content.chars().rev();
assert_eq!(last_chars.next(), Some('\n'));
assert_ne!(last_chars.next(), Some('\n'));
}
#[cargo_test]
fn cargo_lock_gitignored_if_lib1() {
fs::create_dir(&paths::root().join(".git")).unwrap();
cargo_process("init --lib --vcs git")
.env("USER", "foo")
.run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(contents.contains(r#"Cargo.lock"#));
}
#[cargo_test]
fn cargo_lock_gitignored_if_lib2() {
fs::create_dir(&paths::root().join(".git")).unwrap();
File::create(&paths::root().join("lib.rs"))
.unwrap()
.write_all(br#""#)
.unwrap();
cargo_process("init --vcs git").env("USER", "foo").run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(contents.contains(r#"Cargo.lock"#));
}
#[cargo_test]
fn cargo_lock_not_gitignored_if_bin1() {
fs::create_dir(&paths::root().join(".git")).unwrap();
cargo_process("init --vcs git --bin")
.env("USER", "foo")
.run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(!contents.contains(r#"Cargo.lock"#));
}
#[cargo_test]
fn cargo_lock_not_gitignored_if_bin2() {
fs::create_dir(&paths::root().join(".git")).unwrap();
File::create(&paths::root().join("main.rs"))
.unwrap()
.write_all(br#""#)
.unwrap();
cargo_process("init --vcs git").env("USER", "foo").run();
assert!(paths::root().join(".gitignore").is_file());
let mut contents = String::new();
File::open(&paths::root().join(".gitignore"))
.unwrap()
.read_to_string(&mut contents)
.unwrap();
assert!(!contents.contains(r#"Cargo.lock"#));
}
#[cargo_test]
fn with_argument() {
cargo_process("init foo --vcs none")
.env("USER", "foo")
.run();
assert!(paths::root().join("foo/Cargo.toml").is_file());
}
#[cargo_test]
fn unknown_flags() {
cargo_process("init foo --flag")
.with_status(1)
.with_stderr_contains(
"error: Found argument '--flag' which wasn't expected, or isn't valid in this context",
)
.run();
}
#[cfg(not(windows))]
#[cargo_test]
fn no_filename() {
cargo_process("init /")
.with_status(101)
.with_stderr(
"[ERROR] cannot auto-detect package name from path \"/\" ; use --name to override"
.to_string(),
)
.run();
}
#[cargo_test]
fn formats_source() {
if !command_is_available("rustfmt") {
return;
}
fs::write(&paths::root().join("rustfmt.toml"), "tab_spaces = 2").unwrap();
cargo_process("init --lib")
.env("USER", "foo")
.with_stderr("[CREATED] library package")
.run();
assert_eq!(
fs::read_to_string(paths::root().join("src/lib.rs")).unwrap(),
r#"#[cfg(test)]
mod tests {
#[test]
fn it_works() {
assert_eq!(2 + 2, 4);
}
}
"#
);
}
#[cargo_test]
fn ignores_failure_to_format_source() {
cargo_process("init --lib")
.env("USER", "foo")
.env("PATH", "") // pretend that `rustfmt` is missing
.with_stderr("[CREATED] library package")
.run();
assert_eq!(
fs::read_to_string(paths::root().join("src/lib.rs")).unwrap(),
r#"#[cfg(test)]
mod tests {
#[test]
fn it_works() {
assert_eq!(2 + 2, 4);
}
}
"#
);
}