#[cfg(windows)]
use processkit::Command;
use processkit::{Error, Mechanism, ProcessGroup, ProcessGroupOptions};
#[tokio::test]
#[ignore = "creates an OS job/cgroup with a resource limit"]
async fn limits_are_enforced_or_rejected_per_platform() {
let res =
ProcessGroup::with_options(ProcessGroupOptions::default().memory_max(64 * 1024 * 1024));
if cfg!(windows) {
let group = res.expect("Windows Job Objects enforce a memory cap");
assert!(matches!(group.mechanism(), Mechanism::JobObject));
} else if cfg!(target_os = "linux") {
match res {
Ok(group) => assert!(matches!(group.mechanism(), Mechanism::CgroupV2)),
Err(Error::ResourceLimit(_)) => {
eprintln!("skipping cgroup enforcement: controller delegation unavailable");
}
Err(other) => panic!("unexpected error: {other:?}"),
}
} else {
assert!(
matches!(res, Err(Error::ResourceLimit(_))),
"a limit on a container-less target must be rejected, not silently dropped"
);
}
}
#[cfg(windows)]
#[tokio::test]
#[ignore = "spawns real subprocesses to prove the active-process cap is enforced"]
async fn windows_process_count_limit_is_enforced() {
let one_proc_sleeper = || Command::new("ping").args(["-n", "30", "127.0.0.1"]);
let group = ProcessGroup::with_options(ProcessGroupOptions::default().max_processes(1))
.expect("create capped group");
assert!(matches!(group.mechanism(), Mechanism::JobObject));
let _first = group
.start(&one_proc_sleeper())
.await
.expect("first child fits the cap");
let second = group.start(&one_proc_sleeper()).await;
assert!(
second.is_err(),
"a second process must not be admitted past max_processes(1)"
);
}
#[cfg(windows)]
#[tokio::test]
#[ignore = "creates a capped Job Object and runs a small child within it"]
async fn windows_memory_and_cpu_limits_accept_and_run() {
let group = ProcessGroup::with_options(
ProcessGroupOptions::default()
.memory_max(512 * 1024 * 1024)
.cpu_quota(0.5),
)
.expect("create capped group");
assert!(matches!(group.mechanism(), Mechanism::JobObject));
let out = group
.start(&Command::new("cmd").args(["/c", "echo hi"]))
.await
.expect("spawn small child")
.output_string()
.await
.expect("collect");
assert!(out.is_success(), "exit {:?}", out.code());
assert!(out.stdout().contains("hi"), "stdout: {:?}", out.stdout());
}