server from db working

This commit is contained in:
2026-05-31 23:59:51 +02:00
parent 2303f06fea
commit b818f7eaa3
12 changed files with 203 additions and 69 deletions

View File

@ -12,9 +12,5 @@ use diesel::r2d2::{
ConnectionManager ConnectionManager
}; };
pub mod schema;
pub static mut DOCKER: LazyLock<Docker> = LazyLock::new(|| Docker::connect_with_local_defaults().expect("Failed to connect to the docker socket") ); pub static mut DOCKER: LazyLock<Docker> = LazyLock::new(|| Docker::connect_with_local_defaults().expect("Failed to connect to the docker socket") );

View File

@ -6,7 +6,7 @@
/* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */ /* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */ /* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/29 21:22:17 by tomoron #+# #+# */ /* Created: 2026/05/29 21:22:17 by tomoron #+# #+# */
/* Updated: 2026/05/29 21:31:39 by tomoron ### ########.fr */ /* Updated: 2026/05/31 15:02:01 by tomoron ### ########.fr */
/* */ /* */
/* ************************************************************************** */ /* ************************************************************************** */
@ -41,10 +41,13 @@ use diesel::{prelude::*, r2d2::{ConnectionManager, Pool}};
use std::env; use std::env;
//pub mod status; pub mod schema;
pub mod models;
//use status::ServerStatus; pub mod status;
//
use status::ServerStatus;
use crate::models::Servers;
pub type DbPool = Pool<ConnectionManager<PgConnection>>; pub type DbPool = Pool<ConnectionManager<PgConnection>>;
@ -66,6 +69,8 @@ async fn main() -> io::Result<()> {
let mc_listener = TcpListener::bind("0.0.0.0:25565").await?; let mc_listener = TcpListener::bind("0.0.0.0:25565").await?;
let rpc_listener = TcpListener::bind("0.0.0.0:8080").await?; let rpc_listener = TcpListener::bind("0.0.0.0:8080").await?;
// let servers = schema::servers::table.select(Servers::as_select()).load(conn).unwrap();
loop { loop {
tokio::select! { tokio::select! {
Ok((socket, _)) = mc_listener.accept() => { Ok((socket, _)) = mc_listener.accept() => {

View File

@ -1,8 +1,6 @@
use crate::minecraft::client::client::Client; use crate::minecraft::client::client::Client;
use crate::minecraft::varint::varint_read; use crate::minecraft::varint::varint_read;
use crate::minecraft::handshake::Handshake;
use std::collections::VecDeque;
use std::io; use std::io;
@ -95,28 +93,4 @@ impl Client {
Ok(()) Ok(())
} }
async fn handle_packet(&mut self, mut packet: VecDeque<u8>) -> Result<(), String> {
let packet_id = varint_read(&mut packet)?;
if self.handshake.is_none() {
if packet_id != 0 {
return Err("packet 0 expected. invalid packet received".to_string());
}
self.handshake = Some(Handshake::from_packet(&mut packet)?);
if self.handshake.as_ref().unwrap().intent == 2 {
self.login_intent_handle(&mut packet, packet_id).await?;
}
return Ok(());
}
let intent = self.handshake.as_ref().unwrap().intent;
if intent == 1 {
self.status_intent_handle(&mut packet, packet_id).await?;
}
Ok(())
}
} }

View File

@ -6,7 +6,7 @@
/* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */ /* By: tomoron <tomoron@student.42angouleme.fr> +#+ +:+ +#+ */
/* +#+#+#+#+#+ +#+ */ /* +#+#+#+#+#+ +#+ */
/* Created: 2026/05/07 17:23:09 by tomoron #+# #+# */ /* Created: 2026/05/07 17:23:09 by tomoron #+# #+# */
/* Updated: 2026/05/30 00:27:19 by tomoron ### ########.fr */ /* Updated: 2026/05/31 23:23:05 by tomoron ### ########.fr */
/* */ /* */
/* ************************************************************************** */ /* ************************************************************************** */
@ -15,6 +15,8 @@ use crate::minecraft::handshake::Handshake;
use crate::minecraft::varint::varint_write; use crate::minecraft::varint::varint_write;
use std::fmt; use std::fmt;
use crate::models::Servers;
pub struct Client { pub struct Client {
pub in_stream: TcpStream, pub in_stream: TcpStream,
@ -22,11 +24,22 @@ pub struct Client {
pub buffer: Vec<u8>, pub buffer: Vec<u8>,
pub out_stream: Option<TcpStream>, pub out_stream: Option<TcpStream>,
pub handshake: Option<Handshake>, pub handshake: Option<Handshake>,
pub closed: bool,
pub server: Option<Result<Servers,diesel::result::Error>>,
pub dbPool: DbPool pub dbPool: DbPool
} }
use crate::DbPool; use crate::DbPool;
use crate::schema;
use crate::models;
use diesel::prelude::*;
use regex::Regex;
impl Client { impl Client {
pub fn create(stream: TcpStream, pool: DbPool) -> Self { pub fn create(stream: TcpStream, pool: DbPool) -> Self {
Self { Self {
@ -37,20 +50,40 @@ impl Client {
out_stream: None, out_stream: None,
handshake: None, handshake: None,
closed: false,
server: None,
dbPool: pool dbPool: pool
} }
} }
pub async fn send_packet(&self, data: Vec<u8>) { pub async fn get_server(&self) -> Result<Servers, diesel::result::Error> {
let mut sent_data: Vec<u8> = varint_write(data.len() as i32); println!("get server name from domain : {}", self.handshake.as_ref().unwrap().server_address);
if let None = self.handshake {
sent_data.extend(data); panic!("get server called without a handshake");
let _ = self.in_stream.writable().await;
match self.in_stream.try_write(sent_data.as_slice()) {
Err(e) => { eprintln!("error while sending response {:?}", e); },
_ => { }
} }
let conn = &mut self.dbPool.get().unwrap();
let reg = Regex::new(r"^(?:([a-zA-Z-_]*)\.)?mc\.tmoron\.fr$").unwrap();
let mut found: Option<String> = None;
if let Some(capture) = reg.captures(&self.handshake.as_ref().unwrap().server_address) {
if let Some(captured_str) = capture.get(1) {
found = Some(captured_str.as_str().to_string());
}
}
if let Some(name) = found {
println!("found server name : {}", name);
schema::servers::table.filter(schema::servers::name.eq(name)).select(models::Servers::as_select()).first(conn)
} else {
schema::servers::table.filter(schema::servers::is_default.eq(true)).select(models::Servers::as_select()).first(conn)
}
} }
} }

View File

@ -1,26 +1,24 @@
use crate::minecraft::client::client::Client; use crate::minecraft::client::client::Client;
use json::object;
use crate::minecraft::varint::varint_write; use crate::minecraft::varint::varint_write;
use std::collections::VecDeque; use std::collections::VecDeque;
use tokio::net::TcpStream; use tokio::net::TcpStream;
impl Client { impl Client {
pub async fn login_intent_handle(&mut self, mut _packet: &mut VecDeque<u8>, _packet_id: i32) -> Result<(),String> { pub async fn login_intent_handle(&mut self, mut _packet: &mut VecDeque<u8>, _packet_id: i32) -> Result<(),String> {
let stream = TcpStream::connect("play.hypixel.net:25565").await.map_err(|_| "failed to connect to remote host".to_string())?; let reason_json = object! {
let mut new_handshake = self.handshake.as_ref().unwrap().clone(); "type": "text",
new_handshake.server_address = "play.hypixel.net".to_string(); "text": "login to start not implemented yet",
new_handshake.server_port = 25565; "color": "red"
};
let mut response: Vec<u8> = vec![];
let response_json = json::stringify(reason_json);
let mut response_json_len = varint_write(response_json.len() as i32);
let handshake_packet = new_handshake.to_packet(); response.append(&mut response_json_len);
let mut packet = Vec::new(); response.append(&mut response_json.as_bytes().into());
packet.extend(varint_write(handshake_packet.len() as i32)); self.send_packet(0x0, response).await;
packet.extend(handshake_packet);
self.out_stream = Some(stream);
let _ = self.out_stream.as_ref().unwrap().writable().await;
let _ = self.out_stream.as_ref().unwrap().try_write(packet.as_slice());
println!("[{}] sent handshake to server", self);
Ok(()) Ok(())
} }
} }

View File

@ -2,3 +2,4 @@ pub mod client;
mod buffer; mod buffer;
mod status; mod status;
mod login; mod login;
mod packet;

View File

@ -0,0 +1,80 @@
use crate::minecraft::client::client::Client;
use crate::minecraft::varint::{varint_read, varint_write};
use crate::minecraft::handshake::Handshake;
use crate::models;
use crate::status::ServerStatus;
use std::collections::VecDeque;
use diesel::result::Error::NotFound;
use tokio::net::TcpStream;
impl Client {
pub async fn handle_packet(&mut self, mut packet: VecDeque<u8>) -> Result<(), String> {
let packet_id = varint_read(&mut packet)?;
if self.handshake.is_none() {
if packet_id != 0 {
return Err("packet 0 expected. invalid packet received".to_string());
}
self.handshake = Some(Handshake::from_packet(&mut packet)?);
self.server = Some(self.get_server().await);
// if self.server.as_ref().unwrap().status == ServerStatus::Running {
// return self.proxy_start().await;
// }
if self.handshake.as_ref().unwrap().intent == 2 {
self.login_intent_handle(&mut packet, packet_id).await?;
}
return Ok(());
}
let intent = self.handshake.as_ref().unwrap().intent;
if intent == 1 {
self.status_intent_handle(&mut packet, packet_id).await?;
}
Ok(())
}
pub async fn send_packet(&self, packet_id: i32, data: Vec<u8>) {
let sent_id = varint_write(packet_id);
let mut sent_data: Vec<u8> = vec![];
sent_data.extend(varint_write((data.len() + sent_id.len()) as i32));
sent_data.extend(sent_id);
sent_data.extend(data);
let _ = self.in_stream.writable().await;
match self.in_stream.try_write(sent_data.as_slice()) {
Err(e) => { eprintln!("error while sending response {:?}", e); },
_ => { }
}
}
async fn proxy_start(&mut self) -> Result<(),String> {
println!("[{}] starting proxy mode", self);
// let stream = TcpStream::connect(&self.server.as_ref().unwrap().redirect_ip.as_ref().unwrap()).await
let stream = TcpStream::connect("play.hypixel.net").await
.map_err(|_| "failed to connect to remote host".to_string())?;
let mut new_handshake = self.handshake.as_ref().unwrap().clone();
new_handshake.server_address = "play.hypixel.net".to_string();
new_handshake.server_port = 25565;
let handshake_packet = new_handshake.to_packet();
let mut packet = Vec::new();
packet.extend(varint_write(handshake_packet.len() as i32));
packet.extend(handshake_packet);
self.out_stream = Some(stream);
let _ = self.out_stream.as_ref().unwrap().writable().await;
let _ = self.out_stream.as_ref().unwrap().try_write(packet.as_slice());
println!("[{}] sent handshake to server", self);
Ok(())
}
}

View File

@ -1,5 +1,10 @@
use crate::minecraft::client::client::Client; use crate::minecraft::client::client::Client;
use diesel::prelude::*;
use diesel::result::Error::NotFound;
use crate::schema;
use crate::models;
use json::object; use json::object;
use crate::minecraft::varint::varint_write; use crate::minecraft::varint::varint_write;
@ -16,27 +21,47 @@ impl Client {
"max": 420, "max": 420,
"online": 69 "online": 69
}, },
"description": "§c".to_string() + &self.handshake.as_ref().unwrap().server_address.clone() + "§r:§a" + &self.handshake.as_ref().unwrap().server_port.to_string(), "description": self.get_description().await,
}; };
if packet_id == 0 { if packet_id == 0 {
let mut response : Vec<u8> = vec![]; let mut response : Vec<u8> = vec![];
let response_json = json::stringify(status_response); let response_json = json::stringify(status_response);
let mut response_json_len = varint_write(response_json.len() as i32); let mut response_json_len = varint_write(response_json.len() as i32);
let mut response_packet_id = varint_write(0);
response.append(&mut response_packet_id);
response.append(&mut response_json_len); response.append(&mut response_json_len);
response.append(&mut response_json.as_bytes().into()); response.append(&mut response_json.as_bytes().into());
println!("[{}] sent status response to client", self); println!("[{}] sent status response to client", self);
self.send_packet(response).await; self.send_packet(0x0, response).await;
} else if packet_id == 1 { } else if packet_id == 1 {
let mut response = varint_write(1); let mut response = vec![];
response.extend(packet.drain(..)); response.extend(packet.drain(..));
self.send_packet(response).await; self.send_packet(0x1, response).await;
println!("[{}] sent ping response to client", self); println!("[{}] sent ping response to client", self);
} }
Ok(()) Ok(())
} }
pub async fn get_description(&self) -> String {
if let None = self.server {
"How did you send a status request without a handshake ???".to_string();
} }
match self.server.as_ref().unwrap() {
Ok(server) => {
let login = match server.last_login {
Some(date) => { format!("{:?}", date) },
None => { "never".to_string() }
};
format!("§fServer is {}\nlast login: {}", server.status, login)
},
Err(NotFound) => { "§cThis server does not exist".to_string() },
Err(error) => { format!("§cAn error occured while retreiving the information:\n{}", error) }
}
}
}

View File

@ -23,6 +23,9 @@ pub async fn process_mc_socket(stream: TcpStream, pool: DbPool) -> Result<(), St
} }
} }
} }
if client.closed {
break;
}
} }
Ok(()) Ok(())
} }

View File

@ -6,9 +6,10 @@ use serde::Deserialize;
use crate::status::ServerStatus; use crate::status::ServerStatus;
use crate::schema;
#[derive(Queryable, Selectable, Debug)] #[derive(Queryable, Selectable, Debug)]
#[diesel(table_name = dockermcmgr::schema::servers)] #[diesel(table_name = schema::servers)]
#[diesel(check_for_backend(diesel::pg::Pg))] #[diesel(check_for_backend(diesel::pg::Pg))]
pub struct Servers { pub struct Servers {
pub id: i64, pub id: i64,
@ -17,17 +18,17 @@ pub struct Servers {
pub last_login: Option<SystemTime>, pub last_login: Option<SystemTime>,
pub container_id: Option<String>, pub container_id: Option<String>,
pub status: ServerStatus, pub status: ServerStatus,
pub redirect_ip: Option<i32> pub redirect_ip: Option<String>
} }
#[derive(Deserialize, Insertable)] #[derive(Deserialize, Insertable)]
#[diesel(table_name = dockermcmgr::schema::servers)] #[diesel(table_name = schema::servers)]
pub struct CreateServer<'a> { pub struct CreateServer<'a> {
pub name: &'a str, pub name: &'a str,
pub volume_path: &'a str, pub volume_path: &'a str,
pub last_login: Option<SystemTime>, pub last_login: Option<SystemTime>,
pub container_id: Option<&'a str>, pub container_id: Option<&'a str>,
pub status: ServerStatus, pub status: ServerStatus,
pub redirect_ip: Option<i32> pub redirect_ip: Option<&'a str>
} }

View File

@ -11,6 +11,8 @@ diesel::table! {
#[max_length = 30] #[max_length = 30]
container_id -> Nullable<Varchar>, container_id -> Nullable<Varchar>,
status -> Int2, status -> Int2,
redirect_ip -> Nullable<Int4>, is_default -> Bool,
#[max_length = 50]
redirect_ip -> Nullable<Varchar>,
} }
} }

View File

@ -4,6 +4,7 @@ use diesel::expression::AsExpression;
use diesel::pg::Pg; use diesel::pg::Pg;
use diesel::serialize::{self, ToSql, Output}; use diesel::serialize::{self, ToSql, Output};
use diesel::sql_types::SmallInt; use diesel::sql_types::SmallInt;
use std::fmt::{self, write};
use std::io::Write; use std::io::Write;
#[derive(Debug, Clone, Copy, PartialEq, Eq, AsExpression, FromSqlRow, serde::Serialize, serde::Deserialize)] #[derive(Debug, Clone, Copy, PartialEq, Eq, AsExpression, FromSqlRow, serde::Serialize, serde::Deserialize)]
@ -14,7 +15,7 @@ pub enum ServerStatus {
Stopping = 2, Stopping = 2,
Starting = 3, Starting = 3,
Running = 4, Running = 4,
Unknown = 5, Unknown = 5
} }
impl ToSql<SmallInt, Pg> for ServerStatus { impl ToSql<SmallInt, Pg> for ServerStatus {
@ -42,3 +43,18 @@ impl FromSql<SmallInt, Pg> for ServerStatus {
} }
} }
} }
impl fmt::Display for ServerStatus {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let str_status = match self {
ServerStatus::Archived => "archieved",
ServerStatus::Stopped => "stopped",
ServerStatus::Stopping => "stopping",
ServerStatus::Starting => "starting",
ServerStatus::Running => "running",
ServerStatus::Unknown => "(error : unknown status)"
};
write!(f, "{}", str_status)
}
}