AppFlowy/frontend/rust-lib/flowy-database2/src/entities/row_entities.rs

370 lines
7.9 KiB
Rust
Raw Normal View History

use std::collections::HashMap;
use collab_database::rows::{Row, RowId, RowMeta};
use collab_database::views::RowOrder;
use flowy_derive::ProtoBuf;
use flowy_error::ErrorCode;
use crate::entities::parser::NotEmptyStr;
use crate::services::database::{InsertedRow, UpdatedRow};
/// [RowPB] Describes a row. Has the id of the parent Block. Has the metadata of the row.
#[derive(Debug, Default, Clone, ProtoBuf, Eq, PartialEq)]
pub struct RowPB {
#[pb(index = 1)]
pub id: String,
#[pb(index = 2)]
pub height: i32,
}
impl std::convert::From<&Row> for RowPB {
fn from(row: &Row) -> Self {
Self {
id: row.id.clone().into_inner(),
height: row.height,
}
}
}
impl std::convert::From<Row> for RowPB {
fn from(row: Row) -> Self {
Self {
id: row.id.into_inner(),
height: row.height,
}
}
}
impl From<RowOrder> for RowPB {
fn from(data: RowOrder) -> Self {
Self {
id: data.id.into_inner(),
height: data.height,
}
}
}
#[derive(Debug, Default, Clone, ProtoBuf)]
pub struct RowMetaPB {
#[pb(index = 1)]
pub id: String,
#[pb(index = 2)]
pub document_id: String,
#[pb(index = 3, one_of)]
pub icon: Option<String>,
#[pb(index = 4, one_of)]
pub cover: Option<String>,
}
impl std::convert::From<&RowMeta> for RowMetaPB {
fn from(row_meta: &RowMeta) -> Self {
Self {
id: row_meta.row_id.clone(),
document_id: row_meta.document_id.clone(),
icon: row_meta.icon_url.clone(),
cover: row_meta.cover_url.clone(),
}
}
}
impl std::convert::From<RowMeta> for RowMetaPB {
fn from(row_meta: RowMeta) -> Self {
Self {
id: row_meta.row_id,
document_id: row_meta.document_id,
icon: row_meta.icon_url,
cover: row_meta.cover_url,
}
}
}
#[derive(Debug, Default, Clone, ProtoBuf)]
pub struct UpdateRowMetaChangesetPB {
#[pb(index = 1)]
pub id: String,
#[pb(index = 2)]
pub view_id: String,
#[pb(index = 3, one_of)]
pub icon_url: Option<String>,
#[pb(index = 4, one_of)]
pub cover_url: Option<String>,
}
#[derive(Debug)]
pub struct UpdateRowMetaParams {
pub id: String,
pub view_id: String,
pub icon_url: Option<String>,
pub cover_url: Option<String>,
}
impl TryInto<UpdateRowMetaParams> for UpdateRowMetaChangesetPB {
type Error = ErrorCode;
fn try_into(self) -> Result<UpdateRowMetaParams, Self::Error> {
let row_id = NotEmptyStr::parse(self.id)
.map_err(|_| ErrorCode::RowIdIsEmpty)?
.0;
let view_id = NotEmptyStr::parse(self.view_id)
.map_err(|_| ErrorCode::ViewIdIsInvalid)?
.0;
Ok(UpdateRowMetaParams {
id: row_id,
view_id,
icon_url: self.icon_url,
cover_url: self.cover_url,
})
}
}
#[derive(Debug, Default, Clone, ProtoBuf)]
pub struct UpdateRowPayloadPB {
#[pb(index = 1)]
pub row_id: String,
#[pb(index = 2, one_of)]
pub insert_document: Option<bool>,
#[pb(index = 3, one_of)]
pub insert_comment: Option<RowCommentPayloadPB>,
}
#[derive(Debug, Default, Clone)]
pub struct UpdateRowParams {
pub row_id: String,
pub insert_comment: Option<RowCommentParams>,
}
impl TryInto<UpdateRowParams> for UpdateRowPayloadPB {
type Error = ErrorCode;
fn try_into(self) -> Result<UpdateRowParams, Self::Error> {
let row_id = NotEmptyStr::parse(self.row_id)
.map_err(|_| ErrorCode::RowIdIsEmpty)?
.0;
let insert_comment = self
.insert_comment
.map(|comment| comment.try_into())
.transpose()?;
Ok(UpdateRowParams {
row_id,
insert_comment,
})
}
}
#[derive(Debug, Default, Clone, ProtoBuf)]
pub struct RowCommentPayloadPB {
#[pb(index = 1)]
pub uid: String,
#[pb(index = 2)]
pub comment: String,
}
#[derive(Debug, Default, Clone)]
pub struct RowCommentParams {
pub uid: String,
pub comment: String,
}
impl TryInto<RowCommentParams> for RowCommentPayloadPB {
type Error = ErrorCode;
fn try_into(self) -> Result<RowCommentParams, Self::Error> {
let uid = NotEmptyStr::parse(self.uid)
.map_err(|_| ErrorCode::RowIdIsEmpty)?
.0;
let comment = NotEmptyStr::parse(self.comment)
.map_err(|_| ErrorCode::RowIdIsEmpty)?
.0;
Ok(RowCommentParams { uid, comment })
}
}
#[derive(Debug, Default, ProtoBuf)]
pub struct OptionalRowPB {
#[pb(index = 1, one_of)]
pub row: Option<RowPB>,
}
#[derive(Debug, Default, ProtoBuf)]
pub struct RepeatedRowPB {
#[pb(index = 1)]
pub items: Vec<RowPB>,
}
impl std::convert::From<Vec<RowPB>> for RepeatedRowPB {
fn from(items: Vec<RowPB>) -> Self {
Self { items }
}
}
#[derive(Debug, Clone, Default, ProtoBuf)]
pub struct InsertedRowPB {
#[pb(index = 1)]
pub row_meta: RowMetaPB,
#[pb(index = 2, one_of)]
pub index: Option<i32>,
#[pb(index = 3)]
pub is_new: bool,
}
impl InsertedRowPB {
pub fn new(row_meta: RowMetaPB) -> Self {
Self {
row_meta,
index: None,
is_new: false,
}
}
pub fn with_index(mut self, index: i32) -> Self {
self.index = Some(index);
self
}
}
impl std::convert::From<RowMetaPB> for InsertedRowPB {
fn from(row_meta: RowMetaPB) -> Self {
Self {
row_meta,
index: None,
is_new: false,
}
}
}
impl From<InsertedRow> for InsertedRowPB {
fn from(data: InsertedRow) -> Self {
Self {
row_meta: data.row_meta.into(),
index: data.index,
is_new: data.is_new,
}
}
}
#[derive(Debug, Clone, Default, ProtoBuf)]
pub struct UpdatedRowPB {
#[pb(index = 1)]
pub row_id: String,
// Indicates the field ids of the cells that were updated in this row.
#[pb(index = 2)]
pub field_ids: Vec<String>,
/// The meta of row was updated if this is Some.
#[pb(index = 3, one_of)]
pub row_meta: Option<RowMetaPB>,
}
impl From<UpdatedRow> for UpdatedRowPB {
fn from(data: UpdatedRow) -> Self {
let row_meta = data.row_meta.map(RowMetaPB::from);
Self {
row_id: data.row_id,
field_ids: data.field_ids,
row_meta,
}
}
}
#[derive(Debug, Default, Clone, ProtoBuf)]
pub struct RowIdPB {
#[pb(index = 1)]
pub view_id: String,
#[pb(index = 2)]
pub row_id: String,
#[pb(index = 3, one_of)]
pub group_id: Option<String>,
}
pub struct RowIdParams {
pub view_id: String,
pub row_id: RowId,
pub group_id: Option<String>,
}
impl TryInto<RowIdParams> for RowIdPB {
type Error = ErrorCode;
fn try_into(self) -> Result<RowIdParams, Self::Error> {
let view_id = NotEmptyStr::parse(self.view_id).map_err(|_| ErrorCode::DatabaseIdIsEmpty)?;
let row_id = NotEmptyStr::parse(self.row_id).map_err(|_| ErrorCode::RowIdIsEmpty)?;
let group_id = match self.group_id {
Some(group_id) => Some(
NotEmptyStr::parse(group_id)
.map_err(|_| ErrorCode::GroupIdIsEmpty)?
.0,
),
None => None,
};
Ok(RowIdParams {
view_id: view_id.0,
row_id: RowId::from(row_id.0),
group_id,
})
}
}
#[derive(ProtoBuf, Default)]
pub struct CreateRowPayloadPB {
#[pb(index = 1)]
pub view_id: String,
#[pb(index = 2, one_of)]
pub start_row_id: Option<String>,
#[pb(index = 3, one_of)]
pub group_id: Option<String>,
#[pb(index = 4, one_of)]
pub data: Option<RowDataPB>,
}
#[derive(ProtoBuf, Default)]
pub struct RowDataPB {
#[pb(index = 1)]
pub cell_data_by_field_id: HashMap<String, String>,
}
#[derive(Default)]
pub struct CreateRowParams {
pub view_id: String,
pub start_row_id: Option<RowId>,
pub group_id: Option<String>,
pub cell_data_by_field_id: Option<HashMap<String, String>>,
}
impl TryInto<CreateRowParams> for CreateRowPayloadPB {
type Error = ErrorCode;
fn try_into(self) -> Result<CreateRowParams, Self::Error> {
let view_id = NotEmptyStr::parse(self.view_id).map_err(|_| ErrorCode::ViewIdIsInvalid)?;
let start_row_id = self.start_row_id.map(RowId::from);
Ok(CreateRowParams {
view_id: view_id.0,
start_row_id,
group_id: self.group_id,
cell_data_by_field_id: self.data.map(|data| data.cell_data_by_field_id),
})
}
}