7 stable releases (3 major)
5.3.1 | Aug 4, 2023 |
---|---|
5.2.2 | Apr 11, 2023 |
5.1.1 |
|
4.2.0 | Mar 13, 2023 |
1.1.0 | Jan 19, 2022 |
#8 in #osu
Used in bancho-packets-derive
81KB
2K
SLoC
bancho-packets
osu! bancho packet Reading & Writing library.
docs https://docs.rs/bancho-packets
Usage
Add to your cargo.toml
[dependencies]
bancho-packets = "4"
Or run the following Cargo command in your project directory:
cargo add bancho-packets
Examples
see more: examples, src/tests.rs
Reading packets
use bancho_packets::*;
// Packets from osu! bancho
let data = &[
4, 0, 0, 0, 0, 0, 0, 24, 0, 0, 19, 0, 0, 0, 11, 17, 72, 101, 108, 108,
111, 44, 32, 87, 111, 114, 108, 100, 33, 240, 159, 146, 150, 4, 0, 0,
0, 0, 0, 0, 24, 0, 0, 18, 0, 0, 0, 11, 16, 229, 147, 136, 229, 147,
136, 227, 128, 144, 240, 159, 152, 131, 227, 128, 145, 104, 0, 0, 0, 0,
0, 0, 24, 0, 0, 23, 0, 0, 0, 11, 21, 232, 175, 187, 229, 143, 150, 229,
174, 140, 228, 186, 134, 239, 188, 129, 239, 188, 129, 226, 156, 168,
];
// Create reader
let reader = PacketReader::new(data);
// Read packets
for packet in reader {
print!("{:?}: ", packet.id);
match packet.payload {
Some(payload) => {
// Read payload
let mut payload_reader = PayloadReader::new(payload);
println!("{:?}", payload_reader.read::<String>());
},
None => println!("Non-payload"),
}
}
/* Results
packet id: OSU_PING: Non-payload
packet id: BANCHO_NOTIFICATION: payload: Some("Hello, World!💖")
packet id: OSU_PING: Non-payload
packet id: BANCHO_NOTIFICATION: payload: Some("哈哈【😃】")
packet id: BANCHO_ACCOUNT_RESTRICTED: Non-payload
packet id: BANCHO_NOTIFICATION: payload: Some("读取完了!!✨")
*/
Writing packets
use bancho_packets::*;
// Single packet
let login_reply_from_server = server::LoginReply::pack(LoginResult::Failed(
LoginFailedReason::InvalidCredentials,
));
let serverside_notification = server::Notification::pack("hello".into());
// Multiple packets with Builder
let packets = PacketBuilder::new()
.add(server::LoginReply::pack(LoginResult::Success(1000)))
.add(server::ProtocolVersion::pack(19))
.add(server::Notification::pack("Welcome to osu!".into()))
.add(server::MainMenuIcon::pack(
"https://image.png".into(),
"https://url.link".into(),
))
.add(server::SilenceEnd::pack(0))
.add(server::ChannelInfoEnd::pack())
.build();
Build your own packet
use bancho_packets::*;
// Build simple packet
let number_data: i32 = 1;
let packet = packet!(PacketId::BANCHO_MATCH_PLAYER_SKIPPED, number_data)
// Complex (old)
pub fn user_stats(
user_id: i32,
action: u8,
info: String,
beatmap_md5: String,
mods: u32,
mode: u8,
beatmap_id: i32,
ranked_score: i64,
accuracy: f32,
playcount: i32,
total_score: i64,
rank: i32,
pp: i16,
) -> Vec<u8> {
packet!(
PacketId::BANCHO_USER_STATS,
data!(
user_id,
action,
info,
beatmap_md5,
mods,
mode,
beatmap_id,
ranked_score,
accuracy / 100f32,
playcount,
total_score,
rank,
pp
)
)
}
// Complex (new)
packet_struct!(
PacketId::BANCHO_SPECTATOR_JOINED,
/// #13: BANCHO_SPECTATOR_JOINED
SpectatorJoined { user_id: i32 }
);
// Complex (new)
packet_struct!(
PacketId::BANCHO_USER_STATS,
/// #11: BANCHO_USER_STATS
UserStats<'a> {
user_id: i32,
online_status: u8,
description: CowStr<'a>,
beatmap_md5: CowStr<'a>,
mods: u32,
mode: u8,
beatmap_id: i32,
ranked_score: i64,
accuracy: f32,
playcount: i32,
total_score: i64,
rank: i32,
pp: i16,
},
fn into_packet_data(self) -> Vec<u8> {
packet!(
Self::ID,
self.user_id,
self.online_status,
self.description,
self.beatmap_md5,
self.mods,
self.mode,
self.beatmap_id,
self.ranked_score,
self.accuracy / 100f32,
self.playcount,
self.total_score,
self.rank,
self.pp
)
}
);
Available attributies
ReadPacket
: This derive macro will implement theBanchoPacketRead
trait for the struct.WritePacket
: This derive macro will implement theBanchoPacketRead
trait for the struct.PacketLength
: This derive macro will implement theBanchoPacketLength
trait for the struct.
example
use bancho_packets::{ReadPacket, PacketReader, PayloadReader};
#[derive(Debug, Clone, ReadPacket)]
/// [`BanchoMessage`] is the message structure of the bancho client.
pub struct BanchoMessage {
pub sender: String,
pub content: String,
pub target: String,
pub sender_id: i32,
}
// Now we can use [`PayloadReader`] to read the [`BanchoMessage`] from bytes.
let mut reader = PacketReader::new(&[
1, 0, 0, 20, 0, 0, 0, 11, 0, 11, 6, 228, 189, 160, 229, 165, 189,
11, 4, 35, 111, 115, 117, 0, 0, 0, 0,
]);
let packet = reader.next().unwrap();
let mut payload_reader = PayloadReader::new(packet.payload.unwrap());
let message = payload_reader.read::<BanchoMessage>();
println!("{:?}: {:?}", packet.id, message);
Run Test
cargo test
Run Benchmark
cargo bench
Dependencies
~0.4–1MB
~24K SLoC