5 个版本 (3 个破坏性更新)

0.5.1 2023年6月22日
0.5.0 2023年2月26日
0.3.0 2023年1月5日
0.2.0 2022年12月25日
0.1.0 2022年12月25日

#758并发

Download history • Rust 包仓库 27/week @ 2024-03-11 • Rust 包仓库 14/week @ 2024-03-18 • Rust 包仓库 16/week @ 2024-03-25 • Rust 包仓库 39/week @ 2024-04-01 • Rust 包仓库 4/week @ 2024-04-08 • Rust 包仓库 9/week @ 2024-04-15 • Rust 包仓库 14/week @ 2024-04-22 • Rust 包仓库 10/week @ 2024-04-29 • Rust 包仓库 15/week @ 2024-05-06 • Rust 包仓库 12/week @ 2024-05-13 • Rust 包仓库 14/week @ 2024-05-20 • Rust 包仓库 8/week @ 2024-05-27 • Rust 包仓库 12/week @ 2024-06-03 • Rust 包仓库 13/week @ 2024-06-10 • Rust 包仓库 10/week @ 2024-06-17 • Rust 包仓库 20/week @ 2024-06-24 • Rust 包仓库

每月55次 下载
5 个 Crates 中使用 (3个直接使用)

MIT 许可证

64KB
1K SLoC

一个用于真正零拷贝消息处理的类似演员的 RPC 框架。

一个用于真正零拷贝消息处理的类似演员的 RPC 框架。

该框架受 tonic 启发,但不是 GRPC 框架。相反,它利用了令人难以置信的 rkyv(反)序列化框架,该框架为我们提供了闪电般的(反)序列化速度,并允许我们执行真正的零拷贝反序列化,这可以在一次处理大量大消息时带来巨大的性能提升。

特性

  • 拥有类型的快速(反)序列化。
  • 真正的零拷贝反序列化,避免大量分配。
  • 动态添加和删除消息处理器/服务。

基本示例

use std::net::SocketAddr;                                                                      
                                                                                                
use datacake_rpc::{                                                                             
    Channel,                                                                                    
    Handler,                                                                                    
    Request,                                                                                    
    RpcClient,                                                                                  
    RpcService,                                                                                 
    Server,                                                                                     
    ServiceRegistry,                                                                            
    Status,                                                                                     
};                                                                                              
use rkyv::{Archive, Deserialize, Serialize};                                                    
                                                                                                
// The framework accepts any messages which implement `Archive` and `Serialize` along           
// with the archived values implementing `CheckBytes` from the `bytecheck` crate.               
// This is to ensure safe, validated deserialization of the values.                             
//                                                                                              
// Checkout rkyv for more information!                                                          
#[repr(C)]                                                                                      
#[derive(Serialize, Deserialize, Archive, PartialEq, Debug)]                                    
#[archive(compare(PartialEq), check_bytes)]                                                                  
#[archive_attr(derive(PartialEq, Debug))]                                           
pub struct MyMessage {                                                                          
    name: String,                                                                               
    age: u32,                                                                                   
}                                                                                               
                                                                                                
pub struct MyService;                                                                           
                                                                                                
impl RpcService for MyService {                                                                 
    // The `register_handlers` is used to mark messages as something                            
    // the given service can handle and process.                                                
    //                                                                                          
    // Messages which are not registered will not be dispatched to the handler.                 
    fn register_handlers(registry: &mut ServiceRegistry<Self>) {                                
        registry.add_handler::<MyMessage>();                                                    
    }                                                                                           
}                                                                                               
                                                                                                
#[datacake_rpc::async_trait]                                                                    
impl Handler<MyMessage> for MyService {                                                         
    type Reply = String;                                                                        
                                                                                                
    // Our `Request` gives us a zero-copy view to our message, this doesn't actually            
    // allocate the message type.                                                               
    async fn on_message(&self, msg: Request<MyMessage>) -> Result<Self::Reply, Status> {        
        Ok(msg.to_owned().unwrap().name)                                                        
    }                                                                                           
}                                                                                               
                                                                                                
#[tokio::main]                                                                                  
async fn main() -> anyhow::Result<()> {                                                         
    let address = "127.0.0.1:8000".parse::<SocketAddr>()?;                                      
                                                                                                
    let server = Server::listen(address).await?;                                                
    // Services can be added and removed at runtime once the server is started.                 
    server.add_service(MyService);                                                              
    println!("Listening to address {}!", address);                                              
                                                                                                
    // Channels are cheap to clone similar to tonic.                                            
    let client = Channel::connect(address);                                              
    println!("Connected to address {}!", address);                                              
                                                                                                
    let rpc_client = RpcClient::<MyService>::new(client);                                   
                                                                                                
    let msg1 = MyMessage {                                                                      
        name: "Bobby".to_string(),                                                              
        age: 12,                                                                                
    };                                                                                          
                                                                                                
    // Clients only need references to the message which helps                                  
    // reduce allocations.                                                                      
    let resp = rpc_client.send(&msg1).await?;                                                   
    assert_eq!(resp, msg1.name);                                                                     
    Ok(())                                                                                      
}                                                                                               

依赖项

~6–14MB
~164K SLoC