cleaned up code, condensed code and fixed a couple of things
This commit is contained in:
parent
40ca10651f
commit
38bcad37bd
10 changed files with 178 additions and 95 deletions
37
src/main.rs
37
src/main.rs
|
|
@ -10,12 +10,14 @@ use once_cell::sync::Lazy;
|
||||||
use rc4::{KeyInit, Rc4, StreamCipher};
|
use rc4::{KeyInit, Rc4, StreamCipher};
|
||||||
use rc4::consts::U5;
|
use rc4::consts::U5;
|
||||||
use simplelog::{ColorChoice, CombinedLogger, Config, LevelFilter, TerminalMode, TermLogger, WriteLogger};
|
use simplelog::{ColorChoice, CombinedLogger, Config, LevelFilter, TerminalMode, TermLogger, WriteLogger};
|
||||||
|
use crate::protocols::auth;
|
||||||
|
use crate::protocols::server::RMCProtocolServer;
|
||||||
use crate::prudp::socket::{Socket, SocketData};
|
use crate::prudp::socket::{Socket, SocketData};
|
||||||
use crate::prudp::packet::{PRUDPPacket, VirtualPort};
|
use crate::prudp::packet::{PRUDPPacket, VirtualPort};
|
||||||
use crate::prudp::router::Router;
|
use crate::prudp::router::Router;
|
||||||
use crate::rmc::message::RMCMessage;
|
use crate::rmc::message::RMCMessage;
|
||||||
use crate::rmc::response::{RMCResponse, RMCResponseResult, send_response};
|
use crate::rmc::response::{RMCResponse, RMCResponseResult, send_response};
|
||||||
use crate::rmc::response::ErrorCode::Core_InvalidIndex;
|
use crate::rmc::response::ErrorCode::{Core_InvalidIndex, Core_NotImplemented};
|
||||||
|
|
||||||
mod endianness;
|
mod endianness;
|
||||||
mod prudp;
|
mod prudp;
|
||||||
|
|
@ -67,6 +69,10 @@ async fn start_servers(){
|
||||||
info!("setting up endpoints");
|
info!("setting up endpoints");
|
||||||
|
|
||||||
// dont assign it to the name _ as that will make it drop right here and now
|
// dont assign it to the name _ as that will make it drop right here and now
|
||||||
|
let rmcserver = RMCProtocolServer::new(Box::new([
|
||||||
|
Box::new(auth::protocol)
|
||||||
|
]));
|
||||||
|
|
||||||
let mut _socket =
|
let mut _socket =
|
||||||
Socket::new(
|
Socket::new(
|
||||||
auth_server_router.clone(),
|
auth_server_router.clone(),
|
||||||
|
|
@ -87,30 +93,9 @@ async fn start_servers(){
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}),
|
}),
|
||||||
Box::new(|p, socket, connection|{
|
Box::new(move |packet, socket, connection|{
|
||||||
Box::pin(
|
let rmcserver = rmcserver.clone();
|
||||||
async move {
|
Box::pin(async move { rmcserver.process_message(packet, &socket, connection).await; })
|
||||||
println!("{:?}",p);
|
|
||||||
let Ok(rmc) = RMCMessage::new(&mut Cursor::new(&p.payload)) else {
|
|
||||||
error!("error reading rmc message");
|
|
||||||
return;
|
|
||||||
};
|
|
||||||
|
|
||||||
println!("recieved rmc message: {{ protocol: {}, method: {}}}", rmc.protocol_id, rmc.method_id);
|
|
||||||
|
|
||||||
if let Some(response) = protocols::auth::try_process_via_protocol(&rmc){
|
|
||||||
send_response(&p, &socket, connection, response).await;
|
|
||||||
}
|
|
||||||
|
|
||||||
send_response(&p, &socket, connection, RMCResponse{
|
|
||||||
protocol_id: rmc.protocol_id as u8,
|
|
||||||
response_result: RMCResponseResult::Error {
|
|
||||||
call_id: rmc.call_id,
|
|
||||||
error_code: Core_InvalidIndex
|
|
||||||
}
|
|
||||||
}).await;
|
|
||||||
}
|
|
||||||
)
|
|
||||||
})
|
})
|
||||||
).await.expect("unable to create socket");
|
).await.expect("unable to create socket");
|
||||||
|
|
||||||
|
|
@ -155,7 +140,7 @@ mod test{
|
||||||
|
|
||||||
let mut a = Cursor::new(&rmc_packet.rest_of_data);
|
let mut a = Cursor::new(&rmc_packet.rest_of_data);
|
||||||
|
|
||||||
let pid = rmc::structures::string::read(&mut a).expect("unable to read pid");
|
//let pid = rmc::structures::string::read(&mut a).expect("unable to read pid");
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test]
|
#[tokio::test]
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,8 @@ use std::io::Cursor;
|
||||||
use log::{error, info};
|
use log::{error, info};
|
||||||
use crate::rmc::message::RMCMessage;
|
use crate::rmc::message::RMCMessage;
|
||||||
use crate::rmc::response::{ErrorCode, RMCResponse, RMCResponseResult};
|
use crate::rmc::response::{ErrorCode, RMCResponse, RMCResponseResult};
|
||||||
use crate::rmc::structures::{string, any};
|
use crate::rmc::structures::{string, any, RmcSerialize};
|
||||||
|
use crate::rmc::structures::any::Any;
|
||||||
|
|
||||||
pub fn login_ex(name: &str) -> RMCResponseResult{
|
pub fn login_ex(name: &str) -> RMCResponseResult{
|
||||||
// todo: figure out how the AuthenticationInfo struct works, parse it and validate login info
|
// todo: figure out how the AuthenticationInfo struct works, parse it and validate login info
|
||||||
|
|
@ -14,12 +15,12 @@ pub fn login_ex(name: &str) -> RMCResponseResult{
|
||||||
pub fn login_ex_raw_params(rmcmessage: &RMCMessage) -> RMCResponseResult{
|
pub fn login_ex_raw_params(rmcmessage: &RMCMessage) -> RMCResponseResult{
|
||||||
let mut reader = Cursor::new(&rmcmessage.rest_of_data);
|
let mut reader = Cursor::new(&rmcmessage.rest_of_data);
|
||||||
|
|
||||||
let Ok(str) = string::read(&mut reader) else {
|
let Ok(str) = String::deserialize(&mut reader) else {
|
||||||
error!("error reading packet");
|
error!("error reading packet");
|
||||||
return rmcmessage.error_result_with_code(ErrorCode::Core_InvalidArgument);
|
return rmcmessage.error_result_with_code(ErrorCode::Core_InvalidArgument);
|
||||||
};
|
};
|
||||||
|
|
||||||
let Ok(any) = any::read(&mut reader) else {
|
let Ok(any) = Any::deserialize(&mut reader) else {
|
||||||
error!("error reading packet");
|
error!("error reading packet");
|
||||||
return rmcmessage.error_result_with_code(ErrorCode::Core_InvalidArgument);
|
return rmcmessage.error_result_with_code(ErrorCode::Core_InvalidArgument);
|
||||||
};
|
};
|
||||||
|
|
@ -35,5 +36,5 @@ pub fn login_ex_raw_params(rmcmessage: &RMCMessage) -> RMCResponseResult{
|
||||||
}
|
}
|
||||||
|
|
||||||
//login_ex(&str)
|
//login_ex(&str)
|
||||||
rmcmessage.error_result_with_code(ErrorCode::Core_AccessDenied)
|
rmcmessage.error_result_with_code(ErrorCode::Authentication_UnderMaintenance)
|
||||||
}
|
}
|
||||||
|
|
@ -1,25 +1,14 @@
|
||||||
mod method_login_ex;
|
mod method_login_ex;
|
||||||
|
|
||||||
use log::{error, info};
|
use log::{error, info};
|
||||||
|
use crate::define_protocol;
|
||||||
use crate::protocols::auth::method_login_ex::{login_ex, login_ex_raw_params};
|
use crate::protocols::auth::method_login_ex::{login_ex, login_ex_raw_params};
|
||||||
use crate::rmc::message::RMCMessage;
|
use crate::rmc::message::RMCMessage;
|
||||||
use crate::rmc::response::{ErrorCode, RMCResponse, RMCResponseResult};
|
use crate::rmc::response::{ErrorCode, RMCResponse, RMCResponseResult};
|
||||||
|
|
||||||
pub fn try_process_via_protocol(rmcmessage: &RMCMessage) -> Option<RMCResponse>{
|
|
||||||
if rmcmessage.protocol_id != 10{
|
define_protocol!{
|
||||||
return None;
|
10 => {
|
||||||
|
0x02 => login_ex_raw_params
|
||||||
}
|
}
|
||||||
|
|
||||||
let response_result = match rmcmessage.method_id{
|
|
||||||
0x02 => login_ex_raw_params(rmcmessage),
|
|
||||||
_ => {
|
|
||||||
error!("invalid method id sent to ticket-granting protocol: {:?}", rmcmessage.method_id);
|
|
||||||
rmcmessage.error_result_with_code(ErrorCode::Core_Exception)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Some(RMCResponse{
|
|
||||||
protocol_id: 10,
|
|
||||||
response_result
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
@ -1 +1,27 @@
|
||||||
pub mod auth;
|
pub mod auth;
|
||||||
|
pub mod server;
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! define_protocol {
|
||||||
|
($id:literal => {$($func_id:literal => $func:path),*} ) => {
|
||||||
|
pub fn protocol(rmcmessage: &RMCMessage) -> Option<RMCResponse>{
|
||||||
|
if rmcmessage.protocol_id != $id{
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let response_result = match rmcmessage.method_id{
|
||||||
|
$(
|
||||||
|
$func_id => $func(rmcmessage),
|
||||||
|
),*
|
||||||
|
_ => {
|
||||||
|
error!("invalid method id sent to protocol {}: {:?}", $id, rmcmessage.method_id);
|
||||||
|
rmcmessage.error_result_with_code(ErrorCode::Core_NotImplemented)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Some(RMCResponse{
|
||||||
|
protocol_id: $id,
|
||||||
|
response_result
|
||||||
|
})
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
42
src/protocols/server.rs
Normal file
42
src/protocols/server.rs
Normal file
|
|
@ -0,0 +1,42 @@
|
||||||
|
use std::io::Cursor;
|
||||||
|
use std::sync::Arc;
|
||||||
|
use log::error;
|
||||||
|
use crate::prudp::packet::PRUDPPacket;
|
||||||
|
use crate::prudp::socket::{ConnectionData, SocketData};
|
||||||
|
use crate::rmc::message::RMCMessage;
|
||||||
|
use crate::rmc::response::{RMCResponse, RMCResponseResult, send_response};
|
||||||
|
use crate::rmc::response::ErrorCode::Core_NotImplemented;
|
||||||
|
|
||||||
|
type ContainedProtocolList = Box<[Box<dyn Fn(&RMCMessage) -> Option<RMCResponse> + Send + Sync>]>;
|
||||||
|
|
||||||
|
pub struct RMCProtocolServer(ContainedProtocolList);
|
||||||
|
|
||||||
|
impl RMCProtocolServer{
|
||||||
|
pub fn new(protocols: ContainedProtocolList) -> Arc<Self>{
|
||||||
|
Arc::new(Self(protocols))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn process_message(&self, packet: PRUDPPacket, socket: &SocketData, connection: &mut ConnectionData){
|
||||||
|
let Ok(rmc) = RMCMessage::new(&mut Cursor::new(&packet.payload)) else {
|
||||||
|
error!("error reading rmc message");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
println!("recieved rmc message: {{ protocol: {}, method: {}}}", rmc.protocol_id, rmc.method_id);
|
||||||
|
|
||||||
|
for proto in &self.0 {
|
||||||
|
if let Some(response) = proto(&rmc) {
|
||||||
|
send_response(&packet, &socket, connection, response).await;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
send_response(&packet, &socket, connection, RMCResponse{
|
||||||
|
protocol_id: rmc.protocol_id as u8,
|
||||||
|
response_result: RMCResponseResult::Error {
|
||||||
|
call_id: rmc.call_id,
|
||||||
|
error_code: Core_NotImplemented
|
||||||
|
}
|
||||||
|
}).await;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -54,8 +54,8 @@ pub struct ActiveConnectionData {
|
||||||
pub reliable_server_counter: u16,
|
pub reliable_server_counter: u16,
|
||||||
pub reliable_client_queue: VecDeque<PRUDPPacket>,
|
pub reliable_client_queue: VecDeque<PRUDPPacket>,
|
||||||
pub connection_data_channel: Sender<Vec<u8>>,
|
pub connection_data_channel: Sender<Vec<u8>>,
|
||||||
pub server_encryption: Box<dyn StreamCipher + Send + Sync>,
|
server_encryption: Box<dyn StreamCipher + Send + Sync>,
|
||||||
pub client_decryption: Box<dyn StreamCipher + Send + Sync>,
|
client_decryption: Box<dyn StreamCipher + Send + Sync>,
|
||||||
pub server_session_id: u8,
|
pub server_session_id: u8,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -383,6 +383,37 @@ impl SocketData {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ConnectionData{
|
||||||
|
pub async fn finish_and_send_packet_to(&mut self, socket: &SocketData, mut packet: PRUDPPacket){
|
||||||
|
if (packet.header.types_and_flags.get_flags() & RELIABLE) != 0{
|
||||||
|
let Some(active_connection) = self.active_connection_data.as_mut() else {
|
||||||
|
error!("tried to send a secure packet to an inactive connection");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
packet.header.sequence_id = active_connection.reliable_server_counter;
|
||||||
|
active_connection.reliable_server_counter += 1;
|
||||||
|
|
||||||
|
active_connection.server_encryption.apply_keystream(&mut packet.payload);
|
||||||
|
}
|
||||||
|
|
||||||
|
packet.header.source_port = socket.virtual_port;
|
||||||
|
packet.header.destination_port = self.sock_addr.virtual_port;
|
||||||
|
|
||||||
|
packet.set_sizes();
|
||||||
|
|
||||||
|
packet.calculate_and_assign_signature(socket.access_key, None, Some(self.server_signature));
|
||||||
|
|
||||||
|
let mut vec = Vec::new();
|
||||||
|
|
||||||
|
packet.write_to(&mut vec).expect("somehow failed to convert backet to bytes");
|
||||||
|
|
||||||
|
if let Err(e) = socket.socket.send_to(&vec, self.sock_addr.regular_socket_addr).await{
|
||||||
|
error!("unable to send packet to destination: {}", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use std::io::Cursor;
|
use std::io::Cursor;
|
||||||
|
|
|
||||||
|
|
@ -49,7 +49,7 @@ pub fn generate_response(protocol_id: u8, response: RMCResponseResult) -> io::Re
|
||||||
let u32_size: u32 = size as _;
|
let u32_size: u32 = size as _;
|
||||||
|
|
||||||
data_out.write_all(bytes_of(&u32_size))?;
|
data_out.write_all(bytes_of(&u32_size))?;
|
||||||
data_out.push(protocol_id | 0x80);
|
data_out.push(protocol_id);
|
||||||
|
|
||||||
match response{
|
match response{
|
||||||
RMCResponseResult::Success {
|
RMCResponseResult::Success {
|
||||||
|
|
@ -59,7 +59,8 @@ pub fn generate_response(protocol_id: u8, response: RMCResponseResult) -> io::Re
|
||||||
} => {
|
} => {
|
||||||
data_out.push(1);
|
data_out.push(1);
|
||||||
data_out.write_all(bytes_of(&call_id))?;
|
data_out.write_all(bytes_of(&call_id))?;
|
||||||
data_out.write_all(bytes_of(&method_id))?;
|
let ored_method_id = method_id | 0x8000;
|
||||||
|
data_out.write_all(bytes_of(&ored_method_id))?;
|
||||||
data_out.write_all(&data)?;
|
data_out.write_all(&data)?;
|
||||||
},
|
},
|
||||||
RMCResponseResult::Error {
|
RMCResponseResult::Error {
|
||||||
|
|
@ -81,8 +82,6 @@ pub async fn send_response(original_packet: &PRUDPPacket, socket: &SocketData, c
|
||||||
|
|
||||||
let ConnectionData{
|
let ConnectionData{
|
||||||
active_connection_data,
|
active_connection_data,
|
||||||
sock_addr,
|
|
||||||
server_signature,
|
|
||||||
..
|
..
|
||||||
} = connection;
|
} = connection;
|
||||||
|
|
||||||
|
|
@ -90,15 +89,12 @@ pub async fn send_response(original_packet: &PRUDPPacket, socket: &SocketData, c
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
let mut packet = original_packet.base_response_packet();
|
let mut packet = original_packet.base_response_packet();
|
||||||
|
|
||||||
|
|
||||||
packet.header.types_and_flags.set_types(DATA);
|
packet.header.types_and_flags.set_types(DATA);
|
||||||
packet.header.types_and_flags.set_flag(RELIABLE | HAS_SIZE | NEED_ACK);
|
packet.header.types_and_flags.set_flag((original_packet.header.types_and_flags.get_flags() & RELIABLE) | NEED_ACK);
|
||||||
|
|
||||||
packet.header.sequence_id = active_connection.reliable_server_counter;
|
|
||||||
active_connection.reliable_server_counter += 1;
|
|
||||||
packet.header.session_id = active_connection.server_session_id;
|
packet.header.session_id = active_connection.server_session_id;
|
||||||
packet.header.substream_id = 0;
|
packet.header.substream_id = 0;
|
||||||
|
|
||||||
|
|
@ -106,18 +102,7 @@ pub async fn send_response(original_packet: &PRUDPPacket, socket: &SocketData, c
|
||||||
|
|
||||||
packet.payload = rmcresponse.to_data();
|
packet.payload = rmcresponse.to_data();
|
||||||
|
|
||||||
|
connection.finish_and_send_packet_to(socket, packet).await;
|
||||||
active_connection.server_encryption.apply_keystream(&mut packet.payload);
|
|
||||||
|
|
||||||
packet.set_sizes();
|
|
||||||
packet.calculate_and_assign_signature(socket.access_key, None, Some(*server_signature));
|
|
||||||
|
|
||||||
let mut vec = Vec::new();
|
|
||||||
|
|
||||||
packet.write_to(&mut vec).expect("somehow failed to convert backet to bytes");
|
|
||||||
|
|
||||||
socket.socket.send_to(&vec, sock_addr.regular_socket_addr).await.expect("failed to send data back");
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//taken from kinnays error list directly
|
//taken from kinnays error list directly
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
use std::io::{Read, Seek};
|
use std::io::{Read, Seek, Write};
|
||||||
use crate::endianness::{IS_BIG_ENDIAN, ReadExtensions};
|
use crate::endianness::{IS_BIG_ENDIAN, ReadExtensions};
|
||||||
use super::{string, Result};
|
use super::{string, Result, RmcSerialize};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct Any{
|
pub struct Any{
|
||||||
|
|
@ -8,21 +8,26 @@ pub struct Any{
|
||||||
pub data: Vec<u8>
|
pub data: Vec<u8>
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn read(reader: &mut (impl Read + Seek)) -> Result<Any>{
|
impl RmcSerialize for Any{
|
||||||
let name = string::read(reader)?;
|
fn serialize(&self, writer: &mut dyn Write) -> Result<()> {
|
||||||
|
todo!()
|
||||||
|
}
|
||||||
|
fn deserialize(mut reader: &mut dyn Read) -> Result<Self> {
|
||||||
|
let name = String::deserialize(reader)?;
|
||||||
|
|
||||||
// also length ?
|
// also length ?
|
||||||
let len2: u32 = reader.read_struct(IS_BIG_ENDIAN)?;
|
let len2: u32 = reader.read_struct(IS_BIG_ENDIAN)?;
|
||||||
let length: u32 = reader.read_struct(IS_BIG_ENDIAN)?;
|
let length: u32 = reader.read_struct(IS_BIG_ENDIAN)?;
|
||||||
|
|
||||||
let mut data = vec![0; length as usize];
|
let mut data = vec![0; length as usize];
|
||||||
|
|
||||||
reader.read_exact(&mut data)?;
|
reader.read_exact(&mut data)?;
|
||||||
|
|
||||||
Ok(
|
Ok(
|
||||||
Any{
|
Any{
|
||||||
name,
|
name,
|
||||||
data
|
data
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -1,6 +1,8 @@
|
||||||
use std::io;
|
use std::io;
|
||||||
|
use std::io::{Read, Seek, Write};
|
||||||
use std::str::Utf8Error;
|
use std::str::Utf8Error;
|
||||||
use std::string::FromUtf8Error;
|
use std::string::FromUtf8Error;
|
||||||
|
use md5::digest::impl_oid_carrier;
|
||||||
use thiserror::Error;
|
use thiserror::Error;
|
||||||
|
|
||||||
//ideas for the future: make a proc macro library which allows generation of struct reads
|
//ideas for the future: make a proc macro library which allows generation of struct reads
|
||||||
|
|
@ -16,4 +18,9 @@ pub enum Error{
|
||||||
type Result<T> = std::result::Result<T, Error>;
|
type Result<T> = std::result::Result<T, Error>;
|
||||||
|
|
||||||
pub mod string;
|
pub mod string;
|
||||||
pub mod any;
|
pub mod any;
|
||||||
|
|
||||||
|
pub trait RmcSerialize: Sized{
|
||||||
|
fn serialize(&self, writer: &mut dyn Write) -> Result<()>;
|
||||||
|
fn deserialize(reader: &mut dyn Read) -> Result<Self>;
|
||||||
|
}
|
||||||
|
|
@ -1,18 +1,30 @@
|
||||||
use std::ffi::CString;
|
use std::ffi::CString;
|
||||||
use std::io::{Read, Seek};
|
use std::io::{Read, Seek, Write};
|
||||||
|
use bytemuck::bytes_of;
|
||||||
use log::error;
|
use log::error;
|
||||||
use crate::endianness::{IS_BIG_ENDIAN, ReadExtensions};
|
use crate::endianness::{IS_BIG_ENDIAN, ReadExtensions};
|
||||||
use super::Result;
|
use super::{Result, RmcSerialize};
|
||||||
|
|
||||||
pub fn read(reader: &mut (impl Read + Seek)) -> Result<String>{
|
impl RmcSerialize for String{
|
||||||
let len: u16 = reader.read_struct(IS_BIG_ENDIAN)?;
|
fn deserialize(mut reader: &mut dyn Read) -> Result<Self> {
|
||||||
let mut data = vec![0; len as usize - 1];
|
let len: u16 = reader.read_struct(IS_BIG_ENDIAN)?;
|
||||||
reader.read_exact(&mut data)?;
|
let mut data = vec![0; len as usize - 1];
|
||||||
|
reader.read_exact(&mut data)?;
|
||||||
|
|
||||||
let null: u8 = reader.read_struct(IS_BIG_ENDIAN)?;
|
let null: u8 = reader.read_struct(IS_BIG_ENDIAN)?;
|
||||||
if null != 0{
|
if null != 0{
|
||||||
error!("unable to find null terminator... continuing anyways");
|
error!("unable to find null terminator... continuing anyways");
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(String::from_utf8(data)?)
|
||||||
}
|
}
|
||||||
|
fn serialize(&self, writer: &mut dyn Write) -> Result<()> {
|
||||||
|
let u16_len: u16 = self.len() as u16;
|
||||||
|
writer.write(bytes_of(&u16_len))?;
|
||||||
|
|
||||||
Ok(String::from_utf8(data)?)
|
writer.write(self.as_bytes())?;
|
||||||
|
writer.write(&[0])?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue