333 lines
9.9 KiB
Rust
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(())
|
|
}
|