Skip to content

Commit

Permalink
journald: add integration tests for journald subscriber (#1709)
Browse files Browse the repository at this point in the history
Per discussion with @hawkw in #1698 I'm adding a few simple integration
tests for the journald subscriber, to have some safety net when
implementing the actual issue in #1698.

These tests send messages of various complexity to the journal, and then
use `journalctl`'s JSON output to get them back out, to check whether
the message arrives in the systemd journal as it was intended to.

## Motivation

Increase test coverage for the journald subscriber and codify a known
good state before approaching a fix for #1698.
  • Loading branch information
swsnr authored and hawkw committed Dec 20, 2021
1 parent da8dfde commit 772d5c2
Show file tree
Hide file tree
Showing 2 changed files with 177 additions and 0 deletions.
5 changes: 5 additions & 0 deletions tracing-journald/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,8 @@ rust-version = "1.42.0"
[dependencies]
tracing-core = { path = "../tracing-core", version = "0.1.10" }
tracing-subscriber = { path = "../tracing-subscriber", version = "0.3" }

[dev-dependencies]
serde_json = "1.0.68"
serde = { version = "1.0.130", features = ["derive"] }
tracing = { path = "../tracing", version = "0.1" }
172 changes: 172 additions & 0 deletions tracing-journald/tests/journal.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
#![cfg(target_os = "linux")]

use std::collections::HashMap;
use std::process::Command;
use std::time::Duration;

use serde::Deserialize;
use tracing::{debug, error, info, warn};
use tracing_journald::Layer;
use tracing_subscriber::layer::SubscriberExt;
use tracing_subscriber::Registry;

fn journalctl_version() -> std::io::Result<String> {
let output = Command::new("journalctl").arg("--version").output()?;
Ok(String::from_utf8_lossy(&output.stdout).to_string())
}

fn with_journald(f: impl FnOnce()) {
match journalctl_version() {
Ok(_) => {
let sub = Registry::default().with(Layer::new().unwrap().with_field_prefix(None));
tracing::subscriber::with_default(sub, f);
}
Err(error) => eprintln!(
"SKIPPING TEST: journalctl --version failed with error: {}",
error
),
}
}

#[derive(Debug, PartialEq, Deserialize)]
#[serde(untagged)]
enum Field {
Text(String),
Binary(Vec<u8>),
}

// Convenience impls to compare fields against strings and bytes with assert_eq!
impl PartialEq<&str> for Field {
fn eq(&self, other: &&str) -> bool {
match self {
Field::Text(s) => s == other,
Field::Binary(_) => false,
}
}
}

impl PartialEq<[u8]> for Field {
fn eq(&self, other: &[u8]) -> bool {
match self {
Field::Text(s) => s.as_bytes() == other,
Field::Binary(data) => data == other,
}
}
}

/// Retry `f` 10 times 100ms apart.
///
/// When `f` returns an error wait 100ms and try it again, up to ten times.
/// If the last attempt failed return the error returned by that attempt.
///
/// If `f` returns Ok immediately return the result.
fn retry<T, E>(f: impl Fn() -> Result<T, E>) -> Result<T, E> {
let attempts = 10;
let interval = Duration::from_millis(100);
for attempt in (0..attempts).rev() {
match f() {
Ok(result) => return Ok(result),
Err(e) if attempt == 0 => return Err(e),
Err(_) => std::thread::sleep(interval),
}
}
unreachable!()
}

/// Read from journal with `journalctl`.
///
/// `test_name` is a string to match in the `TEST_NAME` field
/// of the `journalctl` call, to make sure to only select journal entries
/// originating from and relevant to the current test.
///
/// Additionally filter by the `_PID` field with the PID of this
/// test process, to make sure this method only reads journal entries
/// created by this test process.
fn read_from_journal(test_name: &str) -> Vec<HashMap<String, Field>> {
let stdout = String::from_utf8(
Command::new("journalctl")
.args(&["--user", "--output=json"])
// Filter by the PID of the current test process
.arg(format!("_PID={}", std::process::id()))
// tracing-journald logs strings in their debug representation
.arg(format!("TEST_NAME={:?}", test_name))
.output()
.unwrap()
.stdout,
)
.unwrap();

stdout
.lines()
.map(|l| {
dbg!(l);
serde_json::from_str(l).unwrap()
})
.collect()
}

/// Read exactly one line from journal for the given test name.
///
/// Try to read lines for `testname` from journal, and `retry()` if the wasn't
/// _exactly_ one matching line.
fn retry_read_one_line_from_journal(testname: &str) -> HashMap<String, Field> {
retry(|| {
let mut messages = read_from_journal(testname);
if messages.len() == 1 {
Ok(messages.pop().unwrap())
} else {
Err(format!(
"one messages expected, got {} messages",
messages.len()
))
}
})
.unwrap()
}

#[test]
fn simple_message() {
with_journald(|| {
info!(test.name = "simple_message", "Hello World");

let message = retry_read_one_line_from_journal("simple_message");
assert_eq!(message["MESSAGE"], "Hello World");
assert_eq!(message["PRIORITY"], "5");
});
}

#[test]
fn multiline_message() {
with_journald(|| {
warn!(test.name = "multiline_message", "Hello\nMultiline\nWorld");

let message = retry_read_one_line_from_journal("multiline_message");
assert_eq!(message["MESSAGE"], "Hello\nMultiline\nWorld");
assert_eq!(message["PRIORITY"], "4");
});
}

#[test]
fn multiline_message_trailing_newline() {
with_journald(|| {
error!(
test.name = "multiline_message_trailing_newline",
"A trailing newline\n"
);

let message = retry_read_one_line_from_journal("multiline_message_trailing_newline");
assert_eq!(message["MESSAGE"], "A trailing newline\n");
assert_eq!(message["PRIORITY"], "3");
});
}

#[test]
fn internal_null_byte() {
with_journald(|| {
debug!(test.name = "internal_null_byte", "An internal\x00byte");

let message = retry_read_one_line_from_journal("internal_null_byte");
assert_eq!(message["MESSAGE"], b"An internal\x00byte"[..]);
assert_eq!(message["PRIORITY"], "6");
});
}

0 comments on commit 772d5c2

Please sign in to comment.