use async closures for tests
This commit is contained in:
parent
6aa72b3c87
commit
2dbb085008
2 changed files with 50 additions and 42 deletions
|
@ -1,41 +1,10 @@
|
||||||
|
#![feature(async_closure)]
|
||||||
extern crate mozaic4_backend;
|
extern crate mozaic4_backend;
|
||||||
|
|
||||||
use diesel;
|
|
||||||
use diesel::prelude::*;
|
|
||||||
use mozaic4_backend::DbConn;
|
|
||||||
use rocket::http::{ContentType, Header, Status};
|
use rocket::http::{ContentType, Header, Status};
|
||||||
use rocket::local::asynchronous::Client;
|
|
||||||
|
|
||||||
// We use a lock to synchronize between tests so DB operations don't collide.
|
mod util;
|
||||||
// For now. In the future, we'll have a nice way to run each test in a DB
|
use util::run_test;
|
||||||
// transaction so we can regain concurrency.
|
|
||||||
static DB_LOCK: parking_lot::Mutex<()> = parking_lot::const_mutex(());
|
|
||||||
|
|
||||||
async fn reset_db(db: &DbConn) {
|
|
||||||
db.run(|conn| {
|
|
||||||
diesel::sql_query("TRUNCATE TABLE users, sessions")
|
|
||||||
.execute(conn)
|
|
||||||
.expect("drop all tables");
|
|
||||||
})
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! run_test {
|
|
||||||
(|$client:ident, $conn:ident| $block:expr) => {{
|
|
||||||
let _lock = DB_LOCK.lock();
|
|
||||||
|
|
||||||
rocket::async_test(async move {
|
|
||||||
let $client = Client::tracked(mozaic4_backend::rocket())
|
|
||||||
.await
|
|
||||||
.expect("Rocket client");
|
|
||||||
let db = mozaic4_backend::DbConn::get_one($client.rocket()).await;
|
|
||||||
let $conn = db.expect("failed to get database connection for testing");
|
|
||||||
reset_db(&$conn).await;
|
|
||||||
|
|
||||||
$block
|
|
||||||
})
|
|
||||||
}};
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct BearerAuth {
|
pub struct BearerAuth {
|
||||||
token: String,
|
token: String,
|
||||||
|
@ -53,9 +22,9 @@ impl<'a> Into<Header<'a>> for BearerAuth {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[rocket::async_test]
|
||||||
fn test_registration() {
|
async fn test_registration() {
|
||||||
run_test!(|client, _conn| {
|
run_test(async move |client, _conn| {
|
||||||
let response = client
|
let response = client
|
||||||
.post("/register")
|
.post("/register")
|
||||||
.header(ContentType::JSON)
|
.header(ContentType::JSON)
|
||||||
|
@ -87,12 +56,12 @@ fn test_registration() {
|
||||||
let resp = response.into_string().await.unwrap();
|
let resp = response.into_string().await.unwrap();
|
||||||
let json: serde_json::Value = serde_json::from_str(&resp).unwrap();
|
let json: serde_json::Value = serde_json::from_str(&resp).unwrap();
|
||||||
assert_eq!(json["username"], "piepkonijn");
|
assert_eq!(json["username"], "piepkonijn");
|
||||||
});
|
}).await
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[rocket::async_test]
|
||||||
fn test_reject_invalid_credentials() {
|
async fn test_reject_invalid_credentials() {
|
||||||
run_test!(|client, _conn| {
|
run_test(async move |client, _conn| {
|
||||||
let response = client
|
let response = client
|
||||||
.post("/login")
|
.post("/login")
|
||||||
.header(ContentType::JSON)
|
.header(ContentType::JSON)
|
||||||
|
@ -102,5 +71,5 @@ fn test_reject_invalid_credentials() {
|
||||||
|
|
||||||
assert_eq!(response.status(), Status::Forbidden);
|
assert_eq!(response.status(), Status::Forbidden);
|
||||||
// assert_eq!(response.content_type(), Some(ContentType::JSON));
|
// assert_eq!(response.content_type(), Some(ContentType::JSON));
|
||||||
});
|
}).await
|
||||||
}
|
}
|
||||||
|
|
39
backend/tests/util/mod.rs
Normal file
39
backend/tests/util/mod.rs
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
use std::future::Future;
|
||||||
|
|
||||||
|
use diesel::RunQueryDsl;
|
||||||
|
use mozaic4_backend::DbConn;
|
||||||
|
use rocket::local::asynchronous::Client;
|
||||||
|
|
||||||
|
// 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: parking_lot::Mutex<()> = parking_lot::const_mutex(());
|
||||||
|
|
||||||
|
async fn reset_db(db: &DbConn) {
|
||||||
|
db.run(|conn| {
|
||||||
|
diesel::sql_query("TRUNCATE TABLE users, sessions")
|
||||||
|
.execute(conn)
|
||||||
|
.expect("drop all tables");
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn run_test<F, R>(test_closure: F)
|
||||||
|
where
|
||||||
|
F: FnOnce(Client, DbConn) -> R,
|
||||||
|
R: Future<Output = ()>,
|
||||||
|
{
|
||||||
|
let _lock = DB_LOCK.lock();
|
||||||
|
|
||||||
|
let client = Client::untracked(mozaic4_backend::rocket())
|
||||||
|
.await
|
||||||
|
.expect("failed to create test client");
|
||||||
|
let db = mozaic4_backend::DbConn::get_one(client.rocket())
|
||||||
|
.await
|
||||||
|
.expect("failed to get db connection");
|
||||||
|
|
||||||
|
// make sure we start with a clean DB
|
||||||
|
reset_db(&db).await;
|
||||||
|
|
||||||
|
test_closure(client, db).await;
|
||||||
|
}
|
Loading…
Reference in a new issue