Compare commits

..

4 Commits

5 changed files with 115 additions and 153 deletions

10
Cargo.lock generated
View File

@ -865,6 +865,15 @@ dependencies = [
"serde_derive", "serde_derive",
] ]
[[package]]
name = "serde_bytes"
version = "0.11.15"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "387cc504cb06bb40a96c8e04e951fe01854cf6bc921053c954e4a606d9675c6a"
dependencies = [
"serde",
]
[[package]] [[package]]
name = "serde_derive" name = "serde_derive"
version = "1.0.204" version = "1.0.204"
@ -1070,6 +1079,7 @@ dependencies = [
"rand", "rand",
"reqwest", "reqwest",
"serde", "serde",
"serde_bytes",
"sha1", "sha1",
"urlencoding", "urlencoding",
] ]

View File

@ -10,6 +10,7 @@ reqwest = { version = "0.12.5", features = ["blocking"] }
rand = "0.8.5" rand = "0.8.5"
urlencoding = "2.1.3" urlencoding = "2.1.3"
serde = { version = "1.0.204", features = ["derive"] } serde = { version = "1.0.204", features = ["derive"] }
serde_bytes = "0.11.15"
[dev-dependencies] [dev-dependencies]
hex-literal = "0.4.1" hex-literal = "0.4.1"

View File

@ -81,12 +81,18 @@ impl<'de> Deserializer<'de> {
impl <'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> { impl <'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
type Error = Error; type Error = Error;
fn deserialize_any<V>(self, _: V) -> Result<V::Value> fn deserialize_any<V>(self, visitor: V) -> Result<V::Value>
where where
V: Visitor<'de> V: Visitor<'de>
{ {
println!("i don't get this API!"); let b = self.peek_byte()? as char;
Err(Error::WontImplement) match b {
'i' => self.deserialize_i64(visitor),
'l' => self.deserialize_seq(visitor),
'd' => self.deserialize_map(visitor),
'0'..='9' => self.deserialize_str(visitor),
_ => Err(Error::Syntax(format!("any: invalid start char: {}", b))),
}
} }
fn deserialize_bool<V>(self, _: V) -> Result<V::Value> fn deserialize_bool<V>(self, _: V) -> Result<V::Value>
@ -274,14 +280,15 @@ impl <'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
where where
V: Visitor<'de> V: Visitor<'de>
{ {
match self.peek_byte()? as char{ let b = self.peek_byte()? as char;
match b {
'l' => self.deserialize_seq(visitor), 'l' => self.deserialize_seq(visitor),
'0'..='9' => { '0'..='9' => {
let str = self.parse_byte_string()?; let str = self.parse_byte_string()?;
let ba = ByteAccess::new(self, str); let ba = ByteAccess::new(self, str);
visitor.visit_seq(ba) visitor.visit_seq(ba)
}, },
_ => Err(Error::Syntax), _ => Err(Error::Syntax(format!("tuple: invalid start char: {}", b))),
} }
} }
@ -329,12 +336,11 @@ impl <'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
self.deserialize_bytes(visitor) self.deserialize_bytes(visitor)
} }
fn deserialize_ignored_any<V>(self, _: V) -> Result<V::Value> fn deserialize_ignored_any<V>(self, visitor: V) -> Result<V::Value>
where where
V: Visitor<'de> V: Visitor<'de>
{ {
println!("i really don't get this API!"); self.deserialize_any(visitor)
Err(Error::WontImplement)
} }
} }
@ -394,12 +400,13 @@ impl<'a, 'de> SeqAccess<'de> for RegularAccess<'a, 'de> {
where where
T: DeserializeSeed<'de> T: DeserializeSeed<'de>
{ {
match self.de.peek_byte()? as char { let b = self.de.peek_byte()? as char;
match b {
'd' | 'i' | 'l' | '0'..='9' => { 'd' | 'i' | 'l' | '0'..='9' => {
seed.deserialize(&mut *self.de).map(Some) seed.deserialize(&mut *self.de).map(Some)
}, },
'e' => Ok(None), 'e' => Ok(None),
_ => Err(Error::Syntax) _ => Err(Error::Syntax(format!("seq: next_element: invalid start/end char: {}", b)))
} }
} }
} }
@ -424,11 +431,12 @@ impl<'a, 'de> MapAccess<'de> for RegularAccess<'a, 'de> {
where where
V: DeserializeSeed<'de> V: DeserializeSeed<'de>
{ {
match self.de.peek_byte()? as char { let b = self.de.peek_byte()? as char;
match b {
'd' | 'i' | 'l' | '0'..='9' => { 'd' | 'i' | 'l' | '0'..='9' => {
seed.deserialize(&mut *self.de) seed.deserialize(&mut *self.de)
}, },
_ => Err(Error::Syntax) _ => Err(Error::Syntax(format!("map: next_value: invalid start char: {}", b)))
} }
} }
} }
@ -516,4 +524,14 @@ mod test {
assert_eq!(de.l, vec![23, 17]); assert_eq!(de.l, vec![23, 17]);
assert_eq!(de.o, Some(42)); assert_eq!(de.o, Some(42));
} }
#[test]
fn test_struct_extra() {
#[derive(Deserialize, Debug)]
struct A {
a: i64,
}
let de: A = from_str("d1:ai42e1:bi18ee").unwrap();
assert_eq!(de.a, 42);
}
} }

View File

@ -9,7 +9,7 @@ pub enum Error {
WontImplement, WontImplement,
Eof, Eof,
Syntax, Syntax(String),
InvalidUtf8, InvalidUtf8,
ExpectedBytes, ExpectedBytes,
ExpectedBytesSep, ExpectedBytesSep,
@ -48,8 +48,8 @@ impl Display for Error {
Error::Message(msg) => formatter.write_str(msg), Error::Message(msg) => formatter.write_str(msg),
Error::WontImplement => formatter.write_str("there is no reasonable way to (de)serialize this type"), Error::WontImplement => formatter.write_str("there is no reasonable way to (de)serialize this type"),
Error::Eof => formatter.write_str("unexpected end of input"), Error::Eof => formatter.write_str("unexpected end of input"),
Error::Syntax => formatter.write_str("syntax error"), Error::Syntax(msg) => formatter.write_fmt(format_args!("syntax error: {}", msg)),
Error::InvalidUtf8 => formatter.write_str("could not decoded as UTF-8"), Error::InvalidUtf8 => formatter.write_str("could not decode as UTF-8"),
Error::ExpectedBytes => formatter.write_str("expected byte string start char: any number"), Error::ExpectedBytes => formatter.write_str("expected byte string start char: any number"),
Error::ExpectedBytesSep => formatter.write_str("expected byte separator char: ':'"), Error::ExpectedBytesSep => formatter.write_str("expected byte separator char: ':'"),
Error::ExpectedInteger => formatter.write_str("expected integer start char 'i'"), Error::ExpectedInteger => formatter.write_str("expected integer start char 'i'"),

View File

@ -2,168 +2,106 @@
#![allow(uncommon_codepoints)] #![allow(uncommon_codepoints)]
use std::{env, fs, path}; use std::{env, fs, path};
use std::collections::HashMap;
use crate::bencode::de::from_bytes;
use crate::bencode::ser::to_bytes;
use anyhow::{anyhow, Result}; use anyhow::{anyhow, Result};
use rand::prelude::*; use rand::prelude::*;
use reqwest::blocking::Client; use reqwest::blocking::Client;
use reqwest::Url; use reqwest::Url;
use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
use crate::bencode::custom::{Bencode, ByteString}; use serde::ser::SerializeSeq;
use serde_bytes::ByteBuf;
use sha1::{Digest, Sha1};
mod bencode; mod bencode;
mod torrent;
#[derive(Debug)] #[derive(Serialize, Deserialize, Debug)]
struct FileInfo { struct FileInfo {
length: i64, length: i64,
#[serde(deserialize_with = "deserialize_path", serialize_with = "serialize_path")]
path: String, path: String,
} }
#[derive(Debug)] fn deserialize_path<'de, D>(deserializer: D) -> Result<String, D::Error>
where D: Deserializer<'de> {
let parts: Vec<String> = Vec::deserialize(deserializer)?;
for p in parts.iter() {
let res = p.find(path::MAIN_SEPARATOR_STR);
if let Some(_) = res {
return Err(de::Error::custom(format!("Unable to deal with {} (platform path separator in path parts", path::MAIN_SEPARATOR_STR)))
}
}
Ok(parts.join(path::MAIN_SEPARATOR_STR))
}
fn serialize_path<S>(v: &String, serializer: S) -> Result<S::Ok, S::Error>
where S: Serializer {
let parts: Vec<&str> = v.split(path::MAIN_SEPARATOR_STR).collect();
let mut seq = serializer.serialize_seq(Some(parts.len()))?;
for p in parts {
seq.serialize_element(p)?;
}
seq.end()
}
#[derive(Serialize, Deserialize, Debug)]
struct TorrentInfo { struct TorrentInfo {
files: Option<Vec<FileInfo>>, files: Option<Vec<FileInfo>>,
length: Option<i64>, length: Option<i64>,
name: String, name: String,
#[serde(rename(serialize = "piece length", deserialize = "piece length"))]
piece_length: i64, piece_length: i64,
#[serde(deserialize_with = "deserialize_pieces", serialize_with = "serialize_pieces")]
pieces: Vec<[u8; 20]>, pieces: Vec<[u8; 20]>,
} }
#[derive(Debug)] fn deserialize_pieces<'de, D>(deserializer: D) -> Result<Vec<[u8; 20]>, D::Error>
struct Torrent { where D: Deserializer<'de> {
announce: String, let all_pieces = ByteBuf::deserialize(deserializer)?.into_vec();
info: TorrentInfo, if all_pieces.len() % 20 != 0 {
info_hash: [u8; 20], return Err(de::Error::custom("Pieces string length not a multiple of 20"))
_extra: HashMap<String, String>
}
impl Torrent {
pub fn from(input: &Bencode) -> Result<Self> {
let Bencode::Dict(decoded) = input else {
return Err(anyhow!("Torrent has wrong type. Not a dict"))
};
let b_info = decoded.get(&ByteString::from_str("info"))
.ok_or(anyhow!("Could not find key \"info\""))?;
let Bencode::Dict(info_dict) = b_info else {
return Err(anyhow!("Info has wrong type. Not a dict"))
};
let Bencode::Bytes(name) = info_dict.get(&ByteString::from_str("name"))
.ok_or(anyhow!("Could not find key \"name\""))? else {
return Err(anyhow!("name not a string"))
};
let Bencode::Integer(piece_length) = *info_dict.get(&ByteString::from_str("piece length"))
.ok_or(anyhow!("Could not find key \"piece length\""))? else {
return Err(anyhow!("piece length not an integer"))
};
let Bencode::Bytes(all_pieces) = info_dict.get(&ByteString::from_str("pieces"))
.ok_or(anyhow!("Could not find key \"pieces\""))? else {
return Err(anyhow!("pieces not an string"))
};
if all_pieces.to_vec().len() % 20 != 0 {
return Err(anyhow!("Pieces string length not a multiple of 20"))
} }
// TODO maybe handle the error, even though it should be checked above // TODO maybe handle the error, even though it should be checked above
let pieces = all_pieces.to_vec().chunks(20).map(|x| x.try_into().unwrap()).collect(); let pieces = all_pieces.chunks(20).map(|x| x.try_into().unwrap()).collect();
Ok(pieces)
}
let info; fn serialize_pieces<S>(v: &Vec<[u8; 20]>, serializer: S) -> Result<S::Ok, S::Error>
if let Some(b_length) = info_dict.get(&ByteString::from_str("length")) { where S: Serializer
let Bencode::Integer(length) = b_length else { {
return Err(anyhow!("length not an integer")) let mut buf = Vec::new();
}; for i in v {
info = TorrentInfo { buf.extend(i)
files: None,
length: Some(*length),
name: name.to_string()?,
piece_length,
pieces,
} }
} else if let Some(b_files) = info_dict.get(&ByteString::from_str("files")) { serializer.serialize_bytes(&buf[..])
let Bencode::List(files_list) = b_files else { }
return Err(anyhow!("files not a list"))
};
let mut files = Vec::new(); #[derive(Serialize, Deserialize, Debug)]
for b_file_info in files_list { struct Torrent {
let Bencode::Dict(file_info) = b_file_info else { announce: String,
return Err(anyhow!("file_info not a dict")) info: TorrentInfo,
}; #[serde(skip)]
let Bencode::List(b_path) = file_info.get(&ByteString::from_str("path")) info_hash: [u8; 20],
.ok_or(anyhow!("Could not find key \"path\""))? else { }
return Err(anyhow!("path not a list"))
}; impl Torrent {
let mut v_path = Vec::new(); fn from(input: &[u8]) -> Result<Self> {
for b_path_part in b_path { let mut torrent: Self = from_bytes(input)?;
let Bencode::Bytes(path_part) = b_path_part else { torrent.compute_info_hash()?;
return Err(anyhow!("path part not a string")) Ok(torrent)
};
v_path.push(path_part.to_string()?);
} }
let Bencode::Integer(length) = *file_info.get(&ByteString::from_str("length")) fn compute_info_hash(&mut self) -> Result<()> {
.ok_or(anyhow!("Could not find key \"length\""))? else { let mut hasher = Sha1::new();
return Err(anyhow!("length not an integer"))
};
files.push(FileInfo{
path: v_path.join(path::MAIN_SEPARATOR_STR),
length
})
}
info = TorrentInfo { let info_str = to_bytes(&self.info)?;
files: Some(files), hasher.update(info_str);
length: None,
name: name.to_string()?,
piece_length,
pieces,
}
} else {
return Err(anyhow!("Could not find key \"length\" or \"files\""))
}
self.info_hash = hasher.finalize().into();
let b_announce = decoded.get(&ByteString::from_str("announce")) Ok(())
.ok_or(anyhow!("Could not find key \"announce\""))?;
let Bencode::Bytes(announce) = b_announce else {
return Err(anyhow!("announce not a string"))
};
let mut _extra = HashMap::new();
for (k, v) in decoded {
if k == &ByteString::from_str("announce") || k == &ByteString::from_str("info") {
continue
}
let Ok(k_str) = k.to_string() else {
println!("Ignoring non-UTF-8 key {k:?}");
continue;
};
match v {
Bencode::Integer(i) => {
_extra.insert(k_str, format!("{}", i));
}
Bencode::Bytes(s) => {
let Ok(s_str) = s.to_string() else {
println!("Ignoring non-UTF-8 value of key {k_str}");
continue;
};
_extra.insert(k_str, s_str);
}
_ => {
println!("Ignoring extra of type list or dict")
}
}
}
let result = Self {
announce: announce.to_string()?,
info_hash: b_info.sha1(),
info,
_extra,
};
Ok(result)
} }
} }
@ -176,12 +114,7 @@ fn main() -> Result<()> {
let torrent_bytes = fs::read(&file_path)?; let torrent_bytes = fs::read(&file_path)?;
let torrent_parsed = Bencode::parse(&torrent_bytes)?; let torrent = Torrent::from(&torrent_bytes)?;
println!("torrent file parsed");
// println!("parsed file: {torrent_parsed}");
let torrent = Torrent::from(&torrent_parsed)?;
// println!("torrent decoded: {torrent:?}"); // println!("torrent decoded: {torrent:?}");
@ -220,7 +153,7 @@ fn main() -> Result<()> {
let resp = client.get(url).send()?; let resp = client.get(url).send()?;
let status = resp.status(); let status = resp.status();
// let body = resp.text()?; let body = resp.text()?;
println!("Response: {status}"); println!("Response: {status}");
Ok(()) Ok(())