Encoding and decoding support for BSON in Rust
This crate works with Cargo and can be found on
crates.io with a Cargo.toml
like:
toml
[dependencies]
bson = "0.14"
Prepare your struct for Serde serialization:
```rust
pub struct Person { #[serde(rename = "_id")] // Use MongoDB's special primary key field name when serializing pub id: bson::oid::ObjectId, pub name: String, pub age: i32 } ```
Serialize the struct:
```rust use bson;
let person = Person { id: "12345", name: "Emma", age: 3 };
let serializedperson = bson::tobson(&person)?; // Serialize
if let bson::Bson::Document(document) = serializedperson { mongoCollection.insertone(document, None)?; // Insert into a MongoDB collection } else { println!("Error converting the BSON object into a MongoDB document"); } ```
Deserialize the struct:
```rust use bson::doc;
// Read the document from a MongoDB collection let persondocument = mongoCollection.findone(Some(doc! { "id": bson::oid::ObjectId::withstring("12345").expect("Id not valid") }), None)? .expect("Document not found");
// Deserialize the document into a Person instance let person = bson::frombson(bson::Bson::Document(persondocument))? ```
In the BSON specification, unsigned integer types are unsupported; for example, u32
. In the older version of this crate (< v0.8.0
), if you uses serde
to serialize unsigned integer types into BSON, it will store them with Bson::FloatingPoint
type. From v0.8.0
, we removed this behavior and simply returned an error when you want to serialize unsigned integer types to BSON. #72
For backward compatibility, we've provided a mod bson::compat::u2f
to explicitly serialize unsigned integer types into BSON's floating point value as follows:
```rust
fn testcompatu2f() { #[derive(Serialize, Deserialize, Eq, PartialEq, Debug)] struct Foo { #[serde(with = "bson::compat::u2f")] x: u32 }
let foo = Foo { x: 20 };
let b = bson::to_bson(&foo).unwrap();
assert_eq!(b, Bson::Document(doc! { "x": Bson::FloatingPoint(20.0) }));
let de_foo = bson::from_bson::<Foo>(b).unwrap();
assert_eq!(de_foo, foo);
} ```
In this example, we added an attribute #[serde(with = "bson::compat::u2f")]
on field x
, which will tell serde
to use the bson::compat::u2f::serialize
and bson::compat::u2f::deserialize
methods to process this field.
We encourage and would happily accept contributions in the form of GitHub pull requests. Before opening one, be sure to run the tests locally; check out the testing section for information on how to do that. Once you open a pull request, your branch will be run against the same testing matrix that we use for our continuous integration system, so it is usually sufficient to only run the integration tests locally against a standalone. Remember to always run the linter tests before opening a pull request.
To actually run the tests, you can use cargo
like you would in any other crate:
bash
cargo test --verbose # runs against localhost:27017
Our linter tests use the nightly version of rustfmt
to verify that the source is formatted properly and the stable version of clippy
to statically detect any common mistakes.
You can use rustup
to install them both:
bash
rustup component add clippy --toolchain stable
rustup component add rustfmt --toolchain nightly
To run the linter tests, run the check-clippy.sh
and check-rustfmt.sh
scripts in the .evergreen
directory:
bash
bash .evergreen/check-clippy.sh && bash .evergreen/check-rustfmt.sh
Commits to master are run automatically on evergreen.