use crate::entities::grid::{FieldChangesetParams, GridSettingChangesetParams}; 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::revision::{ gen_block_id, gen_grid_filter_id, gen_grid_group_id, gen_grid_id, gen_grid_sort_id, FieldRevision, FieldTypeRevision, GridBlockMetaRevision, GridBlockMetaRevisionChangeset, GridFilterRevision, GridGroupRevision, GridLayoutRevision, GridRevision, GridSettingRevision, GridSortRevision, }; use lib_infra::util::move_vec_element; use lib_ot::core::{OperationTransform, PhantomAttributes, TextDelta, TextDeltaBuilder}; use std::collections::HashMap; use std::sync::Arc; pub type GridRevisionDelta = TextDelta; pub type GridRevisionDeltaBuilder = TextDeltaBuilder; pub struct GridRevisionPad { grid_rev: Arc, delta: GridRevisionDelta, } pub trait JsonDeserializer { fn deserialize(&self, type_option_data: Vec) -> CollaborateResult; } impl GridRevisionPad { pub fn grid_id(&self) -> String { self.grid_rev.grid_id.clone() } pub async fn duplicate_grid_block_meta(&self) -> (Vec, Vec) { let fields = self .grid_rev .fields .iter() .map(|field_rev| field_rev.as_ref().clone()) .collect(); let blocks = self .grid_rev .blocks .iter() .map(|block| { let mut duplicated_block = (&*block.clone()).clone(); duplicated_block.block_id = gen_block_id(); duplicated_block }) .collect::>(); (fields, blocks) } pub fn from_delta(delta: GridRevisionDelta) -> CollaborateResult { let content = delta.content()?; let grid: GridRevision = serde_json::from_str(&content) .map_err(|e| CollaborateError::internal().context(format!("Deserialize delta to grid failed: {}", e)))?; Ok(Self { grid_rev: Arc::new(grid), delta, }) } pub fn from_revisions(revisions: Vec) -> CollaborateResult { let grid_delta: GridRevisionDelta = make_delta_from_revisions::(revisions)?; Self::from_delta(grid_delta) } #[tracing::instrument(level = "debug", skip_all, err)] pub fn create_field_rev( &mut self, new_field_rev: FieldRevision, start_field_id: Option, ) -> CollaborateResult> { self.modify_grid(|grid_meta| { // Check if the field exists or not if grid_meta .fields .iter() .any(|field_rev| field_rev.id == new_field_rev.id) { tracing::error!("Duplicate grid field"); return Ok(None); } let insert_index = match start_field_id { None => None, Some(start_field_id) => grid_meta.fields.iter().position(|field| field.id == start_field_id), }; let new_field_rev = Arc::new(new_field_rev); match insert_index { None => grid_meta.fields.push(new_field_rev), Some(index) => grid_meta.fields.insert(index, new_field_rev), } Ok(Some(())) }) } pub fn delete_field_rev(&mut self, field_id: &str) -> CollaborateResult> { self.modify_grid( |grid_meta| match grid_meta.fields.iter().position(|field| field.id == field_id) { None => Ok(None), Some(index) => { grid_meta.fields.remove(index); Ok(Some(())) } }, ) } pub fn duplicate_field_rev( &mut self, field_id: &str, duplicated_field_id: &str, ) -> CollaborateResult> { self.modify_grid( |grid_meta| match grid_meta.fields.iter().position(|field| field.id == field_id) { None => Ok(None), Some(index) => { let mut duplicate_field_rev = grid_meta.fields[index].as_ref().clone(); duplicate_field_rev.id = duplicated_field_id.to_string(); duplicate_field_rev.name = format!("{} (copy)", duplicate_field_rev.name); grid_meta.fields.insert(index + 1, Arc::new(duplicate_field_rev)); Ok(Some(())) } }, ) } pub fn switch_to_field( &mut self, field_id: &str, field_type: T, type_option_json_builder: B, ) -> CollaborateResult> where B: FnOnce(&FieldTypeRevision) -> String, T: Into, { let field_type = field_type.into(); self.modify_grid(|grid_meta| { // match grid_meta.fields.iter_mut().find(|field_rev| field_rev.id == field_id) { None => { tracing::warn!("Can not find the field with id: {}", field_id); Ok(None) } Some(field_rev) => { let mut_field_rev = Arc::make_mut(field_rev); if mut_field_rev.get_type_option_str(field_type).is_none() { let type_option_json = type_option_json_builder(&field_type); mut_field_rev.insert_type_option_str(&field_type, type_option_json); } mut_field_rev.field_type_rev = field_type; Ok(Some(())) } } }) } pub fn update_field_rev( &mut self, changeset: FieldChangesetParams, deserializer: T, ) -> 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_rev = 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_option_data) = changeset.type_option_data { match deserializer.deserialize(type_option_data) { Ok(json_str) => { let field_type = field.field_type_rev; field.insert_type_option_str(&field_type, json_str); is_changed = Some(()) } Err(err) => { tracing::error!("Deserialize data to type option json failed: {}", err); } } } Ok(is_changed) }) } pub fn get_field_rev(&self, field_id: &str) -> Option<(usize, &Arc)> { self.grid_rev .fields .iter() .enumerate() .find(|(_, field)| field.id == field_id) } pub fn replace_field_rev(&mut self, field_rev: Arc) -> CollaborateResult> { self.modify_grid( |grid_meta| match grid_meta.fields.iter().position(|field| field.id == field_rev.id) { None => Ok(None), Some(index) => { grid_meta.fields.remove(index); grid_meta.fields.insert(index, field_rev); Ok(Some(())) } }, ) } pub fn move_field( &mut self, field_id: &str, from_index: usize, to_index: usize, ) -> CollaborateResult> { self.modify_grid(|grid_meta| { match move_vec_element( &mut grid_meta.fields, |field| field.id == field_id, from_index, to_index, ) .map_err(internal_error)? { true => Ok(Some(())), false => Ok(None), } }) } pub fn contain_field(&self, field_id: &str) -> bool { self.grid_rev.fields.iter().any(|field| field.id == field_id) } pub fn get_field_revs(&self, field_ids: Option>) -> CollaborateResult>> { match field_ids { None => Ok(self.grid_rev.fields.clone()), Some(field_ids) => { let field_by_field_id = self .grid_rev .fields .iter() .map(|field| (&field.id, field)) .collect::>>(); let fields = field_ids .iter() .flat_map(|field_id| match field_by_field_id.get(&field_id) { None => { tracing::error!("Can't find the field with id: {}", field_id); None } Some(field) => Some((*field).clone()), }) .collect::>>(); Ok(fields) } } } pub fn create_block_meta_rev(&mut self, block: GridBlockMetaRevision) -> CollaborateResult> { self.modify_grid(|grid_meta| { if grid_meta.blocks.iter().any(|b| b.block_id == block.block_id) { tracing::warn!("Duplicate grid block"); Ok(None) } else { match grid_meta.blocks.last() { None => grid_meta.blocks.push(Arc::new(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_meta.blocks.push(Arc::new(block)); } } Ok(Some(())) } }) } pub fn get_block_meta_revs(&self) -> Vec> { self.grid_rev.blocks.clone() } pub fn update_block_rev( &mut self, changeset: GridBlockMetaRevisionChangeset, ) -> 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 get_grid_setting_rev(&self) -> &GridSettingRevision { &self.grid_rev.setting } /// If layout is None, then the default layout will be the read from GridSettingRevision pub fn get_filters( &self, layout: Option<&GridLayoutRevision>, field_ids: Option>, ) -> Option>> { let mut filter_revs = vec![]; let layout_ty = layout.unwrap_or(&self.grid_rev.setting.layout); let field_revs = self.get_field_revs(None).ok()?; field_revs.iter().for_each(|field_rev| { let mut is_contain = true; if let Some(field_ids) = &field_ids { is_contain = field_ids.contains(&field_rev.id); } if is_contain { // Only return the filters for the current fields' type. let field_id = &field_rev.id; let field_type_rev = &field_rev.field_type_rev; if let Some(mut t_filter_revs) = self.grid_rev.setting.get_filters(layout_ty, field_id, field_type_rev) { filter_revs.append(&mut t_filter_revs); } } }); Some(filter_revs) } pub fn update_grid_setting_rev( &mut self, changeset: GridSettingChangesetParams, ) -> CollaborateResult> { self.modify_grid(|grid_rev| { let mut is_changed = None; let layout_rev = changeset.layout_type; if let Some(params) = changeset.insert_filter { let filter_rev = GridFilterRevision { id: gen_grid_filter_id(), field_id: params.field_id.clone(), condition: params.condition, content: params.content, }; grid_rev .setting .insert_filter(&layout_rev, ¶ms.field_id, ¶ms.field_type_rev, filter_rev); is_changed = Some(()) } if let Some(params) = changeset.delete_filter { match grid_rev .setting .get_mut_filters(&layout_rev, ¶ms.field_id, ¶ms.field_type_rev) { Some(filters) => { filters.retain(|filter| filter.id != params.filter_id); } None => { tracing::warn!("Can't find the filter with {:?}", layout_rev); } } } if let Some(params) = changeset.insert_group { let rev = GridGroupRevision { id: gen_grid_group_id(), field_id: params.field_id, sub_field_id: params.sub_field_id, }; grid_rev .setting .groups .entry(layout_rev.clone()) .or_insert_with(std::vec::Vec::new) .push(rev); is_changed = Some(()) } if let Some(delete_group_id) = changeset.delete_group { match grid_rev.setting.groups.get_mut(&layout_rev) { Some(groups) => groups.retain(|group| group.id != delete_group_id), None => { tracing::warn!("Can't find the group with {:?}", layout_rev); } } } if let Some(sort) = changeset.insert_sort { let rev = GridSortRevision { id: gen_grid_sort_id(), field_id: sort.field_id, }; grid_rev .setting .sorts .entry(layout_rev.clone()) .or_insert_with(std::vec::Vec::new) .push(rev); is_changed = Some(()) } if let Some(delete_sort_id) = changeset.delete_sort { match grid_rev.setting.sorts.get_mut(&layout_rev) { Some(sorts) => sorts.retain(|sort| sort.id != delete_sort_id), None => { tracing::warn!("Can't find the sort with {:?}", layout_rev); } } } Ok(is_changed) }) } pub fn md5(&self) -> String { md5(&self.delta.json_bytes()) } pub fn delta_str(&self) -> String { self.delta.json_str() } pub fn delta_bytes(&self) -> Bytes { self.delta.json_bytes() } pub fn fields(&self) -> &[Arc] { &self.grid_rev.fields } fn modify_grid(&mut self, f: F) -> CollaborateResult> where F: FnOnce(&mut GridRevision) -> CollaborateResult>, { let cloned_grid = self.grid_rev.clone(); match f(Arc::make_mut(&mut self.grid_rev))? { None => Ok(None), Some(_) => { let old = make_grid_rev_json_str(&cloned_grid)?; let new = self.json_str()?; match cal_diff::(old, new) { None => Ok(None), Some(delta) => { self.delta = self.delta.compose(&delta)?; Ok(Some(GridChangeset { delta, md5: self.md5() })) } } } } } fn modify_block(&mut self, block_id: &str, f: F) -> CollaborateResult> where F: FnOnce(&mut GridBlockMetaRevision) -> CollaborateResult>, { self.modify_grid( |grid_rev| match grid_rev.blocks.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) => { let block_rev = Arc::make_mut(&mut grid_rev.blocks[index]); f(block_rev) } }, ) } fn modify_field(&mut self, field_id: &str, f: F) -> CollaborateResult> where F: FnOnce(&mut FieldRevision) -> CollaborateResult>, { self.modify_grid( |grid_rev| match grid_rev.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) => { let mut_field_rev = Arc::make_mut(&mut grid_rev.fields[index]); f(mut_field_rev) } }, ) } pub fn json_str(&self) -> CollaborateResult { make_grid_rev_json_str(&self.grid_rev) } } pub fn make_grid_rev_json_str(grid: &GridRevision) -> 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: GridRevisionDelta, /// md5: the md5 of the grid after applying the change. pub md5: String, } pub fn make_grid_delta(grid_rev: &GridRevision) -> GridRevisionDelta { let json = serde_json::to_string(&grid_rev).unwrap(); TextDeltaBuilder::new().insert(&json).build() } pub fn make_grid_revisions(user_id: &str, grid_rev: &GridRevision) -> RepeatedRevision { let delta = make_grid_delta(grid_rev); let bytes = delta.json_bytes(); let revision = Revision::initial_revision(user_id, &grid_rev.grid_id, bytes); revision.into() } impl std::default::Default for GridRevisionPad { fn default() -> Self { let grid = GridRevision::new(&gen_grid_id()); let delta = make_grid_delta(&grid); GridRevisionPad { grid_rev: Arc::new(grid), delta, } } }