Support for flexibly deserializing database resultsets and rows into rust types, and for serializing rust types into database parameter values.
Being based on serde, this crate can facilitate the data exchange between applications and a database driver. It is meant to be used by the implementors of database drivers, who then can expose a more comfortable driver API.
Add hdbconnect to the dependencies section in your project's Cargo.toml
, with
[dependencies]
serde_db = "0.9"
The below examples assume the DB driver exposes on its resultset type a function
fn into_typed<'de, T: serde::de::Deserialize<'de>>(self) -> Result<T, E>
which is implemented using serde_db
.
The application then can directly assign the database results to appropriate rust variables!
#[macro_use]
extern crate serde_derive;
...
#[derive(Deserialize)]
struct MyStruct {...}
...
let resultset = ...;
let data: Vec<MyStruct> = resultset.into_typed()?;
Note that MyStruct
has to implement serde::de::Deserialize
.
let vec_s: Vec<String> = resultset.into_typed()?;
let s: String = resultset.into_typed()?;
For better streaming of large resultsets, you might want to iterate over the rows, like in
for row in resultset {
let t: (String, NaiveDateTime, i32, Option<i32>) = row.into_typed()?;
}
or
for row in resultset {
let data: MyStruct = row.into_typed()?;
}
serde_db
also when a DB driver needs to translate rust values into DB types.
A Prepared Statement for example might have a generic function
fn add_batch<T>(&mut self, input: &T) -> HdbResult<()>
where T: serde::ser::Serialize
If it is implemented with serde_db
, then the application can hand over
a tuple of rust values that correspond to the parameters of the prepared statement,
or they can hand over an appropriate struct that implements serde::ser::Serialize
.
In both cases they do not need to differentiate between nullable and non-nullable
database values (except that they cannot convert an Option::None
into a non-nullable database
value).
In its implementation of DbvFactory
,
the DB driver can decide to make their life even easier by converting flexibly between
different number types (an example can be found in the tests of this crate).
The implementation of add_batch()
converts input
into
a Vec of the driver's database values that can subsequently be sent to the DB server:
let db_values: Vec<DBValue> = serde_db::ser::to_params(&input, input_metadata)?;
It is required that the prepared statement has metadata about the needed input parameters,
which implement DbvFactory
.