- add: mouse cmd packet handling

- add: change map handler
- update: logout handler to send the reply packet
- update: character id list change from u8 to u32
This commit is contained in:
2025-01-28 22:48:11 -05:00
parent 09c10efb46
commit b5e16f0bef
7 changed files with 124 additions and 25 deletions

View File

@@ -25,6 +25,7 @@ utils = { path = "../utils" }
warp = "0.3.7" warp = "0.3.7"
dashmap = "6.1.0" dashmap = "6.1.0"
uuid = { version = "1.11.0", features = ["v4"] } uuid = { version = "1.11.0", features = ["v4"] }
chrono = "0.4.39"
[build-dependencies] [build-dependencies]
tonic-build = "0.12.3" tonic-build = "0.12.3"

View File

@@ -5,7 +5,7 @@ pub struct ConnectionState {
pub user_id: Option<i32>, pub user_id: Option<i32>,
pub session_id: Option<String>, pub session_id: Option<String>,
pub character_id: Option<i8>, pub character_id: Option<i8>,
pub character_list: Option<Vec<u8>>, pub character_list: Option<Vec<u32>>,
pub additional_data: HashMap<String, String>, // Flexible data storage pub additional_data: HashMap<String, String>, // Flexible data storage
} }

View File

@@ -1,15 +1,18 @@
use crate::auth_client::AuthClient; use crate::auth_client::AuthClient;
use utils::null_string::NullTerminatedString; use crate::connection_service::ConnectionService;
use crate::packet::{send_packet, Packet, PacketPayload}; use crate::packet::{send_packet, Packet, PacketPayload};
use crate::packet_type::PacketType; use crate::packet_type::PacketType;
use crate::packets::cli_accept_req::CliAcceptReq; use crate::packets::cli_accept_req::CliAcceptReq;
use crate::packets::cli_channel_list_req::CliChannelListReq; use crate::packets::cli_channel_list_req::CliChannelListReq;
use crate::packets::cli_join_server_token_req::CliJoinServerTokenReq; use crate::packets::cli_join_server_token_req::CliJoinServerTokenReq;
use crate::packets::cli_login_token_req::CliLoginTokenReq; use crate::packets::cli_login_token_req::CliLoginTokenReq;
use crate::packets::cli_logout_req::CliLogoutReq;
use crate::packets::cli_srv_select_req::CliSrvSelectReq; use crate::packets::cli_srv_select_req::CliSrvSelectReq;
use crate::packets::srv_accept_reply::SrvAcceptReply; use crate::packets::srv_accept_reply::SrvAcceptReply;
use crate::packets::srv_channel_list_reply::{ChannelInfo, SrvChannelListReply}; use crate::packets::srv_channel_list_reply::{ChannelInfo, SrvChannelListReply};
use crate::packets::srv_join_server_reply::SrvJoinServerReply;
use crate::packets::srv_login_reply::{ServerInfo, SrvLoginReply}; use crate::packets::srv_login_reply::{ServerInfo, SrvLoginReply};
use crate::packets::srv_logout_reply::SrvLogoutReply;
use crate::packets::srv_srv_select_reply::SrvSrvSelectReply; use crate::packets::srv_srv_select_reply::SrvSrvSelectReply;
use crate::packets::*; use crate::packets::*;
use std::collections::HashMap; use std::collections::HashMap;
@@ -17,15 +20,13 @@ use std::env;
use std::error::Error; use std::error::Error;
use std::net::SocketAddr; use std::net::SocketAddr;
use std::sync::Arc; use std::sync::Arc;
use tokio::io::AsyncWriteExt;
use tokio::net::TcpStream; use tokio::net::TcpStream;
use tokio::sync::Mutex; use tokio::sync::Mutex;
use tonic::{Code, Status}; use tonic::{Code, Status};
use tracing::{debug, error, info, warn}; use tracing::{debug, error, info, warn};
use utils::null_string::NullTerminatedString;
use utils::service_discovery; use utils::service_discovery;
use crate::connection_service::ConnectionService;
use crate::packets::cli_logout_req::CliLogoutReq;
use crate::packets::srv_join_server_reply::SrvJoinServerReply;
pub(crate) async fn handle_alive_req(stream: &mut TcpStream, packet: Packet, auth_client: Arc<Mutex<AuthClient>>, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> { pub(crate) async fn handle_alive_req(stream: &mut TcpStream, packet: Packet, auth_client: Arc<Mutex<AuthClient>>, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> {
if let Some(mut state) = connection_service.get_connection(&connection_id) { if let Some(mut state) = connection_service.get_connection(&connection_id) {
@@ -78,11 +79,15 @@ pub(crate) async fn handle_join_server_req(stream: &mut TcpStream, packet: Packe
} }
pub(crate) async fn handle_logout_req(stream: &mut TcpStream, packet: Packet, auth_client: Arc<Mutex<AuthClient>>, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> { pub(crate) async fn handle_logout_req(stream: &mut TcpStream, packet: Packet, auth_client: Arc<Mutex<AuthClient>>, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> {
let request = CliLogoutReq::decode(packet.payload.as_slice());
if let Some(mut state) = connection_service.get_connection(&connection_id) { if let Some(mut state) = connection_service.get_connection(&connection_id) {
let session_id = state.session_id.clone().unwrap(); let session_id = state.session_id.clone().unwrap();
let mut auth_client = auth_client.lock().await; let mut auth_client = auth_client.lock().await;
auth_client.logout(&session_id).await?; auth_client.logout(&session_id).await?;
let data = SrvLogoutReply { wait_time: 1 };
let response_packet = Packet::new(PacketType::PakwcLogoutReply, &data)?;
send_packet(stream, &response_packet).await?;
stream.shutdown().await?;
Ok(()) Ok(())
} else { } else {
Err("Unable to find connection state".into()) Err("Unable to find connection state".into())
@@ -134,7 +139,7 @@ pub(crate) async fn handle_login_req(stream: &mut TcpStream, packet: Packet, aut
} else { } else {
name = format!(" {}", name); name = format!(" {}", name);
} }
server_info.push(ServerInfo { test: u8::from(is_test), name: NullTerminatedString::new(&name), id}); server_info.push(ServerInfo { test: u8::from(is_test), name: NullTerminatedString::new(&name), id });
id = id + 1; id = id + 1;
} }
debug!("Server info: {:?}", server_info); debug!("Server info: {:?}", server_info);

View File

@@ -23,7 +23,7 @@ use tonic::{Code, Status};
use tracing::{debug, error, info, warn}; use tracing::{debug, error, info, warn};
use utils::null_string::NullTerminatedString; use utils::null_string::NullTerminatedString;
fn convert_slot(slot: i32) -> srv_char_list_reply::EquippedPosition { pub(crate) fn convert_slot(slot: i32) -> srv_char_list_reply::EquippedPosition {
match enums::EquippedPosition::from_i32(slot) { match enums::EquippedPosition::from_i32(slot) {
Some(enums::EquippedPosition::Goggles) => srv_char_list_reply::EquippedPosition::Googles, Some(enums::EquippedPosition::Goggles) => srv_char_list_reply::EquippedPosition::Googles,
Some(enums::EquippedPosition::Helmet) => srv_char_list_reply::EquippedPosition::Helmet, Some(enums::EquippedPosition::Helmet) => srv_char_list_reply::EquippedPosition::Helmet,
@@ -53,12 +53,12 @@ pub(crate) async fn handle_char_list_req(stream: &mut TcpStream, packet: Packet,
let mut character_client = character_client.lock().await; let mut character_client = character_client.lock().await;
let character_list = character_client.get_character_list(&user_id.to_string()).await?; let character_list = character_client.get_character_list(&user_id.to_string()).await?;
let mut characters = vec![]; let mut characters = vec![];
let mut character_id_list: Vec<u8> = Vec::new(); let mut character_id_list: Vec<u32> = Vec::new();
for character in character_list.characters { for character in character_list.characters {
let mut item_list: [EquippedItem; (MAX_VISIBLE_ITEMS as usize)] = core::array::from_fn(|i| EquippedItem::default()); let mut item_list: [EquippedItem; (MAX_VISIBLE_ITEMS as usize)] = core::array::from_fn(|i| EquippedItem::default());
for item in character.items { for item in character.items {
if item.slot < 10 { if item.slot < MAX_VISIBLE_ITEMS as i32 {
let slot = convert_slot(item.slot) as usize; let slot = convert_slot(item.slot) as usize;
item_list[slot] = EquippedItem { item_list[slot] = EquippedItem {
id: item.item_id as u16, id: item.item_id as u16,
@@ -133,7 +133,7 @@ pub(crate) async fn handle_delete_char_req(stream: &mut TcpStream, packet: Packe
let mut user_id = 0; let mut user_id = 0;
let session_id; let session_id;
let mut character_id_list: Vec<u8> = Vec::new(); let mut character_id_list: Vec<u32> = Vec::new();
if let Some(mut state) = connection_service.get_connection(&connection_id) { if let Some(mut state) = connection_service.get_connection(&connection_id) {
user_id = state.user_id.expect("Missing user id in connection state"); user_id = state.user_id.expect("Missing user id in connection state");
@@ -165,7 +165,7 @@ pub(crate) async fn handle_select_char_req(stream: &mut TcpStream, packet: Packe
debug!("{:?}", request); debug!("{:?}", request);
let mut user_id = 0; let mut user_id = 0;
let mut character_id_list: Vec<u8> = Vec::new(); let mut character_id_list: Vec<u32> = Vec::new();
if let Some(mut state) = connection_service.get_connection_mut(&connection_id) { if let Some(mut state) = connection_service.get_connection_mut(&connection_id) {
user_id = state.user_id.expect("Missing user id in connection state"); user_id = state.user_id.expect("Missing user id in connection state");
character_id_list = state.character_list.clone().expect("Missing character id list"); character_id_list = state.character_list.clone().expect("Missing character id list");
@@ -182,7 +182,7 @@ pub(crate) async fn handle_select_char_req(stream: &mut TcpStream, packet: Packe
send_packet(stream, &response_packet).await?; send_packet(stream, &response_packet).await?;
let mut character_client = character_client.lock().await; let mut character_client = character_client.lock().await;
let character_data = character_client.get_character(&user_id.to_string(), character_id_list[request.char_id as usize]).await?; let character_data = character_client.get_character(&user_id.to_string(), character_id_list[request.char_id as usize] as u8).await?;
let character = character_data.character.unwrap_or_default(); let character = character_data.character.unwrap_or_default();

View File

@@ -1,2 +1,3 @@
pub mod auth; pub mod auth;
pub mod character; pub mod character;
pub mod world;

View File

@@ -0,0 +1,88 @@
use crate::character_client::CharacterClient;
use crate::connection_service::ConnectionService;
use crate::packet::{send_packet, Packet, PacketPayload};
use crate::packet_type::PacketType;
use chrono::{Local, Timelike};
use std::error::Error;
use std::sync::Arc;
use tokio::net::TcpStream;
use tokio::sync::Mutex;
use tracing::debug;
fn distance(x1: f64, y1: f64, x2: f64, y2: f64) -> u16 {
let dist = ((x2 - x1).powi(2) + (y2 - y1).powi(2)).sqrt();
dist.round() as u16
}
pub(crate) async fn handle_change_map_req(stream: &mut TcpStream, packet: Packet, character_client: Arc<Mutex<CharacterClient>>, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> {
use crate::packets::cli_change_map_req::*;
use crate::packets::srv_change_map_reply::*;
let request = CliChangeMapReq::decode(packet.payload.as_slice())?;
debug!("{:?}", request);
let mut user_id = 0;
let mut char_id = 0;
let mut character_id_list: Vec<u32> = Vec::new();
let session_id;
if let Some(mut state) = connection_service.get_connection(&connection_id) {
user_id = state.user_id.expect("Missing user id in connection state");
session_id = state.session_id.expect("Missing session id in connection state");
char_id = state.character_id.expect("Missing character id in connection state");
character_id_list = state.character_list.clone().expect("Missing character id list");
}
let mut character_client = character_client.lock().await;
let character_data = character_client.get_character(&user_id.to_string(), character_id_list[char_id as usize] as u8).await?;
let character = character_data.character.unwrap_or_default();
let stats = character.stats.unwrap();
let now = Local::now();
let time_as_u16 = (now.hour() * 100 + now.minute()) as u16;
let data = SrvChangeMapReply {
object_index: character_id_list[char_id as usize] as u16,
hp: stats.hp as u16,
mp: stats.mp as u16,
xp: stats.xp as u16,
penalize_xp: stats.penalty_xp as u16,
craft_rate: 0,
update_time: 0,
world_rate: 1,
town_rate: 1,
item_rate: [0u8; 11],
flags: 1,
world_time: time_as_u16,
team_number: 10,
};
let response_packet = Packet::new(PacketType::PakwcChangeMapReply, &data)?;
send_packet(stream, &response_packet).await?;
Ok(())
}
pub(crate) async fn handle_mouse_cmd_req(stream: &mut TcpStream, packet: Packet, connection_service: Arc<ConnectionService>, connection_id: String) -> Result<(), Box<dyn Error + Send + Sync>> {
use crate::packets::cli_mouse_cmd::*;
use crate::packets::srv_mouse_cmd::*;
let request = CliMouseCmd::decode(packet.payload.as_slice())?;
debug!("{:?}", request);
let mut char_id = 0;
let mut character_id_list: Vec<u32> = Vec::new();
if let Some(mut state) = connection_service.get_connection(&connection_id) {
char_id = state.character_id.expect("Missing character id in connection state");
character_id_list = state.character_list.clone().expect("Missing character id list");
}
let data = SrvMouseCmd {
id: character_id_list[char_id as usize] as u16,
target_id: request.target_id,
distance: distance(520000 as f64, 520000 as f64, request.x as f64, request.y as f64),
x: request.x,
y: request.y,
z: request.z,
};
let response_packet = Packet::new(PacketType::PakwcMouseCmd, &data)?;
send_packet(stream, &response_packet).await?;
Ok(())
}

View File

@@ -1,5 +1,9 @@
use crate::auth_client::AuthClient; use crate::auth_client::AuthClient;
use crate::bufferpool::BufferPool;
use crate::character_client::CharacterClient;
use crate::connection_service::ConnectionService;
use crate::handlers::*; use crate::handlers::*;
use crate::metrics::{ACTIVE_CONNECTIONS, PACKETS_RECEIVED};
use crate::packet::Packet; use crate::packet::Packet;
use crate::packet_type::PacketType; use crate::packet_type::PacketType;
use std::error::Error; use std::error::Error;
@@ -8,10 +12,6 @@ use tokio::io::AsyncReadExt;
use tokio::net::TcpStream; use tokio::net::TcpStream;
use tokio::sync::Mutex; use tokio::sync::Mutex;
use tracing::{debug, warn}; use tracing::{debug, warn};
use crate::bufferpool::BufferPool;
use crate::character_client::CharacterClient;
use crate::connection_service::ConnectionService;
use crate::metrics::{ACTIVE_CONNECTIONS, PACKETS_RECEIVED};
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub struct PacketRouter { pub struct PacketRouter {
@@ -32,7 +32,7 @@ impl PacketRouter {
} }
let packet_size = u16::from_le_bytes(buffer[0..2].try_into()?) as usize; let packet_size = u16::from_le_bytes(buffer[0..2].try_into()?) as usize;
if packet_size > 6 { if packet_size > 6 {
let mut body_handle = stream.take((packet_size-6) as u64); let mut body_handle = stream.take((packet_size - 6) as u64);
let n = body_handle.read(&mut buffer[6..]).await?; let n = body_handle.read(&mut buffer[6..]).await?;
if n == 0 { if n == 0 {
break; // Connection closed break; // Connection closed
@@ -86,12 +86,16 @@ impl PacketRouter {
PacketType::PakcsDeleteCharReq => character::handle_delete_char_req(stream, packet, self.character_client.clone(), self.connection_service.clone(), connection_id).await, PacketType::PakcsDeleteCharReq => character::handle_delete_char_req(stream, packet, self.character_client.clone(), self.connection_service.clone(), connection_id).await,
PacketType::PakcsSelectCharReq => character::handle_select_char_req(stream, packet, self.character_client.clone(), self.connection_service.clone(), connection_id).await, PacketType::PakcsSelectCharReq => character::handle_select_char_req(stream, packet, self.character_client.clone(), self.connection_service.clone(), connection_id).await,
// World Packets
PacketType::PakcsChangeMapReq => world::handle_change_map_req(stream, packet, self.character_client.clone(), self.connection_service.clone(), connection_id).await,
PacketType::PakcsMouseCmd => world::handle_mouse_cmd_req(stream, packet, self.connection_service.clone(), connection_id).await,
// 1 => chat::handle_chat(packet).await?, // 1 => chat::handle_chat(packet).await?,
// 2 => movement::handle_movement(packet).await?, // 2 => movement::handle_movement(packet).await?,
_ => { _ => {
warn!("Unhandled packet type: {:?}", packet.packet_type); warn!("Unhandled packet type: {:?}", packet.packet_type);
Ok(()) Ok(())
}, }
} }
} }
} }