recipes/backend/tests/http.rs

333 lines
9.9 KiB
Rust

use std::{error::Error, sync::Arc};
use axum::http;
use axum_test::TestServer;
use common::ron_api;
use cookie::Cookie;
use scraper::{ElementRef, Html, Selector};
use serde::Serialize;
use sscanf::sscanf;
use recipes::{app, email};
mod utils;
#[tokio::test]
async fn homepage() -> Result<(), Box<dyn Error>> {
// Arrange.
let state = utils::common_state().await?;
let user_id =
utils::create_user(&state.db_connection, "president@spaceball.planet", "12345").await?;
let _recipe_id = utils::create_recipe(&state.db_connection, user_id, "spaghetti").await?;
let server = TestServer::new(app::make_service(state))?;
// Act.
let response = server.get("/").await;
// Assert.
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let first_recipe_title =
Selector::parse("#recipes-list .recipes-list-public .recipe-item").unwrap();
let elements: Vec<ElementRef> = document.select(&first_recipe_title).collect();
assert_eq!(elements.len(), 1);
assert_eq!(elements[0].inner_html(), "spaghetti");
Ok(())
}
#[tokio::test]
async fn recipe_view() -> Result<(), Box<dyn Error>> {
// Arrange.
let state = utils::common_state().await?;
let user_id =
utils::create_user(&state.db_connection, "president@spaceball.planet", "12345").await?;
let recipe_id = utils::create_recipe(&state.db_connection, user_id, "spaghetti").await?;
let server = TestServer::new(app::make_service(state))?;
// Act.
let response = server.get(&format!("/recipe/view/{}", recipe_id)).await;
// Assert.
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let recipe_title = Selector::parse("#recipe-view .recipe-title").unwrap();
let elements: Vec<ElementRef> = document.select(&recipe_title).collect();
assert_eq!(elements.len(), 1);
assert_eq!(elements[0].inner_html(), "spaghetti");
Ok(())
}
#[tokio::test]
async fn user_edit() -> Result<(), Box<dyn Error>> {
// Arrange.
let state = utils::common_state().await?;
let _user_id =
utils::create_user(&state.db_connection, "president@spaceball.planet", "12345").await?;
let token = utils::sign_in(&state.db_connection, "president@spaceball.planet", "12345").await?;
let server = TestServer::new(app::make_service(state))?;
// Act.
let response = server
.get("/user/edit")
.add_cookie(Cookie::new("auth_token", token))
.await;
// Assert.
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let user_email = Selector::parse("#input-email").unwrap();
let elements: Vec<ElementRef> = document.select(&user_email).collect();
assert_eq!(elements.len(), 1);
assert_eq!(
elements[0].attr("value").unwrap(),
"president@spaceball.planet"
);
Ok(())
}
#[derive(Serialize, Debug)]
pub struct SignUpFormData {
email: String,
password_1: String,
password_2: String,
}
#[tokio::test]
async fn sign_up() -> Result<(), Box<dyn Error>> {
use std::{cell::RefCell, rc::Rc};
// Arrange.
let validation_url: Rc<RefCell<String>> = Rc::new(RefCell::new(String::new()));
let validation_url_clone = validation_url.clone();
let mut mock_email_service = utils::mock_email::MockEmailService::new();
mock_email_service
.expect_send_email()
.withf(|_email_sender, email_receiver, _title, _message| {
email_receiver == "president@spaceball.planet"
})
.times(1)
.returning_st(move |_email_sender, _email_receiver, _title, message| {
let url = sscanf!(
message,
"Follow this link to confirm your inscription, http://127.0.0.1:8000{String}"
)
.unwrap();
*validation_url_clone.borrow_mut() = url;
Ok(())
});
let state = utils::common_state_with_email_service(Arc::new(mock_email_service)).await?;
let server = TestServer::new(app::make_service(state))?;
let tr = recipes::translation::Tr::new("en");
// Sign up page.
{
// Act.
let response = server
.post("/signup")
.form(&SignUpFormData {
email: "president@spaceball.planet".into(),
password_1: "12345678".into(),
password_2: "12345678".into(),
})
.await;
// Assert.
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let message_selector = Selector::parse("#message").unwrap();
let element = document.select(&message_selector).next().unwrap();
assert_eq!(
element.inner_html(),
tr.t(common::translation::Sentence::SignUpEmailSent)
);
}
// Validation page.
{
// Act.
let response = server.get(&validation_url.borrow());
let response = response.await;
// Assert.
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let message_selector = Selector::parse("#message").unwrap();
let element = document.select(&message_selector).next().unwrap();
assert_eq!(
element.inner_html(),
tr.t(common::translation::Sentence::SignUpEmailValidationSuccess)
);
}
Ok(())
}
#[derive(Serialize, Debug)]
pub struct SignInFormData {
email: String,
password: String,
}
#[tokio::test]
async fn sign_in() -> Result<(), Box<dyn Error>> {
// Arrange.
let state = utils::common_state().await?;
let _user_id = utils::create_user(
&state.db_connection,
"president@spaceball.planet",
"12345678",
)
.await?;
let server = TestServer::new(app::make_service(state))?;
// Act.
let response = server
.post("/signin")
.form(&SignInFormData {
email: "president@spaceball.planet".into(),
password: "12345678".into(),
})
.await;
// Assert.
response.assert_status_see_other(); // Redirection after successful sign in.
response.assert_text("");
response.assert_header("location", "/?user_message=16&user_message_icon=0");
Ok(())
}
#[tokio::test]
async fn create_recipe_and_edit_it() -> Result<(), Box<dyn Error>> {
// Arrange.
let state = utils::common_state().await?;
let _user_id = utils::create_user(
&state.db_connection,
"president@spaceball.planet",
"12345678",
)
.await?;
let token = utils::sign_in(
&state.db_connection,
"president@spaceball.planet",
"12345678",
)
.await?;
let server = TestServer::new(app::make_service(state))?;
// Act.
let cookie = Cookie::new("auth_token", token);
let response = server.get("/recipe/new").add_cookie(cookie.clone()).await;
response.assert_status_see_other();
let location = response.header("location").to_str().unwrap().to_string();
let recipe_id = sscanf!(&location, "/en/recipe/edit/{i64}").unwrap();
let response = server.get(&location).add_cookie(cookie.clone()).await;
response.assert_status_ok();
let response = server
.patch("/ron-api/recipe/title")
.add_cookie(cookie.clone())
.add_header(http::header::CONTENT_TYPE, common::consts::MIME_TYPE_RON)
.bytes(
ron_api::to_string(ron_api::SetRecipeTitle {
recipe_id,
title: "AAA".into(),
})
.unwrap()
.into(),
)
.await;
response.assert_status_ok();
let response = server
.patch("/ron-api/recipe/description")
.add_cookie(cookie.clone())
.add_header(http::header::CONTENT_TYPE, common::consts::MIME_TYPE_RON)
.bytes(
ron_api::to_string(ron_api::SetRecipeDescription {
recipe_id,
description: "BBB".into(),
})
.unwrap()
.into(),
)
.await;
response.assert_status_ok();
let response = server
.patch("/ron-api/recipe/servings")
.add_cookie(cookie.clone())
.add_header(http::header::CONTENT_TYPE, common::consts::MIME_TYPE_RON)
.bytes(
ron_api::to_string(ron_api::SetRecipeServings {
recipe_id,
servings: Some(42),
})
.unwrap()
.into(),
)
.await;
response.assert_status_ok();
// TODO: Set other recipe fields.
// Assert.
let response = server
.get(&format!("/recipe/edit/{}", recipe_id))
.add_cookie(cookie.clone())
.await;
response.assert_status_ok();
let document = Html::parse_document(&response.text());
if !document.errors.is_empty() {
panic!("{:?}", document.errors);
}
let title_selector = Selector::parse("#input-title").unwrap();
let element_title = document.select(&title_selector).next().unwrap();
assert_eq!(element_title.attr("value").unwrap(), "AAA");
let description_selector = Selector::parse("#text-area-description").unwrap();
let element_description = document.select(&description_selector).next().unwrap();
assert_eq!(element_description.inner_html(), "BBB");
let servings_selector = Selector::parse("#input-servings").unwrap();
let element_servings = document.select(&servings_selector).next().unwrap();
assert_eq!(element_servings.attr("value").unwrap(), "42");
// dbg!(response);
Ok(())
}