2017-05-26 23:44:53 +00:00
|
|
|
use super::task::Task;
|
2019-06-13 02:41:29 +00:00
|
|
|
|
|
|
|
use parking_lot::Mutex;
|
|
|
|
use rand::{Rng, thread_rng, distributions::Alphanumeric};
|
2017-05-26 23:44:53 +00:00
|
|
|
|
2017-06-06 20:41:04 +00:00
|
|
|
use rocket::local::Client;
|
2017-05-26 23:44:53 +00:00
|
|
|
use rocket::http::{Status, ContentType};
|
|
|
|
|
|
|
|
// We use a lock to synchronize between tests so DB operations don't collide.
|
|
|
|
// For now. In the future, we'll have a nice way to run each test in a DB
|
|
|
|
// transaction so we can regain concurrency.
|
|
|
|
static DB_LOCK: Mutex<()> = Mutex::new(());
|
|
|
|
|
|
|
|
macro_rules! run_test {
|
2017-06-06 20:41:04 +00:00
|
|
|
(|$client:ident, $conn:ident| $block:expr) => ({
|
2017-05-26 23:44:53 +00:00
|
|
|
let _lock = DB_LOCK.lock();
|
2018-12-30 21:52:08 +00:00
|
|
|
|
2019-08-24 17:27:10 +00:00
|
|
|
rocket::async_test(async move {
|
|
|
|
let rocket = super::rocket();
|
|
|
|
let db = super::DbConn::get_one(&rocket);
|
|
|
|
let $client = Client::new(rocket).expect("Rocket client");
|
|
|
|
let $conn = db.expect("failed to get database connection for testing");
|
|
|
|
Task::delete_all(&$conn).expect("failed to delete all tasks for testing");
|
|
|
|
|
|
|
|
$block
|
|
|
|
})
|
2017-05-26 23:44:53 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_insertion_deletion() {
|
2017-06-06 20:41:04 +00:00
|
|
|
run_test!(|client, conn| {
|
2017-05-26 23:44:53 +00:00
|
|
|
// Get the tasks before making changes.
|
2020-02-13 04:11:51 +00:00
|
|
|
let init_tasks = Task::all(&conn).unwrap();
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
// Issue a request to insert a new task.
|
2017-06-06 20:41:04 +00:00
|
|
|
client.post("/todo")
|
2017-05-26 23:44:53 +00:00
|
|
|
.header(ContentType::Form)
|
2017-06-06 20:41:04 +00:00
|
|
|
.body("description=My+first+task")
|
2019-08-24 17:27:10 +00:00
|
|
|
.dispatch().await;
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
// Ensure we have one more task in the database.
|
2020-02-13 04:11:51 +00:00
|
|
|
let new_tasks = Task::all(&conn).unwrap();
|
2017-05-26 23:44:53 +00:00
|
|
|
assert_eq!(new_tasks.len(), init_tasks.len() + 1);
|
|
|
|
|
|
|
|
// Ensure the task is what we expect.
|
|
|
|
assert_eq!(new_tasks[0].description, "My first task");
|
|
|
|
assert_eq!(new_tasks[0].completed, false);
|
|
|
|
|
|
|
|
// Issue a request to delete the task.
|
|
|
|
let id = new_tasks[0].id.unwrap();
|
2019-11-26 16:25:56 +00:00
|
|
|
client.delete(format!("/todo/{}", id)).dispatch().await;
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
// Ensure it's gone.
|
2020-02-13 04:11:51 +00:00
|
|
|
let final_tasks = Task::all(&conn).unwrap();
|
2017-05-26 23:44:53 +00:00
|
|
|
assert_eq!(final_tasks.len(), init_tasks.len());
|
2017-05-27 00:12:57 +00:00
|
|
|
if final_tasks.len() > 0 {
|
|
|
|
assert_ne!(final_tasks[0].description, "My first task");
|
|
|
|
}
|
2017-05-26 23:44:53 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_toggle() {
|
2017-06-06 20:41:04 +00:00
|
|
|
run_test!(|client, conn| {
|
2017-05-26 23:44:53 +00:00
|
|
|
// Issue a request to insert a new task; ensure it's not yet completed.
|
2017-06-06 20:41:04 +00:00
|
|
|
client.post("/todo")
|
2017-05-26 23:44:53 +00:00
|
|
|
.header(ContentType::Form)
|
2017-06-06 20:41:04 +00:00
|
|
|
.body("description=test_for_completion")
|
2019-08-24 17:27:10 +00:00
|
|
|
.dispatch().await;
|
2017-05-26 23:44:53 +00:00
|
|
|
|
2020-02-13 04:11:51 +00:00
|
|
|
let task = Task::all(&conn).unwrap()[0].clone();
|
2017-05-26 23:44:53 +00:00
|
|
|
assert_eq!(task.completed, false);
|
|
|
|
|
|
|
|
// Issue a request to toggle the task; ensure it is completed.
|
2019-11-26 16:25:56 +00:00
|
|
|
client.put(format!("/todo/{}", task.id.unwrap())).dispatch().await;
|
2020-02-13 04:11:51 +00:00
|
|
|
assert_eq!(Task::all(&conn).unwrap()[0].completed, true);
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
// Issue a request to toggle the task; ensure it's not completed again.
|
2019-11-26 16:25:56 +00:00
|
|
|
client.put(format!("/todo/{}", task.id.unwrap())).dispatch().await;
|
2020-02-13 04:11:51 +00:00
|
|
|
assert_eq!(Task::all(&conn).unwrap()[0].completed, false);
|
2017-05-26 23:44:53 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_many_insertions() {
|
|
|
|
const ITER: usize = 100;
|
|
|
|
|
2017-06-06 20:41:04 +00:00
|
|
|
run_test!(|client, conn| {
|
2017-05-26 23:44:53 +00:00
|
|
|
// Get the number of tasks initially.
|
2020-02-13 04:11:51 +00:00
|
|
|
let init_num = Task::all(&conn).unwrap().len();
|
2017-05-26 23:44:53 +00:00
|
|
|
let mut descs = Vec::new();
|
|
|
|
|
|
|
|
for i in 0..ITER {
|
|
|
|
// Issue a request to insert a new task with a random description.
|
2019-08-24 17:27:10 +00:00
|
|
|
let desc: String = thread_rng().sample_iter(&Alphanumeric).take(12).collect();
|
2019-11-26 16:25:56 +00:00
|
|
|
client.post("/todo")
|
2017-05-26 23:44:53 +00:00
|
|
|
.header(ContentType::Form)
|
2019-11-26 16:25:56 +00:00
|
|
|
.body(format!("description={}", desc))
|
|
|
|
.dispatch().await;
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
// Record the description we choose for this iteration.
|
|
|
|
descs.insert(0, desc);
|
|
|
|
|
|
|
|
// Ensure the task was inserted properly and all other tasks remain.
|
2020-02-13 04:11:51 +00:00
|
|
|
let tasks = Task::all(&conn).unwrap();
|
2017-05-26 23:44:53 +00:00
|
|
|
assert_eq!(tasks.len(), init_num + i + 1);
|
|
|
|
|
|
|
|
for j in 0..i {
|
|
|
|
assert_eq!(descs[j], tasks[j].description);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_bad_form_submissions() {
|
2017-06-06 20:41:04 +00:00
|
|
|
run_test!(|client, _conn| {
|
2017-05-26 23:44:53 +00:00
|
|
|
// Submit an empty form. We should get a 422 but no flash error.
|
2017-06-06 20:41:04 +00:00
|
|
|
let res = client.post("/todo")
|
|
|
|
.header(ContentType::Form)
|
2019-08-24 17:27:10 +00:00
|
|
|
.dispatch().await;
|
2017-06-06 20:41:04 +00:00
|
|
|
|
2017-05-26 23:44:53 +00:00
|
|
|
let mut cookies = res.headers().get("Set-Cookie");
|
2017-06-06 20:41:04 +00:00
|
|
|
assert_eq!(res.status(), Status::UnprocessableEntity);
|
2017-05-26 23:44:53 +00:00
|
|
|
assert!(!cookies.any(|value| value.contains("error")));
|
|
|
|
|
2017-06-06 20:41:04 +00:00
|
|
|
// Submit a form with an empty description. We look for 'error' in the
|
|
|
|
// cookies which corresponds to flash message being set as an error.
|
|
|
|
let res = client.post("/todo")
|
2017-05-26 23:44:53 +00:00
|
|
|
.header(ContentType::Form)
|
2017-06-06 20:41:04 +00:00
|
|
|
.body("description=")
|
2019-08-24 17:27:10 +00:00
|
|
|
.dispatch().await;
|
2017-05-26 23:44:53 +00:00
|
|
|
|
|
|
|
let mut cookies = res.headers().get("Set-Cookie");
|
|
|
|
assert!(cookies.any(|value| value.contains("error")));
|
|
|
|
|
|
|
|
// Submit a form without a description. Expect a 422 but no flash error.
|
2017-06-06 20:41:04 +00:00
|
|
|
let res = client.post("/todo")
|
2017-05-26 23:44:53 +00:00
|
|
|
.header(ContentType::Form)
|
2017-06-06 20:41:04 +00:00
|
|
|
.body("evil=smile")
|
2019-08-24 17:27:10 +00:00
|
|
|
.dispatch().await;
|
2017-06-06 20:41:04 +00:00
|
|
|
|
2017-05-26 23:44:53 +00:00
|
|
|
let mut cookies = res.headers().get("Set-Cookie");
|
2017-06-06 20:41:04 +00:00
|
|
|
assert_eq!(res.status(), Status::UnprocessableEntity);
|
2017-05-26 23:44:53 +00:00
|
|
|
assert!(!cookies.any(|value| value.contains("error")));
|
|
|
|
})
|
|
|
|
}
|