test(tvix/store/pathinfo): add more tests for references

This should cover all error cases produced.

Change-Id: If31816d9b087551d86d7913df55df8f9f44bb554
Reviewed-on: https://cl.tvl.fyi/c/depot/+/9546
Autosubmit: flokli <flokli@flokli.de>
Reviewed-by: edef <edef@edef.eu>
Tested-by: BuildkiteCI
This commit is contained in:
Florian Klink 2023-10-05 11:57:05 +03:00 committed by flokli
parent 1f03a520a9
commit d45d6de561
2 changed files with 104 additions and 45 deletions

View file

@ -1,4 +1,4 @@
use crate::proto::{NarInfo, PathInfo, ValidatePathInfoError}; use crate::proto::{PathInfo, ValidatePathInfoError};
use crate::tests::fixtures::*; use crate::tests::fixtures::*;
use bytes::Bytes; use bytes::Bytes;
use nix_compat::store_path::{self, StorePath}; use nix_compat::store_path::{self, StorePath};
@ -150,49 +150,79 @@ fn validate_symlink(
assert_eq!(t_result, p.validate()); assert_eq!(t_result, p.validate());
} }
/// Ensure parsing a correct PathInfo without narinfo populated succeeds.
#[test] #[test]
fn validate_references() { fn validate_references_without_narinfo_ok() {
// create a PathInfo without narinfo field. assert!(PATH_INFO_WITHOUT_NARINFO.validate().is_ok());
let path_info = PathInfo { }
node: Some(castorepb::Node {
node: Some(castorepb::node::Node::Directory(castorepb::DirectoryNode { /// Ensure parsing a correct PathInfo with narinfo populated succeeds.
name: DUMMY_NAME.into(), #[test]
digest: DUMMY_DIGEST.clone().into(), fn validate_references_with_narinfo_ok() {
size: 0, assert!(PATH_INFO_WITH_NARINFO.validate().is_ok());
})), }
}),
references: vec![DUMMY_OUTPUT_HASH.clone().into()], /// Create a PathInfo with a wrong count of narinfo.reference_names,
narinfo: None, /// and ensure validation fails.
}; #[test]
assert!(path_info.validate().is_ok()); fn validate_inconsistent_num_refs_fail() {
let mut path_info = PATH_INFO_WITH_NARINFO.clone();
// create a PathInfo with a narinfo field, but an inconsistent set of references path_info.narinfo.as_mut().unwrap().reference_names = vec![];
let path_info_with_narinfo_missing_refs = PathInfo {
narinfo: Some(NarInfo { match path_info.validate().expect_err("must_fail") {
nar_size: 0, ValidatePathInfoError::InconsistentNumberOfReferences(1, 0) => {}
nar_sha256: DUMMY_DIGEST.clone().into(), e => panic!("unexpected error: {:?}", e),
signatures: vec![], };
reference_names: vec![], }
}),
..path_info.clone() /// Create a PathInfo with a wrong digest length in references.
}; #[test]
match path_info_with_narinfo_missing_refs fn validate_invalid_reference_digest_len() {
.validate() let mut path_info = PATH_INFO_WITHOUT_NARINFO.clone();
.expect_err("must_fail") path_info.references.push(vec![0xff, 0xff].into());
{
ValidatePathInfoError::InconsistentNumberOfReferences(_, _) => {} match path_info.validate().expect_err("must fail") {
_ => panic!("unexpected error"), ValidatePathInfoError::InvalidReferenceDigestLen(
}; 1, // position
2, // unexpected digest len
// create a pathinfo with the correct number of references, should suceed ) => {}
let path_info_with_narinfo = PathInfo { e => panic!("unexpected error: {:?}", e),
narinfo: Some(NarInfo { };
nar_size: 0, }
nar_sha256: DUMMY_DIGEST.clone().into(),
signatures: vec![], /// Create a PathInfo with a narinfo.reference_name[1] that is no valid store path.
reference_names: vec![DUMMY_NAME.to_string()], #[test]
}), fn validate_invalid_narinfo_reference_name() {
..path_info let mut path_info = PATH_INFO_WITH_NARINFO.clone();
};
assert!(path_info_with_narinfo.validate().is_ok()); // This is invalid, as the store prefix is not part of reference_names.
path_info.narinfo.as_mut().unwrap().reference_names[0] =
"/nix/store/00000000000000000000000000000000-dummy".to_string();
match path_info.validate().expect_err("must fail") {
ValidatePathInfoError::InvalidNarinfoReferenceName(0, reference_name) => {
assert_eq!(
"/nix/store/00000000000000000000000000000000-dummy",
reference_name
);
}
e => panic!("unexpected error: {:?}", e),
}
}
/// Create a PathInfo with a narinfo.reference_name[0] that doesn't match references[0].
#[test]
fn validate_inconsistent_narinfo_reference_name_digest() {
let mut path_info = PATH_INFO_WITH_NARINFO.clone();
// mutate the first reference, they were all zeroes before
path_info.references[0] = vec![0xff; store_path::DIGEST_SIZE].into();
match path_info.validate().expect_err("must fail") {
ValidatePathInfoError::InconsistentNarinfoReferenceNameDigest(0, e_expected, e_actual) => {
assert_eq!(path_info.references[0][..], e_expected);
assert_eq!(DUMMY_OUTPUT_HASH[..], e_actual);
}
e => panic!("unexpected error: {:?}", e),
}
} }

View file

@ -1,5 +1,8 @@
use lazy_static::lazy_static; use lazy_static::lazy_static;
pub use tvix_castore::fixtures::*; pub use tvix_castore::fixtures::*;
use tvix_castore::proto as castorepb;
use crate::proto::{NarInfo, PathInfo};
pub const DUMMY_NAME: &str = "00000000000000000000000000000000-dummy"; pub const DUMMY_NAME: &str = "00000000000000000000000000000000-dummy";
@ -94,4 +97,30 @@ lazy_static! {
1, 0, 0, 0, 0, 0, 0, 0, b')', 0, 0, 0, 0, 0, 0, 0, // ")" 1, 0, 0, 0, 0, 0, 0, 0, b')', 0, 0, 0, 0, 0, 0, 0, // ")"
1, 0, 0, 0, 0, 0, 0, 0, b')', 0, 0, 0, 0, 0, 0, 0, // ")" 1, 0, 0, 0, 0, 0, 0, 0, b')', 0, 0, 0, 0, 0, 0, 0, // ")"
]; ];
/// A PathInfo message without .narinfo populated.
pub static ref PATH_INFO_WITHOUT_NARINFO : PathInfo = PathInfo {
node: Some(castorepb::Node {
node: Some(castorepb::node::Node::Directory(castorepb::DirectoryNode {
name: DUMMY_NAME.into(),
digest: DUMMY_DIGEST.clone().into(),
size: 0,
})),
}),
references: vec![DUMMY_OUTPUT_HASH.clone().into()],
narinfo: None,
};
/// A PathInfo message with .narinfo populated.
/// The references in `narinfo.reference_names` aligns with what's in
/// `references`.
pub static ref PATH_INFO_WITH_NARINFO : PathInfo = PathInfo {
narinfo: Some(NarInfo {
nar_size: 0,
nar_sha256: DUMMY_DIGEST.clone().into(),
signatures: vec![],
reference_names: vec![DUMMY_NAME.to_string()],
}),
..PATH_INFO_WITHOUT_NARINFO.clone()
};
} }