I'm trying to provide a generic filter capability to my diesel application. The user would specify a column to filter against, an appropriate value for that column (str, float, int, etc) and then one of the operations:
pub enum FilterOp {
NONE,
EQUAL,
LESS_THAN,
LESS_THAN_EQUAL,
GREATER_THAN,
GREATER_THAN_EQUAL,
CONTAINS,
NOT_EQUAL,
STRING_LIKE,
}
So if my table has:
diesel::table! {
use diesel::sql_types::*;
use postgis_diesel::sql_types::*;
entities (uid) {
uid -> Uuid,
source -> Text,
source_id -> Text,
...
So for the "source_id" column, I end up with code like:
let mut query: schema::entities::BoxedQuery<diesel::pg::Pg> = schema::entities::table.into_boxed();
if "source_id".eq( fitler_column.as_str() ) {
...
match op {
FilterOp::EQUAL => {
query = query.filter( source_id.eq( filter_value.string_value() ) );
},
FilterOp::NOT_EQUAL => {
query = query.filter( source_id.ne( filter_value.string_value() ) );
},
FilterOp::LESS_THAN => {
query = query.filter( source_id.lt( filter_value.string_value() ) );
},
And this works similarly for other column types (a uuid string would be converted to a Uuid instance). But it's a lot of duplicated code (for each column).
So I want to extract out the match, something like: (for the "source" column):
query_filter(op,
&mut query,
&schema::entities::dsl::source,
filter_value.string_value() );
But I'm struggling to properly define query_filter. It needs to look something like:
fn query_filter<T, V> ( op: FilterOp,
query: &mut schema::entities::BoxedQuery<diesel::pg::Pg>,
column: &T,
filter_value: &V )
{
use crate::schema::entities::dsl::*;
match op {
FilterOp::EQUAL => {
*query = query.filter( column.eq( filter_value ) );
},
...
So this parameterizes the column and it's SQL type (T) and the corresponding native Rust type (V). I'm stumped on what the T and V types should be declared as.
Does this need to be macro instead?
Thanks!
As you post does not really contain a minimal example I've built the following one:
Now before we go to the "solution" first a word of warning: Writing abstract code for diesel is possible, but requires handling a lot of trait bounds. So be sure that you have a good understanding of how rusts trait system works before you start toying around with this kind of code.
Now the first thing that needs to be answered is how to restrict the types
C
andV
. We can look at the diesel documentation here and find that there is a trait calledColumn
, which seems to fit our usecase forC
. Otherwise we can also look atExpressionMethods::eq
method and see that it's implemented for all types that implementExpression
.Column
impliesExpression
so we are fine there. ForV
we look again at theeq
method and see that the argument is bound onT: AsExpression<Self::SqlType>,
This indicates that we need to restrictV
on that same trait.This leaves us with the following starting trait bounds:
C: Column
andV: AsExpression<C::SqlType>
.If we try that the compiler complains about "the trait
diesel::sql_types::SqlType
is not implemented for<C as diesel::Expression>::SqlType
", so we add the suggested trait bound:C::SqlType: SqlType
If we try that the compiler complains again about "the method
eq
exists for type parameterC
, but its trait bounds were not satisfied". Now it gives the non meaningful suggestion to restrictC
toIterator
. The help also contains the following block:So lets try that
C: SingleValue
suggestion instead as that is some diesel trait.If we try that the compiler complains again about "the trait bound
C: ValidGrouping<()>
is not satisfied" and suggests to addC: ValidGrouping<()>
.If we try that the compiler complains again about "the trait bound
<V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression: ValidGrouping<()>
is not satisfied" and suggests that to add that bound as well. Now that can be written more shortly asdiesel::dsl::AsExpr<V, C>: ValidGrouping<()>
If we try that the compiler complains again about "the trait bound
<C as ValidGrouping<()>>::IsAggregate: MixedAggregates<<<V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression as ValidGrouping<()>>::IsAggregate>
is not satisfied" and suggests to add another trait bound. The suggested bound can be written more easily by using<C as ValidGrouping<()>>::IsAggregate:MixedAggregates<<dsl::AsExpr<V, C> as ValidGrouping<()>>::IsAggregate>
If we try that again the compiler complain again about "the trait bound
<<C as ValidGrouping<()>>::IsAggregate as MixedAggregates<<<V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression as ValidGrouping<()>>::IsAggregate>>::Output: MixedAggregates<diesel::expression::is_aggregate::No>
is not satisfied" and suggests another trait bound. Unfortunately that suggestion is a trap. Instead we need to modify the last bound to<C as ValidGrouping<()>>::IsAggregate:MixedAggregates<<dsl::AsExpr<V, C> as ValidGrouping<()>>::IsAggregate, Output = is_aggregate::No>
If we do that the compiler complains again about "the trait bound
<<C as diesel::Expression>::SqlType as diesel::sql_types::SqlType>::IsNull: OneIsNullable<<<C as diesel::Expression>::SqlType as diesel::sql_types::SqlType>::IsNull>
is not satisfied" and suggests another long trait bound. This can be simplified by using the bound from the "required forexpression::operators::Eq<C, <V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression>
to implementdiesel::Expression
" line instead. This gives us the following bound:dsl::Eq<C, V>: Expression
If we do that the compiler complain again about "the trait bound
<expression::grouped::Grouped<expression::operators::Eq<C, <V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression>> as diesel::Expression>::SqlType: BoolOrNullableBool
is not satisfied" and suggest another trait bound.If we try that the compiler complains again about "the trait bound
C: AppearsOnTable<users::table>
is not satisfied" and suggests to add another trait bound. That bound is somewhat helpful, but it's better to add a bound based on one of the trait bounds in the following note line: "required forexpression::operators::Eq<C, <V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression>
to implementAppearsOnTable<users::table>
". This results in the bounddsl::Eq<C, V>: AppearsOnTable<users::table>
If we try that the compiler complains again about "the trait bound
C: QueryFragment<Sqlite>
is not satisfied". Again it's better to add the trait bound todsl::Eq
based on the required because … lines.If we try that the compiler complains again about "
C
cannot be sent between threads safely". Again it's better to add the trait bound todsl::Eq
If we try that the compiler complain about "the associated type
<V as AsExpression<<C as diesel::Expression>::SqlType>>::Expression
may not live long enough" and indicates that there is an hidden lifetime bound in theBoxedQuery
type. So we add a named lifetime'a
there and restrict the relevant type to live at least as long as'a
. The same error message is emitted forC
so we add the bound there as well.After adding these bounds the code finally compiles. This gives the following code:
The first 5 bounds are general bounds that are required for all operators. The last 2 bounds (both lines containing
dsl::Eq
) are bounds specific for the operator you use in this function. If you use several operators these lines need to be duplicated and adjusted for the other operators. For example for supporting.ne()
the following additional bounds are required:The overall message of this is: