Skip to content

Commit

Permalink
Refactor tests, update nix to 0.29, bump MSRV to 1.69
Browse files Browse the repository at this point in the history
  • Loading branch information
niklasmohrin committed Oct 24, 2024
1 parent e98597a commit 8f4f0dd
Show file tree
Hide file tree
Showing 3 changed files with 66 additions and 75 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/ci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ jobs:
fail-fast: false
matrix:
os: [ubuntu-latest, windows-latest, macos-latest]
rust: ["1.63.0", stable]
rust: ["1.69.0", stable]
runs-on: ${{ matrix.os }}
steps:
- uses: actions/checkout@v4
Expand All @@ -22,7 +22,7 @@ jobs:
toolchain: ${{ matrix.rust }}
components: clippy
- run: cargo build
- run: cargo test
- run: cargo test -F test-close-again -- --test-threads 1
- run: cargo clippy

build_extra:
Expand Down
7 changes: 4 additions & 3 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ name = "clircle"
version = "0.5.0"
authors = ["Niklas Mohrin <[email protected]>"]
license = "MIT OR Apache-2.0"
rust-version = "1.63"
rust-version = "1.69"
edition = "2021"
description = "Detect IO circles in your CLI apps arguments."
homepage = "https://github.com/niklasmohrin/clircle"
Expand All @@ -16,6 +16,7 @@ keywords = ["cycle", "arguments", "argv", "io"]
[features]
default = ["serde"]
serde = ["dep:serde", "dep:serde_derive"]
test-close-again = []

[dependencies]
serde = { version = "1.0.117", optional = true }
Expand All @@ -28,7 +29,7 @@ winapi = { version = "0.3.9", features = ["winnt", "winbase", "processenv", "han
[dev-dependencies]
tempfile = "3.1.0"

[target.'cfg(not(windows))'.dev-dependencies.nix]
version = "0.24.1"
[target.'cfg(unix)'.dev-dependencies.nix]
version = "0.29"
default-features = false
features = ["term"]
130 changes: 60 additions & 70 deletions src/clircle_unix.rs
Original file line number Diff line number Diff line change
Expand Up @@ -131,88 +131,78 @@ impl hash::Hash for UnixIdentifier {
mod tests {
use super::*;

use std::error::Error;
use std::io::Write;
use std::os::fd::OwnedFd;

use nix::pty::{openpty, OpenptyResult};
use nix::unistd::close;

#[test]
fn test_fd_closing() -> Result<(), Box<dyn Error>> {
let dir = tempfile::tempdir().expect("Couldn't create tempdir.");
let dir_path = dir.path().to_path_buf();

// 1) Check that the file returned by into_inner is still valid
let file = File::create(dir_path.join("myfile"))?;
let ident = UnixIdentifier::try_from(file)?;
fn test_into_inner() {
let file = tempfile::tempfile().expect("failed to create tempfile");
file.metadata().expect("can stat file");
let ident = UnixIdentifier::try_from(file).expect("failed to create identifier");
let mut file = ident
.into_inner()
.ok_or("Did not get file back from identifier")?;
// Check if file can be written to without weird errors
file.write_all(b"Some test content")?;

// 2) Check that dropping the Identifier does not close the file, if owns_fd is false
let fd = file.into_raw_fd();
let ident = unsafe { UnixIdentifier::try_from_raw_fd(fd, false) };
if let Err(e) = ident {
let _ = dbg!(close(fd));
return Err(Box::new(e));
}
let ident = ident.unwrap();
drop(ident);
close(fd).map_err(|e| {
format!(
"Error closing file, that I told UnixIdentifier not to close: {}",
e
)
})?;

// 3) Check that the file is closed on drop, if owns_fd is true
let fd = File::open(dir_path.join("myfile"))?.into_raw_fd();
let ident = unsafe { UnixIdentifier::try_from_raw_fd(fd, true) };
if let Err(e) = ident {
let _ = dbg!(close(fd));
return Err(Box::new(e));
}
let ident = ident.unwrap();
.expect("failed to convert identifier to file");
file.write_all(b"some test content")
.expect("failed to write test content to file");
}

#[test]
fn test_borrowed_fd() {
let file = tempfile::tempfile().expect("failed to create tempfile");
let fd: OwnedFd = file.into();
let ident = unsafe { UnixIdentifier::try_from_raw_fd(fd.as_raw_fd(), false) }
.expect("failed to create identifier");
drop(ident);
close(fd).expect_err("This file descriptor should have been closed already!");
let fd = fd.into_raw_fd();
close(fd).expect("error closing fd");
#[cfg(feature = "test-close-again")]
close(fd).expect_err("closing again should fail");
}

Ok(())
#[test]
fn test_owned_fd() {
let file = tempfile::tempfile().expect("failed to create tempfile");
let fd: OwnedFd = file.into();
let ident = unsafe { UnixIdentifier::try_from_raw_fd(fd.as_raw_fd(), true) }
.expect("failed to create identifier");
drop(ident);
#[cfg(feature = "test-close-again")]
close(fd.into_raw_fd())
.expect_err("the fd should have already been closed by dropping the identifier");
}

#[test]
fn test_pty_equal_but_not_conflicting() -> Result<(), &'static str> {
let OpenptyResult { master, slave } = openpty(None, None).expect("Could not open pty.");
let res = unsafe { UnixIdentifier::try_from_raw_fd(slave, false) }
.map_err(|_| "Error creating UnixIdentifier from pty fd")
.and_then(|ident| {
if !ident.eq(&ident) {
return Err("ident != ident");
}
if ident.surely_conflicts_with(&ident) {
return Err("pty fd does not conflict with itself, but conflict detected");
}

let second_ident = unsafe { UnixIdentifier::try_from_raw_fd(slave, false) }
.map_err(|_| "Error creating second Identifier to pty")?;
if !ident.eq(&second_ident) {
return Err("ident != second_ident");
}
if ident.surely_conflicts_with(&second_ident) {
return Err(
"Two Identifiers to the same pty should not conflict, but they do.",
);
}
Ok(())
});

let r1 = close(master);
let r2 = close(slave);

r1.expect("Error closing master end of pty");
r2.expect("Error closing slave end of pty");

res
fn test_pty_equal_but_not_conflicting() {
let OpenptyResult {
master: parent,
slave: child,
} = openpty(None, None).expect("failed to open pty");

let parent_ident = unsafe { UnixIdentifier::try_from_raw_fd(parent.as_raw_fd(), false) }
.expect("failed to create parent identifier");

assert_eq!(parent_ident, parent_ident);

assert!(!parent_ident.surely_conflicts_with(&parent_ident));

let child_ident = unsafe { UnixIdentifier::try_from_raw_fd(child.as_raw_fd(), false) }
.expect("failed to create child identifier");

assert_ne!(parent_ident, child_ident);
assert!(!parent_ident.surely_conflicts_with(&child_ident));

drop(child_ident);
drop(parent_ident);
let child = child.into_raw_fd();
close(child).expect("failed to close child");
#[cfg(feature = "test-close-again")]
close(child).expect_err("closing child again should fail");
let parent = parent.into_raw_fd();
close(parent).expect("failed to close parent");
#[cfg(feature = "test-close-again")]
close(parent).expect_err("closing parent again should fail");
}
}

0 comments on commit 8f4f0dd

Please sign in to comment.