369 lines
12 KiB
Rust
369 lines
12 KiB
Rust
|
|
/*! 🔌 SocketBox - TCP/UDP Socket networking
|
|||
|
|
*
|
|||
|
|
* ## 📝 概要
|
|||
|
|
* Rustの std::net を基盤とした高性能ネットワーキング Box
|
|||
|
|
* TCP サーバー・クライアント両対応、HTTPサーバー基盤として利用
|
|||
|
|
*
|
|||
|
|
* ## 🛠️ 利用可能メソッド
|
|||
|
|
* ### TCP Server
|
|||
|
|
* - `bind(address, port)` - TCP ソケット bind
|
|||
|
|
* - `listen(backlog)` - 接続待機開始
|
|||
|
|
* - `accept()` - クライアント接続受諾
|
|||
|
|
*
|
|||
|
|
* ### TCP Client
|
|||
|
|
* - `connect(address, port)` - サーバーへ接続
|
|||
|
|
*
|
|||
|
|
* ### IO Operations
|
|||
|
|
* - `read()` - データ読み取り
|
|||
|
|
* - `write(data)` - データ送信
|
|||
|
|
* - `close()` - ソケット閉鎖
|
|||
|
|
*
|
|||
|
|
* ## 💡 使用例
|
|||
|
|
* ```nyash
|
|||
|
|
* // TCP Server
|
|||
|
|
* server = new SocketBox()
|
|||
|
|
* server.bind("0.0.0.0", 8080)
|
|||
|
|
* server.listen(128)
|
|||
|
|
* client = server.accept()
|
|||
|
|
*
|
|||
|
|
* // TCP Client
|
|||
|
|
* client = new SocketBox()
|
|||
|
|
* client.connect("127.0.0.1", 8080)
|
|||
|
|
* client.write("Hello Server!")
|
|||
|
|
* response = client.read()
|
|||
|
|
* ```
|
|||
|
|
*/
|
|||
|
|
|
|||
|
|
use crate::box_trait::{NyashBox, StringBox, IntegerBox, BoolBox, BoxCore, BoxBase};
|
|||
|
|
use std::any::Any;
|
|||
|
|
use std::net::{TcpListener, TcpStream, SocketAddr, ToSocketAddrs};
|
|||
|
|
use std::io::{Read, Write, BufRead, BufReader};
|
|||
|
|
use std::sync::{Arc, Mutex};
|
|||
|
|
use std::time::Duration;
|
|||
|
|
|
|||
|
|
/// TCP/UDP ソケット操作を提供するBox
|
|||
|
|
#[derive(Debug)]
|
|||
|
|
pub struct SocketBox {
|
|||
|
|
base: BoxBase,
|
|||
|
|
// TCP Server
|
|||
|
|
listener: Arc<Mutex<Option<TcpListener>>>,
|
|||
|
|
// TCP Client/Connected Socket
|
|||
|
|
stream: Arc<Mutex<Option<TcpStream>>>,
|
|||
|
|
// Connection state
|
|||
|
|
is_server: Arc<Mutex<bool>>,
|
|||
|
|
is_connected: Arc<Mutex<bool>>,
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
impl Clone for SocketBox {
|
|||
|
|
fn clone(&self) -> Self {
|
|||
|
|
Self {
|
|||
|
|
base: BoxBase::new(), // New unique ID for clone
|
|||
|
|
listener: Arc::clone(&self.listener),
|
|||
|
|
stream: Arc::clone(&self.stream),
|
|||
|
|
is_server: Arc::clone(&self.is_server),
|
|||
|
|
is_connected: Arc::clone(&self.is_connected),
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
impl SocketBox {
|
|||
|
|
pub fn new() -> Self {
|
|||
|
|
Self {
|
|||
|
|
base: BoxBase::new(),
|
|||
|
|
listener: Arc::new(Mutex::new(None)),
|
|||
|
|
stream: Arc::new(Mutex::new(None)),
|
|||
|
|
is_server: Arc::new(Mutex::new(false)),
|
|||
|
|
is_connected: Arc::new(Mutex::new(false)),
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// TCP ソケットをアドレス・ポートにバインド
|
|||
|
|
pub fn bind(&self, address: Box<dyn NyashBox>, port: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|||
|
|
let addr_str = address.to_string_box().value;
|
|||
|
|
let port_str = port.to_string_box().value;
|
|||
|
|
|
|||
|
|
let socket_addr = format!("{}:{}", addr_str, port_str);
|
|||
|
|
|
|||
|
|
match TcpListener::bind(&socket_addr) {
|
|||
|
|
Ok(listener) => {
|
|||
|
|
*self.listener.lock().unwrap() = Some(listener);
|
|||
|
|
*self.is_server.lock().unwrap() = true;
|
|||
|
|
Box::new(BoolBox::new(true))
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox bind error: {}", e);
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// 指定した backlog で接続待機開始
|
|||
|
|
pub fn listen(&self, backlog: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|||
|
|
// TcpListener::bind already sets up listening with default backlog
|
|||
|
|
// This method exists for API compatibility but doesn't need additional setup
|
|||
|
|
let _backlog_num = backlog.to_string_box().value.parse::<i32>().unwrap_or(128);
|
|||
|
|
|
|||
|
|
if self.listener.lock().unwrap().is_some() {
|
|||
|
|
Box::new(BoolBox::new(true))
|
|||
|
|
} else {
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// クライアント接続を受諾(ブロッキング)
|
|||
|
|
pub fn accept(&self) -> Box<dyn NyashBox> {
|
|||
|
|
let listener_guard = self.listener.lock().unwrap();
|
|||
|
|
if let Some(ref listener) = *listener_guard {
|
|||
|
|
match listener.accept() {
|
|||
|
|
Ok((stream, _addr)) => {
|
|||
|
|
drop(listener_guard);
|
|||
|
|
|
|||
|
|
// Create new SocketBox for the client connection
|
|||
|
|
let client_socket = SocketBox::new();
|
|||
|
|
*client_socket.stream.lock().unwrap() = Some(stream);
|
|||
|
|
*client_socket.is_connected.lock().unwrap() = true;
|
|||
|
|
|
|||
|
|
Box::new(client_socket)
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox accept error: {}", e);
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// サーバーに接続(クライアントモード)
|
|||
|
|
pub fn connect(&self, address: Box<dyn NyashBox>, port: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|||
|
|
let addr_str = address.to_string_box().value;
|
|||
|
|
let port_str = port.to_string_box().value;
|
|||
|
|
|
|||
|
|
let socket_addr = format!("{}:{}", addr_str, port_str);
|
|||
|
|
|
|||
|
|
match TcpStream::connect(&socket_addr) {
|
|||
|
|
Ok(stream) => {
|
|||
|
|
// Set timeout for read/write operations
|
|||
|
|
let _ = stream.set_read_timeout(Some(Duration::from_secs(30)));
|
|||
|
|
let _ = stream.set_write_timeout(Some(Duration::from_secs(30)));
|
|||
|
|
|
|||
|
|
*self.stream.lock().unwrap() = Some(stream);
|
|||
|
|
*self.is_connected.lock().unwrap() = true;
|
|||
|
|
*self.is_server.lock().unwrap() = false;
|
|||
|
|
Box::new(BoolBox::new(true))
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox connect error: {}", e);
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// データを読み取り(改行まで or EOF)
|
|||
|
|
pub fn read(&self) -> Box<dyn NyashBox> {
|
|||
|
|
let stream_guard = self.stream.lock().unwrap();
|
|||
|
|
if let Some(ref stream) = *stream_guard {
|
|||
|
|
// Clone the stream to avoid borrowing issues
|
|||
|
|
match stream.try_clone() {
|
|||
|
|
Ok(stream_clone) => {
|
|||
|
|
drop(stream_guard);
|
|||
|
|
|
|||
|
|
let mut reader = BufReader::new(stream_clone);
|
|||
|
|
let mut buffer = String::new();
|
|||
|
|
|
|||
|
|
match reader.read_line(&mut buffer) {
|
|||
|
|
Ok(_) => {
|
|||
|
|
// Remove trailing newline
|
|||
|
|
if buffer.ends_with('\n') {
|
|||
|
|
buffer.pop();
|
|||
|
|
if buffer.ends_with('\r') {
|
|||
|
|
buffer.pop();
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
Box::new(StringBox::new(buffer))
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox read error: {}", e);
|
|||
|
|
Box::new(StringBox::new("".to_string()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox stream clone error: {}", e);
|
|||
|
|
Box::new(StringBox::new("".to_string()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
Box::new(StringBox::new("".to_string()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// HTTP request を読み取り(ヘッダーまで含む)
|
|||
|
|
pub fn read_http_request(&self) -> Box<dyn NyashBox> {
|
|||
|
|
let stream_guard = self.stream.lock().unwrap();
|
|||
|
|
if let Some(ref stream) = *stream_guard {
|
|||
|
|
match stream.try_clone() {
|
|||
|
|
Ok(stream_clone) => {
|
|||
|
|
drop(stream_guard);
|
|||
|
|
|
|||
|
|
let mut reader = BufReader::new(stream_clone);
|
|||
|
|
let mut request = String::new();
|
|||
|
|
let mut line = String::new();
|
|||
|
|
|
|||
|
|
// Read HTTP request line by line until empty line
|
|||
|
|
loop {
|
|||
|
|
line.clear();
|
|||
|
|
match reader.read_line(&mut line) {
|
|||
|
|
Ok(0) => break, // EOF
|
|||
|
|
Ok(_) => {
|
|||
|
|
request.push_str(&line);
|
|||
|
|
// Empty line indicates end of headers
|
|||
|
|
if line.trim().is_empty() {
|
|||
|
|
break;
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox HTTP read error: {}", e);
|
|||
|
|
break;
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
Box::new(StringBox::new(request))
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox stream clone error: {}", e);
|
|||
|
|
Box::new(StringBox::new("".to_string()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
Box::new(StringBox::new("".to_string()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// データを送信
|
|||
|
|
pub fn write(&self, data: Box<dyn NyashBox>) -> Box<dyn NyashBox> {
|
|||
|
|
let data_str = data.to_string_box().value;
|
|||
|
|
|
|||
|
|
let mut stream_guard = self.stream.lock().unwrap();
|
|||
|
|
if let Some(ref mut stream) = *stream_guard {
|
|||
|
|
match stream.write_all(data_str.as_bytes()) {
|
|||
|
|
Ok(_) => {
|
|||
|
|
match stream.flush() {
|
|||
|
|
Ok(_) => Box::new(BoolBox::new(true)),
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox flush error: {}", e);
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
},
|
|||
|
|
Err(e) => {
|
|||
|
|
eprintln!("🚨 SocketBox write error: {}", e);
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
Box::new(BoolBox::new(false))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// ソケット閉鎖
|
|||
|
|
pub fn close(&self) -> Box<dyn NyashBox> {
|
|||
|
|
*self.stream.lock().unwrap() = None;
|
|||
|
|
*self.listener.lock().unwrap() = None;
|
|||
|
|
*self.is_connected.lock().unwrap() = false;
|
|||
|
|
*self.is_server.lock().unwrap() = false;
|
|||
|
|
Box::new(BoolBox::new(true))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// 接続状態確認
|
|||
|
|
pub fn is_connected(&self) -> Box<dyn NyashBox> {
|
|||
|
|
Box::new(BoolBox::new(*self.is_connected.lock().unwrap()))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
/// サーバーモード確認
|
|||
|
|
pub fn is_server(&self) -> Box<dyn NyashBox> {
|
|||
|
|
Box::new(BoolBox::new(*self.is_server.lock().unwrap()))
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
impl NyashBox for SocketBox {
|
|||
|
|
fn clone_box(&self) -> Box<dyn NyashBox> {
|
|||
|
|
Box::new(self.clone())
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn to_string_box(&self) -> StringBox {
|
|||
|
|
let is_server = *self.is_server.lock().unwrap();
|
|||
|
|
let is_connected = *self.is_connected.lock().unwrap();
|
|||
|
|
|
|||
|
|
let status = if is_server {
|
|||
|
|
"Server"
|
|||
|
|
} else if is_connected {
|
|||
|
|
"Connected"
|
|||
|
|
} else {
|
|||
|
|
"Disconnected"
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
StringBox::new(format!("SocketBox(id: {}, status: {})", self.base.id, status))
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn type_name(&self) -> &'static str {
|
|||
|
|
"SocketBox"
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn equals(&self, other: &dyn NyashBox) -> BoolBox {
|
|||
|
|
if let Some(other_socket) = other.as_any().downcast_ref::<SocketBox>() {
|
|||
|
|
BoolBox::new(self.base.id == other_socket.base.id)
|
|||
|
|
} else {
|
|||
|
|
BoolBox::new(false)
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
impl BoxCore for SocketBox {
|
|||
|
|
fn box_id(&self) -> u64 {
|
|||
|
|
self.base.id
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn parent_type_id(&self) -> Option<std::any::TypeId> {
|
|||
|
|
self.base.parent_type_id
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn fmt_box(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|||
|
|
let is_server = *self.is_server.lock().unwrap();
|
|||
|
|
let is_connected = *self.is_connected.lock().unwrap();
|
|||
|
|
|
|||
|
|
let status = if is_server {
|
|||
|
|
"Server"
|
|||
|
|
} else if is_connected {
|
|||
|
|
"Connected"
|
|||
|
|
} else {
|
|||
|
|
"Disconnected"
|
|||
|
|
};
|
|||
|
|
|
|||
|
|
write!(f, "SocketBox(id: {}, status: {})", self.base.id, status)
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn as_any(&self) -> &dyn Any {
|
|||
|
|
self
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
fn as_any_mut(&mut self) -> &mut dyn Any {
|
|||
|
|
self
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
impl std::fmt::Display for SocketBox {
|
|||
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|||
|
|
self.fmt_box(f)
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// Auto-cleanup implementation for proper resource management
|
|||
|
|
impl Drop for SocketBox {
|
|||
|
|
fn drop(&mut self) {
|
|||
|
|
// Ensure sockets are properly closed
|
|||
|
|
let _ = self.close();
|
|||
|
|
}
|
|||
|
|
}
|