This crate generates Rust Code and optionally compatible Protobuf and SQL schema files from ASN.1 definitions. Integration with serde is supported.
The crate can be used as standalone CLI binary or used as library through its API
(for example inside your build.rs
script).
| Feature | Parses | UPER | Protobuf | PSQL | Async PSQL |
| --------------------|:--------|:--------|:------------|:------------|:-----------|
| ...extensible | โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| SEQUENCE OF
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| SET
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...extensible | โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| SET OF
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ENUMERATED
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...extensible | โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| CHOICE
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...extensible | โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| BIT STRING
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| OCTET STRING
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| UTF8String
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| IA5String
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| NumericString
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| PrintableString
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| VisibleString
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...SIZE(A..B)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| ...SIZE(A..B,...)
| โ๏ธ yes | โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored |
| INTEGER
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| ...A..B
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยฒ | โ๏ธ yesยฒ | โ๏ธ yesยฒ |
| ...A..B,...
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยฒ | โ๏ธ yesยฒ | โ๏ธ yesยฒ |
| BOOLEAN
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| OPTIONAL
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes | โ๏ธ yes |
| DEFAULT ...
| โ๏ธ yes | | | | |
| ...INTEGER
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...*String
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...BOOLEAN
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| ...ENUMERATED
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| NULL
| โ๏ธ yes | โ๏ธ yes | โ๏ธ yesยน | โ๏ธ yesยน | โ๏ธ yesยน |
| IMPORTS..FROM..;
| โ๏ธ yes | | | | |
| ObjectIdentifiers
| โ๏ธ yes | | | | |
| Value References | โ๏ธ yes | | | | |
| ... in Range | โ๏ธ yes | | | | |
| ... in Size | โ๏ธ yes | | | | |
| ... in Default | โ๏ธ yes | | | | |
| WITH COMPONENTS
| โ๏ธ yes | ๐ ignored | ๐ ignored | ๐ ignored | ๐ ignored |
*The legacy UPER Reader/Writer is deprecated and will be removed in version 0.3.0
TLDR - The new (v0.2.0) UPER Reader/Writer supports all listed features - Protobuf, sync&async PSQL ignore most constraints - The legacy UPER Reader/Writer does not support all features (pre v0.2.0)
itu-t(0) identified-organization(4) etsi(0) itsDomain(5) wg1(1) ts(102894) cdd(2) version(2)
itu-t(0) identified-organization(4) etsi(0) itsDomain(5) wg1(1) en(302637) cam(2) version(2)
itu-t(0) identified-organization(4) etsi(0) itsDomain(5) wg1(1) en(302637) denm(1) version(2)
It is always helpful to check asn1rs --help
in advance.
The basic usage can be seen blow:
asn1rs -t rust directory/for/rust/files some.asn1 messages.asn1
asn1rs -t proto directory/for/protobuf/files some.asn1 messages.asn1
asn1rs -t sql directory/for/sql/schema/files some.asn1 messages.asn1
The following example generates Rust, Protobuf and SQL files for all .asn1
-files in the asn/
directory of a workspace.
While the generated Rust code is written to the src/
directory, the Protobuf files are written to proto/
and the SQL files are written to sql/
.
Additionally, in this example each generated Rust-Type also receives Serialize
and Deserialize
derive directives (#[derive(Serialize, Deserialize)]
) for serde integration.
Sample build.rs
file:
```rust use asn1rs::converter::Converter; use asn1rs::gen::rust::RustCodeGenerator; use asn1rs::gen::sql::SqlDefGenerator;
pub fn main() { let mut converter = Converter::default();
// collecting all relevant .asn1 files
std::fs::read_dir("../protocol/asn")
.into_iter()
.flat_map(|read_dir| {
read_dir
.into_iter()
.flat_map(|dir_entry| dir_entry.into_iter())
.flat_map(|entry| {
entry
.path()
.as_os_str()
.to_os_string()
.into_string()
.into_iter()
})
.filter(|entry| entry.ends_with(".asn1"))
})
.for_each(|path| {
println!("cargo:rerun-if-changed={}", path);
if let Err(e) = converter.load_file(&path) {
panic!("Loading of .asn1 file failed {}: {:?}", path, e);
}
});
// writing the .rs files into src with serde_derive support
// feature flags decide whether additional code for protobuf and (async) psql is generated
if let Err(e) = converter.to_rust("src/", |generator: &mut RustCodeGenerator| {
generator.add_global_derive("Serialize"); // Adds serde_derive support: #[derive(Serialize)]
generator.add_global_derive("Deserialize"); // Adds serde_derive support: #[derive(Deserialize)]
}) {
panic!("Conversion to rust failed: {:?}", e);
}
// OPTIONAL: writing the .proto representation to ../protocol/proto
if let Err(e) = converter.to_protobuf("../protocol/proto/") {
panic!("Conversion to proto failed: {:?}", e);
}
// OPTIONAL: writing the .sql schema files to ../protocol/sql
if let Err(e) = converter.to_sql_with(
"../protocol/sql/",
SqlDefGenerator::default() // optional parameter, alternatively see Converter::to_sql(&self, &str)
.optimize_tables_for_write_performance() // optional
.wrap_primary_key_on_overflow(), // optional
) {
panic!("Conversion to sql failed: {:?}", e);
}
}
```
Minimal example by inlining the ASN.1 definition. For more examples see tests/. ```rust use asn1rs::prelude::*;
asntorust!( r"BasicInteger DEFINITIONS AUTOMATIC TAGS ::= BEGIN
RangedMax ::= Integer (0..MAX)
NotRanged ::= Integer
END"
);
fn testwriteread() { // inner INTEGER identified as u64 let value = NotRanged(123_u64);
let mut writer = UperWriter::default();
writer.write(&value).expect("Failed to serialize");
let mut reader = writer.into_reader();
let value2 = reader.read::<NotRanged>().expect("Failed to deserialize");
assert_eq!(value, value2);
}
fn testconstrainteq() { // these types should normally not be accessed, but in this exampled they show // the way the ASN.1 constraints are encoded with the Rust type system. use asn1rs::syn::numbers::Constraint; asserteq!( asn1rsRangedMaxField0Constraint::MIN, _asn1rsNotRangedField0Constraint::MIN, ); asserteq!( asn1rsRangedMaxField0Constraint::MAX, _asn1rsNotRangedField0Constraint::MAX, ); } ```
Minimal example showcasing what is being generated from an ASN.1 definition:
```asn MyMessages DEFINITIONS AUTOMATIC TAGS ::= BEGIN
Header ::= SEQUENCE { timestamp INTEGER (0..1209600000) }
END ```
The generated Rust file:
```rust use asn1rs::prelude::*;
pub struct Header { #[asn(integer(0..1209600000))] pub timestamp: u32, }
// only with the feature "async-psql": Insert and query functions for async PostgreSQL
impl Header {
pub async fn apsqlretrievemany(context: &apsql::Context<'>, ids: &[i32]) -> Result
// only with the feature "psql": Insert and query functions for non-async PostgreSQL impl PsqlRepresentable for Header { /../ } impl PsqlInsertable for Header { /../ } impl PsqlQueryable for Header { /../ } ```
The generated protobuf file (optional):
```proto syntax = 'proto3'; package my.messages;
message Header { uint32 timestamp = 1; } ```
The generated SQL file (optional):
```sql DROP TABLE IF EXISTS Header CASCADE;
CREATE TABLE Header ( id SERIAL PRIMARY KEY, timestamp INTEGER NOT NULL ); ```
NOTE: This requires the async-psql
feature.
Using async postgres allows the message - or the batched messages - to take advantage of [pipelining
].
This can provide a significant speedup for deep message types (personal experience this is around 26%) compared to the synchronous/blocking postgres implementation.
```rust use asn1rs::io::asyncpsql::*; use tokiopostgres::NoTls;
async fn main() { let transactional = true; let (mut client, connection) = tokiopostgres::connect( "host=localhost user=postgres applicationname=psqlasyncdemo", NoTls, ) .await .expect("Failed to connect");
tokio::spawn(connection);
let context = if transactional {
let transaction = client
.transaction()
.await
.expect("Failed to open a new transaction");
Cache::default().into_transaction_context(transaction)
} else {
Cache::default().into_client_context(client)
};
// using sample message from above
let message = Header {
timestamp: 1234,
};
// This issues all necessary insert statements on the given Context and
// because it does not require exclusive access to the context, you can
// issue multiple inserts and await them concurrently with for example
// tokio::try_join, futures::try_join_all or the like.
let id = message.apsql_insert(&context).await.expect("Insert failed");
// This disassembles the context, allowing the Transaction to be committed
// or rolled back. This operation also optimizes the read access to
// prepared statements of the Cache. If you do not want to do that, then call
// Context::split_unoptimized instead.
// You can also call `Cache::optimize()` manually to optimize the read access
// to the cached prepared statements.
// See the doc for more information about the usage of cached prepared statements
let (mut cache, transaction) = context.split();
// this is (logically) a nop on a non-transactional context
transaction.commit().await.expect("failed to commit");
let context = if transactional {
let transaction = client
.transaction()
.await
.expect("Failed to open a new transaction");
Cache::default().into_transaction_context(transaction)
} else {
Cache::default().into_client_context(client)
};
let message_from_db = Header::apsql_retrieve(&context, id).await.expect("Failed to load");
assert_eq!(message, message_from_db);
} ```
The module asn1rs::io
exposes (de-)serializers and helpers for direct usage without ASN.1 definition:
```rust
use asn1rs::prelude::*;
use asn1rs::io::per::unaligned::buffer::BitBuffer;
let mut buffer = BitBuffer::default(); buffer.writebit(true).unwrap(); buffer.writeutf8_string("My UTF8 Text").unwrap();
sendtoanother_host(buffer.into::
The module asn1rs::io::protobuf
exposes (de-)serializers for protobuf usage:
```rust
use asn1rs::io::protobuf::*;
let mut buffer = Vec::default(); buffer.writevarint(1337).unwrap(); buffer.writestring("Still UTF8 Text").unwrap();
sendtoanother_host(buffer): ```
Things to do at some point in time (PRs are welcome)
#![no_std]
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.
This crate was initially developed during a research project at IT-Designers GmbH (http://www.it-designers.de).