JSONB data with Rust and Diesel

栏目: IT技术 · 发布时间: 4年前

内容简介:PostgreSQL JSONB columns are quite handy, there many times that I use them actually. I usually employ JSONB for (meta)data that even if they get lost, we are fine, we just need to do more manual work.The usual cases include:Working lately in Rust, I was tr

PostgreSQL JSONB columns are quite handy, there many times that I use them actually. I usually employ JSONB for (meta)data that even if they get lost, we are fine, we just need to do more manual work.

The usual cases include:

meta

Working lately in Rust, I was trying to figure out how can I use a JSONB column with diesel. It turns out that it’s not that difficult actually!

So let’s say that we have the following db-related struct (the context taken from an OCPP -related project):

#[derive(Debug, Queryable, QueryableByName)]
#[table_name = "remote_starts"]
pub struct RemoteStart {
    id: Uuid,
    created_at: chrono::DateTime<Utc>,
    updated_at: chrono::DateTime<Utc>,
}

First we need to add the migration

ALTER TABLE remote_starts ADD COLUMN meta JSONB DEFAULT '{}'::jsonb NOT NULL;

It’s important to define a default, in general nullable columns should be avoided. Also, we can’t set the default to the current structure of the json document we are thinking to build because

  • it’s live system and we don’t even have the data to normalize existing rows
  • we don’t even know how the document will look like in the future, and every time we want to add/remove attributes from the document we also need to update the default migration, to keep things in sync. That’s quite tedious and we should shift type checking in the application level instead. As I said I mostly treat JSONB column data as “nice to have”, rather than crtical application data.

Hence, having the default as an empty json, should be enough, and we can shift type checking in the application layer.

Then we define the struct that will handle the data of the jsonb column. Of course, we need to derive the appropriate traits for it:

#[derive(FromSqlRow, AsExpression, serde::Serialize, serde::Deserialize, Debug, Default)]
#[sql_type = "Jsonb"]
pub struct Meta {
    pub error_info: Option<String>,
    pub suggested_id_tag: Option<String>,
}

Here we have each field of Meta as optional, and probably it’s a good idea since we don’t want to be super strict for the data found inside.

The next part and the most tricky, is to derive the FromSql and ToSql Diesel traits, that Diesel uses internally to serialize and deserialize the column. Just a note, when I say serialize/deserialize, I mean in the sql terms, not JSON, it just happens here that the column we want to serialize/deserialize is a JSON(B) column. After some research I figured out:

impl FromSql<Jsonb, Pg> for RemoteStartMeta {
    fn from_sql(bytes: Option<&[u8]>) -> diesel::deserialize::Result<Self> {
        let value = <serde_json::Value as FromSql<Jsonb, Pg>>::from_sql(bytes)?;
        Ok(serde_json::from_value(value)?)
    }
}

impl ToSql<Jsonb, Pg> for RemoteStartMeta {
    fn to_sql<W: Write>(&self, out: &mut Output<W, Pg>) -> diesel::serialize::Result {
        let value = serde_json::to_value(self)?;
        <serde_json::Value as ToSql<Jsonb, Pg>>::to_sql(&value, out)
    }
}

Yes it’s not that straightforward, even for me personally that I have been working 9 months with Rust. But if you look the big picture it actually makes sense: for the FromSql , we read the value and serialize it using Serde, while for the ToSql trait, we just serialize the value again using Serde. The tricky part is to derive the correct types. Note that Pg is diesel::pg::Pg , and we could actually use the generic trait of diesel::backend::Backend , that is more generic and does work for all 3 sql databases (postgres, mysql and sqlite). We could I said, as I couldn’t figure out actually how, I gave up after 1 hour fight with the compiler..

Anyway, once you have those traits, then you can update the db-struct and you should be good to go:

#[derive(Debug, Queryable, QueryableByName)]
#[table_name = "remote_starts"]
pub struct RemoteStart {
    id: Uuid,
    created_at: chrono::DateTime<Utc>,
    updated_at: chrono::DateTime<Utc>,
    meta: Meta
}

Updating the jsonb attributes

So what can you do when you wanna update a simple field of json attribute? Let’s first see how we would do that in postgreSQL. Let’s say that we get back an error (note: OCPP operations are mostly async), and we need to update our db with the error info for further inspection. Using postgresql we would do:

UPDATE remote_starts SET meta = jsonb_set(meta, '{error_info}', '"ConnectionError"');

If we are fine with loosening type checking a bit (and shift more checks over the db itself), we could use the AsChangeset trait that would enable us to make all fields optional and when updating the struct, diesel will update only the ones that are Some. It’s a quite handy pattern if you want to build up something quickly.

But as I said, that is problematic because:

  • you lose on type checking
  • it has a lot of boilerplate for JSONB columns, because the None/Some that diesel uses to check which columns to update, works only on column level. Which means that if you want to update a single JSONB attribute, you need to make sure that you write the whole column in order to keep intact the rest attributes, so you basically don’t actually run the update above but rather a complete column update.

However, it turns out that Diesel has a magic macro named sql_function! that can help us. As the documentation says:

Diesel only provides support for a very small number of SQL functions. This macro enables you to add additional functions from the SQL standard, as well as any custom functions your application might have. The syntax for this macro is very similar to that of a normal Rust function, except the argument and return types will be the SQL types being used. Typically these types will come from diesel::sql_types This macro will generate two items. A function with the name that you’ve given, and a module with a helper type representing the return type of your function.

So basically we should define the postgres function with the postgres types, but with rust syntax. Let’s look again what function we use to update the meta :

jsonb_set(meta, '{error_info}', '"ConnectionError"')

Let’s find what the postgresql documentation says exactly about this function:

jsonb_set(target jsonb, path text[], new_value jsonb [, create_missing boolean])

So, using sql_function! that would be:

sql_function!{
    fn jsonb_set(target: Jsonb, path: Array<Text>, new_value: Jsonb) -> Jsonb
}

I think it seems quite straightforward, right? At least until you figure out that Diesel provides you that little gem.

So how can we use it? As you would use anything else in diesel, when updating a column. The sql_function! macro above has created a jsonb_set function, with the appropriate types/signature ready to be used.

diesel::update(remote_starts::table.filter(remote_starts::id.eq(id))).set(
    remote_starts::meta.eq(jsonb_set(
        remote_starts::meta,
        vec![String::from("error_info")],
        serde_json::Value::String(error.unwrap_or("")), //we will improve that part later by assigning proper null
    )),
)

Note: error here is an Option<String> argument to that function that does the update, so replace the error with just Some(ConnectionError) , None or any other error you could think.

Ok great, only that having "error_info" string hand-typed in there breaks the whole idea of type checking. A simple miss and the JSONB data is messed up.. What can we do better? We could extract it as a separate function:

pub fn update_error_info_jsonb_expression(
    error_info: Option<String>,
) -> jsonb_set::jsonb_set<
    remote_starts::columns::meta,
    diesel::expression::bound::Bound<
        diesel::sql_types::Array<diesel::sql_types::Text>,
        std::vec::Vec<String>,
    >,
    diesel::expression::bound::Bound<diesel::sql_types::Jsonb, serde_json::value::Value>,
> {
    let error_info = match error_info {
        Some(error_info) => serde_json::Value::String(error_info),
        _ => serde_json::Value::Null,
    };

    super::jsonb_set(
        remote_starts::meta,
        vec![String::from("error_info")],
        error_info,
    )
}

Yeap that’s what Rust does internally with type inference in the previous example. Well almost, previously we would unwrap the error with empty string in case of None , here we set proper null in case of an empty string.

But my point here is that, instead of writing manually the json attribute you want to update each time (like error_info ), we wrap that part in a function to save ourselves from bugs. We still need to explicitly hand-write the error_info , only that we write it once, wrap it in a function and use that. So we increased safety a bit and minimized the possibility of introducing a bug.

Talking about safety, actually in Ruby, that specific part would be “safer” here, mostly because with some metaprogramming, you wouldn’t type the attribute name, but rather you would call a method name, and if it didn’t exist it would throw a runtime error and you wouldn’t mess up your data (but you would have to fix that error though). Of course, given that you are working with a sane library and not a pretty lame one that uses method_missing and allows you to write any attribute… I am noting that here cause there is a lot of hype in Rust regarding safety, but there are many concepts of “safety” when writing software.

But in fact, in Rust you can also make it safer, ruby-equivalent safe at least, by wrapping the whole thing in a macro. However writing that macro is hard, way way harder than Ruby metaprogramming, so that’s all I could get for safety, the macro stuff probably in another blog post..


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

程序员的英语

程序员的英语

[韩]朴栽浒、[韩]李海永 / 颜廷连 / 人民邮电出版社 / 2018-2 / 49.00元

高考以后就把英语都还给老师了? 写代码特顺溜,一到英语就卡壳? 常见的语法书太枯燥,单词书又太宽泛? 不用急,快来加入针对开发人员的英语读解能力训练项目! - 安全与黑客攻击、无人机与机器人、大数据、物联网、云计算,顺应新技术潮流! - 语法、单词、完形填空、阅读理解、翻译,多角度提升读解能力! - 英语母语技术人员审校,提供“语言和技术”双保险!一起来看看 《程序员的英语》 这本书的介绍吧!

HTML 压缩/解压工具
HTML 压缩/解压工具

在线压缩/解压 HTML 代码

RGB转16进制工具
RGB转16进制工具

RGB HEX 互转工具

SHA 加密
SHA 加密

SHA 加密工具