Skip to content

Commit

Permalink
Revert "Merge pull request naomijub#150 from Grinkers/parse"
Browse files Browse the repository at this point in the history
This reverts commit bc5bc8f, reversing
changes made to c60d05b.
  • Loading branch information
Grinkers committed Jun 11, 2024
1 parent e5f341b commit 5d34779
Show file tree
Hide file tree
Showing 17 changed files with 691 additions and 796 deletions.
2 changes: 0 additions & 2 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ repository = "https://github.com/edn-rs/edn-rs"
keywords = ["EDN", "no_std"]
license = "MIT"
edition = "2021"
resolver = "2"

[lints.rust]
rust_2018_idioms = { level = "warn", priority = -1 }
Expand All @@ -20,7 +19,6 @@ unsafe_code = "deny"
nursery = { level = "deny", priority = -1 }
pedantic = { level = "deny", priority = -1 }


[features]
default = ["sets", "std"]
json = ["regex"]
Expand Down
20 changes: 3 additions & 17 deletions bb.edn
Original file line number Diff line number Diff line change
@@ -1,11 +1,7 @@
{:tasks
{:init (do (def code-cov-env
{"CARGO_INCREMENTAL" "0"
"RUSTFLAGS" "-Cinstrument-coverage -Copt-level=0 -Ccodegen-units=1 -Cinline-threshold=0 -Clink-dead-code -Coverflow-checks=off"
"LLVM_PROFILE_FILE" "target/coverage/cargo-test-%p-%m.profraw"}))
:requires ([babashka.fs :as fs])
clean {:doc "Removes target folder"
:task (fs/delete-tree "target")}
{clean {:doc "Removes target folder"
:requires ([babashka.fs :as fs])
:task (fs/delete-tree "target")}
test_lib_features (shell "cargo test --all-features --no-fail-fast")
test_lib_no_default_features (shell "cargo test --features std --no-default-features --no-fail-fast")
example_fast (shell "cargo test --examples --no-fail-fast")
Expand All @@ -27,15 +23,5 @@
:task (shell "cargo clippy --features json --no-default-features -- -W future-incompatible -W rust_2018_idioms -W clippy::all -W clippy::pedantic -W clippy::nursery --deny warnings")}
clippy {:doc "Runs all variations of cargo clippy"
:depends [cargo-clippy-all-features cargo-clippy-no-sets-json cargo-clippy-no-defaults]}
cov-all-features {:doc "Coverage, all features"
:task (shell {:extra-env code-cov-env} "cargo test --all-features")}
cov-std-only {:doc "Coverage, std only"
:task (shell {:extra-env code-cov-env} "cargo test --no-default-features --features std")}
clean-cov {:doc "Cleans all .profraw files and generated html"
:task (fs/delete-tree "target/coverage")}
grcov {:doc "Runs grcov to generate human readable html"
:task (shell "grcov target/coverage --binary-path ./target/debug/deps/ -s . -t html --branch --ignore-not-existing -o target/coverage/html")}
coverage {:doc "Generates coverage in human friendly html in target/coverage/"
:depends [clean-cov cov-all-features cov-std-only grcov]}
test {:doc "Runs all tests and checks"
:depends [cargo-test cargo-fmt clippy]}}}
7 changes: 6 additions & 1 deletion examples/complex_struct_deserialization.rs
Original file line number Diff line number Diff line change
Expand Up @@ -70,7 +70,12 @@ fn complex_wrong() -> Result<(), EdnError> {
let bad_edn_str = "{:list [{:name \"rose\" :age \"some text\" :cool true}, {:name \"josh\" :age 33 :cool false}, {:name \"eva\" :age 296 :cool true}]}";
let complex: Result<Complex, EdnError> = edn_rs::from_str(bad_edn_str);

assert!(complex.is_err());
assert_eq!(
complex,
Err(EdnError::Deserialize(
"couldn't convert `\"some text\"` into `uint`".to_string()
))
);

Ok(())
}
Expand Down
7 changes: 6 additions & 1 deletion examples/from_edn.rs
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,12 @@ fn person_mistyped() -> Result<(), EdnError> {
}));
let person: Result<Person, EdnError> = edn_rs::from_edn(&bad_edn);

assert!(person.is_err());
assert_eq!(
person,
Err(EdnError::Deserialize(
"couldn't convert `\"some text\"` into `uint`".to_string()
))
);

Ok(())
}
Expand Down
14 changes: 12 additions & 2 deletions examples/struct_from_str.rs
Original file line number Diff line number Diff line change
Expand Up @@ -36,15 +36,25 @@ fn person_mistyped() -> Result<(), EdnError> {
let bad_edn_str = "{:name \"rose\" :age \"some text\" }";
let person: Result<Person, EdnError> = edn_rs::from_str(bad_edn_str);

assert!(person.is_err());
assert_eq!(
person,
Err(EdnError::Deserialize(
"couldn't convert `\"some text\"` into `uint`".to_string()
))
);

Ok(())
}

fn person_overflow() -> Result<(), EdnError> {
let overflow_edn_str = " {:name \"rose\" :age 9002 } ";
let person: Result<Person, EdnError> = edn_rs::from_str(overflow_edn_str);

assert!(person.is_err());
assert_eq!(
format!("{person:?}"),
"Err(TryFromInt(TryFromIntError(())))"
);

Ok(())
}

Expand Down
133 changes: 87 additions & 46 deletions src/deserialize/mod.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use alloc::collections::BTreeMap;
#[cfg(feature = "sets")]
use alloc::collections::BTreeSet;
use alloc::format;
use alloc::string::{String, ToString};
use alloc::vec::Vec;
use core::any;
Expand All @@ -11,8 +12,7 @@ use std::collections::HashMap;
#[cfg(all(feature = "sets", feature = "std"))]
use std::collections::HashSet;

use crate::edn::Edn;
use crate::EdnError as Error;
use crate::edn::{Edn, Error};

pub mod parse;

Expand All @@ -23,7 +23,7 @@ use ordered_float::OrderedFloat;
///
/// # Errors
///
/// Error implements Debug. See docs for more information.
/// Error will be like `EdnError::Deserialize("couldn't convert <value> into <type>")`
///
/// ```
/// use crate::edn_rs::{Edn, EdnError, Deserialize};
Expand Down Expand Up @@ -60,22 +60,27 @@ use ordered_float::OrderedFloat;
/// let bad_edn_str = "{:name \"rose\" :age \"some text\" }";
/// let person: Result<Person, EdnError> = edn_rs::from_str(bad_edn_str);
///
/// println!("{:?}", person);
/// assert_eq!(
/// person,
/// Err(EdnError::Deserialize(
/// "couldn't convert `\"some text\"` into `uint`".to_string()
/// ))
/// );
/// ```
#[allow(clippy::missing_errors_doc)]
pub trait Deserialize: Sized {
fn deserialize(edn: &Edn) -> Result<Self, Error>;
}

const fn build_deserialize_error(type_: &'static str) -> Error {
Error::deserialize(type_)
fn build_deserialize_error(edn: &Edn, type_: &str) -> Error {
Error::Deserialize(format!("couldn't convert `{edn}` into `{type_}`"))
}

impl Deserialize for () {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Nil => Ok(()),
_ => Err(build_deserialize_error("unit")),
_ => Err(build_deserialize_error(edn, "unit")),
}
}
}
Expand All @@ -84,15 +89,15 @@ impl Deserialize for () {
impl Deserialize for OrderedFloat<f64> {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
edn.to_float()
.ok_or_else(|| build_deserialize_error("edn_rs::Double"))
.ok_or_else(|| build_deserialize_error(edn, "edn_rs::Double"))
.map(Into::into)
}
}

impl Deserialize for f64 {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
edn.to_float()
.ok_or_else(|| build_deserialize_error("edn_rs::Double"))
.ok_or_else(|| build_deserialize_error(edn, "edn_rs::Double"))
.map(Into::into)
}
}
Expand All @@ -104,7 +109,7 @@ macro_rules! impl_deserialize_int {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
let int = edn
.to_int()
.ok_or_else(|| build_deserialize_error("int"))?;
.ok_or_else(|| build_deserialize_error(edn, "int"))?;
Ok(Self::try_from(int)?)
}
}
Expand All @@ -121,7 +126,7 @@ macro_rules! impl_deserialize_uint {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
let uint = edn
.to_uint()
.ok_or_else(|| build_deserialize_error("uint"))?;
.ok_or_else(|| build_deserialize_error(edn, "uint"))?;
Ok(Self::try_from(uint)?)
}
}
Expand All @@ -133,7 +138,8 @@ impl_deserialize_uint!(u8, u16, u32, u64, usize);

impl Deserialize for bool {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
edn.to_bool().ok_or_else(|| build_deserialize_error("bool"))
edn.to_bool()
.ok_or_else(|| build_deserialize_error(edn, "bool"))
}
}

Expand All @@ -154,7 +160,8 @@ impl Deserialize for String {

impl Deserialize for char {
fn deserialize(edn: &Edn) -> Result<Self, Error> {
edn.to_char().ok_or_else(|| build_deserialize_error("char"))
edn.to_char()
.ok_or_else(|| build_deserialize_error(edn, "char"))
}
}

Expand All @@ -164,23 +171,23 @@ where
{
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Vector(v) => Ok(v
.0
.iter()
Edn::Vector(_) => Ok(edn
.iter_some()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|e| Deserialize::deserialize(e))
.collect::<Result<Self, Error>>()?),
Edn::List(l) => Ok(l
.0
.iter()
Edn::List(_) => Ok(edn
.iter_some()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|e| Deserialize::deserialize(e))
.collect::<Result<Self, Error>>()?),
#[cfg(feature = "sets")]
Edn::Set(s) => Ok(s
.0
.iter()
Edn::Set(_) => Ok(edn
.iter_some()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|e| Deserialize::deserialize(e))
.collect::<Result<Self, Error>>()?),
_ => Err(build_deserialize_error(any::type_name::<Self>())),
_ => Err(build_deserialize_error(edn, any::type_name::<Self>())),
}
}
}
Expand All @@ -193,17 +200,22 @@ where
{
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Map(m) => m
.0
.iter()
Edn::Map(_) => edn
.map_iter()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|(key, e)| {
Ok((
key.to_string(),
Deserialize::deserialize(e).map_err(|_| Error::deserialize("HashMap"))?,
Deserialize::deserialize(e).map_err(|_| {
Error::Deserialize(format!(
"Cannot safely deserialize {:?} to {}",
edn, "HashMap"
))
})?,
))
})
.collect::<Result<Self, Error>>(),
_ => Err(build_deserialize_error(any::type_name::<Self>())),
_ => Err(build_deserialize_error(edn, any::type_name::<Self>())),
}
}
}
Expand All @@ -214,17 +226,22 @@ where
{
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Map(m) => m
.0
.iter()
Edn::Map(_) => edn
.map_iter()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|(key, e)| {
Ok((
key.to_string(),
Deserialize::deserialize(e).map_err(|_| Error::deserialize("BTreeMap"))?,
Deserialize::deserialize(e).map_err(|_| {
Error::Deserialize(format!(
"Cannot safely deserialize {:?} to {}",
edn, "BTreeMap"
))
})?,
))
})
.collect::<Result<Self, Error>>(),
_ => Err(build_deserialize_error(any::type_name::<Self>())),
_ => Err(build_deserialize_error(edn, any::type_name::<Self>())),
}
}
}
Expand All @@ -237,12 +254,19 @@ where
{
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Set(s) => {
s.0.iter()
.map(|e| Deserialize::deserialize(e).map_err(|_| Error::deserialize("HashSet")))
.collect::<Result<Self, Error>>()
}
_ => Err(build_deserialize_error(any::type_name::<Self>())),
Edn::Set(_) => edn
.set_iter()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|e| {
Deserialize::deserialize(e).map_err(|_| {
Error::Deserialize(format!(
"Cannot safely deserialize {:?} to {}",
edn, "HashSet"
))
})
})
.collect::<Result<Self, Error>>(),
_ => Err(build_deserialize_error(edn, any::type_name::<Self>())),
}
}
}
Expand All @@ -254,12 +278,19 @@ where
{
fn deserialize(edn: &Edn) -> Result<Self, Error> {
match edn {
Edn::Set(s) => s
.0
.iter()
.map(|e| Deserialize::deserialize(e).map_err(|_| Error::deserialize("BTreeSet")))
Edn::Set(_) => edn
.set_iter()
.ok_or_else(|| Error::Iter(format!("Could not create iter from {edn:?}")))?
.map(|e| {
Deserialize::deserialize(e).map_err(|_| {
Error::Deserialize(format!(
"Cannot safely deserialize {:?} to {}",
edn, "BTreeSet"
))
})
})
.collect::<Result<Self, Error>>(),
_ => Err(build_deserialize_error(any::type_name::<Self>())),
_ => Err(build_deserialize_error(edn, any::type_name::<Self>())),
}
}
}
Expand Down Expand Up @@ -317,7 +348,12 @@ where
/// let bad_edn_str = "{:name \"rose\" :age \"some text\" }";
/// let person: Result<Person, EdnError> = edn_rs::from_str(bad_edn_str);
///
/// println!("{:?}", person);
/// assert_eq!(
/// person,
/// Err(EdnError::Deserialize(
/// "couldn't convert `\"some text\"` into `uint`".to_string()
/// ))
/// );
/// ```
pub fn from_str<T: Deserialize>(s: &str) -> Result<T, Error> {
let edn = Edn::from_str(s)?;
Expand Down Expand Up @@ -371,7 +407,12 @@ pub fn from_str<T: Deserialize>(s: &str) -> Result<T, Error> {
/// }));
/// let person: Result<Person, EdnError> = edn_rs::from_edn(&bad_edn);
///
/// println!("{:?}", person);
/// assert_eq!(
/// person,
/// Err(EdnError::Deserialize(
/// "couldn't convert `\"some text\"` into `uint`".to_string()
/// ))
/// );
/// ```
pub fn from_edn<T: Deserialize>(edn: &Edn) -> Result<T, Error> {
T::deserialize(edn)
Expand Down
Loading

0 comments on commit 5d34779

Please sign in to comment.