4
0

utils.rs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. //! Utilities to run tests
  2. use std::{
  3. ffi::CString,
  4. io, process,
  5. sync::atomic::{AtomicU64, Ordering},
  6. };
  7. use aya::netlink_set_link_up;
  8. use libc::if_nametoindex;
  9. use netns_rs::{get_from_current_thread, NetNs};
  10. pub struct NetNsGuard {
  11. name: String,
  12. old_ns: NetNs,
  13. ns: Option<NetNs>,
  14. }
  15. impl NetNsGuard {
  16. pub fn new() -> Self {
  17. let old_ns = get_from_current_thread().expect("Failed to get current netns");
  18. static COUNTER: AtomicU64 = AtomicU64::new(0);
  19. let pid = process::id();
  20. let name = format!("aya-test-{pid}-{}", COUNTER.fetch_add(1, Ordering::Relaxed));
  21. // Create and enter netns
  22. let ns = NetNs::new(&name).unwrap_or_else(|e| panic!("Failed to create netns {name}: {e}"));
  23. let netns = Self {
  24. old_ns,
  25. ns: Some(ns),
  26. name,
  27. };
  28. let ns = netns.ns.as_ref().unwrap();
  29. ns.enter()
  30. .unwrap_or_else(|e| panic!("Failed to enter network namespace {}: {e}", netns.name));
  31. println!("Entered network namespace {}", netns.name);
  32. // By default, the loopback in a new netns is down. Set it up.
  33. let lo = CString::new("lo").unwrap();
  34. unsafe {
  35. let idx = if_nametoindex(lo.as_ptr());
  36. if idx == 0 {
  37. panic!(
  38. "Interface `lo` not found in netns {}: {}",
  39. netns.name,
  40. io::Error::last_os_error()
  41. );
  42. }
  43. netlink_set_link_up(idx as i32)
  44. .unwrap_or_else(|e| panic!("Failed to set `lo` up in netns {}: {e}", netns.name));
  45. }
  46. netns
  47. }
  48. }
  49. impl Drop for NetNsGuard {
  50. fn drop(&mut self) {
  51. // Avoid panic in panic
  52. if let Err(e) = self.old_ns.enter() {
  53. eprintln!("Failed to return to original netns: {e}");
  54. }
  55. if let Some(ns) = self.ns.take() {
  56. if let Err(e) = ns.remove() {
  57. eprintln!("Failed to remove netns {}: {e}", self.name);
  58. }
  59. }
  60. println!("Exited network namespace {}", self.name);
  61. }
  62. }
  63. /// Performs `assert!` macro. If the assertion fails and host kernel version
  64. /// is above feature version, then fail test.
  65. macro_rules! kernel_assert {
  66. ($cond:expr, $version:expr $(,)?) => {
  67. let pass: bool = $cond;
  68. if !pass {
  69. let feat_version: aya::util::KernelVersion = $version;
  70. let current = aya::util::KernelVersion::current().unwrap();
  71. let cond_literal = stringify!($cond);
  72. if current >= feat_version {
  73. // Host kernel is expected to have the feat but does not
  74. panic!(
  75. r#" assertion `{cond_literal}` failed: expected host kernel v{current} to have v{feat_version} feature"#,
  76. );
  77. } else {
  78. // Continue with tests since host is not expected to have feat
  79. eprintln!(
  80. r#"ignoring assertion at {}:{}
  81. assertion `{cond_literal}` failed: continuing since host kernel v{current} is not expected to have v{feat_version} feature"#,
  82. file!(), line!(),
  83. );
  84. }
  85. }
  86. };
  87. }
  88. pub(crate) use kernel_assert;
  89. /// Performs `assert_eq!` macro. If the assertion fails and host kernel version
  90. /// is above feature version, then fail test.
  91. macro_rules! kernel_assert_eq {
  92. ($left:expr, $right:expr, $version:expr $(,)?) => {
  93. if $left != $right {
  94. let feat_version: aya::util::KernelVersion = $version;
  95. let current = aya::util::KernelVersion::current().unwrap();
  96. if current >= feat_version {
  97. // Host kernel is expected to have the feat but does not
  98. panic!(
  99. r#" assertion `left == right` failed: expected host kernel v{current} to have v{feat_version} feature
  100. left: {:?}
  101. right: {:?}"#,
  102. $left, $right,
  103. );
  104. } else {
  105. // Continue with tests since host is not expected to have feat
  106. eprintln!(
  107. r#"ignoring assertion at {}:{}
  108. assertion `left == right` failed: continuing since host kernel v{current} is not expected to have v{feat_version} feature
  109. left: {:?}
  110. right: {:?}"#,
  111. file!(), line!(), $left, $right,
  112. );
  113. }
  114. }
  115. };
  116. }
  117. pub(crate) use kernel_assert_eq;