use crate::{
error::{Error, ErrorKind},
version::Version,
};
use serde::{de, ser, Deserialize, Serialize};
use std::{fmt, str::FromStr};
#[derive(Clone, Debug, Deserialize, Eq, PartialEq, Serialize)]
pub struct Package {
pub name: Name,
pub version: Version,
pub source: Option<String>,
#[serde(
default,
serialize_with = "serialize_dependencies",
deserialize_with = "deserialize_dependencies"
)]
pub dependencies: Vec<Package>,
}
pub(crate) fn serialize_dependencies<S>(
dependencies: &[Package],
serializer: S,
) -> Result<S::Ok, S::Error>
where
S: ser::Serializer,
{
dependencies
.iter()
.map(|dependency| {
let mut dependency_string = format!("{} {}", &dependency.name, &dependency.version);
if let Some(source) = &dependency.source {
dependency_string = format!("{} ({})", dependency_string, source);
}
dependency_string
})
.collect::<Vec<_>>()
.serialize(serializer)
}
pub(crate) fn deserialize_dependencies<'de, D>(deserializer: D) -> Result<Vec<Package>, D::Error>
where
D: de::Deserializer<'de>,
{
let dependencies = Vec::<String>::deserialize(deserializer)?;
let mut result = vec![];
for dependency_string in &dependencies {
let mut parts = dependency_string.split_whitespace();
let name_str = parts
.next()
.ok_or_else(|| de::Error::custom("empty dependency string"))?;
let version_str = parts.next().ok_or_else(|| {
de::Error::custom(format!(
"missing version for dependency: {}",
&dependency_string
))
})?;
let source = parts
.next()
.map(|s| {
if s.len() < 2 || !s.starts_with('(') || !s.ends_with(')') {
Err(de::Error::custom(format!(
"malformed source in dependency: {}",
&dependency_string,
)))
} else {
Ok(s[1..(s.len() - 1)].to_owned())
}
})
.transpose()?;
if parts.next().is_some() {
Err(de::Error::custom(format!(
"malformed dependency: {}",
dependency_string,
)))?;
}
result.push(Package {
name: name_str.into(),
version: version_str.parse().map_err(de::Error::custom)?,
source,
dependencies: vec![],
});
}
Ok(result)
}
#[derive(Clone, Debug, Deserialize, Eq, Hash, PartialEq, PartialOrd, Ord, Serialize)]
pub struct Name(String);
impl Name {
pub fn as_str(&self) -> &str {
&self.0
}
}
impl AsRef<Name> for Name {
fn as_ref(&self) -> &Name {
self
}
}
impl fmt::Display for Name {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.0.fmt(f)
}
}
impl<'a> From<&'a str> for Name {
fn from(string: &'a str) -> Name {
Name(string.into())
}
}
impl Into<String> for Name {
fn into(self) -> String {
self.0
}
}
#[derive(Copy, Clone, Debug, Eq, Hash, PartialEq, PartialOrd, Ord)]
pub enum Collection {
Crates,
Rust,
}
impl Collection {
pub fn as_str(&self) -> &str {
match self {
Collection::Crates => "crates",
Collection::Rust => "rust",
}
}
}
impl fmt::Display for Collection {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.as_str())
}
}
impl FromStr for Collection {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
Ok(match s {
"crates" => Collection::Crates,
"rust" => Collection::Rust,
other => fail!(ErrorKind::Parse, "invalid package type: {}", other),
})
}
}
impl<'de> Deserialize<'de> for Collection {
fn deserialize<D: de::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
use de::Error;
let string = String::deserialize(deserializer)?;
string.parse().map_err(D::Error::custom)
}
}
impl Serialize for Collection {
fn serialize<S: ser::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
self.to_string().serialize(serializer)
}
}
#[cfg(test)]
mod tests {
use super::Collection;
#[test]
fn parse_crate() {
let crate_kind = "crates".parse::<Collection>().unwrap();
assert_eq!(Collection::Crates, crate_kind);
assert_eq!("crates", crate_kind.as_str());
}
#[test]
fn parse_rust() {
let rust_kind = "rust".parse::<Collection>().unwrap();
assert_eq!(Collection::Rust, rust_kind);
assert_eq!("rust", rust_kind.as_str());
}
#[test]
fn parse_other() {
assert!("foobar".parse::<Collection>().is_err());
}
}