Move everything to workspaces
This commit is contained in:
parent
75547d3000
commit
ad867e8e3c
42 changed files with 338 additions and 168 deletions
195
crates/bencode/src/bencode_value.rs
Normal file
195
crates/bencode/src/bencode_value.rs
Normal file
|
|
@ -0,0 +1,195 @@
|
|||
use std::{collections::HashMap, marker::PhantomData};
|
||||
|
||||
use buffers::{ByteBuf, ByteString};
|
||||
use clone_to_owned::CloneToOwned;
|
||||
use serde::{Deserialize, Deserializer};
|
||||
|
||||
use crate::serde_bencode_de::from_bytes;
|
||||
|
||||
pub fn dyn_from_bytes<'de, ByteBuf>(buf: &'de [u8]) -> anyhow::Result<BencodeValue<ByteBuf>>
|
||||
where
|
||||
ByteBuf: From<&'de [u8]> + Deserialize<'de> + std::hash::Hash + Eq,
|
||||
{
|
||||
from_bytes(buf)
|
||||
}
|
||||
|
||||
impl<ByteBuf: serde::Serialize + Eq + std::hash::Hash> serde::Serialize for BencodeValue<ByteBuf> {
|
||||
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
||||
where
|
||||
S: serde::Serializer,
|
||||
{
|
||||
match self {
|
||||
BencodeValue::Bytes(b) => b.serialize(serializer),
|
||||
BencodeValue::Integer(v) => v.serialize(serializer),
|
||||
BencodeValue::List(l) => l.serialize(serializer),
|
||||
BencodeValue::Dict(d) => d.serialize(serializer),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<'de, ByteBuf> serde::de::Deserialize<'de> for BencodeValue<ByteBuf>
|
||||
where
|
||||
ByteBuf: From<&'de [u8]> + Deserialize<'de> + std::hash::Hash + Eq,
|
||||
{
|
||||
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
||||
where
|
||||
D: Deserializer<'de>,
|
||||
{
|
||||
struct Visitor<ByteBuf> {
|
||||
buftype: PhantomData<ByteBuf>,
|
||||
}
|
||||
|
||||
impl<'de, ByteBuf> serde::de::Visitor<'de> for Visitor<ByteBuf>
|
||||
where
|
||||
ByteBuf: From<&'de [u8]> + Deserialize<'de> + std::hash::Hash + Eq,
|
||||
{
|
||||
type Value = BencodeValue<ByteBuf>;
|
||||
|
||||
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
|
||||
write!(formatter, "a bencode value")
|
||||
}
|
||||
|
||||
fn visit_i64<E>(self, v: i64) -> Result<Self::Value, E>
|
||||
where
|
||||
E: serde::de::Error,
|
||||
{
|
||||
Ok(BencodeValue::Integer(v))
|
||||
}
|
||||
|
||||
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
|
||||
where
|
||||
A: serde::de::SeqAccess<'de>,
|
||||
{
|
||||
let mut v = Vec::new();
|
||||
while let Some(value) = seq.next_element()? {
|
||||
v.push(value);
|
||||
}
|
||||
Ok(BencodeValue::List(v))
|
||||
}
|
||||
|
||||
fn visit_borrowed_bytes<E>(self, v: &'de [u8]) -> Result<Self::Value, E>
|
||||
where
|
||||
E: serde::de::Error,
|
||||
{
|
||||
Ok(BencodeValue::Bytes(ByteBuf::from(v)))
|
||||
}
|
||||
|
||||
fn visit_map<A>(self, mut map: A) -> Result<Self::Value, A::Error>
|
||||
where
|
||||
A: serde::de::MapAccess<'de>,
|
||||
{
|
||||
let mut hmap = HashMap::new();
|
||||
while let Some(key) = map.next_key()? {
|
||||
let value = map.next_value()?;
|
||||
hmap.insert(key, value);
|
||||
}
|
||||
Ok(BencodeValue::Dict(hmap))
|
||||
}
|
||||
}
|
||||
|
||||
deserializer.deserialize_any(Visitor {
|
||||
buftype: PhantomData,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// A dynamic value when we don't know exactly what we are deserializing.
|
||||
// Useful for debugging.
|
||||
|
||||
#[derive(PartialEq, Eq)]
|
||||
pub enum BencodeValue<ByteBuf: std::hash::Hash + Eq> {
|
||||
Bytes(ByteBuf),
|
||||
Integer(i64),
|
||||
List(Vec<BencodeValue<ByteBuf>>),
|
||||
Dict(HashMap<ByteBuf, BencodeValue<ByteBuf>>),
|
||||
}
|
||||
|
||||
impl<ByteBuf: std::fmt::Debug + std::hash::Hash + Eq> std::fmt::Debug for BencodeValue<ByteBuf> {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
match self {
|
||||
BencodeValue::Bytes(b) => std::fmt::Debug::fmt(b, f),
|
||||
BencodeValue::Integer(i) => std::fmt::Debug::fmt(i, f),
|
||||
BencodeValue::List(l) => std::fmt::Debug::fmt(l, f),
|
||||
BencodeValue::Dict(d) => std::fmt::Debug::fmt(d, f),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<ByteBuf> CloneToOwned for BencodeValue<ByteBuf>
|
||||
where
|
||||
ByteBuf: CloneToOwned + std::hash::Hash + Eq,
|
||||
<ByteBuf as CloneToOwned>::Target: Eq + std::hash::Hash,
|
||||
{
|
||||
type Target = BencodeValue<<ByteBuf as CloneToOwned>::Target>;
|
||||
|
||||
fn clone_to_owned(&self) -> Self::Target {
|
||||
match self {
|
||||
BencodeValue::Bytes(b) => BencodeValue::Bytes(b.clone_to_owned()),
|
||||
BencodeValue::Integer(i) => BencodeValue::Integer(*i),
|
||||
BencodeValue::List(l) => BencodeValue::List(l.clone_to_owned()),
|
||||
BencodeValue::Dict(d) => BencodeValue::Dict(d.clone_to_owned()),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub type BencodeValueBorrowed<'a> = BencodeValue<ByteBuf<'a>>;
|
||||
pub type BencodeValueOwned = BencodeValue<ByteString>;
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::serde_bencode_ser::bencode_serialize_to_writer;
|
||||
|
||||
use super::*;
|
||||
use serde::Serialize;
|
||||
use std::io::Read;
|
||||
|
||||
#[test]
|
||||
fn test_deserialize_torrent_dyn() {
|
||||
let mut buf = Vec::new();
|
||||
let filename = "resources/ubuntu-21.04-desktop-amd64.iso.torrent";
|
||||
std::fs::File::open(filename)
|
||||
.unwrap()
|
||||
.read_to_end(&mut buf)
|
||||
.unwrap();
|
||||
|
||||
let torrent_borrowed: BencodeValueBorrowed = from_bytes(&buf).unwrap();
|
||||
let torrent_owned: BencodeValueOwned = from_bytes(&buf).unwrap();
|
||||
dbg!(torrent_borrowed);
|
||||
dbg!(torrent_owned);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_serialize_torrent_dyn() {
|
||||
let mut buf = Vec::new();
|
||||
let filename = "resources/ubuntu-21.04-desktop-amd64.iso.torrent";
|
||||
std::fs::File::open(filename)
|
||||
.unwrap()
|
||||
.read_to_end(&mut buf)
|
||||
.unwrap();
|
||||
|
||||
let torrent: BencodeValueBorrowed = from_bytes(&buf).unwrap();
|
||||
|
||||
let mut buf = Vec::<u8>::new();
|
||||
bencode_serialize_to_writer(&torrent, &mut buf).unwrap();
|
||||
|
||||
let new_torrent = from_bytes(&buf).unwrap();
|
||||
assert_eq!(torrent, new_torrent);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_serialize_struct_with_option() {
|
||||
#[derive(Serialize)]
|
||||
struct Test {
|
||||
f1: i64,
|
||||
#[serde(skip_serializing_if = "Option::is_none")]
|
||||
missing: Option<i64>,
|
||||
}
|
||||
let test = Test {
|
||||
f1: 100,
|
||||
missing: None,
|
||||
};
|
||||
let mut buf = Vec::<u8>::new();
|
||||
bencode_serialize_to_writer(&test, &mut buf).unwrap();
|
||||
assert_eq!(&buf, b"d2:f1i100ee");
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue