#binary-format #ion #binary-parser #binary #amazon #encode #parser

ion-binary-rs

Pure Rust parser, encoder and hasher for Amazon's Ion binary format

28 releases

0.8.13 Feb 2, 2024
0.8.12 Oct 1, 2023
0.8.11 Oct 21, 2022
0.8.10 Jul 17, 2022
0.7.0 Nov 18, 2020

#258 in Network programming

Download history 5741/week @ 2024-08-15 5626/week @ 2024-08-22 5704/week @ 2024-08-29 5865/week @ 2024-09-05 5995/week @ 2024-09-12 5733/week @ 2024-09-19 6202/week @ 2024-09-26 6175/week @ 2024-10-03 6192/week @ 2024-10-10 6098/week @ 2024-10-17 6092/week @ 2024-10-24 6052/week @ 2024-10-31 6010/week @ 2024-11-07 5999/week @ 2024-11-14 5942/week @ 2024-11-21 4931/week @ 2024-11-28

24,140 downloads per month
Used in qldb

Apache-2.0/MIT

315KB
7K SLoC

Ion Binary parser/encoder & Ion Hash in Rust

Ion binary is a library written in safe rust for parsing, encoding and hashing Amazon's Ion binary format.

Coverage Status Documentation Crates.io

It parses, encodes and hashes anything you throw at it. Any failure to do so is a bug that we will fix and we will be very happy if you report them.

The code is mature and way more tested than Amazon's alternatives, including their js library. Amazon, when implementing the "good tests" only check that it parses. We check that the value is the correct one too. Additionally, we seem to be implementing way more of the ion protocol than Amazon's libraries, as we don't have huge skip lists in the tests files. We test all their test suite, not only a few test.

How to use the library

First of all, you need to be aware of the trade offs that we took for this library:

  • The API returns strings instead of Symbols. If needed we can add symbol, but we think string is the the most ergonomic way.
  • When parsing/decoding you can add shared tables for binary blobs that doesn't have all the required symbols.

We have implemented the whole amazon ion test-suite for parsing. Encoding and Hashing fully tested. We are working in expading the coverage. We would appreciate any bug you can report. You can check all the test for examples.

Example

Parsing


use ion_binary_rs::IonParser;

// This is the response from Amazon's QLDB introduction example using Rusoto
let ion_test = b"\xe0\x01\0\xea\xee\xa6\x81\x83\xde\xa2\x87\xbe\x9f\x83VIN\x84Type\x84Year\x84Make\x85Model\x85Color\xde\xb9\x8a\x8e\x911C4RJFAG0FC625797\x8b\x85Sedan\x8c\"\x07\xe3\x8d\x88Mercedes\x8e\x87CLK 350\x8f\x85White";

let mut parser = IonParser::new(&ion_test[..]);

println!("Decoded Ion: {:?}", parser.consume_all().unwrap())
// Decoded Ion: [Struct({"Color": String("White"), "Year": Integer(2019), "VIN": String("1C4RJFAG0FC625797"), "Make": String("Mercedes"), "Model": String("CLK 350"), "Type": String("Sedan")})]

Encoding


use ion_binary_rs::{IonEncoder, IonParser, IonValue};
use std::collections::HashMap;

let mut ion_struct = HashMap::new();

ion_struct.insert("Model".to_string(), IonValue::String("CLK 350".to_string()));
ion_struct.insert("Type".to_string(), IonValue::String("Sedan".to_string()));
ion_struct.insert("Color".to_string(), IonValue::String("White".to_string()));
ion_struct.insert(
    "VIN".to_string(),
    IonValue::String("1C4RJFAG0FC625797".to_string()),
);
ion_struct.insert("Make".to_string(), IonValue::String("Mercedes".to_string()));
ion_struct.insert("Year".to_string(), IonValue::Integer(2019));

let ion_value = IonValue::Struct(ion_struct);

let mut encoder = IonEncoder::new();

encoder.add(ion_value.clone());
let bytes = encoder.encode();

let resulting_ion_value = IonParser::new(&bytes[..]).consume_value().unwrap().0;

assert_eq!(ion_value, resulting_ion_value);

Hashing

use sha2::Sha256;
use ion_binary_rs::{IonHash, IonValue};
use std::collections::HashMap;

let mut ion_struct = HashMap::new();

ion_struct.insert("Model".to_string(), IonValue::String("CLK 350".to_string()));
ion_struct.insert("Type".to_string(), IonValue::String("Sedan".to_string()));
ion_struct.insert("Color".to_string(), IonValue::String("White".to_string()));
ion_struct.insert(
    "VIN".to_string(),
    IonValue::String("1C4RJFAG0FC625797".to_string()),
);
ion_struct.insert("Make".to_string(), IonValue::String("Mercedes".to_string()));
ion_struct.insert("Year".to_string(), IonValue::Integer(2019));

let ion_value = IonValue::Struct(ion_struct);

let hash = IonHash::digest::<Sha256>(&ion_value);

println!("{:X?}", hash);

Safe Rust

No unsafe code was directly used in this crate. You can check in lib.rs the #![deny(unsafe_code)] line.

Contributing

We would be thrilled if you decide to check the library and/or contribute to it! Just open an issue or pull request and we can check what you would like to implement. Bug hunting and proposals are always welcomed. And of course, feel free to ask anything.

License

Licensed under either of Apache License, Version 2.0 or MIT license at your option.
Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in this crate by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.

Dependencies

~3–4.5MB
~87K SLoC