Easy-peasy support of newtypes inside of Diesel.
diesel-derive-newtype supports Diesel according to its major version -- 0.x through 1.x support the corresponding diesel versions, 2.x supports diesel 2.x.
So for Diesel 1.x:
[dependencies]
diesel-derive-newtype = "1.0"
And for Diesel 2.x:
[dependencies]
diesel-derive-newtype = "2.0.0-rc.0"
Note: this crate requires Rust 1.56 due to the dependency on syn 2.x.
This crate exposes a single custom-derive macro DieselNewType
which
implements ToSql
, FromSql
, FromSqlRow
, Queryable
, AsExpression
and QueryId
for the single-field tuple struct (NewType) it is applied
to.
The goal of this project is that:
derive(DieselNewType)
should be enough for you to use newtypes anywhere you would use their underlying types within Diesel. (plausibly successful)- Should get the same compile-time guarantees when using your newtypes as expression elements in Diesel as you do in other rust code (depends on your desires, see Limitations, below.)
This implementation:
#[macro_use]
extern crate diesel_derive_newtype;
#[derive(DieselNewType)] // Doesn't need to be on its own line
#[derive(Debug, Hash, PartialEq, Eq)] // required by diesel
struct MyId(i64);
Allows you to use the MyId
struct inside your entities as though they were
the underlying type:
table! {
my_items {
id -> Integer,
val -> Integer,
}
}
#[derive(Debug, PartialEq, Identifiable, Queryable)]
struct MyItem {
id: MyId,
val: u8,
}
Oooohhh. Ahhhh.
See the tests for a more complete example.
The DieselNewtype
derive does not create new database types, or Diesel
serialization types. That is, if you have a MyId(i64)
, this will use
Diesel's underlying BigInt
type, which means that even though your
newtypes can be used anywhere the underlying type can be used, the
underlying types, or any other newtypes of the same underlying type, can be
used as well.
At a certain point everything does become bits on the wire, so if we didn't do it this way then Diesel would have to do it somewhere else, and this is reasonable default behavior (it's pretty debuggable), but I'm investigating auto-generating new proxy types as well to make it impossible to construct an insert statement using a tuple or a mis-typed struct.
Here's an example of that this type-hole looks like:
#[derive(Debug, Hash, PartialEq, Eq, DieselNewType)]
struct OneId(i64);
#[derive(Debug, Hash, PartialEq, Eq, DieselNewType)]
struct OtherId(i64);
#[derive(Debug, Clone, PartialEq, Identifiable, Insertable, Queryable)]
#[diesel(table_name = my_entities)]
pub struct MyEntity {
id: OneId,
val: i32,
}
fn darn(conn: &Connection) {
// shouldn't allow constructing the wrong type, but does
let OtherId: Vec<OtherId> = my_entities
.select(id)
.filter(id.eq(OtherId(1))) // shouldn't allow filtering by wrong type
.execute(conn).unwrap();
}
See tests/should-not-compile.rs
for the
things I think should fail to compile.
I believe that the root cause of this is that Diesel implements the various
expression methods for types that implement AsExpression
, based on the
SQL type, not caring about self
and other
's Rust type matching. That
seems like a pretty good decision in general, but it is a bit unfortunate
here.
I hope to find a solution that doesn't involve implementing every
*Expression
trait manually with an extra bound, but for now you have to
keep in mind that the Diesel methods basically auto-transmute your data into
the underlying SQL type.
diesel-derive-newtype supports Diesel according to its major version -- 0.x through 1.x support the corresponding diesel versions, 2.x supports diesel 2.x.
[dependencies]
diesel-derive-newtype = "0.1"
diesel-derive-newtype is licensed under either of
- Apache License, Version 2.0, (LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT license (LICENSE-MIT or http://opensource.org/licenses/MIT)
at your option.
Patches and bug reports welcome!