use tokio::io::AsyncReadExt;
mod nar {
pub use crate::nar::reader::r#async as reader;
}
#[tokio::test]
async fn symlink() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/symlink.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::Symlink { target } => {
assert_eq!(
&b"/nix/store/somewhereelse"[..],
&target,
"target must match"
);
}
_ => panic!("unexpected type"),
}
}
#[tokio::test]
async fn file() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/helloworld.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::File {
executable,
mut reader,
} => {
assert!(!executable);
let mut buf = vec![];
reader
.read_to_end(&mut buf)
.await
.expect("read must succeed");
assert_eq!(&b"Hello World!"[..], &buf);
}
_ => panic!("unexpected type"),
}
}
#[tokio::test]
async fn complicated() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/complicated.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::Directory(mut dir_reader) => {
// first entry is .keep, an empty regular file.
must_read_file(
".keep",
dir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some"),
)
.await;
// second entry is aa, a symlink to /nix/store/somewhereelse
must_be_symlink(
"aa",
"/nix/store/somewhereelse",
dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some"),
);
{
// third entry is a directory called "keep"
let entry = dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some");
assert_eq!(b"keep", entry.name);
match entry.node {
nar::reader::Node::Directory(mut subdir_reader) => {
{
// first entry is .keep, an empty regular file.
let entry = subdir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some");
must_read_file(".keep", entry).await;
}
// we must read the None
assert!(
subdir_reader
.next()
.await
.expect("next must succeed")
.is_none(),
"keep directory contains only .keep"
);
}
_ => panic!("unexpected type for keep/.keep"),
}
};
// reading more entries yields None (and we actually must read until this)
assert!(dir_reader.next().await.expect("must succeed").is_none());
}
_ => panic!("unexpected type"),
}
}
#[tokio::test]
#[should_panic]
#[ignore = "TODO: async poisoning"]
async fn file_read_abandoned() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/complicated.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::Directory(mut dir_reader) => {
// first entry is .keep, an empty regular file.
{
let entry = dir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some");
assert_eq!(b".keep", entry.name);
// don't bother to finish reading it.
};
// this should panic (not return an error), because we are meant to abandon the archive reader now.
assert!(dir_reader.next().await.expect("must succeed").is_none());
}
_ => panic!("unexpected type"),
}
}
#[tokio::test]
#[should_panic]
#[ignore = "TODO: async poisoning"]
async fn dir_read_abandoned() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/complicated.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::Directory(mut dir_reader) => {
// first entry is .keep, an empty regular file.
must_read_file(
".keep",
dir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some"),
)
.await;
// second entry is aa, a symlink to /nix/store/somewhereelse
must_be_symlink(
"aa",
"/nix/store/somewhereelse",
dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some"),
);
{
// third entry is a directory called "keep"
let entry = dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some");
assert_eq!(b"keep", entry.name);
match entry.node {
nar::reader::Node::Directory(_) => {
// don't finish using it, which poisons the archive reader
}
_ => panic!("unexpected type for keep/.keep"),
}
};
// this should panic, because we didn't finish reading the child subdirectory
assert!(dir_reader.next().await.expect("must succeed").is_none());
}
_ => panic!("unexpected type"),
}
}
#[tokio::test]
#[should_panic]
#[ignore = "TODO: async poisoning"]
async fn dir_read_after_none() {
let mut f = std::io::Cursor::new(include_bytes!("../../tests/complicated.nar"));
let node = nar::reader::open(&mut f).await.unwrap();
match node {
nar::reader::Node::Directory(mut dir_reader) => {
// first entry is .keep, an empty regular file.
must_read_file(
".keep",
dir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some"),
)
.await;
// second entry is aa, a symlink to /nix/store/somewhereelse
must_be_symlink(
"aa",
"/nix/store/somewhereelse",
dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some"),
);
{
// third entry is a directory called "keep"
let entry = dir_reader
.next()
.await
.expect("next must be some")
.expect("must be some");
assert_eq!(b"keep", entry.name);
match entry.node {
nar::reader::Node::Directory(mut subdir_reader) => {
// first entry is .keep, an empty regular file.
must_read_file(
".keep",
subdir_reader
.next()
.await
.expect("next must succeed")
.expect("must be some"),
)
.await;
// we must read the None
assert!(
subdir_reader
.next()
.await
.expect("next must succeed")
.is_none(),
"keep directory contains only .keep"
);
}
_ => panic!("unexpected type for keep/.keep"),
}
};
// reading more entries yields None (and we actually must read until this)
assert!(dir_reader.next().await.expect("must succeed").is_none());
// this should panic, because we already got a none so we're meant to stop.
dir_reader.next().await.unwrap();
unreachable!()
}
_ => panic!("unexpected type"),
}
}
async fn must_read_file(name: &'static str, entry: nar::reader::Entry<'_, '_>) {
assert_eq!(name.as_bytes(), entry.name);
match entry.node {
nar::reader::Node::File {
executable,
mut reader,
} => {
assert!(!executable);
assert_eq!(reader.read(&mut [0]).await.unwrap(), 0);
}
_ => panic!("unexpected type for {}", name),
}
}
fn must_be_symlink(
name: &'static str,
exp_target: &'static str,
entry: nar::reader::Entry<'_, '_>,
) {
assert_eq!(name.as_bytes(), entry.name);
match entry.node {
nar::reader::Node::Symlink { target } => {
assert_eq!(exp_target.as_bytes(), &target);
}
_ => panic!("unexpected type for {}", name),
}
}