use std::io::Read; use crate::nar; #[test] fn symlink() { let mut f = std::io::Cursor::new(include_bytes!("../tests/symlink.nar")); let node = nar::reader::open(&mut f).unwrap(); match node { nar::reader::Node::Symlink { target } => { assert_eq!( &b"/nix/store/somewhereelse"[..], &target, "target must match" ); } _ => panic!("unexpected type"), } } #[test] fn file() { let mut f = std::io::Cursor::new(include_bytes!("../tests/helloworld.nar")); let node = nar::reader::open(&mut f).unwrap(); match node { nar::reader::Node::File { executable, mut reader, } => { assert!(!executable); let mut buf = vec![]; reader.read_to_end(&mut buf).expect("read must succeed"); assert_eq!(&b"Hello World!"[..], &buf); } _ => panic!("unexpected type"), } } #[test] fn complicated() { let mut f = std::io::Cursor::new(include_bytes!("../tests/complicated.nar")); let node = nar::reader::open(&mut f).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() .expect("next must succeed") .expect("must be some"), ); // second entry is aa, a symlink to /nix/store/somewhereelse must_be_symlink( "aa", "/nix/store/somewhereelse", dir_reader .next() .expect("next must be some") .expect("must be some"), ); { // third entry is a directory called "keep" let entry = dir_reader .next() .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() .expect("next must succeed") .expect("must be some"); must_read_file(".keep", entry); } // we must read the None assert!( subdir_reader.next().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().expect("must succeed").is_none()); } _ => panic!("unexpected type"), } } #[test] #[should_panic] fn file_read_abandoned() { let mut f = std::io::Cursor::new(include_bytes!("../tests/complicated.nar")); let node = nar::reader::open(&mut f).unwrap(); match node { nar::reader::Node::Directory(mut dir_reader) => { // first entry is .keep, an empty regular file. { let entry = dir_reader .next() .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().expect("must succeed").is_none()); } _ => panic!("unexpected type"), } } #[test] #[should_panic] fn dir_read_abandoned() { let mut f = std::io::Cursor::new(include_bytes!("../tests/complicated.nar")); let node = nar::reader::open(&mut f).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() .expect("next must succeed") .expect("must be some"), ); // second entry is aa, a symlink to /nix/store/somewhereelse must_be_symlink( "aa", "/nix/store/somewhereelse", dir_reader .next() .expect("next must be some") .expect("must be some"), ); { // third entry is a directory called "keep" let entry = dir_reader .next() .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().expect("must succeed").is_none()); } _ => panic!("unexpected type"), } } #[test] #[should_panic] fn dir_read_after_none() { let mut f = std::io::Cursor::new(include_bytes!("../tests/complicated.nar")); let node = nar::reader::open(&mut f).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() .expect("next must succeed") .expect("must be some"), ); // second entry is aa, a symlink to /nix/store/somewhereelse must_be_symlink( "aa", "/nix/store/somewhereelse", dir_reader .next() .expect("next must be some") .expect("must be some"), ); { // third entry is a directory called "keep" let entry = dir_reader .next() .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() .expect("next must succeed") .expect("must be some"), ); // we must read the None assert!( subdir_reader.next().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().expect("must succeed").is_none()); // this should panic, because we already got a none so we're meant to stop. dir_reader.next().unwrap(); unreachable!() } _ => panic!("unexpected type"), } } 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]).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), } }