server from db working
This commit is contained in:
@ -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") );
|
||||||
|
|
||||||
|
|||||||
13
src/main.rs
13
src/main.rs
@ -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() => {
|
||||||
|
|||||||
@ -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(())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -2,3 +2,4 @@ pub mod client;
|
|||||||
mod buffer;
|
mod buffer;
|
||||||
mod status;
|
mod status;
|
||||||
mod login;
|
mod login;
|
||||||
|
mod packet;
|
||||||
|
|||||||
80
src/minecraft/client/packet.rs
Normal file
80
src/minecraft/client/packet.rs
Normal 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(())
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -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) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@ -23,6 +23,9 @@ pub async fn process_mc_socket(stream: TcpStream, pool: DbPool) -> Result<(), St
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if client.closed {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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>
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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>,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -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)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user