I want to use Postgresql's jsonb_path_exists
function with diesel
.
I started with the function definition which compiles.
diesel::sql_function! {
/// https://www.postgresql.org/docs/current/functions-json.html
fn jsonb_path_exists(
jsonb: diesel::sql_types::Nullable<diesel::sql_types::Jsonb>,
path: diesel::sql_types::Text,
) -> diesel::sql_types::Bool;
}
Now I get a runtime error when I use the function:
mytable::table.filter(
jsonb_path_exists(
activities::activity_object,
format!(r#"$.**.id ? (@ == "some-uuid")"#),
),
)
The above fails with diesel error: function jsonb_path_exists(jsonb, text) does not exist
.
Looking at the signature of jsonb_path_exists
I figured the problem is with the second argument: diesel::sql_types::Text
.
It should be a jsonpath not text.
jsonb_path_exists ( target jsonb, path jsonpath [, vars jsonb [, silent boolean ]] ) → boolean
I confirmed that by debug printing the query, replacing double quotes with single quotes and running it manually against the database. That worked.
I am not sure how to define a new diesel SQL type to make diesel correctly treat the String
as Postgres path
type and emit single quotes.
Looking at the existing types I did not find anything I could implement this analogously to.
EDIT:
Adding a new SqlType
is a start. However, how to implement AsExpression<JsonPathType>
for some struct JsonPath(String)
is unclear.
/// Q: How do we know the oid?
/// A: `SELECT typname, oid, typarray FROM pg_type WHERE typname = 'jsonpath';`
#[derive(Debug, Clone, Copy, Default, QueryId, SqlType)]
#[diesel(postgres_type(oid = 4072, array_oid = 4073))]
pub struct JsonPathType;
With some help from weiznich on gitter, I figured out the conversion. In a gist, the binary wire format is the path string as bytes prepended with a version.
I found the correct version in the postgres codebase here: