Expand description
§Efficient bytestring serialization for serde
-supporting types
§What it does
Byte arrays ([u8; N]
, Vec<u8>
, Box<[u8]>
and so on) are treated by serde
as arrays of integers, which leads to inefficient representations in various formats.
For example, this is how serialization works by default for binary and human-readable formats:
use serde::{Serialize, Deserialize};
#[derive(Serialize, Deserialize)]
struct Array([u8; 16]);
let array = Array([
0, 1, 0xf2, 3, 0xf4, 5, 0xf6, 7, 0xf8, 9, 0xfa, 11, 0xfc, 13, 14, 0xff
]);
// Serializing as MessagePack
assert_eq!(
rmp_serde::encode::to_vec(&array).unwrap(),
[
0xdc, 0, 0x10,
0, 1, 0xcc, 0xf2, 3, 0xcc, 0xf4, 5, 0xcc, 0xf6, 7,
0xcc, 0xf8, 9, 0xcc, 0xfa, 11, 0xcc, 0xfc, 13, 14, 0xcc, 0xff,
]
);
// Serializing as JSON
assert_eq!(
serde_json::to_string(&array).unwrap(),
"[0,1,242,3,244,5,246,7,248,9,250,11,252,13,14,255]"
);
Note that in MessagePack the bytes with the value above 0x7f
(that is, the ones with the MSB set) are prefixed by 0xcc
, which makes the bytestring take more space than it should.
And in case of JSON, the bytestring is serialized as an array of integers, which is also not very efficient.
This crate provides methods that can be used in serde(with)
field attribute to amend this behavior and serialize bytestrings efficiently, verbatim in binary formats, or using the selected encoding in human-readable formats.
§Usage
To use, add a serde(with)
annotation with an argument composed of a container type (whether it is array-like, slice-like and so on) and the desired encoding:
use serde::{Deserialize, Serialize};
use serde_encoded_bytes::{ArrayLike, Hex};
#[derive(Debug, PartialEq, Eq, Serialize, Deserialize)]
struct Array(#[serde(with = "ArrayLike::<Hex>")] [u8; 16]);
let array = Array([
0, 1, 0xf2, 3, 0xf4, 5, 0xf6, 7, 0xf8, 9, 0xfa, 11, 0xfc, 13, 14, 0xff,
]);
// Serializing as MessagePack
assert_eq!(
rmp_serde::encode::to_vec(&array).unwrap(),
[0xc4, 0x10, 0, 1, 0xf2, 3, 0xf4, 5, 0xf6, 7, 0xf8, 9, 0xfa, 11, 0xfc, 13, 14, 0xff]
);
// Serializing as JSON
assert_eq!(
serde_json::to_string(&array).unwrap(),
"\"0x0001f203f405f607f809fa0bfc0d0eff\""
);
As you can see, the serialization of the example above is now more efficient in either format.
Note that due to serde
limitations (see https://github.com/serde-rs/serde/issues/2120) fixed-size arrays will still be serialized with their length included in binary formats.
§Tested formats
While this crate is supposed to work for any format that supports serde
, it is specifically tested on:
bincode
v2.0.0-rc.3ciborium
v0.2rmp-serde
v1serde_asn1_der
v0.8serde-json-core
v0.6serde-json
v1toml
v0.8
§Prior art
More established crates with intersecting functionality include:
serde-bytes-repr
- similar capabilities, but a different approach to the API, which can be inconvenient in a number of cases;serde_bytes
- provides efficient serialization in binary formats, but not in human-readable formats;serdect
- focused specifically on constant-time serialization;hex-buffer-serde
- fixed encoding;base64-serde
,serde-hex
,stremio-serde-hex
- fixed encoding, and no support for binary formats.
Structs§
- A container for array-like data, e.g. Rust stack arrays, or
generic_array::GenericArray<...>
. - Base64
base64
Encodes the byte sequence into a base64 representation. - A container for boxed array-like data, e.g.
Box<[u8; 4]>
orBox<generic_array::GenericArray<...>>
. - Hex
hex
Encodes the byte sequence into a0x
-prefixed hexadecimal representation. - A container for slice-like data, e.g.
Vec<u8>
orBox<u8>
.
Traits§
- A trait for encoding bytes into strings.
- A type of a trait alias, to work around https://github.com/rust-lang/rust/issues/113517. If not for that issue, we could just use
TryFrom<[u8; N]>
directly in the bounds. - A type of a trait alias, to work around https://github.com/rust-lang/rust/issues/113517. If not for that issue, we could just use
TryFrom<&'a [u8]>
directly in the bounds.