use super::{Lockfile, ResolveVersion};
use crate::{
metadata, Checksum, Dependency, Error, ErrorKind, Metadata, Name, Package, Patch, SourceId,
Version,
};
use serde::{de, ser, Deserialize, Serialize};
use std::{
convert::{TryFrom, TryInto},
fmt,
str::FromStr,
};
impl<'de> Deserialize<'de> for Lockfile {
fn deserialize<D: de::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
let raw_lockfile = EncodableLockfile::deserialize(deserializer)?;
raw_lockfile.try_into().map_err(de::Error::custom)
}
}
impl Serialize for Lockfile {
fn serialize<S: ser::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
EncodableLockfile::from(self).serialize(serializer)
}
}
#[derive(Debug, Deserialize, Serialize)]
pub(super) struct EncodableLockfile {
#[serde(default)]
pub(super) package: Vec<EncodablePackage>,
pub(super) root: Option<EncodablePackage>,
#[serde(default, skip_serializing_if = "Metadata::is_empty")]
pub(super) metadata: Metadata,
#[serde(default, skip_serializing_if = "Patch::is_empty")]
pub(super) patch: Patch,
}
impl EncodableLockfile {
pub fn find_checksum(&self, package: &Package) -> Option<Checksum> {
for (key, value) in &self.metadata {
if let Ok(dep) = key.checksum_dependency() {
if dep.name == package.name && dep.version == package.version {
return value.checksum().ok();
}
}
}
None
}
}
impl TryFrom<EncodableLockfile> for Lockfile {
type Error = Error;
fn try_from(raw_lockfile: EncodableLockfile) -> Result<Lockfile, Error> {
let version = ResolveVersion::detect(&raw_lockfile.package, &raw_lockfile.metadata)?;
let mut packages = Vec::with_capacity(raw_lockfile.package.len());
for raw_package in &raw_lockfile.package {
packages.push(match version {
ResolveVersion::V1 => {
let mut pkg = Package::try_from(raw_package)?;
pkg.checksum = raw_lockfile.find_checksum(&pkg);
pkg
}
ResolveVersion::V2 => raw_package.resolve(&raw_lockfile.package)?,
});
}
Ok(Lockfile {
version,
packages,
root: raw_lockfile
.root
.as_ref()
.map(|root| root.try_into())
.transpose()?,
metadata: raw_lockfile.metadata,
patch: raw_lockfile.patch,
})
}
}
impl From<&Lockfile> for EncodableLockfile {
fn from(lockfile: &Lockfile) -> EncodableLockfile {
let mut packages = Vec::with_capacity(lockfile.packages.len());
let mut metadata = lockfile.metadata.clone();
for package in &lockfile.packages {
let mut raw_pkg = EncodablePackage::from(package);
let checksum_key = metadata::Key::for_checksum(&Dependency::from(package));
match lockfile.version {
ResolveVersion::V1 => {
if let Some(checksum) = raw_pkg.checksum.take() {
let value = checksum.to_string().parse::<metadata::Value>().unwrap();
metadata.insert(checksum_key, value);
}
}
ResolveVersion::V2 => {
raw_pkg.v2_deps(&lockfile.packages);
metadata.remove(&checksum_key);
}
}
packages.push(raw_pkg);
}
EncodableLockfile {
package: packages,
root: lockfile.root.as_ref().map(|root| root.into()),
metadata,
patch: lockfile.patch.clone(),
}
}
}
#[derive(Debug, Deserialize, Serialize)]
pub(crate) struct EncodablePackage {
pub(super) name: Name,
pub(super) version: Version,
pub(super) source: Option<SourceId>,
pub(super) checksum: Option<Checksum>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub(super) dependencies: Vec<EncodableDependency>,
pub(super) replace: Option<EncodableDependency>,
}
impl EncodablePackage {
fn resolve(&self, packages: &[EncodablePackage]) -> Result<Package, Error> {
let mut dependencies = Vec::with_capacity(self.dependencies.len());
for dep in &self.dependencies {
dependencies.push(dep.resolve(packages)?);
}
Ok(Package {
name: self.name.clone(),
version: self.version.clone(),
source: self.source.clone(),
checksum: self.checksum.clone(),
dependencies,
replace: self
.replace
.as_ref()
.map(|rep| rep.try_into())
.transpose()?,
})
}
fn v2_deps(&mut self, packages: &[Package]) {
for dependency in &mut self.dependencies {
dependency.v2(packages);
}
}
}
impl TryFrom<&EncodablePackage> for Package {
type Error = Error;
fn try_from(raw_package: &EncodablePackage) -> Result<Package, Error> {
raw_package.resolve(&[])
}
}
impl From<&Package> for EncodablePackage {
fn from(package: &Package) -> EncodablePackage {
EncodablePackage {
name: package.name.clone(),
version: package.version.clone(),
source: package.source.clone(),
checksum: package.checksum.clone(),
dependencies: package
.dependencies
.iter()
.map(|dep| dep.into())
.collect::<Vec<_>>(),
replace: package.replace.as_ref().map(|rep| rep.into()),
}
}
}
#[derive(Clone, Debug, Eq, Hash, PartialEq, PartialOrd, Ord)]
pub(crate) struct EncodableDependency {
pub(super) name: Name,
pub(super) version: Option<Version>,
pub(super) source: Option<SourceId>,
}
impl EncodableDependency {
pub fn resolve(&self, packages: &[EncodablePackage]) -> Result<Dependency, Error> {
let mut version = None;
let mut source = None;
if let Some(v) = &self.version {
version = Some(v.clone());
source = self.source.clone();
} else {
for pkg in packages {
if pkg.name == self.name {
if version.is_some() {
fail!(ErrorKind::Parse, "ambiguous dependency: {}", self.name);
}
version = Some(pkg.version.clone());
source = pkg.source.clone();
}
}
};
if version.is_none() {
fail!(
ErrorKind::Parse,
"couldn't resolve dependency: {}",
self.name
);
}
Ok(Dependency {
name: self.name.clone(),
version: version.unwrap(),
source,
})
}
pub fn v2(&mut self, packages: &[Package]) {
let mut matching = vec![];
for package in packages {
if package.name == self.name {
matching.push(package);
}
}
if matching.len() == 1 {
self.version = None;
self.source = None;
}
}
}
impl FromStr for EncodableDependency {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Error> {
let mut parts = s.split_whitespace();
let name = parts
.next()
.ok_or_else(|| format_err!(ErrorKind::Parse, "empty dependency string"))?
.parse()?;
let version = parts.next().map(FromStr::from_str).transpose()?;
let source = parts
.next()
.map(|s| {
if s.len() < 2 || !s.starts_with('(') || !s.ends_with(')') {
Err(format_err!(
ErrorKind::Parse,
"malformed source in dependency: {}",
s
))
} else {
s[1..(s.len() - 1)].parse()
}
})
.transpose()?;
if parts.next().is_some() {
fail!(ErrorKind::Parse, "malformed dependency: {}", s);
}
Ok(Self {
name,
version,
source,
})
}
}
impl fmt::Display for EncodableDependency {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", &self.name)?;
if let Some(version) = &self.version {
write!(f, " {}", version)?;
}
if let Some(source) = &self.source {
write!(f, " ({})", source)?;
}
Ok(())
}
}
impl TryFrom<&EncodableDependency> for Dependency {
type Error = Error;
fn try_from(raw_dependency: &EncodableDependency) -> Result<Dependency, Error> {
raw_dependency.resolve(&[])
}
}
impl From<&Dependency> for EncodableDependency {
fn from(package: &Dependency) -> EncodableDependency {
EncodableDependency {
name: package.name.clone(),
version: Some(package.version.clone()),
source: package.source.clone(),
}
}
}
impl<'de> Deserialize<'de> for EncodableDependency {
fn deserialize<D: de::Deserializer<'de>>(deserializer: D) -> Result<Self, D::Error> {
use de::Error;
let s = String::deserialize(deserializer)?;
s.parse().map_err(D::Error::custom)
}
}
impl Serialize for EncodableDependency {
fn serialize<S: ser::Serializer>(&self, serializer: S) -> Result<S::Ok, S::Error> {
self.to_string().serialize(serializer)
}
}