From 7f2014aaefc9d0a5813098fa27f0ac3164f4bc1a Mon Sep 17 00:00:00 2001 From: Christian Meusel Date: Mon, 6 May 2024 18:42:33 +0200 Subject: [PATCH 1/5] Check test configuration at runtime and on demand This cleans up the initial attempt of specifying the hardware ports to be used for testing by: * Requiring the configuration environment variables only to be present when actually required by a test case * Evaluating these environment variables at runtime, eliminating the footgun that changing them does not trigger rebuilding the tests * No longer requiring to set dummy configuration variables when not running any hardware tests at all (like in CI). The configuration environment variable prefix changes from SERIALPORT_RS_TEST_ to SERIALPORT_TEST_ to match the crate name and not the repository name. The changes add two new dev dependencies which both look worth it: * The envconfig crate eliminates any need for custom error reporting about which environment variable is missing (which not comes for free when just using std::env::var) * The rstest crate conveniently allows to use test fixtures and supports parametrizing tests which will likely be used by the next test cases to come --- .github/workflows/build.yaml | 4 --- .github/workflows/ci.yaml | 5 ---- Cargo.toml | 2 ++ tests/config.rs | 25 +++++++++++++++++++ tests/test_serialport.rs | 48 ++++++++++++++++++++---------------- 5 files changed, 54 insertions(+), 30 deletions(-) create mode 100644 tests/config.rs diff --git a/.github/workflows/build.yaml b/.github/workflows/build.yaml index 6e9cf674..70ab60e3 100644 --- a/.github/workflows/build.yaml +++ b/.github/workflows/build.yaml @@ -51,10 +51,6 @@ env: PKG_CONFIG_ALLOW_CROSS: 1 # Deny warnings. RUSTFLAGS: -D warnings - # Dummy ports for hardware tests. Keep in sync with other workflows (in - # ci.yaml). - SERIALPORT_RS_TEST_PORT_1: DUMMY_PORT_1 - SERIALPORT_RS_TEST_PORT_2: DUMMY_PORT_2 jobs: build: diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 42ded30a..e488faab 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -7,11 +7,6 @@ on: push: workflow_dispatch: -env: - # Dummy ports for hardware tests. Keep in sync with workflows in build.yaml. - SERIALPORT_RS_TEST_PORT_1: DUMMY_PORT_1 - SERIALPORT_RS_TEST_PORT_2: DUMMY_PORT_2 - jobs: # -------------------------------------------------------------------------- # LINT diff --git a/Cargo.toml b/Cargo.toml index 77a2a8c8..7280727e 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -45,6 +45,8 @@ serde = { version = "1.0", features = ["derive"], optional = true } [dev-dependencies] assert_hex = "0.4.1" clap = { version = "3.1.6", features = ["derive"] } +envconfig = "0.10.0" +rstest = { version = "0.12.0", default-features = false } [features] default = ["libudev"] diff --git a/tests/config.rs b/tests/config.rs new file mode 100644 index 00000000..d1f0e90c --- /dev/null +++ b/tests/config.rs @@ -0,0 +1,25 @@ +// Configuration for integration tests. This crate is about interacting with real serial ports and +// so some tests need actual hardware. + +use envconfig::Envconfig; +use rstest::fixture; + +// Configuration for tests requiring acutual hardware. +// +// For conveniently pulling this configuration into a test case as a parameter, you might want to +// use the test fixture [`hw_config`]. +#[derive(Clone, Debug, Envconfig, Eq, PartialEq)] +pub struct HardwareConfig { + #[envconfig(from = "SERIALPORT_TEST_PORT_1")] + pub port_1: String, + #[envconfig(from = "SERIALPORT_TEST_PORT_2")] + pub port_2: String, +} + +// Test fixture for conveniently pulling the actual hardware configuration into test cases. +// +// See [`fixture`](rstest::fixture) for details. +#[fixture] +pub fn hw_config() -> HardwareConfig { + HardwareConfig::init_from_env().unwrap() +} diff --git a/tests/test_serialport.rs b/tests/test_serialport.rs index b33cfb68..59081d17 100644 --- a/tests/test_serialport.rs +++ b/tests/test_serialport.rs @@ -1,13 +1,14 @@ //! Tests for the `SerialPort` trait. #![cfg(unix)] +mod config; + +use config::{hw_config, HardwareConfig}; +use rstest::rstest; use serialport::*; use std::time::Duration; -const PORT_1: &str = env!("SERIALPORT_RS_TEST_PORT_1"); -const PORT_2: &str = env!("SERIALPORT_RS_TEST_PORT_2"); - -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] fn test_listing_ports() { let ports = serialport::available_ports().expect("No ports found!"); @@ -16,32 +17,34 @@ fn test_listing_ports() { } } -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] -fn test_opening_found_ports() { +fn test_opening_found_ports(hw_config: HardwareConfig) { // There is no guarantee that we even might open the ports returned by `available_ports`. But // the ports we are using for testing shall be among them. let ports = serialport::available_ports().unwrap(); - assert!(ports.iter().any(|info| info.port_name == PORT_1)); - assert!(ports.iter().any(|info| info.port_name == PORT_2)); + assert!(ports.iter().any(|info| info.port_name == hw_config.port_1)); + assert!(ports.iter().any(|info| info.port_name == hw_config.port_2)); } -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] -fn test_opening_port() { - serialport::new(PORT_1, 9600).open().unwrap(); +fn test_opening_port(hw_config: HardwareConfig) { + serialport::new(hw_config.port_1, 9600).open().unwrap(); } -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] -fn test_opening_native_port() { - serialport::new(PORT_1, 9600).open_native().unwrap(); +fn test_opening_native_port(hw_config: HardwareConfig) { + serialport::new(hw_config.port_1, 9600) + .open_native() + .unwrap(); } -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] -fn test_configuring_ports() { - serialport::new(PORT_1, 9600) +fn test_configuring_ports(hw_config: HardwareConfig) { + serialport::new(hw_config.port_1, 9600) .data_bits(DataBits::Five) .flow_control(FlowControl::None) .parity(Parity::None) @@ -51,17 +54,20 @@ fn test_configuring_ports() { .unwrap(); } -#[test] +#[rstest] #[cfg_attr(feature = "ignore-hardware-tests", ignore)] -fn test_duplicating_port_config() { - let port1_config = serialport::new(PORT_1, 9600) +fn test_duplicating_port_config(hw_config: HardwareConfig) { + let port1_config = serialport::new(hw_config.port_1, 9600) .data_bits(DataBits::Five) .flow_control(FlowControl::None) .parity(Parity::None) .stop_bits(StopBits::One) .timeout(Duration::from_millis(1)); - let port2_config = port1_config.clone().path(PORT_2).baud_rate(115_200); + let port2_config = port1_config + .clone() + .path(hw_config.port_2) + .baud_rate(115_200); let _port1 = port1_config.open().unwrap(); let _port1 = port2_config.open().unwrap(); From efac9f7c1266359a83d2f629bd6d44f9c1f69be8 Mon Sep 17 00:00:00 2001 From: Christian Meusel Date: Mon, 6 May 2024 18:52:32 +0200 Subject: [PATCH 2/5] Run tests for SerialPort trait on all platforms The expected behavior is the same for all platforms. Let's run these test cases everywhere then. --- tests/test_serialport.rs | 2 -- 1 file changed, 2 deletions(-) diff --git a/tests/test_serialport.rs b/tests/test_serialport.rs index 59081d17..0f441bfb 100644 --- a/tests/test_serialport.rs +++ b/tests/test_serialport.rs @@ -1,6 +1,4 @@ //! Tests for the `SerialPort` trait. -#![cfg(unix)] - mod config; use config::{hw_config, HardwareConfig}; From 95a78eab427a80bec1895d252d4b915b7e974ee9 Mon Sep 17 00:00:00 2001 From: Christian Meusel Date: Mon, 6 May 2024 21:36:59 +0200 Subject: [PATCH 3/5] Add first integration tests for read timeouts --- tests/test_timeout.rs | 98 +++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 98 insertions(+) create mode 100644 tests/test_timeout.rs diff --git a/tests/test_timeout.rs b/tests/test_timeout.rs new file mode 100644 index 00000000..00db709f --- /dev/null +++ b/tests/test_timeout.rs @@ -0,0 +1,98 @@ +mod config; + +use config::{hw_config, HardwareConfig}; +use rstest::rstest; +use serialport::*; +use std::io::Read; +use std::time::{Duration, Instant}; + +#[rstest] +#[case(b"a")] +#[case(b"0123456789")] +#[case(b"0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~")] +#[cfg_attr(feature = "ignore-hardware-tests", ignore)] +fn test_timeout_zero(hw_config: HardwareConfig, #[case] message: &[u8]) { + let timeout = Duration::ZERO; + let margin = Duration::from_millis(100); + + let mut sender = serialport::new(hw_config.port_1, 115200).open().unwrap(); + let mut receiver = serialport::new(hw_config.port_2, 115200) + .timeout(timeout) + .open() + .unwrap(); + let mut buffer: [u8; 1024] = [0xff; 1024]; + + sender.clear(ClearBuffer::All).unwrap(); + receiver.clear(ClearBuffer::All).unwrap(); + + sender.write_all(message).unwrap(); + sender.flush().unwrap(); + let flushed_at = Instant::now(); + + let expected_until = flushed_at + timeout + margin; + let mut timeouts = 0usize; + + loop { + match receiver.read(&mut buffer) { + Ok(read) => { + assert!(read > 0); + println!( + "read: {} bytes of {} after {} timeouts/{} ms", + read, + message.len(), + timeouts, + (Instant::now() - flushed_at).as_millis() + ); + assert_eq!(message[..read], buffer[..read]); + break; + } + Err(e) => { + assert_eq!(e.kind(), std::io::ErrorKind::TimedOut); + timeouts += 1; + } + } + + assert!(expected_until > Instant::now()); + } +} + +#[rstest] +#[case(Duration::from_millis(10))] +#[case(Duration::from_millis(100))] +#[case(Duration::from_millis(1000))] +#[cfg_attr(feature = "ignore-hardware-tests", ignore)] +fn test_timeout_greater_zero(hw_config: HardwareConfig, #[case] timeout: Duration) { + let margin = Duration::from_millis(100); + + let mut sender = serialport::new(hw_config.port_1, 115200).open().unwrap(); + let mut receiver = serialport::new(hw_config.port_2, 115200) + .timeout(timeout) + .open() + .unwrap(); + + let message = + b"0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~"; + let mut buffer: [u8; 1024] = [0xff; 1024]; + + sender.clear(ClearBuffer::All).unwrap(); + receiver.clear(ClearBuffer::All).unwrap(); + + sender.write_all(message).unwrap(); + sender.flush().unwrap(); + + let flushed_at = Instant::now(); + + let read = receiver.read(&mut buffer).unwrap(); + let read_at = Instant::now(); + + println!( + "read: {} bytes of {} after {} ms", + read, + message.len(), + (Instant::now() - flushed_at).as_millis() + ); + + assert!(read > 0); + assert!(flushed_at + timeout + margin > read_at); + assert_eq!(buffer[..read], message[..read]); +} From cb88008a7ce16a975378a0fdc32faeb2868ae1dc Mon Sep 17 00:00:00 2001 From: Christian Meusel Date: Fri, 10 May 2024 11:40:04 +0200 Subject: [PATCH 4/5] Silence Clippy warning on assigning_clones --- src/lib.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/lib.rs b/src/lib.rs index f01363dc..200374e8 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -331,6 +331,8 @@ pub struct SerialPortBuilder { impl SerialPortBuilder { /// Set the path to the serial port + // TODO: Switch to `clone_into` when bumping our MSRV past 1.63 and remove this exemption. + #[allow(clippy::assigning_clones)] #[must_use] pub fn path<'a>(mut self, path: impl Into>) -> Self { self.path = path.into().as_ref().to_owned(); From 685324e74932d6de8cb1cec59ccfa53517456b46 Mon Sep 17 00:00:00 2001 From: Christian Meusel Date: Sun, 12 May 2024 17:25:37 +0200 Subject: [PATCH 5/5] Build version-dependent examples only when supported This apparently does not reduce noise in CI but at least allows to build the examples with Rust 1.59.0. --- Cargo.toml | 1 + examples/loopback.rs | 10 ++++++++++ 2 files changed, 11 insertions(+) diff --git a/Cargo.toml b/Cargo.toml index 7280727e..a3932436 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -47,6 +47,7 @@ assert_hex = "0.4.1" clap = { version = "3.1.6", features = ["derive"] } envconfig = "0.10.0" rstest = { version = "0.12.0", default-features = false } +rustversion = "1.0.16" [features] default = ["libudev"] diff --git a/examples/loopback.rs b/examples/loopback.rs index 65f5bd35..dc519345 100644 --- a/examples/loopback.rs +++ b/examples/loopback.rs @@ -193,6 +193,16 @@ fn loopback_standard<'a>( } } +#[rustversion::before(1.63)] +fn loopback_split<'a>( + _port: &mut Box, + _read_stats: &mut Stats<'a>, + _write_stats: &mut Stats<'a>, +) { + unimplemented!("requires Rust 1.63 or later"); +} + +#[rustversion::since(1.63)] fn loopback_split<'a>( port: &mut Box, read_stats: &mut Stats<'a>,