433 lines
15 KiB
Rust
Raw Normal View History

2022-03-05 10:59:44 +08:00
use crate::entities::revision::{md5, RepeatedRevision, Revision};
2022-03-04 18:11:12 +08:00
use crate::errors::{internal_error, CollaborateError, CollaborateResult};
use crate::util::{cal_diff, make_delta_from_revisions};
2022-03-15 19:00:28 +08:00
use bytes::Bytes;
use flowy_grid_data_model::entities::FieldType;
use flowy_grid_data_model::entities::{FieldChangesetParams, FieldOrder};
use flowy_grid_data_model::revision::{
gen_block_id, gen_grid_id, FieldRevision, GridBlockRevision, GridBlockRevisionChangeset, GridRevision,
2022-03-11 21:36:00 +08:00
};
2022-04-28 16:54:04 +08:00
use lib_infra::util::move_vec_element;
2022-03-06 09:03:02 +08:00
use lib_ot::core::{OperationTransformable, PlainTextAttributes, PlainTextDelta, PlainTextDeltaBuilder};
2022-03-11 21:36:00 +08:00
use std::collections::HashMap;
2022-03-04 18:11:12 +08:00
use std::sync::Arc;
pub type GridRevisionDelta = PlainTextDelta;
pub type GridRevisionDeltaBuilder = PlainTextDeltaBuilder;
2022-03-04 18:11:12 +08:00
pub struct GridRevisionPad {
pub(crate) grid_rev: Arc<GridRevision>,
pub(crate) delta: GridRevisionDelta,
2022-03-04 18:11:12 +08:00
}
2022-04-07 08:33:10 +08:00
pub trait JsonDeserializer {
fn deserialize(&self, type_option_data: Vec<u8>) -> CollaborateResult<String>;
}
impl GridRevisionPad {
pub async fn duplicate_grid_meta(&self) -> (Vec<FieldRevision>, Vec<GridBlockRevision>) {
let fields = self.grid_rev.fields.to_vec();
2022-06-06 20:06:08 +08:00
let blocks = self
.grid_rev
2022-06-06 20:06:08 +08:00
.blocks
.iter()
.map(|block| {
let mut duplicated_block = block.clone();
duplicated_block.block_id = gen_block_id();
duplicated_block
})
.collect::<Vec<GridBlockRevision>>();
2022-06-06 20:06:08 +08:00
(fields, blocks)
}
pub fn from_delta(delta: GridRevisionDelta) -> CollaborateResult<Self> {
2022-03-04 18:11:12 +08:00
let s = delta.to_str()?;
let grid: GridRevision = serde_json::from_str(&s)
2022-03-04 18:11:12 +08:00
.map_err(|e| CollaborateError::internal().context(format!("Deserialize delta to grid failed: {}", e)))?;
Ok(Self {
grid_rev: Arc::new(grid),
2022-03-04 18:11:12 +08:00
delta,
})
}
2022-03-05 10:59:44 +08:00
pub fn from_revisions(_grid_id: &str, revisions: Vec<Revision>) -> CollaborateResult<Self> {
let grid_delta: GridRevisionDelta = make_delta_from_revisions::<PlainTextAttributes>(revisions)?;
2022-03-10 12:01:31 +08:00
Self::from_delta(grid_delta)
2022-03-04 18:11:12 +08:00
}
2022-04-05 14:25:07 +08:00
#[tracing::instrument(level = "debug", skip_all, err)]
pub fn create_field_rev(
2022-03-25 20:55:56 +08:00
&mut self,
new_field_rev: FieldRevision,
2022-03-25 20:55:56 +08:00
start_field_id: Option<String>,
) -> CollaborateResult<Option<GridChangeset>> {
2022-04-07 08:33:10 +08:00
self.modify_grid(|grid_meta| {
2022-03-24 17:09:05 +08:00
// Check if the field exists or not
2022-04-07 08:33:10 +08:00
if grid_meta
.fields
.iter()
.any(|field_rev| field_rev.id == new_field_rev.id)
2022-04-07 08:33:10 +08:00
{
2022-03-31 22:51:46 +08:00
tracing::error!("Duplicate grid field");
return Ok(None);
}
let insert_index = match start_field_id {
None => None,
2022-04-07 08:33:10 +08:00
Some(start_field_id) => grid_meta.fields.iter().position(|field| field.id == start_field_id),
};
match insert_index {
None => grid_meta.fields.push(new_field_rev),
Some(index) => grid_meta.fields.insert(index, new_field_rev),
2022-03-13 11:06:28 +08:00
}
Ok(Some(()))
2022-03-04 18:11:12 +08:00
})
}
pub fn delete_field_rev(&mut self, field_id: &str) -> CollaborateResult<Option<GridChangeset>> {
2022-04-07 08:33:10 +08:00
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(()))
}
},
)
2022-03-04 18:11:12 +08:00
}
pub fn duplicate_field_rev(
2022-04-14 13:29:42 +08:00
&mut self,
field_id: &str,
duplicated_field_id: &str,
) -> CollaborateResult<Option<GridChangeset>> {
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].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, duplicate_field_rev);
2022-04-14 13:29:42 +08:00
Ok(Some(()))
}
},
)
2022-03-11 21:36:00 +08:00
}
2022-04-01 22:46:01 +08:00
pub fn switch_to_field<B>(
&mut self,
field_id: &str,
field_type: FieldType,
type_option_json_builder: B,
) -> CollaborateResult<Option<GridChangeset>>
where
B: FnOnce(&FieldType) -> String,
{
2022-04-07 08:33:10 +08:00
self.modify_grid(|grid_meta| {
2022-04-01 22:46:01 +08:00
//
match grid_meta.fields.iter_mut().find(|field_rev| field_rev.id == field_id) {
2022-04-01 22:46:01 +08:00
None => {
tracing::warn!("Can not find the field with id: {}", field_id);
Ok(None)
}
Some(field_rev) => {
if field_rev.get_type_option_str(&field_type).is_none() {
2022-04-01 22:46:01 +08:00
let type_option_json = type_option_json_builder(&field_type);
field_rev.insert_type_option_str(&field_type, type_option_json);
2022-04-01 22:46:01 +08:00
}
field_rev.field_type = field_type;
2022-04-01 22:46:01 +08:00
Ok(Some(()))
}
}
})
}
pub fn update_field_rev<T: JsonDeserializer>(
&mut self,
changeset: FieldChangesetParams,
deserializer: T,
) -> CollaborateResult<Option<GridChangeset>> {
2022-03-14 17:24:25 +08:00
let field_id = changeset.field_id.clone();
2022-03-11 21:36:00 +08:00
self.modify_field(&field_id, |field| {
let mut is_changed = None;
2022-03-14 17:24:25 +08:00
if let Some(name) = changeset.name {
2022-03-11 21:36:00 +08:00
field.name = name;
is_changed = Some(())
}
2022-03-14 17:24:25 +08:00
if let Some(desc) = changeset.desc {
2022-03-11 21:36:00 +08:00
field.desc = desc;
is_changed = Some(())
}
2022-03-14 17:24:25 +08:00
if let Some(field_type) = changeset.field_type {
2022-03-11 21:36:00 +08:00
field.field_type = field_type;
is_changed = Some(())
}
2022-03-14 17:24:25 +08:00
if let Some(frozen) = changeset.frozen {
2022-03-11 21:36:00 +08:00
field.frozen = frozen;
is_changed = Some(())
}
2022-03-14 17:24:25 +08:00
if let Some(visibility) = changeset.visibility {
2022-03-11 21:36:00 +08:00
field.visibility = visibility;
is_changed = Some(())
}
2022-03-14 17:24:25 +08:00
if let Some(width) = changeset.width {
2022-03-11 21:36:00 +08:00
field.width = width;
is_changed = Some(())
}
2022-03-27 09:35:10 +08:00
if let Some(type_option_data) = changeset.type_option_data {
match deserializer.deserialize(type_option_data) {
2022-04-01 16:38:51 +08:00
Ok(json_str) => {
2022-04-01 22:49:26 +08:00
let field_type = field.field_type.clone();
field.insert_type_option_str(&field_type, json_str);
2022-03-27 09:35:10 +08:00
is_changed = Some(())
}
Err(err) => {
tracing::error!("Deserialize data to type option json failed: {}", err);
}
}
2022-03-11 21:36:00 +08:00
}
Ok(is_changed)
})
}
pub fn get_field_rev(&self, field_id: &str) -> Option<(usize, &FieldRevision)> {
self.grid_rev
2022-04-14 13:29:42 +08:00
.fields
.iter()
.enumerate()
.find(|(_, field)| field.id == field_id)
2022-03-27 11:14:21 +08:00
}
pub fn replace_field_rev(&mut self, field_rev: FieldRevision) -> CollaborateResult<Option<GridChangeset>> {
2022-04-07 08:33:10 +08:00
self.modify_grid(
|grid_meta| match grid_meta.fields.iter().position(|field| field.id == field_rev.id) {
2022-04-07 08:33:10 +08:00
None => Ok(None),
Some(index) => {
grid_meta.fields.remove(index);
grid_meta.fields.insert(index, field_rev);
2022-04-07 08:33:10 +08:00
Ok(Some(()))
}
},
)
}
2022-04-15 19:56:44 +08:00
pub fn move_field(
&mut self,
field_id: &str,
2022-04-28 16:54:04 +08:00
from_index: usize,
2022-04-15 19:56:44 +08:00
to_index: usize,
) -> CollaborateResult<Option<GridChangeset>> {
2022-04-28 16:54:04 +08:00
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),
}
})
2022-04-15 19:56:44 +08:00
}
2022-03-27 11:14:21 +08:00
pub fn contain_field(&self, field_id: &str) -> bool {
self.grid_rev.fields.iter().any(|field| field.id == field_id)
2022-03-27 11:14:21 +08:00
}
pub fn get_field_orders(&self) -> Vec<FieldOrder> {
self.grid_rev.fields.iter().map(FieldOrder::from).collect()
2022-03-27 11:14:21 +08:00
}
pub fn get_field_revs(&self, field_orders: Option<Vec<FieldOrder>>) -> CollaborateResult<Vec<FieldRevision>> {
2022-03-27 11:14:21 +08:00
match field_orders {
None => Ok(self.grid_rev.fields.clone()),
2022-03-27 11:14:21 +08:00
Some(field_orders) => {
let field_by_field_id = self
.grid_rev
2022-03-27 11:14:21 +08:00
.fields
.iter()
.map(|field| (&field.id, field))
.collect::<HashMap<&String, &FieldRevision>>();
2022-03-27 11:14:21 +08:00
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::<Vec<FieldRevision>>();
2022-03-27 11:14:21 +08:00
Ok(fields)
}
}
}
pub fn create_block_meta(&mut self, block: GridBlockRevision) -> CollaborateResult<Option<GridChangeset>> {
2022-04-07 08:33:10 +08:00
self.modify_grid(|grid_meta| {
if grid_meta.blocks.iter().any(|b| b.block_id == block.block_id) {
2022-03-13 11:06:28 +08:00
tracing::warn!("Duplicate grid block");
Ok(None)
} else {
match grid_meta.blocks.last() {
None => grid_meta.blocks.push(block),
2022-03-15 11:07:18 +08:00
Some(last_block) => {
if last_block.start_row_index > block.start_row_index
&& last_block.len() > block.start_row_index
{
2022-03-16 10:02:37 +08:00
let msg = "GridBlock's start_row_index should be greater than the last_block's start_row_index and its len".to_string();
2022-03-15 11:07:18 +08:00
return Err(CollaborateError::internal().context(msg))
}
grid_meta.blocks.push(block);
2022-03-15 11:07:18 +08:00
}
}
2022-03-13 11:06:28 +08:00
Ok(Some(()))
}
2022-03-11 21:36:00 +08:00
})
}
pub fn get_block_metas(&self) -> Vec<GridBlockRevision> {
self.grid_rev.blocks.clone()
2022-03-12 09:30:13 +08:00
}
pub fn update_block_meta(
&mut self,
changeset: GridBlockRevisionChangeset,
) -> CollaborateResult<Option<GridChangeset>> {
2022-03-14 17:24:25 +08:00
let block_id = changeset.block_id.clone();
2022-03-11 21:36:00 +08:00
self.modify_block(&block_id, |block| {
let mut is_changed = None;
2022-03-14 17:24:25 +08:00
if let Some(row_count) = changeset.row_count {
2022-03-11 21:36:00 +08:00
block.row_count = row_count;
is_changed = Some(());
}
2022-03-14 17:24:25 +08:00
if let Some(start_row_index) = changeset.start_row_index {
2022-03-13 11:06:28 +08:00
block.start_row_index = start_row_index;
is_changed = Some(());
}
2022-03-11 21:36:00 +08:00
Ok(is_changed)
})
}
pub fn md5(&self) -> String {
2022-03-15 19:00:28 +08:00
md5(&self.delta.to_delta_bytes())
2022-03-05 10:59:44 +08:00
}
2022-03-05 22:30:42 +08:00
pub fn delta_str(&self) -> String {
self.delta.to_delta_str()
}
2022-03-15 19:00:28 +08:00
pub fn delta_bytes(&self) -> Bytes {
self.delta.to_delta_bytes()
}
pub fn fields(&self) -> &[FieldRevision] {
&self.grid_rev.fields
2022-03-05 17:52:25 +08:00
}
2022-03-16 21:19:51 +08:00
fn modify_grid<F>(&mut self, f: F) -> CollaborateResult<Option<GridChangeset>>
2022-03-04 18:11:12 +08:00
where
F: FnOnce(&mut GridRevision) -> CollaborateResult<Option<()>>,
2022-03-04 18:11:12 +08:00
{
let cloned_grid = self.grid_rev.clone();
match f(Arc::make_mut(&mut self.grid_rev))? {
2022-03-04 18:11:12 +08:00
None => Ok(None),
Some(_) => {
let old = json_from_grid(&cloned_grid)?;
let new = json_from_grid(&self.grid_rev)?;
2022-03-04 18:11:12 +08:00
match cal_diff::<PlainTextAttributes>(old, new) {
None => Ok(None),
Some(delta) => {
self.delta = self.delta.compose(&delta)?;
2022-03-16 21:19:51 +08:00
Ok(Some(GridChangeset { delta, md5: self.md5() }))
2022-03-04 18:11:12 +08:00
}
}
}
}
}
2022-03-11 21:36:00 +08:00
2022-03-16 21:19:51 +08:00
pub fn modify_block<F>(&mut self, block_id: &str, f: F) -> CollaborateResult<Option<GridChangeset>>
2022-03-11 21:36:00 +08:00
where
F: FnOnce(&mut GridBlockRevision) -> CollaborateResult<Option<()>>,
2022-03-11 21:36:00 +08:00
{
self.modify_grid(
|grid_meta| match grid_meta.blocks.iter().position(|block| block.block_id == block_id) {
2022-03-17 17:25:43 +08:00
None => {
tracing::warn!("[GridMetaPad]: Can't find any block with id: {}", block_id);
Ok(None)
}
Some(index) => f(&mut grid_meta.blocks[index]),
},
)
2022-03-11 21:36:00 +08:00
}
2022-03-16 21:19:51 +08:00
pub fn modify_field<F>(&mut self, field_id: &str, f: F) -> CollaborateResult<Option<GridChangeset>>
2022-03-11 21:36:00 +08:00
where
F: FnOnce(&mut FieldRevision) -> CollaborateResult<Option<()>>,
2022-03-11 21:36:00 +08:00
{
2022-04-07 08:33:10 +08:00
self.modify_grid(
|grid_meta| match grid_meta.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_meta.fields[index]),
},
)
2022-03-11 21:36:00 +08:00
}
2022-03-04 18:11:12 +08:00
}
fn json_from_grid(grid: &Arc<GridRevision>) -> CollaborateResult<String> {
2022-03-04 18:11:12 +08:00
let json = serde_json::to_string(grid)
.map_err(|err| internal_error(format!("Serialize grid to json str failed. {:?}", err)))?;
Ok(json)
}
2022-03-16 21:19:51 +08:00
pub struct GridChangeset {
pub delta: GridRevisionDelta,
2022-03-04 18:11:12 +08:00
/// 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();
2022-03-04 18:11:12 +08:00
PlainTextDeltaBuilder::new().insert(&json).build()
}
pub fn make_grid_revisions(user_id: &str, grid_rev: &GridRevision) -> RepeatedRevision {
let delta = make_grid_delta(grid_rev);
2022-03-15 19:00:28 +08:00
let bytes = delta.to_delta_bytes();
let revision = Revision::initial_revision(user_id, &grid_rev.grid_id, bytes);
2022-03-05 10:59:44 +08:00
revision.into()
}
impl std::default::Default for GridRevisionPad {
2022-03-04 18:11:12 +08:00
fn default() -> Self {
2022-06-15 17:24:46 +08:00
let grid = GridRevision::new(&gen_grid_id());
2022-03-05 10:59:44 +08:00
let delta = make_grid_delta(&grid);
GridRevisionPad {
grid_rev: Arc::new(grid),
2022-03-04 18:11:12 +08:00
delta,
}
}
}