Browse Source

Merge pull request #138 from comit-network/simplify-auth-route-testing

fix-bad-api-calls
Thomas Eizinger 3 years ago
committed by GitHub
parent
commit
bd41533db1
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 78
      daemon/src/routes_maker.rs

78
daemon/src/routes_maker.rs

@ -88,23 +88,6 @@ pub async fn post_sell_order(
Ok(status::Accepted(None)) Ok(status::Accepted(None))
} }
/// Test route solely for the purposes of exercising authentication.
/// It validates whether the posted order request was correct, but does not do
/// anything else with it.
#[cfg(test)]
#[rocket::post("/order/test", data = "<order>")]
pub async fn post_test_order(
order: Json<CfdNewOrderRequest>,
_auth: Authenticated,
) -> Result<status::Accepted<()>, status::BadRequest<String>> {
let _order = Order::from_default_with_price(order.price, Origin::Ours)
.map_err(|e| status::BadRequest(Some(e.to_string())))?
.with_min_quantity(order.min_quantity)
.with_max_quantity(order.max_quantity);
Ok(status::Accepted(None))
}
/// A "catcher" for all 401 responses, triggers the browser's basic auth implementation. /// A "catcher" for all 401 responses, triggers the browser's basic auth implementation.
#[rocket::catch(401)] #[rocket::catch(401)]
pub fn unauthorized() -> PromptAuthentication { pub fn unauthorized() -> PromptAuthentication {
@ -143,6 +126,7 @@ pub async fn post_accept_order(
status::Accepted(None) status::Accepted(None)
} }
#[rocket::post("/order/reject", data = "<cfd_reject_order_request>")] #[rocket::post("/order/reject", data = "<cfd_reject_order_request>")]
pub async fn post_reject_order( pub async fn post_reject_order(
cfd_reject_order_request: Json<AcceptOrRejectOrderRequest>, cfd_reject_order_request: Json<AcceptOrRejectOrderRequest>,
@ -182,75 +166,41 @@ pub fn index<'r>(_paths: PathBuf, _auth: Authenticated) -> impl Responder<'r, 's
mod tests { mod tests {
use super::*; use super::*;
use crate::auth::Password; use crate::auth::Password;
use bdk::bitcoin::{Address, Amount, Network, PublicKey};
use rocket::http::{Header, Status}; use rocket::http::{Header, Status};
use rocket::local::blocking::Client; use rocket::local::blocking::Client;
use rocket::{Build, Rocket}; use rocket::{Build, Rocket};
use std::time::SystemTime;
#[test] #[test]
fn routes_are_password_protected() { fn routes_are_password_protected() {
let client = Client::tracked(rocket()).unwrap(); let client = Client::tracked(rocket()).unwrap();
let feed_response = client.get("/feed").dispatch(); let response = client.get("/protected").dispatch();
let new_sell_order_response = client
.post("/order/test")
.body(r#"{"price":"40000", "min_quantity":"100", "max_quantity":"10000"}"#)
.dispatch();
let index_response = client.get("/").header(ContentType::HTML).dispatch();
assert_eq!(feed_response.status(), Status::Unauthorized); assert_eq!(response.status(), Status::Unauthorized);
assert_eq!(new_sell_order_response.status(), Status::Unauthorized); assert_eq!(
assert_eq!(index_response.status(), Status::Unauthorized); response.headers().get_one("WWW-Authenticate"),
Some(r#"Basic charset="UTF-8"#)
);
} }
#[test] #[test]
fn correct_password_grants_access() { fn correct_password_grants_access() {
let client = Client::tracked(rocket()).unwrap(); let client = Client::tracked(rocket()).unwrap();
let feed_response = client.get("/feed").header(auth_header()).dispatch(); let response = client.get("/protected").header(auth_header()).dispatch();
let new_sell_order_response = client
.post("/order/test")
.body(r#"{"price":"40000", "min_quantity":"100", "max_quantity":"10000"}"#)
.header(auth_header())
.dispatch();
let index_response = client
.get("/")
.header(ContentType::HTML)
.header(auth_header())
.dispatch();
assert_eq!(feed_response.status(), Status::Ok); assert_eq!(response.status(), Status::Ok);
assert_eq!(new_sell_order_response.status(), Status::Accepted);
assert!(
index_response.status() == Status::NotFound || index_response.status() == Status::Ok
);
} }
#[rocket::get("/protected")]
async fn protected(_auth: Authenticated) {}
/// Constructs a Rocket instance for testing. /// Constructs a Rocket instance for testing.
fn rocket() -> Rocket<Build> { fn rocket() -> Rocket<Build> {
let (_, state1) = watch::channel::<Vec<Cfd>>(vec![]);
let (_, state2) = watch::channel::<Option<Order>>(None);
let (_, state3) = watch::channel::<WalletInfo>(WalletInfo {
balance: Amount::ZERO,
address: Address::p2wpkh(
&PublicKey::new(
"0286cd889349ebc06b3165505b9c083df0a4147f554614ff207c10f16ff509578c"
.parse()
.unwrap(),
),
Network::Regtest,
)
.unwrap(),
last_updated_at: SystemTime::now(),
});
rocket::build() rocket::build()
.manage(state1)
.manage(state2)
.manage(state3)
.manage(Password::from(*b"Now I'm feelin' so fly like a G6")) .manage(Password::from(*b"Now I'm feelin' so fly like a G6"))
.mount("/", rocket::routes![maker_feed, post_test_order, index]) .mount("/", rocket::routes![protected])
.register("/", rocket::catchers![unauthorized])
} }
/// Creates an "Authorization" header that matches the password above, /// Creates an "Authorization" header that matches the password above,

Loading…
Cancel
Save