use crate::entities::revision::{md5, RepeatedRevision, Revision}; use crate::errors::{internal_error, CollaborateError, CollaborateResult}; use crate::util::{cal_diff, make_delta_from_revisions}; use bytes::Bytes; use flowy_grid_data_model::entities::{ FieldChangeset, FieldMeta, FieldOrder, GridBlockMeta, GridBlockMetaChangeset, GridMeta, RepeatedFieldOrder, }; use lib_infra::uuid; use lib_ot::core::{OperationTransformable, PlainTextAttributes, PlainTextDelta, PlainTextDeltaBuilder}; use std::collections::HashMap; use std::sync::Arc; pub type GridMetaDelta = PlainTextDelta; pub type GridDeltaBuilder = PlainTextDeltaBuilder; pub struct GridMetaPad { pub(crate) grid_meta: Arc, pub(crate) delta: GridMetaDelta, } impl GridMetaPad { pub fn from_delta(delta: GridMetaDelta) -> CollaborateResult { let s = delta.to_str()?; let grid: GridMeta = serde_json::from_str(&s) .map_err(|e| CollaborateError::internal().context(format!("Deserialize delta to grid failed: {}", e)))?; Ok(Self { grid_meta: Arc::new(grid), delta, }) } pub fn from_revisions(_grid_id: &str, revisions: Vec) -> CollaborateResult { let grid_delta: GridMetaDelta = make_delta_from_revisions::(revisions)?; Self::from_delta(grid_delta) } pub fn create_field( &mut self, new_field_meta: FieldMeta, start_field_id: Option, ) -> CollaborateResult> { self.modify_grid(|grid| { // Check if the field exists or not if grid.fields.iter().any(|field_meta| field_meta.id == new_field_meta.id) { tracing::warn!("Duplicate grid field"); return Ok(None); } let insert_index = match start_field_id { None => None, Some(start_field_id) => grid.fields.iter().position(|field| field.id == start_field_id), }; match insert_index { None => grid.fields.push(new_field_meta), Some(index) => grid.fields.insert(index, new_field_meta), } Ok(Some(())) }) } pub fn delete_field(&mut self, field_id: &str) -> CollaborateResult> { self.modify_grid(|grid| match grid.fields.iter().position(|field| field.id == field_id) { None => Ok(None), Some(index) => { grid.fields.remove(index); Ok(Some(())) } }) } pub fn contain_field(&self, field_id: &str) -> bool { self.grid_meta.fields.iter().any(|field| field.id == field_id) } pub fn get_field(&self, field_id: &str) -> Option<&FieldMeta> { self.grid_meta.fields.iter().find(|field| field.id == field_id) } pub fn get_field_orders(&self) -> Vec { self.grid_meta.fields.iter().map(FieldOrder::from).collect() } pub fn get_field_metas(&self, field_orders: Option) -> CollaborateResult> { match field_orders { None => Ok(self.grid_meta.fields.clone()), Some(field_orders) => { let field_by_field_id = self .grid_meta .fields .iter() .map(|field| (&field.id, field)) .collect::>(); let fields = field_orders .iter() .flat_map(|field_order| match field_by_field_id.get(&field_order.field_id) { None => { tracing::error!("Can't find the field with id: {}", field_order.field_id); None } Some(field) => Some((*field).clone()), }) .collect::>(); Ok(fields) } } } pub fn update_field(&mut self, changeset: FieldChangeset) -> CollaborateResult> { let field_id = changeset.field_id.clone(); self.modify_field(&field_id, |field| { let mut is_changed = None; if let Some(name) = changeset.name { field.name = name; is_changed = Some(()) } if let Some(desc) = changeset.desc { field.desc = desc; is_changed = Some(()) } if let Some(field_type) = changeset.field_type { field.field_type = field_type; is_changed = Some(()) } if let Some(frozen) = changeset.frozen { field.frozen = frozen; is_changed = Some(()) } if let Some(visibility) = changeset.visibility { field.visibility = visibility; is_changed = Some(()) } if let Some(width) = changeset.width { field.width = width; is_changed = Some(()) } if let Some(type_options) = changeset.type_options { field.type_option_json = type_options; is_changed = Some(()) } Ok(is_changed) }) } pub fn create_block(&mut self, block: GridBlockMeta) -> CollaborateResult> { self.modify_grid(|grid| { if grid.block_metas.iter().any(|b| b.block_id == block.block_id) { tracing::warn!("Duplicate grid block"); Ok(None) } else { match grid.block_metas.last() { None => grid.block_metas.push(block), Some(last_block) => { if last_block.start_row_index > block.start_row_index && last_block.len() > block.start_row_index { let msg = "GridBlock's start_row_index should be greater than the last_block's start_row_index and its len".to_string(); return Err(CollaborateError::internal().context(msg)) } grid.block_metas.push(block); } } Ok(Some(())) } }) } pub fn get_blocks(&self) -> Vec { self.grid_meta.block_metas.clone() } pub fn update_block(&mut self, changeset: GridBlockMetaChangeset) -> CollaborateResult> { let block_id = changeset.block_id.clone(); self.modify_block(&block_id, |block| { let mut is_changed = None; if let Some(row_count) = changeset.row_count { block.row_count = row_count; is_changed = Some(()); } if let Some(start_row_index) = changeset.start_row_index { block.start_row_index = start_row_index; is_changed = Some(()); } Ok(is_changed) }) } pub fn md5(&self) -> String { md5(&self.delta.to_delta_bytes()) } pub fn delta_str(&self) -> String { self.delta.to_delta_str() } pub fn delta_bytes(&self) -> Bytes { self.delta.to_delta_bytes() } pub fn fields(&self) -> &[FieldMeta] { &self.grid_meta.fields } fn modify_grid(&mut self, f: F) -> CollaborateResult> where F: FnOnce(&mut GridMeta) -> CollaborateResult>, { let cloned_grid = self.grid_meta.clone(); match f(Arc::make_mut(&mut self.grid_meta))? { None => Ok(None), Some(_) => { let old = json_from_grid(&cloned_grid)?; let new = json_from_grid(&self.grid_meta)?; match cal_diff::(old, new) { None => Ok(None), Some(delta) => { self.delta = self.delta.compose(&delta)?; Ok(Some(GridChangeset { delta, md5: self.md5() })) } } } } } pub fn modify_block(&mut self, block_id: &str, f: F) -> CollaborateResult> where F: FnOnce(&mut GridBlockMeta) -> CollaborateResult>, { self.modify_grid( |grid| match grid.block_metas.iter().position(|block| block.block_id == block_id) { None => { tracing::warn!("[GridMetaPad]: Can't find any block with id: {}", block_id); Ok(None) } Some(index) => f(&mut grid.block_metas[index]), }, ) } pub fn modify_field(&mut self, field_id: &str, f: F) -> CollaborateResult> where F: FnOnce(&mut FieldMeta) -> CollaborateResult>, { self.modify_grid(|grid| match grid.fields.iter().position(|field| field.id == field_id) { None => { tracing::warn!("[GridMetaPad]: Can't find any field with id: {}", field_id); Ok(None) } Some(index) => f(&mut grid.fields[index]), }) } } fn json_from_grid(grid: &Arc) -> CollaborateResult { let json = serde_json::to_string(grid) .map_err(|err| internal_error(format!("Serialize grid to json str failed. {:?}", err)))?; Ok(json) } pub struct GridChangeset { pub delta: GridMetaDelta, /// md5: the md5 of the grid after applying the change. pub md5: String, } pub fn make_grid_delta(grid_meta: &GridMeta) -> GridMetaDelta { let json = serde_json::to_string(&grid_meta).unwrap(); PlainTextDeltaBuilder::new().insert(&json).build() } pub fn make_grid_revisions(user_id: &str, grid_meta: &GridMeta) -> RepeatedRevision { let delta = make_grid_delta(grid_meta); let bytes = delta.to_delta_bytes(); let revision = Revision::initial_revision(user_id, &grid_meta.grid_id, bytes); revision.into() } impl std::default::Default for GridMetaPad { fn default() -> Self { let grid = GridMeta { grid_id: uuid(), fields: vec![], block_metas: vec![], }; let delta = make_grid_delta(&grid); GridMetaPad { grid_meta: Arc::new(grid), delta, } } }