refactor(nix-compat/store_path): use AsRef<str>

Implement PartialEq/Eq ourselves instead of deriving, by proxying to
name.as_ref() (and digest of course).

Also implement Hash on our own, clippy doesn't like this to be derived,
while Eq/PartialEq is not.

Change-Id: Idbe289a23ba3bc8dabf893d4d8752792ae2778c3
Reviewed-on: https://cl.tvl.fyi/c/depot/+/12744
Tested-by: BuildkiteCI
Reviewed-by: edef <edef@edef.eu>
Autosubmit: flokli <flokli@flokli.de>
This commit is contained in:
Florian Klink 2024-10-16 02:51:35 +03:00 committed by flokli
parent 1474471327
commit 1428ea4e19
5 changed files with 48 additions and 34 deletions

View file

@ -203,11 +203,7 @@ fn string_to_store_path<'a, 'i, S>(
path_str: &'a str, path_str: &'a str,
) -> Result<StorePath<S>, nom::Err<NomError<&'i [u8]>>> ) -> Result<StorePath<S>, nom::Err<NomError<&'i [u8]>>>
where where
S: std::cmp::Eq S: std::clone::Clone + AsRef<str> + std::convert::From<&'a str>,
+ std::fmt::Display
+ std::clone::Clone
+ std::ops::Deref<Target = str>
+ std::convert::From<&'a str>,
{ {
let path = let path =
StorePath::from_absolute_path(path_str.as_bytes()).map_err(|e: store_path::Error| { StorePath::from_absolute_path(path_str.as_bytes()).map_err(|e: store_path::Error| {

View file

@ -36,14 +36,14 @@ pub(crate) trait AtermWriteable {
impl<S> AtermWriteable for StorePath<S> impl<S> AtermWriteable for StorePath<S>
where where
S: std::cmp::Eq + std::ops::Deref<Target = str>, S: AsRef<str>,
{ {
fn aterm_write(&self, writer: &mut impl Write) -> std::io::Result<()> { fn aterm_write(&self, writer: &mut impl Write) -> std::io::Result<()> {
write_char(writer, QUOTE)?; write_char(writer, QUOTE)?;
writer.write_all(STORE_DIR_WITH_SLASH.as_bytes())?; writer.write_all(STORE_DIR_WITH_SLASH.as_bytes())?;
writer.write_all(nixbase32::encode(self.digest()).as_bytes())?; writer.write_all(nixbase32::encode(self.digest()).as_bytes())?;
write_char(writer, '-')?; write_char(writer, '-')?;
writer.write_all(self.name().as_bytes())?; writer.write_all(self.name().as_ref().as_bytes())?;
write_char(writer, QUOTE)?; write_char(writer, QUOTE)?;
Ok(()) Ok(())
} }

View file

@ -1,5 +1,3 @@
use std::{fmt::Display, ops::Deref};
use nix_compat_derive::{NixDeserialize, NixSerialize}; use nix_compat_derive::{NixDeserialize, NixSerialize};
use crate::{ use crate::{
@ -130,7 +128,7 @@ impl NixDeserialize for StorePath<String> {
// Custom implementation since Display does not use absolute paths. // Custom implementation since Display does not use absolute paths.
impl<S> NixSerialize for StorePath<S> impl<S> NixSerialize for StorePath<S>
where where
S: std::cmp::Eq + Deref<Target = str> + Display + Sync, S: AsRef<str> + Sync,
{ {
async fn serialize<W>(&self, writer: &mut W) -> Result<(), W::Error> async fn serialize<W>(&self, writer: &mut W) -> Result<(), W::Error>
where where

View file

@ -2,8 +2,7 @@ use crate::nixbase32;
use data_encoding::{DecodeError, BASE64}; use data_encoding::{DecodeError, BASE64};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::{ use std::{
fmt::{self, Display}, fmt,
ops::Deref,
path::Path, path::Path,
str::{self, FromStr}, str::{self, FromStr},
}; };
@ -51,21 +50,40 @@ pub enum Error {
/// ///
/// A [StorePath] does not encode any additional subpath "inside" the store /// A [StorePath] does not encode any additional subpath "inside" the store
/// path. /// path.
#[derive(Clone, Debug, PartialEq, Eq, Hash)] #[derive(Clone, Debug)]
pub struct StorePath<S> pub struct StorePath<S> {
where
S: std::cmp::Eq + std::cmp::PartialEq,
{
digest: [u8; DIGEST_SIZE], digest: [u8; DIGEST_SIZE],
name: S, name: S,
} }
impl<S> PartialEq for StorePath<S>
where
S: AsRef<str>,
{
fn eq(&self, other: &Self) -> bool {
self.digest() == other.digest() && self.name().as_ref() == other.name().as_ref()
}
}
impl<S> Eq for StorePath<S> where S: AsRef<str> {}
impl<S> std::hash::Hash for StorePath<S>
where
S: AsRef<str>,
{
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
state.write(&self.digest);
state.write(self.name.as_ref().as_bytes());
}
}
/// Like [StorePath], but without a heap allocation for the name. /// Like [StorePath], but without a heap allocation for the name.
/// Used by [StorePath] for parsing. /// Used by [StorePath] for parsing.
pub type StorePathRef<'a> = StorePath<&'a str>; pub type StorePathRef<'a> = StorePath<&'a str>;
impl<S> StorePath<S> impl<S> StorePath<S>
where where
S: std::cmp::Eq + Deref<Target = str>, S: AsRef<str>,
{ {
pub fn digest(&self) -> &[u8; DIGEST_SIZE] { pub fn digest(&self) -> &[u8; DIGEST_SIZE] {
&self.digest &self.digest
@ -78,14 +96,14 @@ where
pub fn as_ref(&self) -> StorePathRef<'_> { pub fn as_ref(&self) -> StorePathRef<'_> {
StorePathRef { StorePathRef {
digest: self.digest, digest: self.digest,
name: &self.name, name: self.name.as_ref(),
} }
} }
pub fn to_owned(&self) -> StorePath<String> { pub fn to_owned(&self) -> StorePath<String> {
StorePath { StorePath {
digest: self.digest, digest: self.digest,
name: self.name.to_string(), name: self.name.as_ref().to_string(),
} }
} }
@ -183,17 +201,14 @@ where
/// Returns an absolute store path string. /// Returns an absolute store path string.
/// That is just the string representation, prefixed with the store prefix /// That is just the string representation, prefixed with the store prefix
/// ([STORE_DIR_WITH_SLASH]), /// ([STORE_DIR_WITH_SLASH]),
pub fn to_absolute_path(&self) -> String pub fn to_absolute_path(&self) -> String {
where
S: Display,
{
format!("{}{}", STORE_DIR_WITH_SLASH, self) format!("{}{}", STORE_DIR_WITH_SLASH, self)
} }
} }
impl<S> PartialOrd for StorePath<S> impl<S> PartialOrd for StorePath<S>
where where
S: std::cmp::PartialEq + std::cmp::Eq, S: AsRef<str>,
{ {
fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> { fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
Some(self.cmp(other)) Some(self.cmp(other))
@ -204,7 +219,7 @@ where
/// of the nixbase32-encoded string. /// of the nixbase32-encoded string.
impl<S> Ord for StorePath<S> impl<S> Ord for StorePath<S>
where where
S: std::cmp::PartialEq + std::cmp::Eq, S: AsRef<str>,
{ {
fn cmp(&self, other: &Self) -> std::cmp::Ordering { fn cmp(&self, other: &Self) -> std::cmp::Ordering {
self.digest.iter().rev().cmp(other.digest.iter().rev()) self.digest.iter().rev().cmp(other.digest.iter().rev())
@ -223,7 +238,7 @@ impl<'a, 'b: 'a> FromStr for StorePath<String> {
impl<'a, 'de: 'a, S> Deserialize<'de> for StorePath<S> impl<'a, 'de: 'a, S> Deserialize<'de> for StorePath<S>
where where
S: std::cmp::Eq + Deref<Target = str> + From<&'a str>, S: AsRef<str> + From<&'a str>,
{ {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error> fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where where
@ -245,7 +260,7 @@ where
impl<S> Serialize for StorePath<S> impl<S> Serialize for StorePath<S>
where where
S: std::cmp::Eq + Deref<Target = str> + Display, S: AsRef<str>,
{ {
fn serialize<SR>(&self, serializer: SR) -> Result<SR::Ok, SR::Error> fn serialize<SR>(&self, serializer: SR) -> Result<SR::Ok, SR::Error>
where where
@ -305,13 +320,18 @@ pub(crate) fn validate_name(s: &(impl AsRef<[u8]> + ?Sized)) -> Result<&str, Err
impl<S> fmt::Display for StorePath<S> impl<S> fmt::Display for StorePath<S>
where where
S: fmt::Display + std::cmp::Eq, S: AsRef<str>,
{ {
/// The string representation of a store path starts with a digest (20 /// The string representation of a store path starts with a digest (20
/// bytes), [crate::nixbase32]-encoded, followed by a `-`, /// bytes), [crate::nixbase32]-encoded, followed by a `-`,
/// and ends with the name. /// and ends with the name.
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}-{}", nixbase32::encode(&self.digest), self.name) write!(
f,
"{}-{}",
nixbase32::encode(&self.digest),
self.name.as_ref()
)
} }
} }

View file

@ -50,7 +50,7 @@ pub fn build_text_path<'a, S, SP, I, C>(
) -> Result<StorePath<SP>, BuildStorePathError> ) -> Result<StorePath<SP>, BuildStorePathError>
where where
S: AsRef<str>, S: AsRef<str>,
SP: std::cmp::Eq + std::ops::Deref<Target = str> + std::convert::From<&'a str>, SP: AsRef<str> + std::convert::From<&'a str>,
I: IntoIterator<Item = S>, I: IntoIterator<Item = S>,
C: AsRef<[u8]>, C: AsRef<[u8]>,
{ {
@ -69,7 +69,7 @@ pub fn build_ca_path<'a, S, SP, I>(
) -> Result<StorePath<SP>, BuildStorePathError> ) -> Result<StorePath<SP>, BuildStorePathError>
where where
S: AsRef<str>, S: AsRef<str>,
SP: std::cmp::Eq + std::ops::Deref<Target = str> + std::convert::From<&'a str>, SP: AsRef<str> + std::convert::From<&'a str>,
I: IntoIterator<Item = S>, I: IntoIterator<Item = S>,
{ {
// self references are only allowed for CAHash::Nar(NixHash::Sha256(_)). // self references are only allowed for CAHash::Nar(NixHash::Sha256(_)).
@ -129,7 +129,7 @@ pub fn build_output_path<'a, SP>(
output_path_name: &'a str, output_path_name: &'a str,
) -> Result<StorePath<SP>, Error> ) -> Result<StorePath<SP>, Error>
where where
SP: std::cmp::Eq + std::ops::Deref<Target = str> + std::convert::From<&'a str>, SP: AsRef<str> + std::convert::From<&'a str>,
{ {
build_store_path_from_fingerprint_parts( build_store_path_from_fingerprint_parts(
&(String::from("output:") + output_name), &(String::from("output:") + output_name),
@ -154,7 +154,7 @@ fn build_store_path_from_fingerprint_parts<'a, SP>(
name: &'a str, name: &'a str,
) -> Result<StorePath<SP>, Error> ) -> Result<StorePath<SP>, Error>
where where
SP: std::cmp::Eq + std::ops::Deref<Target = str> + std::convert::From<&'a str>, SP: AsRef<str> + std::convert::From<&'a str>,
{ {
let fingerprint = format!( let fingerprint = format!(
"{ty}:sha256:{}:{STORE_DIR}:{name}", "{ty}:sha256:{}:{STORE_DIR}:{name}",