使用 sqlx 操作 PostgreSQL 数组

PostgreSQL 原生支持数组。本章将讨论如何使用 sqlx 操作 PostgreSQL 的数组。

数据表定义

模型定义

// src/post/model.rs

#[derive(Default, Serialize, Deserialize, FromRow)]
pub struct Post {
    pub id: i32,
    pub content: String,
    pub images: Vec<String>,
}
  • PostgreSQL 中的数组可以直接和 Rust 的 Vec 进行映射

数据操作

pub async fn create(e: impl PgExecutor<'_>, m: &Post) -> sqlx::Result<i32> {
    let (id,): (i32,) =
        sqlx::query_as(r#"INSERT INTO posts ("content", images) VALUES ($1, $2) RETURNING id"#)
            .bind(&m.content)
            .bind(&m.images)
            .fetch_one(e)
            .await?;
    Ok(id)
}

pub async fn find(e: impl PgExecutor<'_>, id: i32) -> sqlx::Result<Option<Post>> {
    sqlx::query_as(r#"SELECT id, "content", images FROM posts WHERE id = $1"#)
        .bind(id)
        .fetch_optional(e)
        .await
}

handler

// src/post/handler.rs

#[derive(Deserialize)]
pub struct CreateForm {
    content: String,
    images: Vec<String>,
}
pub async fn create(
    State(state): State<ArcAppState>,
    Json(frm): Json<CreateForm>,
) -> Result<Json<i32>> {
    let id = model::create(
        &state.pool,
        &model::Post {
            content: frm.content,
            images: frm.images,
            ..Default::default()
        },
    )
    .await?;

    Ok(Json(id))
}

pub async fn find(
    State(state): State<ArcAppState>,
    Path(id): Path<i32>,
) -> Result<Json<model::Post>> {
    let post = match model::find(&state.pool, id).await? {
        Some(v) => v,
        None => return Err(Error::new("不存在的记录")),
    };

    Ok(Json(post))
}

测试

本章代码位于 03.array分支。

要查看完整内容,请先登录