Websocket authentication

This commit is contained in:
Matéo Duparc 2021-04-26 18:23:26 +02:00
parent 65b2b8e04d
commit 33937354b7
Signed by: hardcoresushi
GPG Key ID: 007F84120107191E
3 changed files with 133 additions and 132 deletions

View File

@ -516,9 +516,27 @@
} }
} }
//source: https://www.w3schools.com/js/js_cookies.asp
function getCookie(cname) {
var name = cname + "=";
var decodedCookie = decodeURIComponent(document.cookie);
var ca = decodedCookie.split(';');
for(var i = 0; i <ca.length; i++) {
var c = ca[i];
while (c.charAt(0) == ' ') {
c = c.substring(1);
}
if (c.indexOf(name) == 0) {
return c.substring(name.length, c.length);
}
}
return "";
}
socket = new WebSocket("ws://"+location.hostname+":WEBSOCKET_PORT/ws"); socket = new WebSocket("ws://"+location.hostname+":WEBSOCKET_PORT/ws");
socket.onopen = function() { socket.onopen = function() {
console.log("Connected"); console.log("Connected");
socket.send(getCookie("aira_auth")); //authenticating websocket connection
window.onfocus = function() { window.onfocus = function() {
if (current_chat_index != -1) { if (current_chat_index != -1) {
socket.send("set_seen "+current_chat_index); socket.send("set_seen "+current_chat_index);

View File

@ -7,7 +7,7 @@ mod ui_interface;
mod constants; mod constants;
mod discovery; mod discovery;
use std::{env, fs, io, net::{SocketAddr, TcpStream}, path::Path, str::FromStr, sync::{Arc, RwLock}}; use std::{env, fs, io, net::SocketAddr, path::Path, str::FromStr, sync::{Arc, RwLock}};
use tokio::{net::TcpListener, runtime::Handle}; use tokio::{net::TcpListener, runtime::Handle};
use actix_web::{App, HttpMessage, HttpRequest, HttpResponse, HttpServer, http::{header, CookieBuilder}, web, web::Data}; use actix_web::{App, HttpMessage, HttpRequest, HttpResponse, HttpServer, http::{header, CookieBuilder}, web, web::Data};
use actix_multipart::Multipart; use actix_multipart::Multipart;
@ -32,20 +32,34 @@ async fn start_websocket_server(global_vars: Arc<RwLock<GlobalVars>>) -> u16 {
loop { loop {
let (stream, _addr) = server.accept().await.unwrap(); let (stream, _addr) = server.accept().await.unwrap();
if *worker_done.read().unwrap() { if *worker_done.read().unwrap() {
let ui_auth_token = {
global_vars.clone().read().unwrap().ui_auth_token.clone()
};
if let Some(ui_auth_token) = ui_auth_token {
let stream = stream.into_std().unwrap(); let stream = stream.into_std().unwrap();
stream.set_nonblocking(false).unwrap(); stream.set_nonblocking(false).unwrap();
match tungstenite::accept(stream.try_clone().unwrap()) { match tungstenite::accept(stream.try_clone().unwrap()) {
Ok(websocket) => { Ok(mut websocket) => {
if let Ok(message) = websocket.read_message() { //waiting for auth token
match message.into_text() {
Ok(token) => {
if token == ui_auth_token {
let ui_connection = UiConnection::new(websocket); let ui_connection = UiConnection::new(websocket);
let global_vars = global_vars.clone(); let global_vars = global_vars.clone();
global_vars.read().unwrap().session_manager.set_ui_connection(ui_connection); global_vars.read().unwrap().session_manager.set_ui_connection(ui_connection.clone());
*worker_done.write().unwrap() = false; *worker_done.write().unwrap() = false;
websocket_worker(stream, global_vars, worker_done.clone()).await; websocket_worker(ui_connection, global_vars, worker_done.clone()).await;
}
} }
Err(e) => print_error!(e) Err(e) => print_error!(e)
} }
} }
} }
Err(e) => print_error!(e)
}
}
}
}
}); });
websocket_port websocket_port
} }
@ -78,8 +92,7 @@ fn load_msgs(session_manager: Arc<SessionManager>, ui_connection: &mut UiConnect
} }
} }
async fn websocket_worker(websocket_strem: TcpStream, global_vars: Arc<RwLock<GlobalVars>>, worker_done: Arc<RwLock<bool>>) { async fn websocket_worker(mut ui_connection: UiConnection, global_vars: Arc<RwLock<GlobalVars>>, worker_done: Arc<RwLock<bool>>) {
let mut ui_connection = UiConnection::from_raw_socket(websocket_strem.try_clone().unwrap());
let session_manager = global_vars.read().unwrap().session_manager.clone(); let session_manager = global_vars.read().unwrap().session_manager.clone();
ui_connection.set_name(&session_manager.get_my_name()); ui_connection.set_name(&session_manager.get_my_name());
session_manager.list_contacts().into_iter().for_each(|contact|{ session_manager.list_contacts().into_iter().for_each(|contact|{
@ -99,7 +112,7 @@ async fn websocket_worker(websocket_strem: TcpStream, global_vars: Arc<RwLock<Gl
}); });
discover_peers(session_manager.clone()); discover_peers(session_manager.clone());
let handle = Handle::current(); let handle = Handle::current();
std::thread::spawn(move || { std::thread::spawn(move || { //new thread needed to block on read_message() without blocking tokio tasks
loop { loop {
match ui_connection.websocket.read_message() { match ui_connection.websocket.read_message() {
Ok(msg) => { Ok(msg) => {
@ -107,8 +120,8 @@ async fn websocket_worker(websocket_strem: TcpStream, global_vars: Arc<RwLock<Gl
ui_connection.write_message(Message::Pong(Vec::new())); //not sure if I'm doing this right ui_connection.write_message(Message::Pong(Vec::new())); //not sure if I'm doing this right
} else if msg.is_text() { } else if msg.is_text() {
let msg = msg.into_text().unwrap(); let msg = msg.into_text().unwrap();
let mut ui_connection = ui_connection.clone();
let session_manager = session_manager.clone(); let session_manager = session_manager.clone();
let mut ui_connection = UiConnection::from_raw_socket(websocket_strem.try_clone().unwrap());
handle.spawn(async move { handle.spawn(async move {
let args: Vec<&str> = msg.split(" ").collect(); let args: Vec<&str> = msg.split(" ").collect();
match args[0] { match args[0] {
@ -229,6 +242,16 @@ async fn websocket_worker(websocket_strem: TcpStream, global_vars: Arc<RwLock<Gl
}); });
} }
fn is_authenticated(req: &HttpRequest) -> bool {
if let Some(cookie) = req.cookie(constants::HTTP_COOKIE_NAME) {
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
if let Some(token) = &global_vars.read().unwrap().ui_auth_token {
return token == cookie.value();
}
}
false
}
#[derive(Deserialize, Serialize, Debug)] #[derive(Deserialize, Serialize, Debug)]
struct FileInfo { struct FileInfo {
uuid: String, uuid: String,
@ -236,14 +259,11 @@ struct FileInfo {
} }
fn handle_load_file(req: HttpRequest, file_info: web::Query<FileInfo>) -> HttpResponse { fn handle_load_file(req: HttpRequest, file_info: web::Query<FileInfo>) -> HttpResponse {
match req.cookie(constants::HTTP_COOKIE_NAME) { if is_authenticated(&req) {
Some(cookie) => {
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
let global_vars = global_vars.read().unwrap();
if global_vars.http_session_manager.is_registered(cookie.value()) {
match Uuid::from_str(&file_info.uuid) { match Uuid::from_str(&file_info.uuid) {
Ok(uuid) => { Ok(uuid) => {
match global_vars.session_manager.load_file(uuid) { let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
match global_vars.read().unwrap().session_manager.load_file(uuid) {
Some(buffer) => { Some(buffer) => {
return HttpResponse::Ok().header("Content-Disposition", format!("attachment; filename=\"{}\"", escape_double_quote(html_escape::decode_html_entities(&file_info.file_name).to_string()))).content_type("application/octet-stream").body(buffer); return HttpResponse::Ok().header("Content-Disposition", format!("attachment; filename=\"{}\"", escape_double_quote(html_escape::decode_html_entities(&file_info.file_name).to_string()))).content_type("application/octet-stream").body(buffer);
} }
@ -253,19 +273,11 @@ fn handle_load_file(req: HttpRequest, file_info: web::Query<FileInfo>) -> HttpRe
Err(e) => print_error!(e) Err(e) => print_error!(e)
} }
} }
}
None => {}
}
HttpResponse::NotFound().finish() HttpResponse::NotFound().finish()
} }
async fn handle_send_file(req: HttpRequest, mut payload: Multipart) -> HttpResponse { async fn handle_send_file(req: HttpRequest, mut payload: Multipart) -> HttpResponse {
let cookie = req.cookie(constants::HTTP_COOKIE_NAME); if is_authenticated(&req) {
if cookie.is_some() {
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
let global_vars_read = global_vars.read().unwrap();
if global_vars_read.http_session_manager.is_registered(cookie.unwrap().value()) {
drop(global_vars_read); //releasing mutex while uploading the file
let mut session_id: Option<usize> = None; let mut session_id: Option<usize> = None;
while let Ok(Some(mut field)) = payload.try_next().await { while let Ok(Some(mut field)) = payload.try_next().await {
let content_disposition = field.content_disposition().unwrap(); let content_disposition = field.content_disposition().unwrap();
@ -281,6 +293,7 @@ async fn handle_send_file(req: HttpRequest, mut payload: Multipart) -> HttpRespo
buffer.extend(chunk.unwrap()); buffer.extend(chunk.unwrap());
} }
let session_id = session_id.unwrap(); let session_id = session_id.unwrap();
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
let global_vars_read = global_vars.read().unwrap(); let global_vars_read = global_vars.read().unwrap();
match global_vars_read.session_manager.send_to(&session_id, protocol::file(filename, &buffer)).await { match global_vars_read.session_manager.send_to(&session_id, protocol::file(filename, &buffer)).await {
Ok(_) => { Ok(_) => {
@ -299,17 +312,15 @@ async fn handle_send_file(req: HttpRequest, mut payload: Multipart) -> HttpRespo
} }
} }
} }
}
HttpResponse::BadRequest().finish() HttpResponse::BadRequest().finish()
} }
async fn handle_logout(req: HttpRequest) -> HttpResponse { async fn handle_logout(req: HttpRequest) -> HttpResponse {
match req.cookie(constants::HTTP_COOKIE_NAME) { if is_authenticated(&req) {
Some(cookie) => {
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap(); let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
let mut global_vars_write = global_vars.write().unwrap(); let mut global_vars_write = global_vars.write().unwrap();
if global_vars_write.session_manager.is_identity_loaded() { if global_vars_write.session_manager.is_identity_loaded() {
global_vars_write.http_session_manager.remove(cookie.value()); global_vars_write.ui_auth_token = None;
global_vars_write.session_manager.stop().await; global_vars_write.session_manager.stop().await;
} }
if Identity::is_protected().unwrap_or(true) { if Identity::is_protected().unwrap_or(true) {
@ -317,14 +328,13 @@ async fn handle_logout(req: HttpRequest) -> HttpResponse {
} else { } else {
HttpResponse::Ok().body(include_str!("frontend/logout.html")) HttpResponse::Ok().body(include_str!("frontend/logout.html"))
} }
} } else {
None => HttpResponse::Unauthorized().finish() HttpResponse::Unauthorized().finish()
} }
} }
fn login(identity: Identity, global_vars: &Arc<RwLock<GlobalVars>>) -> HttpResponse { fn login(identity: Identity, global_vars: &Arc<RwLock<GlobalVars>>) -> HttpResponse {
let mut global_vars_write = global_vars.write().unwrap(); let mut global_vars_write = global_vars.write().unwrap();
let cookie_value = global_vars_write.http_session_manager.register();
let session_manager = global_vars_write.session_manager.clone(); let session_manager = global_vars_write.session_manager.clone();
if !session_manager.is_identity_loaded() { if !session_manager.is_identity_loaded() {
global_vars_write.session_manager.set_identity(Some(identity)); global_vars_write.session_manager.set_identity(Some(identity));
@ -334,11 +344,15 @@ fn login(identity: Identity, global_vars: &Arc<RwLock<GlobalVars>>) -> HttpRespo
} }
}); });
} }
let cookie = CookieBuilder::new(constants::HTTP_COOKIE_NAME, cookie_value) let mut raw_cookie = [0; 32];
.http_only(true) OsRng.fill_bytes(&mut raw_cookie);
.max_age(time::Duration::hours(4) let cookie_value = base64::encode(raw_cookie);
).finish(); global_vars_write.ui_auth_token = Some(cookie_value.clone());
HttpResponse::Found().header(header::LOCATION, "/").set_header(header::SET_COOKIE, cookie.to_string()).finish() let cookie = CookieBuilder::new(constants::HTTP_COOKIE_NAME, cookie_value).max_age(time::Duration::hours(4)).finish();
HttpResponse::Found()
.header(header::LOCATION, "/")
.set_header(header::SET_COOKIE, cookie.to_string())
.finish()
} }
fn on_identity_loaded(identity: Identity, global_vars: &Arc<RwLock<GlobalVars>>) -> HttpResponse { fn on_identity_loaded(identity: Identity, global_vars: &Arc<RwLock<GlobalVars>>) -> HttpResponse {
@ -428,23 +442,16 @@ fn index_not_logged_in(global_vars: &Arc<RwLock<GlobalVars>>) -> HttpResponse {
async fn handle_index(req: HttpRequest) -> HttpResponse { async fn handle_index(req: HttpRequest) -> HttpResponse {
let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap(); let global_vars = req.app_data::<Data<Arc<RwLock<GlobalVars>>>>().unwrap();
match req.cookie(constants::HTTP_COOKIE_NAME) { if is_authenticated(&req) {
Some(cookie) => {
let global_vars_read = global_vars.read().unwrap();
if global_vars_read.http_session_manager.is_registered(cookie.value()) {
HttpResponse::Ok().body( HttpResponse::Ok().body(
include_str!("frontend/index.html") include_str!("frontend/index.html")
.replace("WEBSOCKET_PORT", &global_vars_read.websocket_port.to_string()) .replace("WEBSOCKET_PORT", &global_vars.read().unwrap().websocket_port.to_string())
.replace("IS_IDENTITY_PROTECTED", &Identity::is_protected().unwrap().to_string()) .replace("IS_IDENTITY_PROTECTED", &Identity::is_protected().unwrap().to_string())
) )
} else { } else {
drop(global_vars_read);
index_not_logged_in(global_vars) index_not_logged_in(global_vars)
} }
} }
None => index_not_logged_in(global_vars)
}
}
const JS_CONTENT_TYPE: &str = "text/javascript"; const JS_CONTENT_TYPE: &str = "text/javascript";
@ -564,38 +571,10 @@ struct CreateParams {
password_confirm: String, password_confirm: String,
} }
struct HttpSessionsManager {
http_sessions: Vec<String>,
}
impl HttpSessionsManager {
fn get_index(&self, cookie: &str) -> Option<usize> {
self.http_sessions.iter().position(|c| c == cookie)
}
pub fn new() -> HttpSessionsManager {
HttpSessionsManager {
http_sessions: Vec::new()
}
}
pub fn register(&mut self) -> String {
let mut raw_cookie = [0; 32];
OsRng.fill_bytes(&mut raw_cookie);
let cookie = base64::encode(raw_cookie);
self.http_sessions.push(cookie.clone());
cookie
}
pub fn remove(&mut self, cookie: &str) -> Option<String> {
Some(self.http_sessions.remove(self.get_index(cookie)?))
}
pub fn is_registered(&self, cookie: &str) -> bool {
self.get_index(cookie).is_some()
}
}
struct GlobalVars { struct GlobalVars {
session_manager: Arc<SessionManager>, session_manager: Arc<SessionManager>,
websocket_port: u16, websocket_port: u16,
http_session_manager: HttpSessionsManager, ui_auth_token: Option<String>,
tokio_handle: Handle, tokio_handle: Handle,
} }
@ -612,7 +591,7 @@ async fn main() {
let global_vars = Arc::new(RwLock::new(GlobalVars { let global_vars = Arc::new(RwLock::new(GlobalVars {
session_manager: Arc::new(SessionManager::new()), session_manager: Arc::new(SessionManager::new()),
websocket_port: 0, websocket_port: 0,
http_session_manager: HttpSessionsManager::new(), ui_auth_token: None,
tokio_handle: Handle::current(), tokio_handle: Handle::current(),
})); }));
let websocket_port = start_websocket_server(global_vars.clone()).await; let websocket_port = start_websocket_server(global_vars.clone()).await;

View File

@ -91,11 +91,6 @@ pub struct UiConnection{
} }
impl UiConnection { impl UiConnection {
pub fn from_raw_socket(stream: TcpStream) -> UiConnection {
let websocket = WebSocket::from_raw_socket(stream, Role::Server, None);
UiConnection::new(websocket)
}
pub fn new(websocket: WebSocket<TcpStream>) -> UiConnection { pub fn new(websocket: WebSocket<TcpStream>) -> UiConnection {
UiConnection { UiConnection {
websocket: websocket, websocket: websocket,
@ -162,3 +157,12 @@ impl UiConnection {
self.write_message(Message::from("logout")); self.write_message(Message::from("logout"));
} }
} }
impl Clone for UiConnection {
fn clone(&self) -> Self {
UiConnection {
websocket: WebSocket::from_raw_socket(self.websocket.get_ref().try_clone().unwrap(), Role::Server, None),
is_valid: self.is_valid
}
}
}