Skip to content

Commit

Permalink
Restrict piece length
Browse files Browse the repository at this point in the history
- Must be greater than zero
- Must be a power of two (but can override with `--allow uneven-piece-length`
- Must be greater than 16KiB (but can override with `--allow small-piece-length`
- Must be less than u32 max

type: changed
  • Loading branch information
casey committed Apr 8, 2020
1 parent 85f02d9 commit 6df45e0
Show file tree
Hide file tree
Showing 7 changed files with 246 additions and 13 deletions.
6 changes: 6 additions & 0 deletions src/bytes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,12 @@ const YI: u128 = ZI << 10;
#[derive(Debug, PartialEq, Copy, Clone)]
pub(crate) struct Bytes(pub(crate) u128);

impl Bytes {
pub(crate) fn is_power_of_two(self) -> bool {
self.0 == 0 || self.0 & (self.0 - 1) == 0
}
}

fn float_to_int(x: f64) -> u128 {
#![allow(
clippy::as_conversions,
Expand Down
4 changes: 2 additions & 2 deletions src/common.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
pub(crate) use std::{
borrow::Cow,
cmp::{Ordering, Reverse},
collections::{BTreeMap, HashMap},
collections::{BTreeMap, BTreeSet, HashMap},
convert::{Infallible, TryInto},
env,
ffi::{OsStr, OsString},
Expand Down Expand Up @@ -44,7 +44,7 @@ pub(crate) use crate::{
// structs and enums
pub(crate) use crate::{
bytes::Bytes, env::Env, error::Error, file_info::FileInfo, hasher::Hasher, info::Info,
metainfo::Metainfo, mode::Mode, opt::Opt, platform::Platform, style::Style,
lint::Lint, metainfo::Metainfo, mode::Mode, opt::Opt, platform::Platform, style::Style,
subcommand::Subcommand, torrent::Torrent, use_color::UseColor,
};

Expand Down
11 changes: 11 additions & 0 deletions src/env.rs
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,17 @@ impl Env {
self.err_style.message().suffix(),
)
.ok();

if let Some(lint) = error.lint() {
writeln!(
&mut self.err,
"{}: This check can be disabled with `--allow {}`.",
self.err_style.message().paint("note"),
lint.name()
)
.ok();
}

Err(EXIT_FAILURE)
}
} else {
Expand Down
20 changes: 19 additions & 1 deletion src/error.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,13 @@ pub(crate) enum Error {
bytes,
Bytes(u32::max_value().into())
))]
PieceLength { bytes: Bytes },
PieceLengthTooLarge { bytes: Bytes },
#[snafu(display("Piece length `{}` is not an even power of two", bytes))]
PieceLengthUneven { bytes: Bytes },
#[snafu(display("Piece length must be at least 16 KiB"))]
PieceLengthSmall,
#[snafu(display("Piece length cannot be zero"))]
PieceLengthZero,
#[snafu(display("Serialization failed: {}", source))]
Serialize { source: serde_bencode::Error },
#[snafu(display("Failed to write to standard error: {}", source))]
Expand All @@ -51,6 +57,18 @@ pub(crate) enum Error {
feature
))]
Unstable { feature: &'static str },
#[snafu(display("Unknown lint: {}", text))]
LintUnknown { text: String },
}

impl Error {
pub(crate) fn lint(&self) -> Option<Lint> {
match self {
Self::PieceLengthUneven { .. } => Some(Lint::UnevenPieceLength),
Self::PieceLengthSmall { .. } => Some(Lint::SmallPieceLength),
_ => None,
}
}
}

impl From<clap::Error> for Error {
Expand Down
70 changes: 70 additions & 0 deletions src/lint.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
use crate::common::*;

#[derive(Eq, PartialEq, Debug, Copy, Clone, Ord, PartialOrd)]
pub(crate) enum Lint {
UnevenPieceLength,
SmallPieceLength,
}

const UNEVEN_PIECE_LENGTH: &str = "uneven-piece-length";
const SMALL_PIECE_LENGTH: &str = "small-piece-length";

impl Lint {
pub(crate) fn name(self) -> &'static str {
match self {
Self::UnevenPieceLength => UNEVEN_PIECE_LENGTH,
Self::SmallPieceLength => SMALL_PIECE_LENGTH,
}
}
}

impl FromStr for Lint {
type Err = Error;

fn from_str(text: &str) -> Result<Self, Self::Err> {
match text.replace('_', "-").to_lowercase().as_str() {
UNEVEN_PIECE_LENGTH => Ok(Self::UnevenPieceLength),
SMALL_PIECE_LENGTH => Ok(Self::SmallPieceLength),
_ => Err(Error::LintUnknown {
text: text.to_string(),
}),
}
}
}

impl Display for Lint {
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
write!(f, "{}", self.name())
}
}

#[cfg(test)]
mod tests {
use super::*;

#[test]
fn from_str_ok() {
assert_eq!(
Lint::UnevenPieceLength,
"uneven_piece_length".parse().unwrap()
);

assert_eq!(
Lint::UnevenPieceLength,
"uneven-piece-length".parse().unwrap()
);

assert_eq!(
Lint::UnevenPieceLength,
"UNEVEN-piece-length".parse().unwrap()
);
}

#[test]
fn from_str_err() {
assert_matches!(
"foo".parse::<Lint>(),
Err(Error::LintUnknown { text }) if text == "foo"
);
}
}
1 change: 1 addition & 0 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ mod hasher;
mod info;
mod into_u64;
mod into_usize;
mod lint;
mod metainfo;
mod mode;
mod opt;
Expand Down
147 changes: 137 additions & 10 deletions src/torrent/create.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,13 @@ pub(crate) struct Create {
long_help = "Use `ANNOUNCE` as the primary tracker announce URL. To supply multiple announce URLs, also use `--announce-tier`."
)]
announce: Url,
#[structopt(
name = "ALLOW",
long = "allow",
help = "Use `ANNOUNCE` as the primary tracker announce URL.",
long_help = "Use `ANNOUNCE` as the primary tracker announce URL. To supply multiple announce URLs, also use `--announce-tier`."
)]
allowed_lints: Vec<Lint>,
#[structopt(
long = "announce-tier",
name = "ANNOUNCE-TIER",
Expand Down Expand Up @@ -98,15 +105,57 @@ Note: Many BitTorrent clients do not implement the behavior described in BEP 12.
private: bool,
}

struct Linter {
allowed: BTreeSet<Lint>,
}

impl Linter {
fn new() -> Linter {
Linter {
allowed: BTreeSet::new(),
}
}

fn allow(&mut self, allowed: impl IntoIterator<Item = Lint>) {
self.allowed.extend(allowed)
}

fn is_allowed(&self, lint: Lint) -> bool {
self.allowed.contains(&lint)
}

fn is_denied(&self, lint: Lint) -> bool {
!self.is_allowed(lint)
}
}

impl Create {
pub(crate) fn run(self, env: &Env) -> Result<(), Error> {
let piece_length: u32 = self
.piece_length
.0
.try_into()
.map_err(|_| Error::PieceLength {
let mut linter = Linter::new();
linter.allow(self.allowed_lints.iter().cloned());

if linter.is_denied(Lint::UnevenPieceLength) && !self.piece_length.is_power_of_two() {
return Err(Error::PieceLengthUneven {
bytes: self.piece_length,
})?;
});
}

let piece_length: u32 =
self
.piece_length
.0
.try_into()
.map_err(|_| Error::PieceLengthTooLarge {
bytes: self.piece_length,
})?;

if piece_length == 0 {
return Err(Error::PieceLengthZero);
}

if linter.is_denied(Lint::SmallPieceLength) && piece_length < 16 * 1024 {
return Err(Error::PieceLengthSmall);
}

let input = env.resolve(&self.input);

Expand Down Expand Up @@ -405,14 +454,14 @@ mod tests {
"--announce",
"http://bar",
"--piece-length",
"1",
"64KiB",
]);
fs::write(env.resolve("foo"), "").unwrap();
env.run().unwrap();
let torrent = env.resolve("foo.torrent");
let bytes = fs::read(torrent).unwrap();
let metainfo = serde_bencode::de::from_bytes::<Metainfo>(&bytes).unwrap();
assert_eq!(metainfo.info.piece_length, 1);
assert_eq!(metainfo.info.piece_length, 64 * 1024);
}

#[test]
Expand Down Expand Up @@ -441,7 +490,7 @@ mod tests {
"--announce",
"http://bar",
"--piece-length",
"1",
"16KiB",
]);
fs::write(env.resolve("foo"), "").unwrap();
env.run().unwrap();
Expand All @@ -459,7 +508,7 @@ mod tests {
"--announce",
"http://bar",
"--piece-length",
"1",
"32KiB",
]);
let dir = env.resolve("foo");
fs::create_dir(&dir).unwrap();
Expand Down Expand Up @@ -588,6 +637,8 @@ mod tests {
"http://bar",
"--piece-length",
"1",
"--allow",
"small-piece-length",
]);
let contents = "bar";
fs::write(env.resolve("foo"), contents).unwrap();
Expand Down Expand Up @@ -732,4 +783,80 @@ mod tests {

panic!("Failed to read `opened.txt`.");
}

#[test]
fn uneven_piece_length() {
let mut env = environment(&[
"--input",
"foo",
"--announce",
"http://bar",
"--piece-length",
"17KiB",
]);
assert_matches!(
env.run(),
Err(Error::PieceLengthUneven { bytes }) if bytes.0 == 17 * 1024
);
}

#[test]
fn uneven_piece_length_allow() {
let mut env = environment(&[
"--input",
"foo",
"--announce",
"http://bar",
"--piece-length",
"17KiB",
"--allow",
"uneven-piece-length",
]);
let dir = env.resolve("foo");
fs::create_dir(&dir).unwrap();
env.run().unwrap();
}

#[test]
fn zero_piece_length() {
let mut env = environment(&[
"--input",
"foo",
"--announce",
"http://bar",
"--piece-length",
"0",
]);
assert_matches!(env.run(), Err(Error::PieceLengthZero));
}

#[test]
fn small_piece_length() {
let mut env = environment(&[
"--input",
"foo",
"--announce",
"http://bar",
"--piece-length",
"8KiB",
]);
assert_matches!(env.run(), Err(Error::PieceLengthSmall));
}

#[test]
fn small_piece_length_allow() {
let mut env = environment(&[
"--input",
"foo",
"--announce",
"http://bar",
"--piece-length",
"8KiB",
"--allow",
"small-piece-length",
]);
let dir = env.resolve("foo");
fs::create_dir(&dir).unwrap();
env.run().unwrap();
}
}

0 comments on commit 6df45e0

Please sign in to comment.