mirror of
https://github.com/LukeMathWalker/zero-to-production.git
synced 2024-12-20 23:16:45 +00:00
230 lines
7.6 KiB
Rust
230 lines
7.6 KiB
Rust
use crate::helpers::{assert_is_redirect_to, spawn_app, ConfirmationLinks, TestApp};
|
|
use fake::faker::internet::en::SafeEmail;
|
|
use fake::faker::name::en::Name;
|
|
use fake::Fake;
|
|
use std::time::Duration;
|
|
use wiremock::matchers::{any, method, path};
|
|
use wiremock::{Mock, ResponseTemplate};
|
|
|
|
async fn create_unconfirmed_subscriber(app: &TestApp) -> ConfirmationLinks {
|
|
// We are working with multiple subscribers now,
|
|
// their details must be randomised to avoid conflicts!
|
|
let name: String = Name().fake();
|
|
let email: String = SafeEmail().fake();
|
|
let body = serde_urlencoded::to_string(&serde_json::json!({
|
|
"name": name,
|
|
"email": email
|
|
}))
|
|
.unwrap();
|
|
|
|
let _mock_guard = Mock::given(path("/email"))
|
|
.and(method("POST"))
|
|
.respond_with(ResponseTemplate::new(200))
|
|
.named("Create unconfirmed subscriber")
|
|
.expect(1)
|
|
.mount_as_scoped(&app.email_server)
|
|
.await;
|
|
app.post_subscriptions(body.into())
|
|
.await
|
|
.error_for_status()
|
|
.unwrap();
|
|
|
|
let email_request = &app
|
|
.email_server
|
|
.received_requests()
|
|
.await
|
|
.unwrap()
|
|
.pop()
|
|
.unwrap();
|
|
app.get_confirmation_links(email_request)
|
|
}
|
|
|
|
async fn create_confirmed_subscriber(app: &TestApp) {
|
|
let confirmation_link = create_unconfirmed_subscriber(app).await.html;
|
|
reqwest::get(confirmation_link)
|
|
.await
|
|
.unwrap()
|
|
.error_for_status()
|
|
.unwrap();
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn newsletters_are_not_delivered_to_unconfirmed_subscribers() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
create_unconfirmed_subscriber(&app).await;
|
|
app.test_user.login(&app).await;
|
|
|
|
Mock::given(any())
|
|
.respond_with(ResponseTemplate::new(200))
|
|
.expect(0)
|
|
.mount(&app.email_server)
|
|
.await;
|
|
|
|
// Act - Part 1 - Submit newsletter form
|
|
let newsletter_request_body = serde_json::json!({
|
|
"title": "Newsletter title",
|
|
"text_content": "Newsletter body as plain text",
|
|
"html_content": "<p>Newsletter body as HTML</p>",
|
|
"idempotency_key": uuid::Uuid::new_v4().to_string()
|
|
});
|
|
let response = app.post_publish_newsletter(&newsletter_request_body).await;
|
|
assert_is_redirect_to(&response, "/admin/newsletters");
|
|
|
|
// Act - Part 2 - Follow the redirect
|
|
let html_page = app.get_publish_newsletter_html().await;
|
|
assert!(html_page.contains(
|
|
"<p><i>The newsletter issue has been accepted - \
|
|
emails will go out shortly.</i></p>"
|
|
));
|
|
app.dispatch_all_pending_emails().await;
|
|
// Mock verifies on Drop that we haven't sent the newsletter email
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn newsletters_are_delivered_to_confirmed_subscribers() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
create_confirmed_subscriber(&app).await;
|
|
app.test_user.login(&app).await;
|
|
|
|
Mock::given(path("/email"))
|
|
.and(method("POST"))
|
|
.respond_with(ResponseTemplate::new(200))
|
|
.expect(1)
|
|
.mount(&app.email_server)
|
|
.await;
|
|
|
|
// Act - Part 1 - Submit newsletter form
|
|
let newsletter_request_body = serde_json::json!({
|
|
"title": "Newsletter title",
|
|
"text_content": "Newsletter body as plain text",
|
|
"html_content": "<p>Newsletter body as HTML</p>",
|
|
"idempotency_key": uuid::Uuid::new_v4().to_string()
|
|
});
|
|
let response = app.post_publish_newsletter(&newsletter_request_body).await;
|
|
assert_is_redirect_to(&response, "/admin/newsletters");
|
|
|
|
// Act - Part 2 - Follow the redirect
|
|
let html_page = app.get_publish_newsletter_html().await;
|
|
assert!(html_page.contains(
|
|
"<p><i>The newsletter issue has been accepted - \
|
|
emails will go out shortly.</i></p>"
|
|
));
|
|
app.dispatch_all_pending_emails().await;
|
|
// Mock verifies on Drop that we have sent the newsletter email
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn you_must_be_logged_in_to_see_the_newsletter_form() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
|
|
// Act
|
|
let response = app.get_publish_newsletter().await;
|
|
|
|
// Assert
|
|
assert_is_redirect_to(&response, "/login");
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn you_must_be_logged_in_to_publish_a_newsletter() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
|
|
// Act
|
|
let newsletter_request_body = serde_json::json!({
|
|
"title": "Newsletter title",
|
|
"text_content": "Newsletter body as plain text",
|
|
"html_content": "<p>Newsletter body as HTML</p>",
|
|
"idempotency_key": uuid::Uuid::new_v4().to_string()
|
|
});
|
|
let response = app.post_publish_newsletter(&newsletter_request_body).await;
|
|
|
|
// Assert
|
|
assert_is_redirect_to(&response, "/login");
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn newsletter_creation_is_idempotent() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
create_confirmed_subscriber(&app).await;
|
|
app.test_user.login(&app).await;
|
|
|
|
Mock::given(path("/email"))
|
|
.and(method("POST"))
|
|
.respond_with(ResponseTemplate::new(200))
|
|
.expect(1)
|
|
.mount(&app.email_server)
|
|
.await;
|
|
|
|
// Act - Part 1 - Submit newsletter form
|
|
let newsletter_request_body = serde_json::json!({
|
|
"title": "Newsletter title",
|
|
"text_content": "Newsletter body as plain text",
|
|
"html_content": "<p>Newsletter body as HTML</p>",
|
|
// We expect the idempotency key as part of the
|
|
// form data, not as an header
|
|
"idempotency_key": uuid::Uuid::new_v4().to_string()
|
|
});
|
|
let response = app.post_publish_newsletter(&newsletter_request_body).await;
|
|
assert_is_redirect_to(&response, "/admin/newsletters");
|
|
|
|
// Act - Part 2 - Follow the redirect
|
|
let html_page = app.get_publish_newsletter_html().await;
|
|
assert!(html_page.contains(
|
|
"<p><i>The newsletter issue has been accepted - \
|
|
emails will go out shortly.</i></p>"
|
|
));
|
|
|
|
// Act - Part 3 - Submit newsletter form **again**
|
|
let response = app.post_publish_newsletter(&newsletter_request_body).await;
|
|
assert_is_redirect_to(&response, "/admin/newsletters");
|
|
|
|
// Act - Part 4 - Follow the redirect
|
|
let html_page = app.get_publish_newsletter_html().await;
|
|
assert!(html_page.contains(
|
|
"<p><i>The newsletter issue has been accepted - \
|
|
emails will go out shortly.</i></p>"
|
|
));
|
|
|
|
app.dispatch_all_pending_emails().await;
|
|
// Mock verifies on Drop that we have sent the newsletter email **once**
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn concurrent_form_submission_is_handled_gracefully() {
|
|
// Arrange
|
|
let app = spawn_app().await;
|
|
create_confirmed_subscriber(&app).await;
|
|
app.test_user.login(&app).await;
|
|
|
|
Mock::given(path("/email"))
|
|
.and(method("POST"))
|
|
// Setting a long delay to ensure that the second request
|
|
// arrives before the first one completes
|
|
.respond_with(ResponseTemplate::new(200).set_delay(Duration::from_secs(2)))
|
|
.expect(1)
|
|
.mount(&app.email_server)
|
|
.await;
|
|
|
|
// Act - Submit two newsletter forms concurrently
|
|
let newsletter_request_body = serde_json::json!({
|
|
"title": "Newsletter title",
|
|
"text_content": "Newsletter body as plain text",
|
|
"html_content": "<p>Newsletter body as HTML</p>",
|
|
"idempotency_key": uuid::Uuid::new_v4().to_string()
|
|
});
|
|
let response1 = app.post_publish_newsletter(&newsletter_request_body);
|
|
let response2 = app.post_publish_newsletter(&newsletter_request_body);
|
|
let (response1, response2) = tokio::join!(response1, response2);
|
|
|
|
assert_eq!(response1.status(), response2.status());
|
|
assert_eq!(
|
|
response1.text().await.unwrap(),
|
|
response2.text().await.unwrap()
|
|
);
|
|
app.dispatch_all_pending_emails().await;
|
|
// Mock verifies on Drop that we have sent the newsletter email **once**
|
|
}
|