about summary refs log tree commit diff
path: root/tvix/nix-compat/src/nar/reader/async/test.rs
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),
    }
}