util.rs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. use crate::errors::{CollaborateError, CollaborateResult};
  2. use crate::server_folder::FolderOperations;
  3. use dissimilar::Chunk;
  4. use flowy_http_model::document::DocumentPayloadPB;
  5. use flowy_http_model::folder::FolderInfo;
  6. use flowy_http_model::revision::RepeatedRevision;
  7. use flowy_http_model::revision::Revision;
  8. use lib_ot::core::{DeltaOperationBuilder, OTString, OperationAttributes};
  9. use lib_ot::{
  10. core::{DeltaOperations, OperationTransform, NEW_LINE, WHITESPACE},
  11. text_delta::DeltaTextOperations,
  12. };
  13. use serde::de::DeserializeOwned;
  14. #[inline]
  15. pub fn find_newline(s: &str) -> Option<usize> {
  16. s.find(NEW_LINE)
  17. }
  18. #[inline]
  19. pub fn is_newline(s: &str) -> bool {
  20. s == NEW_LINE
  21. }
  22. #[inline]
  23. pub fn is_whitespace(s: &str) -> bool {
  24. s == WHITESPACE
  25. }
  26. #[inline]
  27. pub fn contain_newline(s: &str) -> bool {
  28. s.contains(NEW_LINE)
  29. }
  30. #[tracing::instrument(level = "trace", skip(revisions), err)]
  31. pub fn make_operations_from_revisions<T>(revisions: Vec<Revision>) -> CollaborateResult<DeltaOperations<T>>
  32. where
  33. T: OperationAttributes + DeserializeOwned + OperationAttributes + serde::Serialize,
  34. {
  35. let mut new_operations = DeltaOperations::<T>::new();
  36. for revision in revisions {
  37. if revision.bytes.is_empty() {
  38. tracing::warn!("revision delta_data is empty");
  39. continue;
  40. }
  41. let operations = DeltaOperations::<T>::from_bytes(revision.bytes).map_err(|e| {
  42. let err_msg = format!("Deserialize revision failed: {:?}", e);
  43. CollaborateError::internal().context(err_msg)
  44. })?;
  45. match new_operations.compose(&operations) {
  46. Ok(composed_operations) => {
  47. new_operations = composed_operations;
  48. // if composed_operations.content().is_ok() {
  49. // new_operations = composed_operations;
  50. // } else {
  51. // tracing::error!(
  52. // "Compose operation failed: rev_id: {}, object_id: {} {:?}",
  53. // revision.rev_id,
  54. // revision.object_id,
  55. // operations
  56. // );
  57. // return Ok(new_operations);
  58. // }
  59. }
  60. Err(e) => {
  61. tracing::error!("Compose operation failed: {}, {:?}", e, operations);
  62. return Ok(new_operations);
  63. }
  64. }
  65. }
  66. Ok(new_operations)
  67. }
  68. pub fn pair_rev_id_from_revision_pbs(revisions: &[Revision]) -> (i64, i64) {
  69. let mut rev_id = 0;
  70. revisions.iter().for_each(|revision| {
  71. if rev_id < revision.rev_id {
  72. rev_id = revision.rev_id;
  73. }
  74. });
  75. if rev_id > 0 {
  76. (rev_id - 1, rev_id)
  77. } else {
  78. (0, rev_id)
  79. }
  80. }
  81. #[inline]
  82. pub fn make_folder_from_revisions_pb(
  83. folder_id: &str,
  84. revisions: RepeatedRevision,
  85. ) -> Result<Option<FolderInfo>, CollaborateError> {
  86. let revisions = revisions.into_inner();
  87. if revisions.is_empty() {
  88. return Ok(None);
  89. }
  90. let mut folder_delta = FolderOperations::new();
  91. let mut base_rev_id = 0;
  92. let mut rev_id = 0;
  93. for revision in revisions {
  94. base_rev_id = revision.base_rev_id;
  95. rev_id = revision.rev_id;
  96. if revision.bytes.is_empty() {
  97. tracing::warn!("revision delta_data is empty");
  98. }
  99. let delta = FolderOperations::from_bytes(revision.bytes)?;
  100. folder_delta = folder_delta.compose(&delta)?;
  101. }
  102. let text = folder_delta.json_str();
  103. Ok(Some(FolderInfo {
  104. folder_id: folder_id.to_string(),
  105. text,
  106. rev_id,
  107. base_rev_id,
  108. }))
  109. }
  110. #[inline]
  111. pub fn make_document_from_revision_pbs(
  112. doc_id: &str,
  113. revisions: RepeatedRevision,
  114. ) -> Result<Option<DocumentPayloadPB>, CollaborateError> {
  115. let revisions = revisions.into_inner();
  116. if revisions.is_empty() {
  117. return Ok(None);
  118. }
  119. let mut delta = DeltaTextOperations::new();
  120. let mut base_rev_id = 0;
  121. let mut rev_id = 0;
  122. for revision in revisions {
  123. base_rev_id = revision.base_rev_id;
  124. rev_id = revision.rev_id;
  125. if revision.bytes.is_empty() {
  126. tracing::warn!("revision delta_data is empty");
  127. }
  128. let new_delta = DeltaTextOperations::from_bytes(revision.bytes)?;
  129. delta = delta.compose(&new_delta)?;
  130. }
  131. Ok(Some(DocumentPayloadPB {
  132. doc_id: doc_id.to_owned(),
  133. data: delta.json_bytes().to_vec(),
  134. rev_id,
  135. base_rev_id,
  136. }))
  137. }
  138. #[inline]
  139. pub fn rev_id_from_str(s: &str) -> Result<i64, CollaborateError> {
  140. let rev_id = s
  141. .to_owned()
  142. .parse::<i64>()
  143. .map_err(|e| CollaborateError::internal().context(format!("Parse rev_id from {} failed. {}", s, e)))?;
  144. Ok(rev_id)
  145. }
  146. pub fn cal_diff<T: OperationAttributes>(old: String, new: String) -> Option<DeltaOperations<T>> {
  147. let chunks = dissimilar::diff(&old, &new);
  148. let mut delta_builder = DeltaOperationBuilder::<T>::new();
  149. for chunk in &chunks {
  150. match chunk {
  151. Chunk::Equal(s) => {
  152. delta_builder = delta_builder.retain(OTString::from(*s).utf16_len());
  153. }
  154. Chunk::Delete(s) => {
  155. delta_builder = delta_builder.delete(OTString::from(*s).utf16_len());
  156. }
  157. Chunk::Insert(s) => {
  158. delta_builder = delta_builder.insert(*s);
  159. }
  160. }
  161. }
  162. let delta = delta_builder.build();
  163. if delta.is_empty() {
  164. None
  165. } else {
  166. Some(delta)
  167. }
  168. }