A Client/Server game networking plugin using QUIC, for the Bevy game engine.
QUIC was really attractive to me as a game networking protocol because most of the hard-work is done by the protocol specification and the implementation (here Quinn). No need to reinvent the wheel once again on error-prones subjects such as a UDP reliability wrapper, some encryption & authentication mechanisms, congestion-control, and so on.
Most of the features proposed by the big networking libs are supported by default through QUIC. As an example, here is the list of features presented in GameNetworkingSockets:
-> Roughly 9 points out of 11 by default.
(*) Kinda, when sharing a QUIC stream, reliable messages need to be framed.
Quinnet does not have many features, I made it mostly to satisfy my own needs for my own game projects.
It currently features:
Although Quinn and parts of Quinnet are asynchronous, the APIs exposed by Quinnet for the client and server are synchronous. This makes the surface API easy to work with and adapted to a Bevy usage. The implementation uses tokio channels to communicate with the networking async tasks.
Those are the features/tasks that will probably come next (in no particular order):
QuinnetClientPlugin
to the bevy app and give it a ClientConfigurationData
resource:rust
App::new()
// ...
.add_plugin(QuinnetClientPlugin::default())
.insert_resource(ClientConfigurationData::new(
"127.0.0.1".to_string(),
6000,
"0.0.0.0".to_string(),
0,
))
// ...
.run();
Client
resource to connect, send & receive messages:```rust
fn start_connection(client: ResMut
// You can already send message(s) even before being connected, they will be buffered. Else, just wait for client.is_connected()
client
.send_message(...)
.unwrap();
} ```
receive_message
is generic, here ServerMessage
is a user provided enum deriving Serialize
and Deserialize
.rust
fn handle_server_messages(
mut client: ResMut<Client>,
/*...*/
) {
while let Ok(Some(message)) = client.receive_message::<ServerMessage>() {
match message {
// Match on your own message types ...
ServerMessage::ClientConnected { client_id, username} => {/*...*/}
ServerMessage::ClientDisconnected { client_id } => {/*...*/}
ServerMessage::ChatMessage { client_id, message } => {/*...*/}
}
}
}
QuinnetServerPlugin
to the bevy app and give it a ServerConfigurationData
resource:rust
App::new()
/*...*/
.add_plugin(QuinnetServerPlugin::default())
.insert_resource(ServerConfigurationData::new(
"127.0.0.1".to_string(),
6000,
"0.0.0.0".to_string(),
))
/*...*/
.run();
receive_message
is generic, here ClientMessage
is a user provided enum deriving Serialize
and Deserialize
.rust
fn handle_client_messages(
mut server: ResMut<Server>,
/*...*/
) {
while let Ok(Some(message)) = server.receive_message::<ClientMessage>() {
// Retrieve the assigned ClientId.
let client_id = message.1;
match message.0 {
// Match on your own message types ...
ClientMessage::Join { username} => {
// Send a messsage to 1 client
server.send_message(client_id, ServerMessage::InitClient {/*...*/}).unwrap();
/*...*/
}
ClientMessage::Disconnect { } => {
// Disconnect a client
server.disconnect_client(client_id);
/*...*/
}
ClientMessage::ChatMessage { message } => {
// Send a message to a group of clients
server.send_group_message(
client_group, // Iterator of ClientId
ServerMessage::ChatMessage {/*...*/}
)
.unwrap();
/*...*/
}
}
}
}
You can also use server.broadcast_message
, which will send a message to all connected clients. "Connected" here means connected to the server plugin, which happens before your own app handshakes/verifications if you have any. Use send_group_message
if you want to control the recipients.
Examples can be found in the examples directory.
This demo comes with an headless server, a terminal client and a shared protocol
Start the server with cargo run --example chat_server
and as many clients as needed with cargo run --example terminal_chat_client
. Type quit
to disconnect with a client.
For logs configuration, see the unoffical bevy cheatbook.
Compatibility of bevy_quinnet
versions:
| bevy_quinnet
| bevy
|
| :-- | :-- |
| 0.1
| 0.8
|
Thanks to the Renet crate for the inspiration on the high level API.
bevy-quinnet is free and open source! All code in this repository is dual-licensed under either:
at your option. This means you can select the license you prefer! This dual-licensing approach is the de-facto standard in the Rust ecosystem and there are very good reasons to include both.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.