From 551617de4f09370e34a279884a1b38f5fb7f702c Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Tue, 4 Feb 2020 21:55:19 -0800 Subject: [PATCH] Don't overwrite destination torrent without `--force` type: changed --- src/opt/torrent/create.rs | 44 ++++++++++++++++++++++++++++++++++++++- 1 file changed, 43 insertions(+), 1 deletion(-) diff --git a/src/opt/torrent/create.rs b/src/opt/torrent/create.rs index 80041c9..d6a0dfc 100644 --- a/src/opt/torrent/create.rs +++ b/src/opt/torrent/create.rs @@ -43,6 +43,12 @@ Note: Many BitTorrent clients do not implement the behavior described in BEP 12. long_help = "Include `COMMENT` in generated `.torrent` file. Stored under `comment` key of top-level metainfo dictionary." )] comment: Option, + #[structopt( + name = "FORCE", + long = "force", + help = "Overwrite the destination `.torrent` file, if it exists." + )] + force: bool, #[structopt( name = "INPUT", long = "input", @@ -232,7 +238,19 @@ impl Create { match &output { Target::File(path) => { - fs::write(path, &bytes).context(error::Filesystem { path })?; + let mut open_options = fs::OpenOptions::new(); + + if self.force { + open_options.write(true).create(true).truncate(true); + } else { + open_options.write(true).create_new(true); + } + + open_options + .open(path) + .and_then(|mut file| file.write_all(&bytes)) + .context(error::Filesystem { path })?; + TorrentSummary::from_metainfo(metainfo)?.write(env)?; if self.open { Platform::open(&path)?; @@ -912,4 +930,28 @@ Content Size 0 bytes let value = bencode::Value::decode(&bytes).unwrap(); assert!(matches!(value, bencode::Value::Dict(_))); } + + #[test] + fn force_default() { + let mut env = environment(&["--input", "foo", "--announce", "http://bar"]); + fs::write(env.resolve("foo"), "").unwrap(); + fs::write(env.resolve("foo.torrent"), "").unwrap(); + assert_matches!( + env.run().unwrap_err(), + Error::Filesystem {source, path} + if path == env.resolve("foo.torrent") && source.kind() == io::ErrorKind::AlreadyExists + ) + } + + #[test] + fn force_true() { + let mut env = environment(&["--input", "foo", "--announce", "http://bar", "--force"]); + fs::write(env.resolve("foo"), "").unwrap(); + fs::write(env.resolve("foo.torrent"), "foo").unwrap(); + env.run().unwrap(); + let torrent = env.resolve("foo.torrent"); + let bytes = fs::read(torrent).unwrap(); + let value = bencode::Value::decode(&bytes).unwrap(); + assert!(matches!(value, bencode::Value::Dict(_))); + } }