123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395 |
- use crate::entities::revision::{md5, RepeatedRevision, Revision};
- use crate::errors::{CollaborateError, CollaborateResult};
- use crate::util::{cal_diff, make_delta_from_revisions};
- use flowy_grid_data_model::entities::{gen_block_id, CellMeta, GridBlockMetaData, RowMeta, RowMetaChangeset};
- use lib_ot::core::{OperationTransformable, PlainTextAttributes, PlainTextDelta, PlainTextDeltaBuilder};
- use serde::{Deserialize, Serialize};
- use std::borrow::Cow;
- use std::collections::HashMap;
- use std::sync::Arc;
- pub type GridBlockMetaDelta = PlainTextDelta;
- pub type GridBlockMetaDeltaBuilder = PlainTextDeltaBuilder;
- #[derive(Debug, Deserialize, Serialize, Clone)]
- pub struct GridBlockMetaPad {
- block_id: String,
- rows: Vec<Arc<RowMeta>>,
- #[serde(skip)]
- pub(crate) delta: GridBlockMetaDelta,
- }
- impl GridBlockMetaPad {
- pub fn from_delta(delta: GridBlockMetaDelta) -> CollaborateResult<Self> {
- let s = delta.to_str()?;
- let meta_data: GridBlockMetaData = serde_json::from_str(&s).map_err(|e| {
- let msg = format!("Deserialize delta to block meta failed: {}", e);
- tracing::error!("{}", s);
- CollaborateError::internal().context(msg)
- })?;
- let block_id = meta_data.block_id;
- let rows = meta_data.rows.into_iter().map(Arc::new).collect::<Vec<Arc<RowMeta>>>();
- Ok(Self { block_id, rows, delta })
- }
- pub fn from_revisions(_grid_id: &str, revisions: Vec<Revision>) -> CollaborateResult<Self> {
- let block_delta: GridBlockMetaDelta = make_delta_from_revisions::<PlainTextAttributes>(revisions)?;
- Self::from_delta(block_delta)
- }
- #[tracing::instrument(level = "trace", skip(self, row), err)]
- pub fn add_row_meta(
- &mut self,
- row: RowMeta,
- start_row_id: Option<String>,
- ) -> CollaborateResult<Option<GridBlockMetaChange>> {
- self.modify(|rows| {
- if let Some(start_row_id) = start_row_id {
- if !start_row_id.is_empty() {
- if let Some(index) = rows.iter().position(|row| row.id == start_row_id) {
- rows.insert(index + 1, Arc::new(row));
- return Ok(Some(()));
- }
- }
- }
- rows.push(Arc::new(row));
- Ok(Some(()))
- })
- }
- pub fn delete_rows(&mut self, row_ids: Vec<Cow<'_, String>>) -> CollaborateResult<Option<GridBlockMetaChange>> {
- self.modify(|rows| {
- rows.retain(|row| !row_ids.contains(&Cow::Borrowed(&row.id)));
- Ok(Some(()))
- })
- }
- pub fn get_row_metas<T>(&self, row_ids: Option<Vec<Cow<'_, T>>>) -> CollaborateResult<Vec<Arc<RowMeta>>>
- where
- T: AsRef<str> + ToOwned + ?Sized,
- {
- match row_ids {
- None => Ok(self.rows.to_vec()),
- Some(row_ids) => {
- let row_map = self
- .rows
- .iter()
- .map(|row| (row.id.as_str(), row.clone()))
- .collect::<HashMap<&str, Arc<RowMeta>>>();
- Ok(row_ids
- .iter()
- .flat_map(|row_id| {
- let row_id = row_id.as_ref().as_ref();
- match row_map.get(row_id) {
- None => {
- tracing::error!("Can't find the row with id: {}", row_id);
- None
- }
- Some(row) => Some(row.clone()),
- }
- })
- .collect::<Vec<_>>())
- }
- }
- }
- pub fn get_cell_metas(
- &self,
- field_id: &str,
- row_ids: Option<Vec<Cow<'_, String>>>,
- ) -> CollaborateResult<Vec<CellMeta>> {
- let rows = self.get_row_metas(row_ids)?;
- let cell_metas = rows
- .iter()
- .flat_map(|row| {
- let cell_meta = row.cells.get(field_id)?;
- Some(cell_meta.clone())
- })
- .collect::<Vec<CellMeta>>();
- Ok(cell_metas)
- }
- pub fn number_of_rows(&self) -> i32 {
- self.rows.len() as i32
- }
- pub fn index_of_row(&self, row_id: &str) -> Option<i32> {
- self.rows
- .iter()
- .position(|row| row.id == row_id)
- .map(|index| index as i32)
- }
- pub fn update_row(&mut self, changeset: RowMetaChangeset) -> CollaborateResult<Option<GridBlockMetaChange>> {
- let row_id = changeset.row_id.clone();
- self.modify_row(&row_id, |row| {
- let mut is_changed = None;
- if let Some(height) = changeset.height {
- row.height = height;
- is_changed = Some(());
- }
- if let Some(visibility) = changeset.visibility {
- row.visibility = visibility;
- is_changed = Some(());
- }
- if !changeset.cell_by_field_id.is_empty() {
- is_changed = Some(());
- changeset.cell_by_field_id.into_iter().for_each(|(field_id, cell)| {
- row.cells.insert(field_id, cell);
- })
- }
- Ok(is_changed)
- })
- }
- pub fn modify<F>(&mut self, f: F) -> CollaborateResult<Option<GridBlockMetaChange>>
- where
- F: for<'a> FnOnce(&'a mut Vec<Arc<RowMeta>>) -> CollaborateResult<Option<()>>,
- {
- let cloned_self = self.clone();
- match f(&mut self.rows)? {
- None => Ok(None),
- Some(_) => {
- let old = cloned_self.to_json()?;
- let new = self.to_json()?;
- match cal_diff::<PlainTextAttributes>(old, new) {
- None => Ok(None),
- Some(delta) => {
- tracing::debug!("[GridBlockMeta] Composing delta {}", delta.to_delta_str());
- tracing::debug!(
- "[GridBlockMeta] current delta: {}",
- self.delta.to_str().unwrap_or_else(|_| "".to_string())
- );
- self.delta = self.delta.compose(&delta)?;
- Ok(Some(GridBlockMetaChange { delta, md5: self.md5() }))
- }
- }
- }
- }
- }
- fn modify_row<F>(&mut self, row_id: &str, f: F) -> CollaborateResult<Option<GridBlockMetaChange>>
- where
- F: FnOnce(&mut RowMeta) -> CollaborateResult<Option<()>>,
- {
- self.modify(|rows| {
- if let Some(row_meta) = rows.iter_mut().find(|row_meta| row_id == row_meta.id) {
- f(Arc::make_mut(row_meta))
- } else {
- tracing::warn!("[BlockMetaPad]: Can't find any row with id: {}", row_id);
- Ok(None)
- }
- })
- }
- pub fn to_json(&self) -> CollaborateResult<String> {
- serde_json::to_string(self)
- .map_err(|e| CollaborateError::internal().context(format!("serial trash to json failed: {}", e)))
- }
- pub fn md5(&self) -> String {
- md5(&self.delta.to_delta_bytes())
- }
- pub fn delta_str(&self) -> String {
- self.delta.to_delta_str()
- }
- }
- pub struct GridBlockMetaChange {
- pub delta: GridBlockMetaDelta,
- /// md5: the md5 of the grid after applying the change.
- pub md5: String,
- }
- pub fn make_block_meta_delta(grid_block_meta_data: &GridBlockMetaData) -> GridBlockMetaDelta {
- let json = serde_json::to_string(&grid_block_meta_data).unwrap();
- PlainTextDeltaBuilder::new().insert(&json).build()
- }
- pub fn make_block_meta_revisions(user_id: &str, grid_block_meta_data: &GridBlockMetaData) -> RepeatedRevision {
- let delta = make_block_meta_delta(grid_block_meta_data);
- let bytes = delta.to_delta_bytes();
- let revision = Revision::initial_revision(user_id, &grid_block_meta_data.block_id, bytes);
- revision.into()
- }
- impl std::default::Default for GridBlockMetaPad {
- fn default() -> Self {
- let block_meta_data = GridBlockMetaData {
- block_id: gen_block_id(),
- rows: vec![],
- };
- let delta = make_block_meta_delta(&block_meta_data);
- GridBlockMetaPad {
- block_id: block_meta_data.block_id,
- rows: block_meta_data.rows.into_iter().map(Arc::new).collect::<Vec<_>>(),
- delta,
- }
- }
- }
- #[cfg(test)]
- mod tests {
- use crate::client_grid::{GridBlockMetaDelta, GridBlockMetaPad};
- use flowy_grid_data_model::entities::{RowMeta, RowMetaChangeset};
- use std::borrow::Cow;
- #[test]
- fn block_meta_add_row() {
- let mut pad = test_pad();
- let row = RowMeta {
- id: "1".to_string(),
- block_id: pad.block_id.clone(),
- cells: Default::default(),
- height: 0,
- visibility: false,
- };
- let change = pad.add_row_meta(row.clone(), None).unwrap().unwrap();
- assert_eq!(pad.rows.first().unwrap().as_ref(), &row);
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":24},{"insert":"{\"id\":\"1\",\"block_id\":\"1\",\"cells\":{},\"height\":0,\"visibility\":false}"},{"retain":2}]"#
- );
- }
- #[test]
- fn block_meta_insert_row() {
- let mut pad = test_pad();
- let row_1 = test_row_meta("1", &pad);
- let row_2 = test_row_meta("2", &pad);
- let row_3 = test_row_meta("3", &pad);
- let change = pad.add_row_meta(row_1.clone(), None).unwrap().unwrap();
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":24},{"insert":"{\"id\":\"1\",\"block_id\":\"1\",\"cells\":{},\"height\":0,\"visibility\":false}"},{"retain":2}]"#
- );
- let change = pad.add_row_meta(row_2.clone(), None).unwrap().unwrap();
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":90},{"insert":",{\"id\":\"2\",\"block_id\":\"1\",\"cells\":{},\"height\":0,\"visibility\":false}"},{"retain":2}]"#
- );
- let change = pad.add_row_meta(row_3.clone(), Some("2".to_string())).unwrap().unwrap();
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":157},{"insert":",{\"id\":\"3\",\"block_id\":\"1\",\"cells\":{},\"height\":0,\"visibility\":false}"},{"retain":2}]"#
- );
- assert_eq!(*pad.rows[0], row_1);
- assert_eq!(*pad.rows[1], row_2);
- assert_eq!(*pad.rows[2], row_3);
- }
- fn test_row_meta(id: &str, pad: &GridBlockMetaPad) -> RowMeta {
- RowMeta {
- id: id.to_string(),
- block_id: pad.block_id.clone(),
- cells: Default::default(),
- height: 0,
- visibility: false,
- }
- }
- #[test]
- fn block_meta_insert_row2() {
- let mut pad = test_pad();
- let row_1 = test_row_meta("1", &pad);
- let row_2 = test_row_meta("2", &pad);
- let row_3 = test_row_meta("3", &pad);
- let _ = pad.add_row_meta(row_1.clone(), None).unwrap().unwrap();
- let _ = pad.add_row_meta(row_2.clone(), None).unwrap().unwrap();
- let _ = pad.add_row_meta(row_3.clone(), Some("1".to_string())).unwrap().unwrap();
- assert_eq!(*pad.rows[0], row_1);
- assert_eq!(*pad.rows[1], row_3);
- assert_eq!(*pad.rows[2], row_2);
- }
- #[test]
- fn block_meta_insert_row3() {
- let mut pad = test_pad();
- let row_1 = test_row_meta("1", &pad);
- let row_2 = test_row_meta("2", &pad);
- let row_3 = test_row_meta("3", &pad);
- let _ = pad.add_row_meta(row_1.clone(), None).unwrap().unwrap();
- let _ = pad.add_row_meta(row_2.clone(), None).unwrap().unwrap();
- let _ = pad.add_row_meta(row_3.clone(), Some("".to_string())).unwrap().unwrap();
- assert_eq!(*pad.rows[0], row_1);
- assert_eq!(*pad.rows[1], row_2);
- assert_eq!(*pad.rows[2], row_3);
- }
- #[test]
- fn block_meta_delete_row() {
- let mut pad = test_pad();
- let pre_delta_str = pad.delta_str();
- let row = RowMeta {
- id: "1".to_string(),
- block_id: pad.block_id.clone(),
- cells: Default::default(),
- height: 0,
- visibility: false,
- };
- let _ = pad.add_row_meta(row.clone(), None).unwrap().unwrap();
- let change = pad.delete_rows(vec![Cow::Borrowed(&row.id)]).unwrap().unwrap();
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":24},{"delete":66},{"retain":2}]"#
- );
- assert_eq!(pad.delta_str(), pre_delta_str);
- }
- #[test]
- fn block_meta_update_row() {
- let mut pad = test_pad();
- let row = RowMeta {
- id: "1".to_string(),
- block_id: pad.block_id.clone(),
- cells: Default::default(),
- height: 0,
- visibility: false,
- };
- let changeset = RowMetaChangeset {
- row_id: row.id.clone(),
- height: Some(100),
- visibility: Some(true),
- cell_by_field_id: Default::default(),
- };
- let _ = pad.add_row_meta(row, None).unwrap().unwrap();
- let change = pad.update_row(changeset).unwrap().unwrap();
- assert_eq!(
- change.delta.to_delta_str(),
- r#"[{"retain":69},{"insert":"10"},{"retain":15},{"insert":"tru"},{"delete":4},{"retain":4}]"#
- );
- assert_eq!(
- pad.to_json().unwrap(),
- r#"{"block_id":"1","rows":[{"id":"1","block_id":"1","cells":[],"height":100,"visibility":true}]}"#
- );
- }
- fn test_pad() -> GridBlockMetaPad {
- let delta = GridBlockMetaDelta::from_delta_str(r#"[{"insert":"{\"block_id\":\"1\",\"rows\":[]}"}]"#).unwrap();
- GridBlockMetaPad::from_delta(delta).unwrap()
- }
- }
|