nidus-testing 1.0.1

Test application harness, request builders, response helpers, and provider overrides for Nidus.
Documentation
use axum::body::Bytes;
use http::{HeaderMap, HeaderValue, StatusCode, header::ToStrError};
use serde::de::DeserializeOwned;
use serde_json::Value;
use std::str;

/// Captured in-memory HTTP response.
///
/// `TestResponse` owns the status, headers, and collected body returned by
/// [`crate::TestRequest::send`]. Assertion helpers panic with ordinary test
/// assertion failures, while `try_json`, `text`, and `header_str` expose
/// fallible parsing.
///
/// ```
/// # use axum::{Json, Router, routing::get};
/// use http::StatusCode;
/// # use nidus_testing::TestApp;
/// use serde_json::json;
/// # #[tokio::main]
/// # async fn main() -> Result<(), http::header::ToStrError> {
/// # let app = TestApp::from_router(
/// #     Router::new().route("/health", get(|| async { Json(json!({ "ok": true })) })),
/// # );
///
/// let response = app.get("/health").send().await;
/// response.assert_status(StatusCode::OK);
/// assert_eq!(response.header_str("content-type")?.unwrap(), "application/json");
/// response.assert_json(json!({ "ok": true }));
/// # Ok(())
/// # }
/// ```
pub struct TestResponse {
    status: StatusCode,
    headers: HeaderMap,
    body: Bytes,
}

impl TestResponse {
    pub(crate) fn new(status: StatusCode, headers: HeaderMap, body: Bytes) -> Self {
        Self {
            status,
            headers,
            body,
        }
    }

    /// Returns the response status code.
    pub fn status(&self) -> StatusCode {
        self.status
    }

    /// Returns the raw response body bytes.
    pub fn body(&self) -> &[u8] {
        &self.body
    }

    /// Returns the response headers.
    pub fn headers(&self) -> &HeaderMap {
        &self.headers
    }

    /// Returns a response header by name.
    pub fn header(&self, name: impl AsRef<str>) -> Option<&HeaderValue> {
        self.headers.get(name.as_ref())
    }

    /// Returns a response header as a UTF-8 string when present.
    pub fn header_str(&self, name: impl AsRef<str>) -> Result<Option<&str>, ToStrError> {
        self.header(name).map(HeaderValue::to_str).transpose()
    }

    /// Asserts the response status code.
    pub fn assert_status(&self, expected: StatusCode) {
        assert_eq!(self.status, expected);
    }

    /// Asserts a response header as a UTF-8 string.
    pub fn assert_header(&self, name: impl AsRef<str>, expected: &str) {
        let name = name.as_ref();
        let actual = self
            .header(name)
            .unwrap_or_else(|| panic!("missing response header `{name}`"));
        assert_eq!(
            actual
                .to_str()
                .unwrap_or_else(|_| panic!("response header `{name}` was not valid UTF-8")),
            expected
        );
    }

    /// Decodes the response body as JSON.
    ///
    /// Panics when the body is not valid JSON for `T`. Use [`Self::try_json`]
    /// when asserting malformed responses.
    pub fn json<T>(&self) -> T
    where
        T: DeserializeOwned,
    {
        self.try_json().expect("test response was not valid JSON")
    }

    /// Tries to decode the response body as JSON.
    pub fn try_json<T>(&self) -> serde_json::Result<T>
    where
        T: DeserializeOwned,
    {
        serde_json::from_slice(&self.body)
    }

    /// Returns the response body as UTF-8 text.
    pub fn text(&self) -> std::result::Result<&str, str::Utf8Error> {
        str::from_utf8(&self.body)
    }

    /// Asserts the response body as UTF-8 text.
    ///
    /// This consumes the response so tests cannot accidentally assert against a
    /// body after moving it into another helper.
    pub fn assert_text(self, expected: &str) {
        let text = self.text().expect("test response was not UTF-8");
        assert_eq!(text, expected);
    }

    /// Asserts the response body as JSON.
    ///
    /// This consumes the response and compares against a `serde_json::Value`.
    pub fn assert_json(self, expected: Value) {
        let actual: Value = self.json();
        assert_eq!(actual, expected);
    }
}