#![feature(proc_macro_hygiene, decl_macro)] mod data; mod ruma_wrapper; mod utils; pub use data::Data; use log::debug; use rocket::{get, post, put, routes, State}; use ruma_client_api::{ error::{Error, ErrorKind}, r0::{ account::register, alias::get_alias, membership::join_room_by_id, message::create_message_event, session::login, }, unversioned::get_supported_versions, }; use ruma_events::collections::all::Event; use ruma_events::room::message::MessageEvent; use ruma_identifiers::{EventId, UserId}; use ruma_wrapper::{MatrixResult, Ruma}; use serde_json::map::Map; use std::convert::TryFrom; use std::{collections::HashMap, convert::TryInto}; #[get("/_matrix/client/versions")] fn get_supported_versions_route() -> MatrixResult { MatrixResult(Ok(get_supported_versions::Response { versions: vec![ "r0.0.1".to_owned(), "r0.1.0".to_owned(), "r0.2.0".to_owned(), "r0.3.0".to_owned(), "r0.4.0".to_owned(), "r0.5.0".to_owned(), "r0.6.0".to_owned(), ], unstable_features: HashMap::new(), })) } #[post("/_matrix/client/r0/register", data = "")] fn register_route( data: State, body: Ruma, ) -> MatrixResult { // Validate user id let user_id: UserId = match (*format!( "@{}:{}", body.username.clone().unwrap_or("randomname".to_owned()), data.hostname() )) .try_into() { Err(_) => { debug!("Username invalid"); return MatrixResult(Err(Error { kind: ErrorKind::InvalidUsername, message: "Username was invalid.".to_owned(), status_code: http::StatusCode::BAD_REQUEST, })); } Ok(user_id) => user_id, }; // Check if username is creative enough if data.user_exists(&user_id) { debug!("ID already taken"); return MatrixResult(Err(Error { kind: ErrorKind::UserInUse, message: "Desired user ID is already taken.".to_owned(), status_code: http::StatusCode::BAD_REQUEST, })); } // Create user data.user_add(&user_id, body.password.clone()); // Generate new device id if the user didn't specify one let device_id = body .device_id .clone() .unwrap_or_else(|| "TODO:randomdeviceid".to_owned()); // Add device data.device_add(&user_id, &device_id); // Generate new token for the device let token = "TODO:randomtoken".to_owned(); data.token_replace(&user_id, &device_id, token.clone()); MatrixResult(Ok(register::Response { access_token: token, home_server: data.hostname(), user_id, device_id, })) } #[post("/_matrix/client/r0/login", data = "")] fn login_route(data: State, body: Ruma) -> MatrixResult { // Validate login method let user_id = if let (login::UserInfo::MatrixId(mut username), login::LoginInfo::Password { password }) = (body.user.clone(), body.login_info.clone()) { if !username.contains(':') { username = format!("@{}:{}", username, data.hostname()); } if let Ok(user_id) = (*username).try_into() { if !data.user_exists(&user_id) {} // Check password if let Some(correct_password) = data.password_get(&user_id) { if password == correct_password { // Success! user_id } else { debug!("Invalid password."); return MatrixResult(Err(Error { kind: ErrorKind::Unknown, message: "".to_owned(), status_code: http::StatusCode::FORBIDDEN, })); } } else { debug!("UserId does not exist (has no assigned password). Can't log in."); return MatrixResult(Err(Error { kind: ErrorKind::Forbidden, message: "".to_owned(), status_code: http::StatusCode::FORBIDDEN, })); } } else { debug!("Invalid UserId."); return MatrixResult(Err(Error { kind: ErrorKind::Unknown, message: "Bad login type.".to_owned(), status_code: http::StatusCode::BAD_REQUEST, })); } } else { debug!("Bad login type"); return MatrixResult(Err(Error { kind: ErrorKind::Unknown, message: "Bad login type.".to_owned(), status_code: http::StatusCode::BAD_REQUEST, })); }; // Generate new device id if the user didn't specify one let device_id = body .device_id .clone() .unwrap_or("TODO:randomdeviceid".to_owned()); // Add device (TODO: We might not want to call it when using an existing device) data.device_add(&user_id, &device_id); // Generate a new token for the device let token = "TODO:randomtoken".to_owned(); data.token_replace(&user_id, &device_id, token.clone()); return MatrixResult(Ok(login::Response { user_id, access_token: token, home_server: Some(data.hostname()), device_id, well_known: None, })); } #[get("/_matrix/client/r0/directory/room/")] fn get_alias_route(room_alias: String) -> MatrixResult { // TODO let room_id = match &*room_alias { "#room:localhost" => "!xclkjvdlfj:localhost", _ => { debug!("Room not found."); return MatrixResult(Err(Error { kind: ErrorKind::NotFound, message: "Room not found.".to_owned(), status_code: http::StatusCode::NOT_FOUND, })); } } .try_into() .unwrap(); MatrixResult(Ok(get_alias::Response { room_id, servers: vec!["localhost".to_owned()], })) } #[post("/_matrix/client/r0/rooms/<_room_id>/join", data = "")] fn join_room_by_id_route( _room_id: String, body: Ruma, ) -> MatrixResult { // TODO MatrixResult(Ok(join_room_by_id::Response { room_id: body.room_id.clone(), })) } #[put( "/_matrix/client/r0/rooms/<_room_id>/send/<_event_type>/<_txn_id>", data = "" )] fn create_message_event_route( data: State, _room_id: String, _event_type: String, _txn_id: String, body: Ruma, ) -> MatrixResult { // Generate event id let event_id = EventId::try_from("$TODOrandomeventid:localhost").unwrap(); data.event_add( &Event::RoomMessage(MessageEvent { content: body.data.clone().into_result().unwrap(), event_id: event_id.clone(), origin_server_ts: utils::millis_since_unix_epoch(), room_id: Some(body.room_id.clone()), sender: body.user_id.clone().expect("user is authenticated"), unsigned: Map::default(), }), &body.room_id, &event_id, ); MatrixResult(Ok(create_message_event::Response { event_id })) } fn main() { // Log info by default if let Err(_) = std::env::var("RUST_LOG") { std::env::set_var("RUST_LOG", "info"); } pretty_env_logger::init(); let data = Data::load_or_create(); data.set_hostname("localhost"); rocket::ignite() .mount( "/", routes![ get_supported_versions_route, register_route, login_route, get_alias_route, join_room_by_id_route, create_message_event_route, ], ) .manage(data) .launch(); }