/// API handlers, the ends of each filter chain use blake2::{Blake2s, Digest}; use jsonwebtoken::errors::ErrorKind; use jsonwebtoken::{decode, Algorithm, DecodingKey, TokenData, Validation}; use log::{debug, warn}; use md5::Md5; use parking_lot::RwLockUpgradableReadGuard; use serde::Serialize; use serde_json; use std::convert::Infallible; use std::fs; use warp::{http::StatusCode, reply}; #[derive(Serialize, Debug)] struct GradeCoinResponse { res: ResponseType, message: String, } #[derive(Debug, Serialize)] enum ResponseType { Success, Error, } use crate::schema::{AuthRequest, Block, Claims, Db, MetuId, NakedBlock, Transaction, User}; const BEARER: &str = "Bearer "; /// POST request to /register endpoint /// /// Lets a [`User`] (=student) to authenticate themselves to the system /// This `request` can be rejected if the payload is malformed (= not authenticated properly) or if /// the [`AuthRequest.user_id`] of the `request` is not in the list of users that can hold a Gradecoin account pub async fn authenticate_user( request: AuthRequest, db: Db, ) -> Result { debug!("POST request to /register, authenticate_user"); let provided_id = request.student_id.clone(); let priv_student_id = match MetuId::new(request.student_id) { Some(id) => id, None => { let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "This user cannot have a gradecoin account".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } }; let userlist = db.users.upgradable_read(); if userlist.contains_key(&provided_id) { let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "This user is already authenticated".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } // TODO: audit public key, is it valid? <13-04-21, yigit> // let new_user = User { user_id: priv_student_id, public_key: request.public_key, balance: 0, }; let user_json = serde_json::to_string(&new_user).unwrap(); fs::write(format!("users/{}.guy", new_user.user_id), user_json).unwrap(); let mut userlist = RwLockUpgradableReadGuard::upgrade(userlist); userlist.insert(provided_id, new_user); // TODO: signature of the public key, please <11-04-21, yigit> // let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Success, message: "User authenticated to use Gradecoin".to_owned(), }); Ok(warp::reply::with_status(res_json, StatusCode::CREATED)) } /// GET /transaction /// Returns JSON array of transactions /// Cannot fail pub async fn list_transactions(db: Db) -> Result { debug!("GET request to /transaction, list_transactions"); let mut result = Vec::new(); let transactions = db.pending_transactions.read(); // let transactions = transactions.clone().into_iter().collect(); for (_, value) in transactions.iter() { result.push(value) } Ok(reply::with_status(reply::json(&result), StatusCode::OK)) } /// POST /block /// /// Proposes a new block for the next round. /// Can reject the block /// /// TODO: WHO IS PROPOSING THIS BLOCK OH GOD <13-04-21, yigit> // ok let's say the proposer has /// to put their transaction as the first transaction of the transaction_list /// that's not going to backfire in any way /// /// TODO: after a block is accepted, it's transactions should play out and the proposer should /// get something for their efforts <13-04-21, yigit> // pub async fn authorized_propose_block( new_block: Block, token: String, db: Db, ) -> Result { debug!("POST request to /block, authorized_propose_block"); let users_store = db.users.read(); println!("{:?}", &new_block); let internal_user = match users_store.get(&new_block.transaction_list[0]) { Some(existing_user) => existing_user, None => { debug!( "User with public key signature {:?} is not found in the database", new_block.transaction_list[0] ); let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "User with the given public key signature is not found in the database" .to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } }; let proposer_public_key = &internal_user.public_key; let token_payload = match authorize_proposer(token, &proposer_public_key) { Ok(data) => data, Err(below) => { debug!("Something went wrong below {:?}", below); let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: below, }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } }; debug!("authorized for block proposal"); if token_payload.claims.tha != new_block.hash { debug!( "The Hash of the block {:?} did not match the hash given in jwt {:?}", new_block.hash, token_payload.claims.tha ); let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "The hash of the block did not match the hash given in JWT".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } debug!("clear for block proposal"); let pending_transactions = db.pending_transactions.upgradable_read(); let blockchain = db.blockchain.upgradable_read(); for transaction_hash in new_block.transaction_list.iter() { if !pending_transactions.contains_key(transaction_hash) { let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "Block contains unknown transaction".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } } let naked_block = NakedBlock { transaction_list: new_block.transaction_list.clone(), nonce: new_block.nonce.clone(), timestamp: new_block.timestamp.clone(), }; let naked_block_flat = serde_json::to_vec(&naked_block).unwrap(); let hashvalue = Blake2s::digest(&naked_block_flat); let hash_string = format!("{:x}", hashvalue); // 6 rightmost bits are zero? let should_zero = hashvalue[31] as i32 + hashvalue[30] as i32 + hashvalue[29] as i32; // TODO: this can be offloaded to validator <13-04-21, yigit> // if should_zero != 0 { debug!("the hash does not have 6 rightmost zero bits"); let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "Given block hash is larger than target value".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } // one last check to see if block is telling the truth if hash_string != new_block.hash { debug!("request was not telling the truth, hash values do not match"); // TODO: does this condition make more sense _before_ the hash 0s check? <13-04-21, yigit> // let res_json = warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "Given hash value does not match the actual block hash".to_owned(), }); return Ok(warp::reply::with_status(res_json, StatusCode::BAD_REQUEST)); } let mut blockchain = RwLockUpgradableReadGuard::upgrade(blockchain); let block_json = serde_json::to_string(&new_block).unwrap(); fs::write( format!("blocks/{}.block", new_block.timestamp.timestamp()), block_json, ) .unwrap(); *blockchain = new_block; let mut pending_transactions = RwLockUpgradableReadGuard::upgrade(pending_transactions); pending_transactions.clear(); Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Success, message: "Block accepted".to_owned(), }), StatusCode::CREATED, )) } /// POST /transaction /// /// Handles the new transaction requests /// Can reject the block if; /// # Arguments /// * `new_transaction` - Valid JSON of a [`Transaction`] /// * `token` - An Authorization header value such as `Bearer aaa.bbb.ccc` /// * `db` - Global [`Db`] instance /// /// TODO This method should check if the user has enough balance for the transaction pub async fn authorized_propose_transaction( new_transaction: Transaction, token: String, db: Db, ) -> Result { debug!("POST request to /transaction, authorized_propose_transaction"); debug!("The transaction request: {:?}", new_transaction); let users_store = db.users.read(); // Is this transaction from an authorized source? let internal_user = match users_store.get(&new_transaction.by) { Some(existing_user) => existing_user, None => { debug!( "User with public key signature {:?} is not found in the database", new_transaction.by ); return Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "User with the given public key signature is not authorized" .to_owned(), }), StatusCode::BAD_REQUEST, )); } }; // `user` is an authenticated student, can propose // check if user can afford the transaction if new_transaction.by == new_transaction.source { // Propose to transact with another user if internal_user.balance < new_transaction.amount { return Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "User does not have enough balance in their account" .to_owned(), }), StatusCode::BAD_REQUEST, )); } } else { // todo: add bank mechanism return Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "Invalid by field for the proposed transaction" .to_owned(), }), StatusCode::BAD_REQUEST, )); } // This public key was already written to the database, we can panic if it's not valid at // *this* point let proposer_public_key = &internal_user.public_key; let token_payload = match authorize_proposer(token, &proposer_public_key) { Ok(data) => data, Err(below) => { debug!("Something went wrong below {:?}", below); return Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: below, }), StatusCode::BAD_REQUEST, )); } }; // this transaction was already checked for correctness at custom_filters, we can panic // here if it has been changed since debug!("authorized for transaction proposal"); let hashed_transaction = Md5::digest(&serde_json::to_vec(&new_transaction).unwrap()); if token_payload.claims.tha != format!("{:x}", hashed_transaction) { debug!( "the hash of the request {:x} did not match the hash given in jwt {:?}", hashed_transaction, token_payload.claims.tha ); return Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Error, message: "The hash of the block did not match the hash given in JWT".to_owned(), }), StatusCode::BAD_REQUEST, )); } debug!("clear for transaction proposal"); let mut transactions = db.pending_transactions.write(); transactions.insert(new_transaction.source.to_owned(), new_transaction); Ok(warp::reply::with_status( warp::reply::json(&GradeCoinResponse { res: ResponseType::Success, message: "Transaction accepted".to_owned(), }), StatusCode::CREATED, )) } /// GET /block /// Returns the last block's JSON /// Cannot fail /// Mostly around for debug purposes pub async fn list_blocks(db: Db) -> Result { debug!("GET request to /block, list_blocks"); let block = db.blockchain.read(); Ok(reply::with_status(reply::json(&*block), StatusCode::OK)) } /// Handles the JWT Authorization /// /// *[`jwt_token`]: The raw JWT token, "Bearer aaa.bbb.ccc" /// *[`user_pem`]: User Public Key, "BEGIN RSA" /// NOT async, might look into it if this becomes a bottleneck fn authorize_proposer(jwt_token: String, user_pem: &String) -> Result, String> { // Throw away the "Bearer " part let raw_jwt = jwt_token.trim_start_matches(BEARER).to_owned(); debug!("raw_jwt: {:?}", raw_jwt); // Extract a jsonwebtoken compatible decoding_key from user's public key // TODO: just use this for reading users pem key <13-04-21, yigit> // let decoding_key = match DecodingKey::from_rsa_pem(user_pem.as_bytes()) { Ok(key) => key, Err(j) => { warn!( "user has invalid RSA key we should crash and burn here {:?}", j ); return Err(String::from("This User's RSA key is invalid")); } }; // Extract the payload inside the JWT let token_payload = match decode::(&raw_jwt, &decoding_key, &Validation::new(Algorithm::RS256)) { Ok(decoded) => decoded, Err(err) => match *err.kind() { ErrorKind::InvalidToken => { debug!("raw_jwt={:?} was malformed err={:?}", raw_jwt, err); return Err(String::from("Invalid Token")); } ErrorKind::InvalidRsaKey => { debug!("The RSA key does not have a valid format, {:?}", err); return Err(String::from("The RSA key does not have a valid format")); } ErrorKind::ExpiredSignature => { debug!("this token has expired {:?}", err); return Err(String::from("This token has expired")); } _ => { warn!("AN UNSPECIFIED ERROR: {:?}", err); return Err(String::from("Unspecified error")); } }, }; Ok(token_payload) }