mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	refactor the Task a little bit
This commit is contained in:
		
							
								
								
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -2370,6 +2370,7 @@ dependencies = [ | |||||||
|  "serde_json", |  "serde_json", | ||||||
|  "time 0.3.14", |  "time 0.3.14", | ||||||
|  "tokio", |  "tokio", | ||||||
|  |  "uuid 1.1.2", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| [[package]] | [[package]] | ||||||
|   | |||||||
| @@ -72,15 +72,23 @@ impl CompatV5ToV6 { | |||||||
|                         v5::Status::Succeeded => v6::Status::Succeeded, |                         v5::Status::Succeeded => v6::Status::Succeeded, | ||||||
|                         v5::Status::Failed => v6::Status::Failed, |                         v5::Status::Failed => v6::Status::Failed, | ||||||
|                     }, |                     }, | ||||||
|                     kind: match &task.content { |                     kind: match task.content.clone() { | ||||||
|                         v5::tasks::TaskContent::IndexCreation { .. } => v6::Kind::IndexCreation, |                         v5::tasks::TaskContent::IndexCreation { primary_key, .. } => { | ||||||
|                         v5::tasks::TaskContent::IndexUpdate { .. } => v6::Kind::IndexUpdate, |                             v6::Kind::IndexCreation { primary_key } | ||||||
|  |                         } | ||||||
|  |                         v5::tasks::TaskContent::IndexUpdate { primary_key, .. } => { | ||||||
|  |                             v6::Kind::IndexUpdate { primary_key } | ||||||
|  |                         } | ||||||
|                         v5::tasks::TaskContent::IndexDeletion { .. } => v6::Kind::IndexDeletion, |                         v5::tasks::TaskContent::IndexDeletion { .. } => v6::Kind::IndexDeletion, | ||||||
|                         v5::tasks::TaskContent::DocumentAddition { |                         v5::tasks::TaskContent::DocumentAddition { | ||||||
|                             merge_strategy, |                             merge_strategy, | ||||||
|                             allow_index_creation, |                             allow_index_creation, | ||||||
|  |                             primary_key, | ||||||
|  |                             documents_count, | ||||||
|                             .. |                             .. | ||||||
|                         } => v6::Kind::DocumentImport { |                         } => v6::Kind::DocumentImport { | ||||||
|  |                             primary_key, | ||||||
|  |                             documents_count: documents_count as u64, | ||||||
|                             method: match merge_strategy { |                             method: match merge_strategy { | ||||||
|                                 v5::tasks::IndexDocumentsMethod::ReplaceDocuments => { |                                 v5::tasks::IndexDocumentsMethod::ReplaceDocuments => { | ||||||
|                                     v6::milli::update::IndexDocumentsMethod::ReplaceDocuments |                                     v6::milli::update::IndexDocumentsMethod::ReplaceDocuments | ||||||
| @@ -91,14 +99,22 @@ impl CompatV5ToV6 { | |||||||
|                             }, |                             }, | ||||||
|                             allow_index_creation: allow_index_creation.clone(), |                             allow_index_creation: allow_index_creation.clone(), | ||||||
|                         }, |                         }, | ||||||
|                         v5::tasks::TaskContent::DocumentDeletion { .. } => { |                         v5::tasks::TaskContent::DocumentDeletion { deletion, .. } => match deletion | ||||||
|                             v6::Kind::DocumentDeletion |                         { | ||||||
|                         } |                             v5::tasks::DocumentDeletion::Clear => v6::Kind::DocumentClear, | ||||||
|  |                             v5::tasks::DocumentDeletion::Ids(documents_ids) => { | ||||||
|  |                                 v6::Kind::DocumentDeletion { documents_ids } | ||||||
|  |                             } | ||||||
|  |                         }, | ||||||
|                         v5::tasks::TaskContent::SettingsUpdate { |                         v5::tasks::TaskContent::SettingsUpdate { | ||||||
|                             allow_index_creation, |                             allow_index_creation, | ||||||
|  |                             is_deletion, | ||||||
|  |                             settings, | ||||||
|                             .. |                             .. | ||||||
|                         } => v6::Kind::Settings { |                         } => v6::Kind::Settings { | ||||||
|                             allow_index_creation: allow_index_creation.clone(), |                             is_deletion, | ||||||
|  |                             allow_index_creation, | ||||||
|  |                             settings: settings.into(), | ||||||
|                         }, |                         }, | ||||||
|                         v5::tasks::TaskContent::Dump { .. } => v6::Kind::DumpExport, |                         v5::tasks::TaskContent::Dump { .. } => v6::Kind::DumpExport, | ||||||
|                     }, |                     }, | ||||||
|   | |||||||
| @@ -1,200 +0,0 @@ | |||||||
| use std::{ |  | ||||||
|     fs::{self, File}, |  | ||||||
|     io::{BufRead, BufReader}, |  | ||||||
|     path::Path, |  | ||||||
|     str::FromStr, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| use tempfile::TempDir; |  | ||||||
| use time::OffsetDateTime; |  | ||||||
| use uuid::Uuid; |  | ||||||
|  |  | ||||||
| use crate::{Error, IndexMetadata, Result, Version}; |  | ||||||
|  |  | ||||||
| pub use meilisearch_types::milli; |  | ||||||
|  |  | ||||||
| use super::Document; |  | ||||||
|  |  | ||||||
| pub type Metadata = crate::Metadata; |  | ||||||
|  |  | ||||||
| pub type Settings<T> = meilisearch_types::settings::Settings<T>; |  | ||||||
| pub type Checked = meilisearch_types::settings::Checked; |  | ||||||
| pub type Unchecked = meilisearch_types::settings::Unchecked; |  | ||||||
|  |  | ||||||
| pub type Task = meilisearch_types::tasks::TaskDump; |  | ||||||
| pub type Key = meilisearch_auth::Key; |  | ||||||
|  |  | ||||||
| // ===== Other types to clarify the code of the compat module |  | ||||||
| // everything related to the tasks |  | ||||||
| pub type Status = meilisearch_types::tasks::Status; |  | ||||||
| pub type Kind = meilisearch_types::tasks::Kind; |  | ||||||
| pub type Details = meilisearch_types::tasks::Details; |  | ||||||
|  |  | ||||||
| // everything related to the settings |  | ||||||
| pub type Setting<T> = meilisearch_types::milli::update::Setting<T>; |  | ||||||
| pub type TypoTolerance = meilisearch_types::settings::TypoSettings; |  | ||||||
| pub type MinWordSizeForTypos = meilisearch_types::settings::MinWordSizeTyposSetting; |  | ||||||
| pub type FacetingSettings = meilisearch_types::settings::FacetingSettings; |  | ||||||
| pub type PaginationSettings = meilisearch_types::settings::PaginationSettings; |  | ||||||
|  |  | ||||||
| // everything related to the api keys |  | ||||||
| pub type Action = meilisearch_auth::Action; |  | ||||||
| pub type StarOr<T> = meilisearch_types::star_or::StarOr<T>; |  | ||||||
| pub type IndexUid = meilisearch_types::index_uid::IndexUid; |  | ||||||
|  |  | ||||||
| // everything related to the errors |  | ||||||
| pub type ResponseError = meilisearch_types::error::ResponseError; |  | ||||||
| pub type Code = meilisearch_types::error::Code; |  | ||||||
|  |  | ||||||
| pub struct V6Reader { |  | ||||||
|     dump: TempDir, |  | ||||||
|     instance_uid: Uuid, |  | ||||||
|     metadata: Metadata, |  | ||||||
|     tasks: BufReader<File>, |  | ||||||
|     keys: BufReader<File>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl V6Reader { |  | ||||||
|     pub fn open(dump: TempDir) -> Result<Self> { |  | ||||||
|         let meta_file = fs::read(dump.path().join("metadata.json"))?; |  | ||||||
|         let instance_uid = fs::read_to_string(dump.path().join("instance_uid.uuid"))?; |  | ||||||
|         let instance_uid = Uuid::from_str(&instance_uid)?; |  | ||||||
|  |  | ||||||
|         Ok(V6Reader { |  | ||||||
|             metadata: serde_json::from_reader(&*meta_file)?, |  | ||||||
|             instance_uid, |  | ||||||
|             tasks: BufReader::new(File::open(dump.path().join("tasks").join("queue.jsonl"))?), |  | ||||||
|             keys: BufReader::new(File::open(dump.path().join("keys.jsonl"))?), |  | ||||||
|             dump, |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn version(&self) -> Version { |  | ||||||
|         Version::V6 |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn date(&self) -> Option<OffsetDateTime> { |  | ||||||
|         Some(self.metadata.dump_date) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn instance_uid(&self) -> Result<Option<Uuid>> { |  | ||||||
|         Ok(Some(self.instance_uid)) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn indexes(&self) -> Result<Box<dyn Iterator<Item = Result<V6IndexReader>> + '_>> { |  | ||||||
|         let entries = fs::read_dir(self.dump.path().join("indexes"))?; |  | ||||||
|         Ok(Box::new( |  | ||||||
|             entries |  | ||||||
|                 .map(|entry| -> Result<Option<_>> { |  | ||||||
|                     let entry = entry?; |  | ||||||
|                     if entry.file_type()?.is_dir() { |  | ||||||
|                         let index = V6IndexReader::new( |  | ||||||
|                             entry |  | ||||||
|                                 .file_name() |  | ||||||
|                                 .to_str() |  | ||||||
|                                 .ok_or(Error::BadIndexName)? |  | ||||||
|                                 .to_string(), |  | ||||||
|                             &entry.path(), |  | ||||||
|                         )?; |  | ||||||
|                         Ok(Some(index)) |  | ||||||
|                     } else { |  | ||||||
|                         Ok(None) |  | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|                 .filter_map(|entry| entry.transpose()), |  | ||||||
|         )) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn tasks( |  | ||||||
|         &mut self, |  | ||||||
|     ) -> Box<dyn Iterator<Item = Result<(Task, Option<Box<super::UpdateFile>>)>> + '_> { |  | ||||||
|         Box::new((&mut self.tasks).lines().map(|line| -> Result<_> { |  | ||||||
|             let task: Task = serde_json::from_str(&line?)?; |  | ||||||
|  |  | ||||||
|             let update_file_path = self |  | ||||||
|                 .dump |  | ||||||
|                 .path() |  | ||||||
|                 .join("tasks") |  | ||||||
|                 .join("update_files") |  | ||||||
|                 .join(format!("{}.jsonl", task.uid.to_string())); |  | ||||||
|  |  | ||||||
|             if update_file_path.exists() { |  | ||||||
|                 Ok(( |  | ||||||
|                     task, |  | ||||||
|                     Some(Box::new(UpdateFile::new(&update_file_path)?) as Box<super::UpdateFile>), |  | ||||||
|                 )) |  | ||||||
|             } else { |  | ||||||
|                 Ok((task, None)) |  | ||||||
|             } |  | ||||||
|         })) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn keys(&mut self) -> Box<dyn Iterator<Item = Result<Key>> + '_> { |  | ||||||
|         Box::new( |  | ||||||
|             (&mut self.keys) |  | ||||||
|                 .lines() |  | ||||||
|                 .map(|line| -> Result<_> { Ok(serde_json::from_str(&line?)?) }), |  | ||||||
|         ) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub struct UpdateFile { |  | ||||||
|     reader: BufReader<File>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl UpdateFile { |  | ||||||
|     fn new(path: &Path) -> Result<Self> { |  | ||||||
|         Ok(UpdateFile { |  | ||||||
|             reader: BufReader::new(File::open(path)?), |  | ||||||
|         }) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl Iterator for UpdateFile { |  | ||||||
|     type Item = Result<Document>; |  | ||||||
|  |  | ||||||
|     fn next(&mut self) -> Option<Self::Item> { |  | ||||||
|         (&mut self.reader) |  | ||||||
|             .lines() |  | ||||||
|             .map(|line| { |  | ||||||
|                 line.map_err(Error::from) |  | ||||||
|                     .and_then(|line| serde_json::from_str(&line).map_err(Error::from)) |  | ||||||
|             }) |  | ||||||
|             .next() |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| pub struct V6IndexReader { |  | ||||||
|     metadata: IndexMetadata, |  | ||||||
|     documents: BufReader<File>, |  | ||||||
|     settings: BufReader<File>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl V6IndexReader { |  | ||||||
|     pub fn new(_name: String, path: &Path) -> Result<Self> { |  | ||||||
|         let metadata = File::open(path.join("metadata.json"))?; |  | ||||||
|  |  | ||||||
|         let ret = V6IndexReader { |  | ||||||
|             metadata: serde_json::from_reader(metadata)?, |  | ||||||
|             documents: BufReader::new(File::open(path.join("documents.jsonl"))?), |  | ||||||
|             settings: BufReader::new(File::open(path.join("settings.json"))?), |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         Ok(ret) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn metadata(&self) -> &IndexMetadata { |  | ||||||
|         &self.metadata |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn documents(&mut self) -> Result<impl Iterator<Item = Result<Document>> + '_> { |  | ||||||
|         Ok((&mut self.documents) |  | ||||||
|             .lines() |  | ||||||
|             .map(|line| -> Result<_> { Ok(serde_json::from_str(&line?)?) })) |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn settings(&mut self) -> Result<Settings<Checked>> { |  | ||||||
|         let settings: Settings<Unchecked> = serde_json::from_reader(&mut self.settings)?; |  | ||||||
|         Ok(settings.check()) |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -6,8 +6,10 @@ use std::{ | |||||||
|  |  | ||||||
| use flate2::{write::GzEncoder, Compression}; | use flate2::{write::GzEncoder, Compression}; | ||||||
| use meilisearch_auth::Key; | use meilisearch_auth::Key; | ||||||
| use meilisearch_types::settings::{Checked, Settings}; | use meilisearch_types::{ | ||||||
| use meilisearch_types::tasks::TaskDump; |     settings::{Checked, Settings}, | ||||||
|  |     tasks::Task, | ||||||
|  | }; | ||||||
| use serde_json::{Map, Value}; | use serde_json::{Map, Value}; | ||||||
| use tempfile::TempDir; | use tempfile::TempDir; | ||||||
| use time::OffsetDateTime; | use time::OffsetDateTime; | ||||||
| @@ -105,7 +107,7 @@ impl TaskWriter { | |||||||
|  |  | ||||||
|     /// Pushes tasks in the dump. |     /// Pushes tasks in the dump. | ||||||
|     /// If the tasks has an associated `update_file` it'll use the `task_id` as its name. |     /// If the tasks has an associated `update_file` it'll use the `task_id` as its name. | ||||||
|     pub fn push_task(&mut self, task: &TaskDump) -> Result<UpdateFile> { |     pub fn push_task(&mut self, task: &Task) -> Result<UpdateFile> { | ||||||
|         self.queue.write_all(&serde_json::to_vec(task)?)?; |         self.queue.write_all(&serde_json::to_vec(task)?)?; | ||||||
|         self.queue.write_all(b"\n")?; |         self.queue.write_all(b"\n")?; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -4,6 +4,61 @@ use meilisearch_types::milli::update::IndexDocumentsMethod::{ | |||||||
| use meilisearch_types::tasks::{Kind, TaskId}; | use meilisearch_types::tasks::{Kind, TaskId}; | ||||||
| use std::ops::ControlFlow::{self, Break, Continue}; | use std::ops::ControlFlow::{self, Break, Continue}; | ||||||
|  |  | ||||||
|  | use crate::KindWithContent; | ||||||
|  |  | ||||||
|  | /// This enum contain the minimal necessary informations | ||||||
|  | /// to make the autobatcher works. | ||||||
|  | enum AutobatchKind { | ||||||
|  |     DocumentImport { | ||||||
|  |         method: IndexDocumentsMethod, | ||||||
|  |         allow_index_creation: bool, | ||||||
|  |     }, | ||||||
|  |     DocumentDeletion, | ||||||
|  |     DocumentClear, | ||||||
|  |     Settings { | ||||||
|  |         allow_index_creation: bool, | ||||||
|  |     }, | ||||||
|  |     IndexCreation, | ||||||
|  |     IndexDeletion, | ||||||
|  |     IndexUpdate, | ||||||
|  |     IndexSwap, | ||||||
|  |     CancelTask, | ||||||
|  |     DeleteTasks, | ||||||
|  |     DumpExport, | ||||||
|  |     Snapshot, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<KindWithContent> for AutobatchKind { | ||||||
|  |     fn from(kind: KindWithContent) -> Self { | ||||||
|  |         match kind { | ||||||
|  |             KindWithContent::DocumentImport { | ||||||
|  |                 method, | ||||||
|  |                 allow_index_creation, | ||||||
|  |                 .. | ||||||
|  |             } => AutobatchKind::DocumentImport { | ||||||
|  |                 method, | ||||||
|  |                 allow_index_creation, | ||||||
|  |             }, | ||||||
|  |             KindWithContent::DocumentDeletion { .. } => AutobatchKind::DocumentDeletion, | ||||||
|  |             KindWithContent::DocumentClear { .. } => AutobatchKind::DocumentClear, | ||||||
|  |             KindWithContent::Settings { | ||||||
|  |                 allow_index_creation, | ||||||
|  |                 .. | ||||||
|  |             } => AutobatchKind::Settings { | ||||||
|  |                 allow_index_creation, | ||||||
|  |             }, | ||||||
|  |             KindWithContent::IndexDeletion { .. } => AutobatchKind::IndexDeletion, | ||||||
|  |             KindWithContent::IndexCreation { .. } => AutobatchKind::IndexCreation, | ||||||
|  |             KindWithContent::IndexUpdate { .. } => AutobatchKind::IndexUpdate, | ||||||
|  |             KindWithContent::IndexSwap { lhs, rhs } => AutobatchKind::IndexSwap, | ||||||
|  |             KindWithContent::CancelTask { .. } => AutobatchKind::CancelTask, | ||||||
|  |             KindWithContent::DeleteTasks { .. } => AutobatchKind::DeleteTasks, | ||||||
|  |             KindWithContent::DumpExport { .. } => AutobatchKind::DumpExport, | ||||||
|  |             KindWithContent::Snapshot => AutobatchKind::Snapshot, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Debug)] | #[derive(Debug)] | ||||||
| pub enum BatchKind { | pub enum BatchKind { | ||||||
|     DocumentClear { |     DocumentClear { | ||||||
| @@ -48,14 +103,16 @@ pub enum BatchKind { | |||||||
|  |  | ||||||
| impl BatchKind { | impl BatchKind { | ||||||
|     /// Returns a `ControlFlow::Break` if you must stop right now. |     /// Returns a `ControlFlow::Break` if you must stop right now. | ||||||
|     pub fn new(task_id: TaskId, kind: Kind) -> ControlFlow<BatchKind, BatchKind> { |     pub fn new(task_id: TaskId, kind: KindWithContent) -> ControlFlow<BatchKind, BatchKind> { | ||||||
|         match kind { |         use AutobatchKind as K; | ||||||
|             Kind::IndexCreation => Break(BatchKind::IndexCreation { id: task_id }), |  | ||||||
|             Kind::IndexDeletion => Break(BatchKind::IndexDeletion { ids: vec![task_id] }), |         match AutobatchKind::from(kind) { | ||||||
|             Kind::IndexUpdate => Break(BatchKind::IndexUpdate { id: task_id }), |             K::IndexCreation => Break(BatchKind::IndexCreation { id: task_id }), | ||||||
|             Kind::IndexSwap => Break(BatchKind::IndexSwap { id: task_id }), |             K::IndexDeletion => Break(BatchKind::IndexDeletion { ids: vec![task_id] }), | ||||||
|             Kind::DocumentClear => Continue(BatchKind::DocumentClear { ids: vec![task_id] }), |             K::IndexUpdate => Break(BatchKind::IndexUpdate { id: task_id }), | ||||||
|             Kind::DocumentImport { |             K::IndexSwap => Break(BatchKind::IndexSwap { id: task_id }), | ||||||
|  |             K::DocumentClear => Continue(BatchKind::DocumentClear { ids: vec![task_id] }), | ||||||
|  |             K::DocumentImport { | ||||||
|                 method, |                 method, | ||||||
|                 allow_index_creation, |                 allow_index_creation, | ||||||
|             } => Continue(BatchKind::DocumentImport { |             } => Continue(BatchKind::DocumentImport { | ||||||
| @@ -63,16 +120,16 @@ impl BatchKind { | |||||||
|                 allow_index_creation, |                 allow_index_creation, | ||||||
|                 import_ids: vec![task_id], |                 import_ids: vec![task_id], | ||||||
|             }), |             }), | ||||||
|             Kind::DocumentDeletion => Continue(BatchKind::DocumentDeletion { |             K::DocumentDeletion => Continue(BatchKind::DocumentDeletion { | ||||||
|                 deletion_ids: vec![task_id], |                 deletion_ids: vec![task_id], | ||||||
|             }), |             }), | ||||||
|             Kind::Settings { |             K::Settings { | ||||||
|                 allow_index_creation, |                 allow_index_creation, | ||||||
|             } => Continue(BatchKind::Settings { |             } => Continue(BatchKind::Settings { | ||||||
|                 allow_index_creation, |                 allow_index_creation, | ||||||
|                 settings_ids: vec![task_id], |                 settings_ids: vec![task_id], | ||||||
|             }), |             }), | ||||||
|             Kind::DumpExport | Kind::Snapshot | Kind::CancelTask | Kind::DeleteTasks => { |             K::DumpExport | K::Snapshot | K::CancelTask | K::DeleteTasks => { | ||||||
|                 unreachable!() |                 unreachable!() | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -80,17 +137,19 @@ impl BatchKind { | |||||||
|  |  | ||||||
|     /// Returns a `ControlFlow::Break` if you must stop right now. |     /// Returns a `ControlFlow::Break` if you must stop right now. | ||||||
|     #[rustfmt::skip] |     #[rustfmt::skip] | ||||||
|     fn accumulate(self, id: TaskId, kind: Kind) -> ControlFlow<BatchKind, BatchKind> { |     fn accumulate(self, id: TaskId, kind: AutobatchKind) -> ControlFlow<BatchKind, BatchKind> { | ||||||
|  |         use AutobatchKind as K; | ||||||
|  |  | ||||||
|         match (self, kind) { |         match (self, kind) { | ||||||
|             // We don't batch any of these operations |             // We don't batch any of these operations | ||||||
|             (this, Kind::IndexCreation | Kind::IndexUpdate | Kind::IndexSwap) => Break(this), |             (this, K::IndexCreation | K::IndexUpdate | K::IndexSwap) => Break(this), | ||||||
|             // The index deletion can batch with everything but must stop after |             // The index deletion can batch with everything but must stop after | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentClear { mut ids } |                 BatchKind::DocumentClear { mut ids } | ||||||
|                 | BatchKind::DocumentDeletion { deletion_ids: mut ids } |                 | BatchKind::DocumentDeletion { deletion_ids: mut ids } | ||||||
|                 | BatchKind::DocumentImport { method: _, allow_index_creation: _, import_ids: mut ids } |                 | BatchKind::DocumentImport { method: _, allow_index_creation: _, import_ids: mut ids } | ||||||
|                 | BatchKind::Settings { allow_index_creation: _, settings_ids: mut ids }, |                 | BatchKind::Settings { allow_index_creation: _, settings_ids: mut ids }, | ||||||
|                 Kind::IndexDeletion, |                 K::IndexDeletion, | ||||||
|             ) => { |             ) => { | ||||||
|                 ids.push(id); |                 ids.push(id); | ||||||
|                 Break(BatchKind::IndexDeletion { ids }) |                 Break(BatchKind::IndexDeletion { ids }) | ||||||
| @@ -98,7 +157,7 @@ impl BatchKind { | |||||||
|             ( |             ( | ||||||
|                 BatchKind::ClearAndSettings { settings_ids: mut ids, allow_index_creation: _, mut other } |                 BatchKind::ClearAndSettings { settings_ids: mut ids, allow_index_creation: _, mut other } | ||||||
|                 | BatchKind::SettingsAndDocumentImport { import_ids: mut ids, method: _, allow_index_creation: _, settings_ids: mut other }, |                 | BatchKind::SettingsAndDocumentImport { import_ids: mut ids, method: _, allow_index_creation: _, settings_ids: mut other }, | ||||||
|                 Kind::IndexDeletion, |                 K::IndexDeletion, | ||||||
|             ) => { |             ) => { | ||||||
|                 ids.push(id); |                 ids.push(id); | ||||||
|                 ids.append(&mut other); |                 ids.append(&mut other); | ||||||
| @@ -107,18 +166,18 @@ impl BatchKind { | |||||||
|  |  | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentClear { mut ids }, |                 BatchKind::DocumentClear { mut ids }, | ||||||
|                 Kind::DocumentClear | Kind::DocumentDeletion, |                 K::DocumentClear | K::DocumentDeletion, | ||||||
|             ) => { |             ) => { | ||||||
|                 ids.push(id); |                 ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentClear { ids }) |                 Continue(BatchKind::DocumentClear { ids }) | ||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::DocumentClear { .. }, |                 this @ BatchKind::DocumentClear { .. }, | ||||||
|                 Kind::DocumentImport { .. } | Kind::Settings { .. }, |                 K::DocumentImport { .. } | K::Settings { .. }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentImport { method: _, allow_index_creation: _, import_ids: mut ids }, |                 BatchKind::DocumentImport { method: _, allow_index_creation: _, import_ids: mut ids }, | ||||||
|                 Kind::DocumentClear, |                 K::DocumentClear, | ||||||
|             ) => { |             ) => { | ||||||
|                 ids.push(id); |                 ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentClear { ids }) |                 Continue(BatchKind::DocumentClear { ids }) | ||||||
| @@ -128,13 +187,13 @@ impl BatchKind { | |||||||
|             // or document imports not allowed to create an index if the first operation can. |             // or document imports not allowed to create an index if the first operation can. | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::DocumentImport { method: _, allow_index_creation: false, .. }, |                 this @ BatchKind::DocumentImport { method: _, allow_index_creation: false, .. }, | ||||||
|                 Kind::DocumentImport { method: _, allow_index_creation: true }, |                 K::DocumentImport { method: _, allow_index_creation: true }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|  |  | ||||||
|             // we can autobatch the same kind of document additions / updates |             // we can autobatch the same kind of document additions / updates | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentImport { method: ReplaceDocuments, allow_index_creation, mut import_ids }, |                 BatchKind::DocumentImport { method: ReplaceDocuments, allow_index_creation, mut import_ids }, | ||||||
|                 Kind::DocumentImport { method: ReplaceDocuments, .. }, |                 K::DocumentImport { method: ReplaceDocuments, .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 import_ids.push(id); |                 import_ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentImport { |                 Continue(BatchKind::DocumentImport { | ||||||
| @@ -145,7 +204,7 @@ impl BatchKind { | |||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentImport { method: UpdateDocuments, allow_index_creation, mut import_ids }, |                 BatchKind::DocumentImport { method: UpdateDocuments, allow_index_creation, mut import_ids }, | ||||||
|                 Kind::DocumentImport { method: UpdateDocuments, .. }, |                 K::DocumentImport { method: UpdateDocuments, .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 import_ids.push(id); |                 import_ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentImport { |                 Continue(BatchKind::DocumentImport { | ||||||
| @@ -159,18 +218,18 @@ impl BatchKind { | |||||||
|             // this match branch MUST be AFTER the previous one |             // this match branch MUST be AFTER the previous one | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::DocumentImport { .. }, |                 this @ BatchKind::DocumentImport { .. }, | ||||||
|                 Kind::DocumentDeletion | Kind::DocumentImport { .. }, |                 K::DocumentDeletion | K::DocumentImport { .. }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|  |  | ||||||
|             // We only want to batch together document imports that are allowed to create the index |             // We only want to batch together document imports that are allowed to create the index | ||||||
|             // or document imports not allowed to create an index if the first operation can. |             // or document imports not allowed to create an index if the first operation can. | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::DocumentImport { allow_index_creation: false, .. }, |                 this @ BatchKind::DocumentImport { allow_index_creation: false, .. }, | ||||||
|                 Kind::Settings { allow_index_creation: true }, |                 K::Settings { allow_index_creation: true }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::DocumentImport { method, allow_index_creation, import_ids }, |                 BatchKind::DocumentImport { method, allow_index_creation, import_ids }, | ||||||
|                 Kind::Settings { .. }, |                 K::Settings { .. }, | ||||||
|             ) => Continue(BatchKind::SettingsAndDocumentImport { |             ) => Continue(BatchKind::SettingsAndDocumentImport { | ||||||
|                 settings_ids: vec![id], |                 settings_ids: vec![id], | ||||||
|                 method, |                 method, | ||||||
| @@ -178,20 +237,20 @@ impl BatchKind { | |||||||
|                 import_ids, |                 import_ids, | ||||||
|             }), |             }), | ||||||
|  |  | ||||||
|             (BatchKind::DocumentDeletion { mut deletion_ids }, Kind::DocumentClear) => { |             (BatchKind::DocumentDeletion { mut deletion_ids }, K::DocumentClear) => { | ||||||
|                 deletion_ids.push(id); |                 deletion_ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentClear { ids: deletion_ids }) |                 Continue(BatchKind::DocumentClear { ids: deletion_ids }) | ||||||
|             } |             } | ||||||
|             (this @ BatchKind::DocumentDeletion { .. }, Kind::DocumentImport { .. }) => Break(this), |             (this @ BatchKind::DocumentDeletion { .. }, K::DocumentImport { .. }) => Break(this), | ||||||
|             (BatchKind::DocumentDeletion { mut deletion_ids }, Kind::DocumentDeletion) => { |             (BatchKind::DocumentDeletion { mut deletion_ids }, K::DocumentDeletion) => { | ||||||
|                 deletion_ids.push(id); |                 deletion_ids.push(id); | ||||||
|                 Continue(BatchKind::DocumentDeletion { deletion_ids }) |                 Continue(BatchKind::DocumentDeletion { deletion_ids }) | ||||||
|             } |             } | ||||||
|             (this @ BatchKind::DocumentDeletion { .. }, Kind::Settings { .. }) => Break(this), |             (this @ BatchKind::DocumentDeletion { .. }, K::Settings { .. }) => Break(this), | ||||||
|  |  | ||||||
|             ( |             ( | ||||||
|                 BatchKind::Settings { settings_ids, allow_index_creation }, |                 BatchKind::Settings { settings_ids, allow_index_creation }, | ||||||
|                 Kind::DocumentClear, |                 K::DocumentClear, | ||||||
|             ) => Continue(BatchKind::ClearAndSettings { |             ) => Continue(BatchKind::ClearAndSettings { | ||||||
|                 settings_ids: settings_ids, |                 settings_ids: settings_ids, | ||||||
|                 allow_index_creation, |                 allow_index_creation, | ||||||
| @@ -199,15 +258,15 @@ impl BatchKind { | |||||||
|             }), |             }), | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::Settings { .. }, |                 this @ BatchKind::Settings { .. }, | ||||||
|                 Kind::DocumentImport { .. } | Kind::DocumentDeletion, |                 K::DocumentImport { .. } | K::DocumentDeletion, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::Settings { allow_index_creation: false, .. }, |                 this @ BatchKind::Settings { allow_index_creation: false, .. }, | ||||||
|                 Kind::Settings { allow_index_creation: true }, |                 K::Settings { allow_index_creation: true }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::Settings { mut settings_ids, allow_index_creation }, |                 BatchKind::Settings { mut settings_ids, allow_index_creation }, | ||||||
|                 Kind::Settings { .. }, |                 K::Settings { .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 settings_ids.push(id); |                 settings_ids.push(id); | ||||||
|                 Continue(BatchKind::Settings { |                 Continue(BatchKind::Settings { | ||||||
| @@ -218,7 +277,7 @@ impl BatchKind { | |||||||
|  |  | ||||||
|             ( |             ( | ||||||
|                 BatchKind::ClearAndSettings { mut other, settings_ids, allow_index_creation }, |                 BatchKind::ClearAndSettings { mut other, settings_ids, allow_index_creation }, | ||||||
|                 Kind::DocumentClear, |                 K::DocumentClear, | ||||||
|             ) => { |             ) => { | ||||||
|                 other.push(id); |                 other.push(id); | ||||||
|                 Continue(BatchKind::ClearAndSettings { |                 Continue(BatchKind::ClearAndSettings { | ||||||
| @@ -227,14 +286,14 @@ impl BatchKind { | |||||||
|                     allow_index_creation, |                     allow_index_creation, | ||||||
|                 }) |                 }) | ||||||
|             } |             } | ||||||
|             (this @ BatchKind::ClearAndSettings { .. }, Kind::DocumentImport { .. }) => Break(this), |             (this @ BatchKind::ClearAndSettings { .. }, K::DocumentImport { .. }) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::ClearAndSettings { |                 BatchKind::ClearAndSettings { | ||||||
|                     mut other, |                     mut other, | ||||||
|                     settings_ids, |                     settings_ids, | ||||||
|                     allow_index_creation, |                     allow_index_creation, | ||||||
|                 }, |                 }, | ||||||
|                 Kind::DocumentDeletion, |                 K::DocumentDeletion, | ||||||
|             ) => { |             ) => { | ||||||
|                 other.push(id); |                 other.push(id); | ||||||
|                 Continue(BatchKind::ClearAndSettings { |                 Continue(BatchKind::ClearAndSettings { | ||||||
| @@ -245,13 +304,13 @@ impl BatchKind { | |||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::ClearAndSettings { allow_index_creation: false, .. }, |                 this @ BatchKind::ClearAndSettings { allow_index_creation: false, .. }, | ||||||
|                 Kind::Settings { |                 K::Settings { | ||||||
|                     allow_index_creation: true, |                     allow_index_creation: true, | ||||||
|                 }, |                 }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::ClearAndSettings { mut settings_ids, other, allow_index_creation }, |                 BatchKind::ClearAndSettings { mut settings_ids, other, allow_index_creation }, | ||||||
|                 Kind::Settings { .. }, |                 K::Settings { .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 settings_ids.push(id); |                 settings_ids.push(id); | ||||||
|                 Continue(BatchKind::ClearAndSettings { |                 Continue(BatchKind::ClearAndSettings { | ||||||
| @@ -262,7 +321,7 @@ impl BatchKind { | |||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
|                 BatchKind::SettingsAndDocumentImport { settings_ids, method: _, import_ids: mut other, allow_index_creation }, |                 BatchKind::SettingsAndDocumentImport { settings_ids, method: _, import_ids: mut other, allow_index_creation }, | ||||||
|                 Kind::DocumentClear, |                 K::DocumentClear, | ||||||
|             ) => { |             ) => { | ||||||
|                 other.push(id); |                 other.push(id); | ||||||
|                 Continue(BatchKind::ClearAndSettings { |                 Continue(BatchKind::ClearAndSettings { | ||||||
| @@ -275,11 +334,11 @@ impl BatchKind { | |||||||
|             // we can batch the settings with a kind of document operation with the same kind of document operation |             // we can batch the settings with a kind of document operation with the same kind of document operation | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::SettingsAndDocumentImport { allow_index_creation: false, .. }, |                 this @ BatchKind::SettingsAndDocumentImport { allow_index_creation: false, .. }, | ||||||
|                 Kind::DocumentImport { allow_index_creation: true, .. }, |                 K::DocumentImport { allow_index_creation: true, .. }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::SettingsAndDocumentImport { settings_ids, method: ReplaceDocuments, mut import_ids, allow_index_creation }, |                 BatchKind::SettingsAndDocumentImport { settings_ids, method: ReplaceDocuments, mut import_ids, allow_index_creation }, | ||||||
|                 Kind::DocumentImport { method: ReplaceDocuments, .. }, |                 K::DocumentImport { method: ReplaceDocuments, .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 import_ids.push(id); |                 import_ids.push(id); | ||||||
|                 Continue(BatchKind::SettingsAndDocumentImport { |                 Continue(BatchKind::SettingsAndDocumentImport { | ||||||
| @@ -291,7 +350,7 @@ impl BatchKind { | |||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
|                 BatchKind::SettingsAndDocumentImport { settings_ids, method: UpdateDocuments, allow_index_creation, mut import_ids }, |                 BatchKind::SettingsAndDocumentImport { settings_ids, method: UpdateDocuments, allow_index_creation, mut import_ids }, | ||||||
|                 Kind::DocumentImport { method: UpdateDocuments, .. }, |                 K::DocumentImport { method: UpdateDocuments, .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 import_ids.push(id); |                 import_ids.push(id); | ||||||
|                 Continue(BatchKind::SettingsAndDocumentImport { |                 Continue(BatchKind::SettingsAndDocumentImport { | ||||||
| @@ -305,15 +364,15 @@ impl BatchKind { | |||||||
|             // this MUST be AFTER the two previous branch |             // this MUST be AFTER the two previous branch | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::SettingsAndDocumentImport { .. }, |                 this @ BatchKind::SettingsAndDocumentImport { .. }, | ||||||
|                 Kind::DocumentDeletion | Kind::DocumentImport { .. }, |                 K::DocumentDeletion | K::DocumentImport { .. }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 this @ BatchKind::SettingsAndDocumentImport { allow_index_creation: false, .. }, |                 this @ BatchKind::SettingsAndDocumentImport { allow_index_creation: false, .. }, | ||||||
|                 Kind::Settings { allow_index_creation: true }, |                 K::Settings { allow_index_creation: true }, | ||||||
|             ) => Break(this), |             ) => Break(this), | ||||||
|             ( |             ( | ||||||
|                 BatchKind::SettingsAndDocumentImport { mut settings_ids, method, allow_index_creation, import_ids }, |                 BatchKind::SettingsAndDocumentImport { mut settings_ids, method, allow_index_creation, import_ids }, | ||||||
|                 Kind::Settings { .. }, |                 K::Settings { .. }, | ||||||
|             ) => { |             ) => { | ||||||
|                 settings_ids.push(id); |                 settings_ids.push(id); | ||||||
|                 Continue(BatchKind::SettingsAndDocumentImport { |                 Continue(BatchKind::SettingsAndDocumentImport { | ||||||
| @@ -323,7 +382,7 @@ impl BatchKind { | |||||||
|                     import_ids, |                     import_ids, | ||||||
|                 }) |                 }) | ||||||
|             } |             } | ||||||
|             (_, Kind::CancelTask | Kind::DeleteTasks | Kind::DumpExport | Kind::Snapshot) => { |             (_, K::CancelTask | K::DeleteTasks | K::DumpExport | K::Snapshot) => { | ||||||
|                 unreachable!() |                 unreachable!() | ||||||
|             } |             } | ||||||
|             ( |             ( | ||||||
| @@ -339,7 +398,7 @@ impl BatchKind { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn autobatch(enqueued: Vec<(TaskId, Kind)>) -> Option<BatchKind> { | pub fn autobatch(enqueued: Vec<(TaskId, KindWithContent)>) -> Option<BatchKind> { | ||||||
|     let mut enqueued = enqueued.into_iter(); |     let mut enqueued = enqueued.into_iter(); | ||||||
|     let (id, kind) = enqueued.next()?; |     let (id, kind) = enqueued.next()?; | ||||||
|     let mut acc = match BatchKind::new(id, kind) { |     let mut acc = match BatchKind::new(id, kind) { | ||||||
| @@ -348,7 +407,7 @@ pub fn autobatch(enqueued: Vec<(TaskId, Kind)>) -> Option<BatchKind> { | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     for (id, kind) in enqueued { |     for (id, kind) in enqueued { | ||||||
|         acc = match acc.accumulate(id, kind) { |         acc = match acc.accumulate(id, kind.into()) { | ||||||
|             Continue(acc) => acc, |             Continue(acc) => acc, | ||||||
|             Break(acc) => return Some(acc), |             Break(acc) => return Some(acc), | ||||||
|         }; |         }; | ||||||
| @@ -362,144 +421,204 @@ mod tests { | |||||||
|     use crate::assert_smol_debug_snapshot; |     use crate::assert_smol_debug_snapshot; | ||||||
|  |  | ||||||
|     use super::*; |     use super::*; | ||||||
|     use Kind::*; |     use uuid::Uuid; | ||||||
|  |  | ||||||
|     fn autobatch_from(input: impl IntoIterator<Item = Kind>) -> Option<BatchKind> { |     fn autobatch_from(input: impl IntoIterator<Item = KindWithContent>) -> Option<BatchKind> { | ||||||
|         autobatch( |         autobatch( | ||||||
|             input |             input | ||||||
|                 .into_iter() |                 .into_iter() | ||||||
|                 .enumerate() |                 .enumerate() | ||||||
|                 .map(|(id, kind)| (id as TaskId, kind)) |                 .map(|(id, kind)| (id as TaskId, kind.into())) | ||||||
|                 .collect(), |                 .collect(), | ||||||
|         ) |         ) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     fn doc_imp(method: IndexDocumentsMethod, allow_index_creation: bool) -> KindWithContent { | ||||||
|  |         KindWithContent::DocumentImport { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |             primary_key: None, | ||||||
|  |             method, | ||||||
|  |             content_file: Uuid::new_v4(), | ||||||
|  |             documents_count: 0, | ||||||
|  |             allow_index_creation, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn doc_del() -> KindWithContent { | ||||||
|  |         KindWithContent::DocumentDeletion { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |             documents_ids: Vec::new(), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn doc_clr() -> KindWithContent { | ||||||
|  |         KindWithContent::DocumentClear { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn settings(allow_index_creation: bool) -> KindWithContent { | ||||||
|  |         KindWithContent::Settings { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |             new_settings: Default::default(), | ||||||
|  |             is_deletion: false, | ||||||
|  |             allow_index_creation, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn idx_create() -> KindWithContent { | ||||||
|  |         KindWithContent::IndexCreation { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |             primary_key: None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn idx_update() -> KindWithContent { | ||||||
|  |         KindWithContent::IndexUpdate { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |             primary_key: None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn idx_del() -> KindWithContent { | ||||||
|  |         KindWithContent::IndexDeletion { | ||||||
|  |             index_uid: String::from("doggo"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     fn idx_swap() -> KindWithContent { | ||||||
|  |         KindWithContent::IndexSwap { | ||||||
|  |             lhs: String::from("doggo"), | ||||||
|  |             rhs: String::from("catto"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn autobatch_simple_operation_together() { |     fn autobatch_simple_operation_together() { | ||||||
|         // we can autobatch one or multiple DocumentAddition together |         // we can autobatch one or multiple DocumentAddition together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true)]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp( ReplaceDocuments, true ), doc_imp(ReplaceDocuments, true )]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 1, 2] })"); | ||||||
|         // we can autobatch one or multiple DocumentUpdate together |         // we can autobatch one or multiple DocumentUpdate together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true)]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_imp(UpdateDocuments, true), doc_imp(UpdateDocuments, true)]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0, 1, 2] })"); | ||||||
|         // we can autobatch one or multiple DocumentDeletion together |         // we can autobatch one or multiple DocumentDeletion together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del()]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, DocumentDeletion, DocumentDeletion]), @"Some(DocumentDeletion { deletion_ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), doc_del(), doc_del()]), @"Some(DocumentDeletion { deletion_ids: [0, 1, 2] })"); | ||||||
|         // we can autobatch one or multiple Settings together |         // we can autobatch one or multiple Settings together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([Settings { allow_index_creation: true }]), @"Some(Settings { allow_index_creation: true, settings_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([settings(true)]), @"Some(Settings { allow_index_creation: true, settings_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([Settings { allow_index_creation: true }, Settings { allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(Settings { allow_index_creation: true, settings_ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([settings(true), settings(true), settings(true)]), @"Some(Settings { allow_index_creation: true, settings_ids: [0, 1, 2] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn simple_document_operation_dont_autobatch_with_other() { |     fn simple_document_operation_dont_autobatch_with_other() { | ||||||
|         // addition, updates and deletion can't batch together |         // addition, updates and deletion can't batch together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(UpdateDocuments, true)]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentDeletion]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_del()]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_imp(ReplaceDocuments, true)]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentDeletion]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_del()]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), doc_imp(ReplaceDocuments, true)]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), doc_imp(UpdateDocuments, true)]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|  |  | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, IndexCreation]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), idx_create()]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, IndexCreation]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), idx_create()]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, IndexCreation]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), idx_create()]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|  |  | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, IndexUpdate]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), idx_update()]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, IndexUpdate]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), idx_update()]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, IndexUpdate]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), idx_update()]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|  |  | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, IndexSwap]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), idx_swap()]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, IndexSwap]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), idx_swap()]), @"Some(DocumentImport { method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, IndexSwap]), @"Some(DocumentDeletion { deletion_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), idx_swap()]), @"Some(DocumentDeletion { deletion_ids: [0] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn document_addition_batch_with_settings() { |     fn document_addition_batch_with_settings() { | ||||||
|         // simple case |         // simple case | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true)]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true)]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|  |  | ||||||
|         // multiple settings and doc addition |         // multiple settings and doc addition | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [2, 3], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true), settings(true), settings(true)]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [2, 3], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true), settings(true), settings(true)]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0, 1] })"); | ||||||
|  |  | ||||||
|         // addition and setting unordered |         // addition and setting unordered | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1, 3], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), doc_imp(ReplaceDocuments, true), settings(true)]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0, 2] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1, 3], method: UpdateDocuments, allow_index_creation: true, import_ids: [0, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), doc_imp(UpdateDocuments, true), settings(true)]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0, 2] })"); | ||||||
|  |  | ||||||
|         // We ensure this kind of batch doesn't batch with forbidden operations |         // We ensure this kind of batch doesn't batch with forbidden operations | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), doc_imp(UpdateDocuments, true)]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), doc_imp(ReplaceDocuments, true)]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentDeletion]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), doc_del()]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentDeletion]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), doc_del()]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexCreation]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), idx_create()]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexCreation]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), idx_create()]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexUpdate]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), idx_update()]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexUpdate]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), idx_update()]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexSwap]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), idx_swap()]), @"Some(settingsAnddoc_im(Repl)eDocuments)allow_index_creation: true, import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexSwap]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: UpdateDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), idx_swap()]), @"Some(settingsAnddoc_im(Upda)Documents)allow_index_creation: true, import_ids: [0] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn clear_and_additions() { |     fn clear_and_additions() { | ||||||
|         // these two doesn't need to batch |         // these two doesn't need to batch | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentClear, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentClear { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_clr(), doc_imp(ReplaceDocuments, true)]), @"Some(doc_clr() { ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentClear, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentClear { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_clr(), doc_imp(UpdateDocuments, true)]), @"Some(doc_clr() { ids: [0] })"); | ||||||
|  |  | ||||||
|         // Basic use case |         // Basic use case | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentClear]), @"Some(DocumentClear { ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true), doc_clr()]), @"Some(doc_clr() { ids: [0, 1, 2] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentClear]), @"Some(DocumentClear { ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_imp(UpdateDocuments, true), doc_clr()]), @"Some(doc_clr() { ids: [0, 1, 2] })"); | ||||||
|  |  | ||||||
|         // This batch kind doesn't mix with other document addition |         // This batch kind doesn't mix with other document addition | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentClear, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentClear { ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true), doc_clr(), doc_imp(ReplaceDocuments, true)]), @"Some(doc_clr() { ids: [0, 1, 2] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentClear, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(DocumentClear { ids: [0, 1, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_imp(UpdateDocuments, true), doc_clr(), doc_imp(UpdateDocuments, true)]), @"Some(doc_clr() { ids: [0, 1, 2] })"); | ||||||
|  |  | ||||||
|         // But you can batch multiple clear together |         // But you can batch multiple clear together | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentClear, DocumentClear, DocumentClear]), @"Some(DocumentClear { ids: [0, 1, 2, 3, 4] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true), doc_clr(), doc_clr(), doc_clr()]), @"Some(doc_clr() { ids: [0, 1, 2, 3, 4] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentImport { method: UpdateDocuments, allow_index_creation: true }, DocumentClear, DocumentClear, DocumentClear]), @"Some(DocumentClear { ids: [0, 1, 2, 3, 4] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), doc_imp(UpdateDocuments, true), doc_clr(), doc_clr(), doc_clr()]), @"Some(doc_clr() { ids: [0, 1, 2, 3, 4] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn clear_and_additions_and_settings() { |     fn clear_and_additions_and_settings() { | ||||||
|         // A clear don't need to autobatch the settings that happens AFTER there is no documents |         // A clear don't need to autobatch the settings that happens AFTER there is no documents | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentClear, Settings { allow_index_creation: true }]), @"Some(DocumentClear { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_clr(), settings(true)]), @"Some(doc_clr() { ids: [0] })"); | ||||||
|  |  | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([Settings { allow_index_creation: true }, DocumentClear, Settings { allow_index_creation: true }]), @"Some(ClearAndSettings { other: [1], allow_index_creation: true, settings_ids: [0, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([settings(true), doc_clr(), settings(true)]), @"Some(clearAndSettings([1) allow_index_creation: true, settings_ids: [0, 2] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentClear]), @"Some(ClearAndSettings { other: [0, 2], allow_index_creation: true, settings_ids: [1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), doc_clr()]), @"Some(clearAndSettings([0)2], allow_index_creation: true, settings_ids: [1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentClear]), @"Some(ClearAndSettings { other: [0, 2], allow_index_creation: true, settings_ids: [1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), doc_clr()]), @"Some(clearAndSettings([0)2], allow_index_creation: true, settings_ids: [1] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn anything_and_index_deletion() { |     fn anything_and_index_deletion() { | ||||||
|         // The indexdeletion doesn't batch with anything that happens AFTER |         // The indexdeletion doesn't batch with anything that happens AFTER | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([IndexDeletion, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(IndexDeletion { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([idx_del(), doc_imp(ReplaceDocuments, true)]), @"Some(idx_del() { ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([IndexDeletion, DocumentImport { method: UpdateDocuments, allow_index_creation: true }]), @"Some(IndexDeletion { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([idx_del(), doc_imp(UpdateDocuments, true)]), @"Some(idx_del() { ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([IndexDeletion, DocumentDeletion]), @"Some(IndexDeletion { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([idx_del(), doc_del()]), @"Some(idx_del() { ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([IndexDeletion, DocumentClear]), @"Some(IndexDeletion { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([idx_del(), doc_clr()]), @"Some(idx_del() { ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([IndexDeletion, Settings { allow_index_creation: true }]), @"Some(IndexDeletion { ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([idx_del(), settings(true)]), @"Some(idx_del() { ids: [0] })"); | ||||||
|  |  | ||||||
|         // The index deletion can accept almost any type of BatchKind and transform it to an IndexDeletion |         // The index deletion can accept almost any type of BatchKind and transform it to an idx_del() | ||||||
|         // First, the basic cases |         // First, the basic cases | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), idx_del()]), @"Some(idx_del() { ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), idx_del()]), @"Some(idx_del() { ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentDeletion, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_del(), idx_del()]), @"Some(idx_del() { ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentClear, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_clr(), idx_del()]), @"Some(idx_del() { ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([Settings { allow_index_creation: true }, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([settings(true), idx_del()]), @"Some(idx_del() { ids: [0, 1] })"); | ||||||
|  |  | ||||||
|         // Then the mixed cases |         // Then the mixed cases | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 2, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), idx_del()]), @"Some(idx_del() { ids: [0, 2, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, IndexDeletion]), @"Some(IndexDeletion { ids: [0, 2, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), idx_del()]), @"Some(idx_del() { ids: [0, 2, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentClear, IndexDeletion]), @"Some(IndexDeletion { ids: [1, 3, 0, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true), doc_clr(), idx_del()]), @"Some(idx_del() { ids: [1, 3, 0, 2] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: UpdateDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }, DocumentClear, IndexDeletion]), @"Some(IndexDeletion { ids: [1, 3, 0, 2] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(UpdateDocuments, true), settings(true), doc_clr(), idx_del()]), @"Some(idx_del() { ids: [1, 3, 0, 2] })"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     #[test] |     #[test] | ||||||
|     fn allowed_and_disallowed_index_creation() { |     fn allowed_and_disallowed_index_creation() { | ||||||
|         // DocumentImport that can create indexes can't be mixed with those disallowed to do so |         // doc_imp(indexes canbe)ixed with those disallowed to do so | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: false }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: false, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, false), doc_imp(ReplaceDocuments, true)]), @"Some(doc_imp(ReplaceDocuments, false)import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, DocumentImport { method: ReplaceDocuments, allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: true, import_ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), doc_imp(ReplaceDocuments, true)]), @"Some(doc_imp(ReplaceDocuments, true)import_ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: false }, DocumentImport { method: ReplaceDocuments, allow_index_creation: false }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: false, import_ids: [0, 1] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, false), doc_imp(ReplaceDocuments, false)]), @"Some(doc_imp(ReplaceDocuments, false)import_ids: [0, 1] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: true }, Settings { allow_index_creation: true }]), @"Some(SettingsAndDocumentImport { settings_ids: [1], method: ReplaceDocuments, allow_index_creation: true, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, true), settings(true)]), @"Some(settingsAnddoc_imp(: ReplaceDocuments: true)import_ids: [0] })"); | ||||||
|         assert_smol_debug_snapshot!(autobatch_from([DocumentImport { method: ReplaceDocuments, allow_index_creation: false }, Settings { allow_index_creation: true }]), @"Some(DocumentImport { method: ReplaceDocuments, allow_index_creation: false, import_ids: [0] })"); |         assert_smol_debug_snapshot!(autobatch_from([doc_imp(ReplaceDocuments, false), settings(true)]), @"Some(doc_imp(ReplaceDocuments, false)import_ids: [0] })"); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +1,6 @@ | |||||||
| use crate::{ | use crate::{autobatcher::BatchKind, Error, IndexScheduler, Result, TaskId}; | ||||||
|     autobatcher::BatchKind, |  | ||||||
|     task::{KindWithContent, Task}, |  | ||||||
|     Error, IndexScheduler, Result, TaskId, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| use meilisearch_types::tasks::{Details, Kind, Status}; | use meilisearch_types::tasks::{Details, Kind, KindWithContent, Status, Task}; | ||||||
|  |  | ||||||
| use log::{debug, info}; | use log::{debug, info}; | ||||||
| use meilisearch_types::milli::update::IndexDocumentsConfig; | use meilisearch_types::milli::update::IndexDocumentsConfig; | ||||||
| @@ -432,7 +428,7 @@ impl IndexScheduler { | |||||||
|                 .map(|task_id| { |                 .map(|task_id| { | ||||||
|                     self.get_task(rtxn, task_id) |                     self.get_task(rtxn, task_id) | ||||||
|                         .and_then(|task| task.ok_or(Error::CorruptedTaskQueue)) |                         .and_then(|task| task.ok_or(Error::CorruptedTaskQueue)) | ||||||
|                         .map(|task| (task.uid, task.kind.as_kind())) |                         .map(|task| (task.uid, task.kind)) | ||||||
|                 }) |                 }) | ||||||
|                 .collect::<Result<Vec<_>>>()?; |                 .collect::<Result<Vec<_>>>()?; | ||||||
|  |  | ||||||
| @@ -862,9 +858,7 @@ impl IndexScheduler { | |||||||
|             (bitmap.remove(task.uid)); |             (bitmap.remove(task.uid)); | ||||||
|         })?; |         })?; | ||||||
|  |  | ||||||
|         task.remove_data()?; |  | ||||||
|         self.all_tasks.delete(wtxn, &task_id)?; |         self.all_tasks.delete(wtxn, &task_id)?; | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,8 +11,7 @@ pub type Result<T> = std::result::Result<T, Error>; | |||||||
| pub type TaskId = u32; | pub type TaskId = u32; | ||||||
|  |  | ||||||
| pub use error::Error; | pub use error::Error; | ||||||
| use meilisearch_types::tasks::{Kind, Status, TaskView}; | use meilisearch_types::tasks::{Kind, KindWithContent, Status, Task}; | ||||||
| pub use task::KindWithContent; |  | ||||||
|  |  | ||||||
| use std::path::PathBuf; | use std::path::PathBuf; | ||||||
| use std::sync::{Arc, RwLock}; | use std::sync::{Arc, RwLock}; | ||||||
| @@ -31,7 +30,6 @@ use meilisearch_types::milli::update::IndexerConfig; | |||||||
| use meilisearch_types::milli::{Index, RoaringBitmapCodec, BEU32}; | use meilisearch_types::milli::{Index, RoaringBitmapCodec, BEU32}; | ||||||
|  |  | ||||||
| use crate::index_mapper::IndexMapper; | use crate::index_mapper::IndexMapper; | ||||||
| use crate::task::Task; |  | ||||||
|  |  | ||||||
| const DEFAULT_LIMIT: fn() -> u32 = || 20; | const DEFAULT_LIMIT: fn() -> u32 = || 20; | ||||||
|  |  | ||||||
| @@ -246,7 +244,7 @@ impl IndexScheduler { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /// Returns the tasks corresponding to the query. |     /// Returns the tasks corresponding to the query. | ||||||
|     pub fn get_tasks(&self, query: Query) -> Result<Vec<TaskView>> { |     pub fn get_tasks(&self, query: Query) -> Result<Vec<Task>> { | ||||||
|         let rtxn = self.env.read_txn()?; |         let rtxn = self.env.read_txn()?; | ||||||
|         let last_task_id = match self.last_task_id(&rtxn)? { |         let last_task_id = match self.last_task_id(&rtxn)? { | ||||||
|             Some(tid) => query.from.map(|from| from.min(tid)).unwrap_or(tid), |             Some(tid) => query.from.map(|from| from.min(tid)).unwrap_or(tid), | ||||||
| @@ -292,13 +290,13 @@ impl IndexScheduler { | |||||||
|             .map_err(|_| Error::CorruptedTaskQueue)? |             .map_err(|_| Error::CorruptedTaskQueue)? | ||||||
|             .clone(); |             .clone(); | ||||||
|  |  | ||||||
|         let ret = tasks.into_iter().map(|task| task.as_task_view()); |         let ret = tasks.into_iter(); | ||||||
|         if processing.is_empty() { |         if processing.is_empty() { | ||||||
|             Ok(ret.collect()) |             Ok(ret.collect()) | ||||||
|         } else { |         } else { | ||||||
|             Ok(ret |             Ok(ret | ||||||
|                 .map(|task| match processing.contains(task.uid) { |                 .map(|task| match processing.contains(task.uid) { | ||||||
|                     true => TaskView { |                     true => Task { | ||||||
|                         status: Status::Processing, |                         status: Status::Processing, | ||||||
|                         started_at: Some(started_at), |                         started_at: Some(started_at), | ||||||
|                         ..task |                         ..task | ||||||
| @@ -311,7 +309,7 @@ impl IndexScheduler { | |||||||
|  |  | ||||||
|     /// Register a new task in the scheduler. If it fails and data was associated with the task |     /// Register a new task in the scheduler. If it fails and data was associated with the task | ||||||
|     /// it tries to delete the file. |     /// it tries to delete the file. | ||||||
|     pub fn register(&self, task: KindWithContent) -> Result<TaskView> { |     pub fn register(&self, task: KindWithContent) -> Result<Task> { | ||||||
|         let mut wtxn = self.env.write_txn()?; |         let mut wtxn = self.env.write_txn()?; | ||||||
|  |  | ||||||
|         let task = Task { |         let task = Task { | ||||||
| @@ -343,13 +341,10 @@ impl IndexScheduler { | |||||||
|             (bitmap.insert(task.uid)); |             (bitmap.insert(task.uid)); | ||||||
|         })?; |         })?; | ||||||
|  |  | ||||||
|         // we persist the file in last to be sure everything before was applied successfuly |  | ||||||
|         task.persist()?; |  | ||||||
|  |  | ||||||
|         match wtxn.commit() { |         match wtxn.commit() { | ||||||
|             Ok(()) => (), |             Ok(()) => (), | ||||||
|             e @ Err(_) => { |             e @ Err(_) => { | ||||||
|                 task.remove_data()?; |                 todo!("remove the data associated with the task"); | ||||||
|                 e?; |                 e?; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -357,7 +352,7 @@ impl IndexScheduler { | |||||||
|         // notify the scheduler loop to execute a new tick |         // notify the scheduler loop to execute a new tick | ||||||
|         self.wake_up.signal(); |         self.wake_up.signal(); | ||||||
|  |  | ||||||
|         Ok(task.as_task_view()) |         Ok(task) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     pub fn create_update_file(&self) -> Result<(Uuid, File)> { |     pub fn create_update_file(&self) -> Result<(Uuid, File)> { | ||||||
| @@ -416,7 +411,6 @@ impl IndexScheduler { | |||||||
|                     task.finished_at = Some(finished_at); |                     task.finished_at = Some(finished_at); | ||||||
|                     // TODO the info field should've been set by the process_batch function |                     // TODO the info field should've been set by the process_batch function | ||||||
|                     self.update_task(&mut wtxn, &task)?; |                     self.update_task(&mut wtxn, &task)?; | ||||||
|                     task.remove_data()?; |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             // In case of a failure we must get back and patch all the tasks with the error. |             // In case of a failure we must get back and patch all the tasks with the error. | ||||||
| @@ -430,7 +424,6 @@ impl IndexScheduler { | |||||||
|                     task.error = Some(error.clone()); |                     task.error = Some(error.clone()); | ||||||
|  |  | ||||||
|                     self.update_task(&mut wtxn, &task)?; |                     self.update_task(&mut wtxn, &task)?; | ||||||
|                     task.remove_data()?; |  | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -585,7 +578,7 @@ mod tests { | |||||||
|  |  | ||||||
|             assert_eq!(task.uid, idx as u32); |             assert_eq!(task.uid, idx as u32); | ||||||
|             assert_eq!(task.status, Status::Enqueued); |             assert_eq!(task.status, Status::Enqueued); | ||||||
|             assert_eq!(task.kind, k); |             assert_eq!(task.kind.as_kind(), k); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         assert_snapshot!(snapshot_index_scheduler(&index_scheduler)); |         assert_snapshot!(snapshot_index_scheduler(&index_scheduler)); | ||||||
|   | |||||||
| @@ -3,278 +3,10 @@ use meilisearch_types::error::ResponseError; | |||||||
| use meilisearch_types::milli::update::IndexDocumentsMethod; | use meilisearch_types::milli::update::IndexDocumentsMethod; | ||||||
| use meilisearch_types::settings::{Settings, Unchecked}; | use meilisearch_types::settings::{Settings, Unchecked}; | ||||||
|  |  | ||||||
| use meilisearch_types::tasks::{Details, Kind, Status, TaskView}; | use meilisearch_types::tasks::{Details, Kind, Status}; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use std::path::PathBuf; | use std::path::PathBuf; | ||||||
| use time::OffsetDateTime; | use time::OffsetDateTime; | ||||||
| use uuid::Uuid; | use uuid::Uuid; | ||||||
|  |  | ||||||
| use crate::TaskId; | use crate::TaskId; | ||||||
|  |  | ||||||
| #[derive(Debug, PartialEq, Serialize, Deserialize)] |  | ||||||
| #[serde(rename_all = "camelCase")] |  | ||||||
| pub struct Task { |  | ||||||
|     pub uid: TaskId, |  | ||||||
|  |  | ||||||
|     #[serde(with = "time::serde::rfc3339")] |  | ||||||
|     pub enqueued_at: OffsetDateTime, |  | ||||||
|     #[serde(with = "time::serde::rfc3339::option")] |  | ||||||
|     pub started_at: Option<OffsetDateTime>, |  | ||||||
|     #[serde(with = "time::serde::rfc3339::option")] |  | ||||||
|     pub finished_at: Option<OffsetDateTime>, |  | ||||||
|  |  | ||||||
|     pub error: Option<ResponseError>, |  | ||||||
|     pub details: Option<Details>, |  | ||||||
|  |  | ||||||
|     pub status: Status, |  | ||||||
|     pub kind: KindWithContent, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl Task { |  | ||||||
|     /// Persist all the temp files associated with the task. |  | ||||||
|     pub fn persist(&self) -> Result<()> { |  | ||||||
|         self.kind.persist() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// Delete all the files associated with the task. |  | ||||||
|     pub fn remove_data(&self) -> Result<()> { |  | ||||||
|         self.kind.remove_data() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// Return the list of indexes updated by this tasks. |  | ||||||
|     pub fn indexes(&self) -> Option<Vec<&str>> { |  | ||||||
|         self.kind.indexes() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// Convert a Task to a TaskView |  | ||||||
|     pub fn as_task_view(&self) -> TaskView { |  | ||||||
|         TaskView { |  | ||||||
|             uid: self.uid, |  | ||||||
|             index_uid: self |  | ||||||
|                 .indexes() |  | ||||||
|                 .and_then(|vec| vec.first().map(|i| i.to_string())), |  | ||||||
|             status: self.status, |  | ||||||
|             kind: self.kind.as_kind(), |  | ||||||
|             details: self.details.as_ref().map(Details::as_details_view), |  | ||||||
|             error: self.error.clone(), |  | ||||||
|             duration: self |  | ||||||
|                 .started_at |  | ||||||
|                 .zip(self.finished_at) |  | ||||||
|                 .map(|(start, end)| end - start), |  | ||||||
|             enqueued_at: self.enqueued_at, |  | ||||||
|             started_at: self.started_at, |  | ||||||
|             finished_at: self.finished_at, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Debug, PartialEq, Serialize, Deserialize)] |  | ||||||
| #[serde(rename_all = "camelCase")] |  | ||||||
| pub enum KindWithContent { |  | ||||||
|     DocumentImport { |  | ||||||
|         index_uid: String, |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|         method: IndexDocumentsMethod, |  | ||||||
|         content_file: Uuid, |  | ||||||
|         documents_count: u64, |  | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     DocumentDeletion { |  | ||||||
|         index_uid: String, |  | ||||||
|         documents_ids: Vec<String>, |  | ||||||
|     }, |  | ||||||
|     DocumentClear { |  | ||||||
|         index_uid: String, |  | ||||||
|     }, |  | ||||||
|     Settings { |  | ||||||
|         index_uid: String, |  | ||||||
|         new_settings: Settings<Unchecked>, |  | ||||||
|         is_deletion: bool, |  | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     IndexDeletion { |  | ||||||
|         index_uid: String, |  | ||||||
|     }, |  | ||||||
|     IndexCreation { |  | ||||||
|         index_uid: String, |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|     }, |  | ||||||
|     IndexUpdate { |  | ||||||
|         index_uid: String, |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|     }, |  | ||||||
|     IndexSwap { |  | ||||||
|         lhs: String, |  | ||||||
|         rhs: String, |  | ||||||
|     }, |  | ||||||
|     CancelTask { |  | ||||||
|         tasks: Vec<TaskId>, |  | ||||||
|     }, |  | ||||||
|     DeleteTasks { |  | ||||||
|         query: String, |  | ||||||
|         tasks: Vec<TaskId>, |  | ||||||
|     }, |  | ||||||
|     DumpExport { |  | ||||||
|         output: PathBuf, |  | ||||||
|     }, |  | ||||||
|     Snapshot, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| impl KindWithContent { |  | ||||||
|     pub fn as_kind(&self) -> Kind { |  | ||||||
|         match self { |  | ||||||
|             KindWithContent::DocumentImport { |  | ||||||
|                 method, |  | ||||||
|                 allow_index_creation, |  | ||||||
|                 .. |  | ||||||
|             } => Kind::DocumentImport { |  | ||||||
|                 method: *method, |  | ||||||
|                 allow_index_creation: *allow_index_creation, |  | ||||||
|             }, |  | ||||||
|             KindWithContent::DocumentDeletion { .. } => Kind::DocumentDeletion, |  | ||||||
|             KindWithContent::DocumentClear { .. } => Kind::DocumentClear, |  | ||||||
|             KindWithContent::Settings { |  | ||||||
|                 allow_index_creation, |  | ||||||
|                 .. |  | ||||||
|             } => Kind::Settings { |  | ||||||
|                 allow_index_creation: *allow_index_creation, |  | ||||||
|             }, |  | ||||||
|             KindWithContent::IndexCreation { .. } => Kind::IndexCreation, |  | ||||||
|             KindWithContent::IndexDeletion { .. } => Kind::IndexDeletion, |  | ||||||
|             KindWithContent::IndexUpdate { .. } => Kind::IndexUpdate, |  | ||||||
|             KindWithContent::IndexSwap { .. } => Kind::IndexSwap, |  | ||||||
|             KindWithContent::CancelTask { .. } => Kind::CancelTask, |  | ||||||
|             KindWithContent::DeleteTasks { .. } => Kind::DeleteTasks, |  | ||||||
|             KindWithContent::DumpExport { .. } => Kind::DumpExport, |  | ||||||
|             KindWithContent::Snapshot => Kind::Snapshot, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn persist(&self) -> Result<()> { |  | ||||||
|         use KindWithContent::*; |  | ||||||
|  |  | ||||||
|         match self { |  | ||||||
|             DocumentImport { .. } => { |  | ||||||
|                 // TODO: TAMO: persist the file |  | ||||||
|                 // content_file.persist(); |  | ||||||
|                 Ok(()) |  | ||||||
|             } |  | ||||||
|             DocumentDeletion { .. } |  | ||||||
|             | DocumentClear { .. } |  | ||||||
|             | Settings { .. } |  | ||||||
|             | IndexCreation { .. } |  | ||||||
|             | IndexDeletion { .. } |  | ||||||
|             | IndexUpdate { .. } |  | ||||||
|             | IndexSwap { .. } |  | ||||||
|             | CancelTask { .. } |  | ||||||
|             | DeleteTasks { .. } |  | ||||||
|             | DumpExport { .. } |  | ||||||
|             | Snapshot => Ok(()), // There is nothing to persist for all these tasks |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn remove_data(&self) -> Result<()> { |  | ||||||
|         use KindWithContent::*; |  | ||||||
|  |  | ||||||
|         match self { |  | ||||||
|             DocumentImport { .. } => { |  | ||||||
|                 // TODO: TAMO: delete the file |  | ||||||
|                 // content_file.delete(); |  | ||||||
|                 Ok(()) |  | ||||||
|             } |  | ||||||
|             IndexCreation { .. } |  | ||||||
|             | DocumentDeletion { .. } |  | ||||||
|             | DocumentClear { .. } |  | ||||||
|             | Settings { .. } |  | ||||||
|             | IndexDeletion { .. } |  | ||||||
|             | IndexUpdate { .. } |  | ||||||
|             | IndexSwap { .. } |  | ||||||
|             | CancelTask { .. } |  | ||||||
|             | DeleteTasks { .. } |  | ||||||
|             | DumpExport { .. } |  | ||||||
|             | Snapshot => Ok(()), // There is no data associated with all these tasks |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     pub fn indexes(&self) -> Option<Vec<&str>> { |  | ||||||
|         use KindWithContent::*; |  | ||||||
|  |  | ||||||
|         match self { |  | ||||||
|             DumpExport { .. } | Snapshot | CancelTask { .. } | DeleteTasks { .. } => None, |  | ||||||
|             DocumentImport { index_uid, .. } |  | ||||||
|             | DocumentDeletion { index_uid, .. } |  | ||||||
|             | DocumentClear { index_uid } |  | ||||||
|             | Settings { index_uid, .. } |  | ||||||
|             | IndexCreation { index_uid, .. } |  | ||||||
|             | IndexUpdate { index_uid, .. } |  | ||||||
|             | IndexDeletion { index_uid } => Some(vec![index_uid]), |  | ||||||
|             IndexSwap { lhs, rhs } => Some(vec![lhs, rhs]), |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /// Returns the default `Details` that correspond to this `KindWithContent`, |  | ||||||
|     /// `None` if it cannot be generated. |  | ||||||
|     pub fn default_details(&self) -> Option<Details> { |  | ||||||
|         match self { |  | ||||||
|             KindWithContent::DocumentImport { |  | ||||||
|                 documents_count, .. |  | ||||||
|             } => Some(Details::DocumentAddition { |  | ||||||
|                 received_documents: *documents_count, |  | ||||||
|                 indexed_documents: 0, |  | ||||||
|             }), |  | ||||||
|             KindWithContent::DocumentDeletion { |  | ||||||
|                 index_uid: _, |  | ||||||
|                 documents_ids, |  | ||||||
|             } => Some(Details::DocumentDeletion { |  | ||||||
|                 received_document_ids: documents_ids.len(), |  | ||||||
|                 deleted_documents: None, |  | ||||||
|             }), |  | ||||||
|             KindWithContent::DocumentClear { .. } => Some(Details::ClearAll { |  | ||||||
|                 deleted_documents: None, |  | ||||||
|             }), |  | ||||||
|             KindWithContent::Settings { new_settings, .. } => Some(Details::Settings { |  | ||||||
|                 settings: new_settings.clone(), |  | ||||||
|             }), |  | ||||||
|             KindWithContent::IndexDeletion { .. } => None, |  | ||||||
|             KindWithContent::IndexCreation { primary_key, .. } |  | ||||||
|             | KindWithContent::IndexUpdate { primary_key, .. } => Some(Details::IndexInfo { |  | ||||||
|                 primary_key: primary_key.clone(), |  | ||||||
|             }), |  | ||||||
|             KindWithContent::IndexSwap { .. } => { |  | ||||||
|                 todo!() |  | ||||||
|             } |  | ||||||
|             KindWithContent::CancelTask { .. } => { |  | ||||||
|                 None // TODO: check correctness of this return value |  | ||||||
|             } |  | ||||||
|             KindWithContent::DeleteTasks { query, tasks } => Some(Details::DeleteTasks { |  | ||||||
|                 matched_tasks: tasks.len(), |  | ||||||
|                 deleted_tasks: None, |  | ||||||
|                 original_query: query.clone(), |  | ||||||
|             }), |  | ||||||
|             KindWithContent::DumpExport { .. } => None, |  | ||||||
|             KindWithContent::Snapshot => None, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[cfg(test)] |  | ||||||
| mod tests { |  | ||||||
|     use meilisearch_types::heed::{types::SerdeJson, BytesDecode, BytesEncode}; |  | ||||||
|  |  | ||||||
|     use crate::assert_smol_debug_snapshot; |  | ||||||
|  |  | ||||||
|     use super::Details; |  | ||||||
|  |  | ||||||
|     #[test] |  | ||||||
|     fn bad_deser() { |  | ||||||
|         let details = Details::DeleteTasks { |  | ||||||
|             matched_tasks: 1, |  | ||||||
|             deleted_tasks: None, |  | ||||||
|             original_query: "hello".to_owned(), |  | ||||||
|         }; |  | ||||||
|         let serialised = SerdeJson::<Details>::bytes_encode(&details).unwrap(); |  | ||||||
|         let deserialised = SerdeJson::<Details>::bytes_decode(&serialised).unwrap(); |  | ||||||
|         assert_smol_debug_snapshot!(details, @r###"DeleteTasks { matched_tasks: 1, deleted_tasks: None, original_query: "hello" }"###); |  | ||||||
|         assert_smol_debug_snapshot!(deserialised, @r###"DeleteTasks { matched_tasks: 1, deleted_tasks: None, original_query: "hello" }"###); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| use actix_web::web::Data; | use actix_web::web::Data; | ||||||
| use actix_web::{web, HttpRequest, HttpResponse}; | use actix_web::{web, HttpRequest, HttpResponse}; | ||||||
| use index_scheduler::IndexScheduler; | use index_scheduler::IndexScheduler; | ||||||
| use index_scheduler::KindWithContent; |  | ||||||
| use log::debug; | use log::debug; | ||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
|  | use meilisearch_types::tasks::KindWithContent; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
|  |  | ||||||
| use crate::analytics::Analytics; | use crate::analytics::Analytics; | ||||||
|   | |||||||
| @@ -6,14 +6,14 @@ use actix_web::HttpMessage; | |||||||
| use actix_web::{web, HttpRequest, HttpResponse}; | use actix_web::{web, HttpRequest, HttpResponse}; | ||||||
| use bstr::ByteSlice; | use bstr::ByteSlice; | ||||||
| use futures::StreamExt; | use futures::StreamExt; | ||||||
| use index_scheduler::{IndexScheduler, KindWithContent}; | use index_scheduler::IndexScheduler; | ||||||
| use log::debug; | use log::debug; | ||||||
| use meilisearch_types::document_formats::{read_csv, read_json, read_ndjson, PayloadType}; | use meilisearch_types::document_formats::{read_csv, read_json, read_ndjson, PayloadType}; | ||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
| use meilisearch_types::heed::RoTxn; | use meilisearch_types::heed::RoTxn; | ||||||
| use meilisearch_types::milli::update::IndexDocumentsMethod; | use meilisearch_types::milli::update::IndexDocumentsMethod; | ||||||
| use meilisearch_types::star_or::StarOr; | use meilisearch_types::star_or::StarOr; | ||||||
| use meilisearch_types::tasks::TaskView; | use meilisearch_types::tasks::KindWithContent; | ||||||
| use meilisearch_types::{milli, Document, Index}; | use meilisearch_types::{milli, Document, Index}; | ||||||
| use mime::Mime; | use mime::Mime; | ||||||
| use once_cell::sync::Lazy; | use once_cell::sync::Lazy; | ||||||
| @@ -26,7 +26,7 @@ use crate::error::MeilisearchHttpError; | |||||||
| use crate::extractors::authentication::{policies::*, GuardedData}; | use crate::extractors::authentication::{policies::*, GuardedData}; | ||||||
| use crate::extractors::payload::Payload; | use crate::extractors::payload::Payload; | ||||||
| use crate::extractors::sequential_extractor::SeqHandler; | use crate::extractors::sequential_extractor::SeqHandler; | ||||||
| use crate::routes::{fold_star_or, PaginationView}; | use crate::routes::{fold_star_or, PaginationView, SummarizedTaskView}; | ||||||
|  |  | ||||||
| static ACCEPTED_CONTENT_TYPE: Lazy<Vec<String>> = Lazy::new(|| { | static ACCEPTED_CONTENT_TYPE: Lazy<Vec<String>> = Lazy::new(|| { | ||||||
|     vec![ |     vec![ | ||||||
| @@ -216,7 +216,7 @@ async fn document_addition( | |||||||
|     mut body: Payload, |     mut body: Payload, | ||||||
|     method: IndexDocumentsMethod, |     method: IndexDocumentsMethod, | ||||||
|     allow_index_creation: bool, |     allow_index_creation: bool, | ||||||
| ) -> Result<TaskView, MeilisearchHttpError> { | ) -> Result<SummarizedTaskView, MeilisearchHttpError> { | ||||||
|     let format = match mime_type |     let format = match mime_type | ||||||
|         .as_ref() |         .as_ref() | ||||||
|         .map(|m| (m.type_().as_str(), m.subtype().as_str())) |         .map(|m| (m.type_().as_str(), m.subtype().as_str())) | ||||||
| @@ -292,7 +292,7 @@ async fn document_addition( | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     debug!("returns: {:?}", task); |     debug!("returns: {:?}", task); | ||||||
|     Ok(task) |     Ok(task.into()) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub async fn delete_documents( | pub async fn delete_documents( | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| use actix_web::web::Data; | use actix_web::web::Data; | ||||||
| use actix_web::{web, HttpRequest, HttpResponse}; | use actix_web::{web, HttpRequest, HttpResponse}; | ||||||
| use index_scheduler::{IndexScheduler, KindWithContent, Query}; | use index_scheduler::{IndexScheduler, Query}; | ||||||
| use log::debug; | use log::debug; | ||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
| use meilisearch_types::milli::{self, FieldDistribution, Index}; | use meilisearch_types::milli::{self, FieldDistribution, Index}; | ||||||
| use meilisearch_types::tasks::Status; | use meilisearch_types::tasks::{KindWithContent, Status}; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
| use time::OffsetDateTime; | use time::OffsetDateTime; | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ use fst::IntoStreamer; | |||||||
| use log::debug; | use log::debug; | ||||||
|  |  | ||||||
| use actix_web::{web, HttpRequest, HttpResponse}; | use actix_web::{web, HttpRequest, HttpResponse}; | ||||||
| use index_scheduler::{IndexScheduler, KindWithContent}; | use index_scheduler::IndexScheduler; | ||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
| use meilisearch_types::heed::RoTxn; | use meilisearch_types::heed::RoTxn; | ||||||
| use meilisearch_types::milli::update::Setting; | use meilisearch_types::milli::update::Setting; | ||||||
| @@ -15,6 +15,7 @@ use meilisearch_types::settings::{ | |||||||
|     Checked, FacetingSettings, MinWordSizeTyposSetting, PaginationSettings, Settings, TypoSettings, |     Checked, FacetingSettings, MinWordSizeTyposSetting, PaginationSettings, Settings, TypoSettings, | ||||||
|     Unchecked, |     Unchecked, | ||||||
| }; | }; | ||||||
|  | use meilisearch_types::tasks::KindWithContent; | ||||||
| use meilisearch_types::Index; | use meilisearch_types::Index; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
|  |  | ||||||
| @@ -30,9 +31,10 @@ macro_rules! make_setting_route { | |||||||
|             use actix_web::{web, HttpRequest, HttpResponse, Resource}; |             use actix_web::{web, HttpRequest, HttpResponse, Resource}; | ||||||
|             use log::debug; |             use log::debug; | ||||||
|  |  | ||||||
|             use index_scheduler::{IndexScheduler, KindWithContent}; |             use index_scheduler::IndexScheduler; | ||||||
|             use meilisearch_types::milli::update::Setting; |             use meilisearch_types::milli::update::Setting; | ||||||
|             use meilisearch_types::settings::Settings; |             use meilisearch_types::settings::Settings; | ||||||
|  |             use meilisearch_types::tasks::KindWithContent; | ||||||
|  |  | ||||||
|             use meilisearch_types::error::ResponseError; |             use meilisearch_types::error::ResponseError; | ||||||
|             use $crate::analytics::Analytics; |             use $crate::analytics::Analytics; | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ use log::debug; | |||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
| use meilisearch_types::settings::{Settings, Unchecked}; | use meilisearch_types::settings::{Settings, Unchecked}; | ||||||
| use meilisearch_types::star_or::StarOr; | use meilisearch_types::star_or::StarOr; | ||||||
| use meilisearch_types::tasks::Status; | use meilisearch_types::tasks::{Kind, Status, Task, TaskId}; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
| use time::OffsetDateTime; | use time::OffsetDateTime; | ||||||
| @@ -49,6 +49,30 @@ where | |||||||
|  |  | ||||||
| const PAGINATION_DEFAULT_LIMIT: fn() -> usize = || 20; | const PAGINATION_DEFAULT_LIMIT: fn() -> usize = || 20; | ||||||
|  |  | ||||||
|  | #[derive(Debug, Serialize)] | ||||||
|  | #[serde(rename_all = "camelCase")] | ||||||
|  | pub struct SummarizedTaskView { | ||||||
|  |     task_uid: TaskId, | ||||||
|  |     index_uid: Option<String>, | ||||||
|  |     status: Status, | ||||||
|  |     #[serde(rename = "type")] | ||||||
|  |     kind: Kind, | ||||||
|  |     #[serde(serialize_with = "time::serde::rfc3339::serialize")] | ||||||
|  |     enqueued_at: OffsetDateTime, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<Task> for SummarizedTaskView { | ||||||
|  |     fn from(task: Task) -> Self { | ||||||
|  |         SummarizedTaskView { | ||||||
|  |             task_uid: task.uid, | ||||||
|  |             index_uid: task.index_uid().map(|s| s.to_string()), | ||||||
|  |             status: task.status, | ||||||
|  |             kind: task.kind.as_kind(), | ||||||
|  |             enqueued_at: task.enqueued_at, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Copy, Deserialize)] | #[derive(Debug, Clone, Copy, Deserialize)] | ||||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||||
| pub struct Pagination { | pub struct Pagination { | ||||||
| @@ -266,7 +290,7 @@ async fn get_stats( | |||||||
|     )?; |     )?; | ||||||
|     let processing_index = processing_task |     let processing_index = processing_task | ||||||
|         .first() |         .first() | ||||||
|         .and_then(|task| task.index_uid.clone()); |         .and_then(|task| task.index_uid().clone()); | ||||||
|  |  | ||||||
|     for (name, index) in index_scheduler.indexes()? { |     for (name, index) in index_scheduler.indexes()? { | ||||||
|         if !search_rules.is_index_authorized(&name) { |         if !search_rules.is_index_authorized(&name) { | ||||||
|   | |||||||
| @@ -3,11 +3,13 @@ use actix_web::{web, HttpRequest, HttpResponse}; | |||||||
| use index_scheduler::{IndexScheduler, TaskId}; | use index_scheduler::{IndexScheduler, TaskId}; | ||||||
| use meilisearch_types::error::ResponseError; | use meilisearch_types::error::ResponseError; | ||||||
| use meilisearch_types::index_uid::IndexUid; | use meilisearch_types::index_uid::IndexUid; | ||||||
|  | use meilisearch_types::settings::{Settings, Unchecked}; | ||||||
| use meilisearch_types::star_or::StarOr; | use meilisearch_types::star_or::StarOr; | ||||||
| use meilisearch_types::tasks::{Kind, Status}; | use meilisearch_types::tasks::{serialize_duration, Details, Kind, Status, Task}; | ||||||
| use serde::Deserialize; | use serde::{Deserialize, Serialize}; | ||||||
| use serde_cs::vec::CS; | use serde_cs::vec::CS; | ||||||
| use serde_json::json; | use serde_json::json; | ||||||
|  | use time::{Duration, OffsetDateTime}; | ||||||
|  |  | ||||||
| use crate::analytics::Analytics; | use crate::analytics::Analytics; | ||||||
| use crate::extractors::authentication::{policies::*, GuardedData}; | use crate::extractors::authentication::{policies::*, GuardedData}; | ||||||
| @@ -22,6 +24,140 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | |||||||
|         .service(web::resource("/{task_id}").route(web::get().to(SeqHandler(get_task)))); |         .service(web::resource("/{task_id}").route(web::get().to(SeqHandler(get_task)))); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[derive(Debug, Clone, PartialEq, Serialize)] | ||||||
|  | #[serde(rename_all = "camelCase")] | ||||||
|  | pub struct TaskView { | ||||||
|  |     pub uid: TaskId, | ||||||
|  |     #[serde(default)] | ||||||
|  |     pub index_uid: Option<String>, | ||||||
|  |     pub status: Status, | ||||||
|  |     #[serde(rename = "type")] | ||||||
|  |     pub kind: Kind, | ||||||
|  |  | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub details: Option<DetailsView>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub error: Option<ResponseError>, | ||||||
|  |  | ||||||
|  |     #[serde( | ||||||
|  |         serialize_with = "serialize_duration", | ||||||
|  |         skip_serializing_if = "Option::is_none", | ||||||
|  |         default | ||||||
|  |     )] | ||||||
|  |     pub duration: Option<Duration>, | ||||||
|  |     #[serde(with = "time::serde::rfc3339")] | ||||||
|  |     pub enqueued_at: OffsetDateTime, | ||||||
|  |     #[serde( | ||||||
|  |         with = "time::serde::rfc3339::option", | ||||||
|  |         skip_serializing_if = "Option::is_none", | ||||||
|  |         default | ||||||
|  |     )] | ||||||
|  |     pub started_at: Option<OffsetDateTime>, | ||||||
|  |     #[serde( | ||||||
|  |         with = "time::serde::rfc3339::option", | ||||||
|  |         skip_serializing_if = "Option::is_none", | ||||||
|  |         default | ||||||
|  |     )] | ||||||
|  |     pub finished_at: Option<OffsetDateTime>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<Task> for TaskView { | ||||||
|  |     fn from(task: Task) -> Self { | ||||||
|  |         TaskView { | ||||||
|  |             uid: task.uid, | ||||||
|  |             index_uid: task | ||||||
|  |                 .indexes() | ||||||
|  |                 .and_then(|vec| vec.first().map(|i| i.to_string())), | ||||||
|  |             status: task.status, | ||||||
|  |             kind: task.kind.as_kind(), | ||||||
|  |             details: task.details.map(DetailsView::from), | ||||||
|  |             error: task.error.clone(), | ||||||
|  |             duration: task | ||||||
|  |                 .started_at | ||||||
|  |                 .zip(task.finished_at) | ||||||
|  |                 .map(|(start, end)| end - start), | ||||||
|  |             enqueued_at: task.enqueued_at, | ||||||
|  |             started_at: task.started_at, | ||||||
|  |             finished_at: task.finished_at, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Default, Debug, PartialEq, Clone, Serialize)] | ||||||
|  | #[serde(rename_all = "camelCase")] | ||||||
|  | pub struct DetailsView { | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub received_documents: Option<u64>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub indexed_documents: Option<u64>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub primary_key: Option<Option<String>>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub received_document_ids: Option<usize>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub deleted_documents: Option<Option<u64>>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub matched_tasks: Option<usize>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub deleted_tasks: Option<Option<usize>>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub original_query: Option<String>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub dump_uid: Option<String>, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     #[serde(flatten)] | ||||||
|  |     pub settings: Option<Settings<Unchecked>>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<Details> for DetailsView { | ||||||
|  |     fn from(details: Details) -> Self { | ||||||
|  |         match details.clone() { | ||||||
|  |             Details::DocumentAddition { | ||||||
|  |                 received_documents, | ||||||
|  |                 indexed_documents, | ||||||
|  |             } => DetailsView { | ||||||
|  |                 received_documents: Some(received_documents), | ||||||
|  |                 indexed_documents: Some(indexed_documents), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::Settings { settings } => DetailsView { | ||||||
|  |                 settings: Some(settings), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::IndexInfo { primary_key } => DetailsView { | ||||||
|  |                 primary_key: Some(primary_key), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::DocumentDeletion { | ||||||
|  |                 received_document_ids, | ||||||
|  |                 deleted_documents, | ||||||
|  |             } => DetailsView { | ||||||
|  |                 received_document_ids: Some(received_document_ids), | ||||||
|  |                 deleted_documents: Some(deleted_documents), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::ClearAll { deleted_documents } => DetailsView { | ||||||
|  |                 deleted_documents: Some(deleted_documents), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::DeleteTasks { | ||||||
|  |                 matched_tasks, | ||||||
|  |                 deleted_tasks, | ||||||
|  |                 original_query, | ||||||
|  |             } => DetailsView { | ||||||
|  |                 matched_tasks: Some(matched_tasks), | ||||||
|  |                 deleted_tasks: Some(deleted_tasks), | ||||||
|  |                 original_query: Some(original_query), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |             Details::Dump { dump_uid } => DetailsView { | ||||||
|  |                 dump_uid: Some(dump_uid), | ||||||
|  |                 ..DetailsView::default() | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Deserialize, Debug)] | #[derive(Deserialize, Debug)] | ||||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||||
| pub struct TasksFilterQuery { | pub struct TasksFilterQuery { | ||||||
|   | |||||||
							
								
								
									
										694
									
								
								meilisearch-http/src/search.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										694
									
								
								meilisearch-http/src/search.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,694 @@ | |||||||
|  | use std::cmp::min; | ||||||
|  | use std::collections::{BTreeMap, BTreeSet, HashSet}; | ||||||
|  | use std::str::FromStr; | ||||||
|  | use std::time::Instant; | ||||||
|  |  | ||||||
|  | use either::Either; | ||||||
|  | use meilisearch_types::{milli, Document}; | ||||||
|  | use milli::tokenizer::TokenizerBuilder; | ||||||
|  | use milli::{ | ||||||
|  |     AscDesc, FieldId, FieldsIdsMap, Filter, FormatOptions, Index, MatchBounds, MatcherBuilder, | ||||||
|  |     SortError, TermsMatchingStrategy, DEFAULT_VALUES_PER_FACET, | ||||||
|  | }; | ||||||
|  | use regex::Regex; | ||||||
|  | use serde::{Deserialize, Serialize}; | ||||||
|  | use serde_json::{json, Value}; | ||||||
|  |  | ||||||
|  | use crate::error::MeilisearchHttpError; | ||||||
|  |  | ||||||
|  | type MatchesPosition = BTreeMap<String, Vec<MatchBounds>>; | ||||||
|  |  | ||||||
|  | pub const DEFAULT_SEARCH_LIMIT: fn() -> usize = || 20; | ||||||
|  | pub const DEFAULT_CROP_LENGTH: fn() -> usize = || 10; | ||||||
|  | pub const DEFAULT_CROP_MARKER: fn() -> String = || "…".to_string(); | ||||||
|  | pub const DEFAULT_HIGHLIGHT_PRE_TAG: fn() -> String = || "<em>".to_string(); | ||||||
|  | pub const DEFAULT_HIGHLIGHT_POST_TAG: fn() -> String = || "</em>".to_string(); | ||||||
|  |  | ||||||
|  | /// The maximimum number of results that the engine | ||||||
|  | /// will be able to return in one search call. | ||||||
|  | pub const DEFAULT_PAGINATION_MAX_TOTAL_HITS: usize = 1000; | ||||||
|  |  | ||||||
|  | #[derive(Deserialize, Debug, Clone, PartialEq, Eq)] | ||||||
|  | #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||||
|  | pub struct SearchQuery { | ||||||
|  |     pub q: Option<String>, | ||||||
|  |     pub offset: Option<usize>, | ||||||
|  |     #[serde(default = "DEFAULT_SEARCH_LIMIT")] | ||||||
|  |     pub limit: usize, | ||||||
|  |     pub attributes_to_retrieve: Option<BTreeSet<String>>, | ||||||
|  |     pub attributes_to_crop: Option<Vec<String>>, | ||||||
|  |     #[serde(default = "DEFAULT_CROP_LENGTH")] | ||||||
|  |     pub crop_length: usize, | ||||||
|  |     pub attributes_to_highlight: Option<HashSet<String>>, | ||||||
|  |     // Default to false | ||||||
|  |     #[serde(default = "Default::default")] | ||||||
|  |     pub show_matches_position: bool, | ||||||
|  |     pub filter: Option<Value>, | ||||||
|  |     pub sort: Option<Vec<String>>, | ||||||
|  |     pub facets: Option<Vec<String>>, | ||||||
|  |     #[serde(default = "DEFAULT_HIGHLIGHT_PRE_TAG")] | ||||||
|  |     pub highlight_pre_tag: String, | ||||||
|  |     #[serde(default = "DEFAULT_HIGHLIGHT_POST_TAG")] | ||||||
|  |     pub highlight_post_tag: String, | ||||||
|  |     #[serde(default = "DEFAULT_CROP_MARKER")] | ||||||
|  |     pub crop_marker: String, | ||||||
|  |     #[serde(default)] | ||||||
|  |     pub matching_strategy: MatchingStrategy, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Deserialize, Debug, Clone, PartialEq, Eq)] | ||||||
|  | #[serde(rename_all = "camelCase")] | ||||||
|  | pub enum MatchingStrategy { | ||||||
|  |     /// Remove query words from last to first | ||||||
|  |     Last, | ||||||
|  |     /// All query words are mandatory | ||||||
|  |     All, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl Default for MatchingStrategy { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         Self::Last | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | impl From<MatchingStrategy> for TermsMatchingStrategy { | ||||||
|  |     fn from(other: MatchingStrategy) -> Self { | ||||||
|  |         match other { | ||||||
|  |             MatchingStrategy::Last => Self::Last, | ||||||
|  |             MatchingStrategy::All => Self::All, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Debug, Clone, Serialize, PartialEq)] | ||||||
|  | pub struct SearchHit { | ||||||
|  |     #[serde(flatten)] | ||||||
|  |     pub document: Document, | ||||||
|  |     #[serde(rename = "_formatted", skip_serializing_if = "Document::is_empty")] | ||||||
|  |     pub formatted: Document, | ||||||
|  |     #[serde(rename = "_matchesPosition", skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub matches_position: Option<MatchesPosition>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Serialize, Debug, Clone, PartialEq)] | ||||||
|  | #[serde(rename_all = "camelCase")] | ||||||
|  | pub struct SearchResult { | ||||||
|  |     pub hits: Vec<SearchHit>, | ||||||
|  |     pub estimated_total_hits: u64, | ||||||
|  |     pub query: String, | ||||||
|  |     pub limit: usize, | ||||||
|  |     pub offset: usize, | ||||||
|  |     pub processing_time_ms: u128, | ||||||
|  |     #[serde(skip_serializing_if = "Option::is_none")] | ||||||
|  |     pub facet_distribution: Option<BTreeMap<String, BTreeMap<String, u64>>>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | pub fn perform_search( | ||||||
|  |     index: &Index, | ||||||
|  |     query: SearchQuery, | ||||||
|  | ) -> Result<SearchResult, MeilisearchHttpError> { | ||||||
|  |     let before_search = Instant::now(); | ||||||
|  |     let rtxn = index.read_txn()?; | ||||||
|  |  | ||||||
|  |     let mut search = index.search(&rtxn); | ||||||
|  |  | ||||||
|  |     if let Some(ref query) = query.q { | ||||||
|  |         search.query(query); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     search.terms_matching_strategy(query.matching_strategy.into()); | ||||||
|  |  | ||||||
|  |     let max_total_hits = index | ||||||
|  |         .pagination_max_total_hits(&rtxn) | ||||||
|  |         .map_err(milli::Error::from)? | ||||||
|  |         .unwrap_or(DEFAULT_PAGINATION_MAX_TOTAL_HITS); | ||||||
|  |  | ||||||
|  |     // Make sure that a user can't get more documents than the hard limit, | ||||||
|  |     // we align that on the offset too. | ||||||
|  |     let offset = min(query.offset.unwrap_or(0), max_total_hits); | ||||||
|  |     let limit = min(query.limit, max_total_hits.saturating_sub(offset)); | ||||||
|  |  | ||||||
|  |     search.offset(offset); | ||||||
|  |     search.limit(limit); | ||||||
|  |  | ||||||
|  |     if let Some(ref filter) = query.filter { | ||||||
|  |         if let Some(facets) = parse_filter(filter)? { | ||||||
|  |             search.filter(facets); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if let Some(ref sort) = query.sort { | ||||||
|  |         let sort = match sort.iter().map(|s| AscDesc::from_str(s)).collect() { | ||||||
|  |             Ok(sorts) => sorts, | ||||||
|  |             Err(asc_desc_error) => { | ||||||
|  |                 return Err(milli::Error::from(SortError::from(asc_desc_error)).into()) | ||||||
|  |             } | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         search.sort_criteria(sort); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let milli::SearchResult { | ||||||
|  |         documents_ids, | ||||||
|  |         matching_words, | ||||||
|  |         candidates, | ||||||
|  |         .. | ||||||
|  |     } = search.execute()?; | ||||||
|  |  | ||||||
|  |     let fields_ids_map = index.fields_ids_map(&rtxn).unwrap(); | ||||||
|  |  | ||||||
|  |     let displayed_ids = index | ||||||
|  |         .displayed_fields_ids(&rtxn)? | ||||||
|  |         .map(|fields| fields.into_iter().collect::<BTreeSet<_>>()) | ||||||
|  |         .unwrap_or_else(|| fields_ids_map.iter().map(|(id, _)| id).collect()); | ||||||
|  |  | ||||||
|  |     let fids = |attrs: &BTreeSet<String>| { | ||||||
|  |         let mut ids = BTreeSet::new(); | ||||||
|  |         for attr in attrs { | ||||||
|  |             if attr == "*" { | ||||||
|  |                 ids = displayed_ids.clone(); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if let Some(id) = fields_ids_map.id(attr) { | ||||||
|  |                 ids.insert(id); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         ids | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // The attributes to retrieve are the ones explicitly marked as to retrieve (all by default), | ||||||
|  |     // but these attributes must be also be present | ||||||
|  |     // - in the fields_ids_map | ||||||
|  |     // - in the the displayed attributes | ||||||
|  |     let to_retrieve_ids: BTreeSet<_> = query | ||||||
|  |         .attributes_to_retrieve | ||||||
|  |         .as_ref() | ||||||
|  |         .map(fids) | ||||||
|  |         .unwrap_or_else(|| displayed_ids.clone()) | ||||||
|  |         .intersection(&displayed_ids) | ||||||
|  |         .cloned() | ||||||
|  |         .collect(); | ||||||
|  |  | ||||||
|  |     let attr_to_highlight = query.attributes_to_highlight.unwrap_or_default(); | ||||||
|  |  | ||||||
|  |     let attr_to_crop = query.attributes_to_crop.unwrap_or_default(); | ||||||
|  |  | ||||||
|  |     // Attributes in `formatted_options` correspond to the attributes that will be in `_formatted` | ||||||
|  |     // These attributes are: | ||||||
|  |     // - the attributes asked to be highlighted or cropped (with `attributesToCrop` or `attributesToHighlight`) | ||||||
|  |     // - the attributes asked to be retrieved: these attributes will not be highlighted/cropped | ||||||
|  |     // But these attributes must be also present in displayed attributes | ||||||
|  |     let formatted_options = compute_formatted_options( | ||||||
|  |         &attr_to_highlight, | ||||||
|  |         &attr_to_crop, | ||||||
|  |         query.crop_length, | ||||||
|  |         &to_retrieve_ids, | ||||||
|  |         &fields_ids_map, | ||||||
|  |         &displayed_ids, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     let tokenizer = TokenizerBuilder::default().build(); | ||||||
|  |  | ||||||
|  |     let mut formatter_builder = MatcherBuilder::new(matching_words, tokenizer); | ||||||
|  |     formatter_builder.crop_marker(query.crop_marker); | ||||||
|  |     formatter_builder.highlight_prefix(query.highlight_pre_tag); | ||||||
|  |     formatter_builder.highlight_suffix(query.highlight_post_tag); | ||||||
|  |  | ||||||
|  |     let mut documents = Vec::new(); | ||||||
|  |  | ||||||
|  |     let documents_iter = index.documents(&rtxn, documents_ids)?; | ||||||
|  |  | ||||||
|  |     for (_id, obkv) in documents_iter { | ||||||
|  |         // First generate a document with all the displayed fields | ||||||
|  |         let displayed_document = make_document(&displayed_ids, &fields_ids_map, obkv)?; | ||||||
|  |  | ||||||
|  |         // select the attributes to retrieve | ||||||
|  |         let attributes_to_retrieve = to_retrieve_ids | ||||||
|  |             .iter() | ||||||
|  |             .map(|&fid| fields_ids_map.name(fid).expect("Missing field name")); | ||||||
|  |         let mut document = | ||||||
|  |             permissive_json_pointer::select_values(&displayed_document, attributes_to_retrieve); | ||||||
|  |  | ||||||
|  |         let (matches_position, formatted) = format_fields( | ||||||
|  |             &displayed_document, | ||||||
|  |             &fields_ids_map, | ||||||
|  |             &formatter_builder, | ||||||
|  |             &formatted_options, | ||||||
|  |             query.show_matches_position, | ||||||
|  |             &displayed_ids, | ||||||
|  |         )?; | ||||||
|  |  | ||||||
|  |         if let Some(sort) = query.sort.as_ref() { | ||||||
|  |             insert_geo_distance(sort, &mut document); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         let hit = SearchHit { | ||||||
|  |             document, | ||||||
|  |             formatted, | ||||||
|  |             matches_position, | ||||||
|  |         }; | ||||||
|  |         documents.push(hit); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let estimated_total_hits = candidates.len(); | ||||||
|  |  | ||||||
|  |     let facet_distribution = match query.facets { | ||||||
|  |         Some(ref fields) => { | ||||||
|  |             let mut facet_distribution = index.facets_distribution(&rtxn); | ||||||
|  |  | ||||||
|  |             let max_values_by_facet = index | ||||||
|  |                 .max_values_per_facet(&rtxn) | ||||||
|  |                 .map_err(milli::Error::from)? | ||||||
|  |                 .unwrap_or(DEFAULT_VALUES_PER_FACET); | ||||||
|  |             facet_distribution.max_values_per_facet(max_values_by_facet); | ||||||
|  |  | ||||||
|  |             if fields.iter().all(|f| f != "*") { | ||||||
|  |                 facet_distribution.facets(fields); | ||||||
|  |             } | ||||||
|  |             let distribution = facet_distribution.candidates(candidates).execute()?; | ||||||
|  |  | ||||||
|  |             Some(distribution) | ||||||
|  |         } | ||||||
|  |         None => None, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let result = SearchResult { | ||||||
|  |         hits: documents, | ||||||
|  |         estimated_total_hits, | ||||||
|  |         query: query.q.clone().unwrap_or_default(), | ||||||
|  |         limit: query.limit, | ||||||
|  |         offset: query.offset.unwrap_or_default(), | ||||||
|  |         processing_time_ms: before_search.elapsed().as_millis(), | ||||||
|  |         facet_distribution, | ||||||
|  |     }; | ||||||
|  |     Ok(result) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn insert_geo_distance(sorts: &[String], document: &mut Document) { | ||||||
|  |     lazy_static::lazy_static! { | ||||||
|  |         static ref GEO_REGEX: Regex = | ||||||
|  |             Regex::new(r"_geoPoint\(\s*([[:digit:].\-]+)\s*,\s*([[:digit:].\-]+)\s*\)").unwrap(); | ||||||
|  |     }; | ||||||
|  |     if let Some(capture_group) = sorts.iter().find_map(|sort| GEO_REGEX.captures(sort)) { | ||||||
|  |         // TODO: TAMO: milli encountered an internal error, what do we want to do? | ||||||
|  |         let base = [ | ||||||
|  |             capture_group[1].parse().unwrap(), | ||||||
|  |             capture_group[2].parse().unwrap(), | ||||||
|  |         ]; | ||||||
|  |         let geo_point = &document.get("_geo").unwrap_or(&json!(null)); | ||||||
|  |         if let Some((lat, lng)) = geo_point["lat"].as_f64().zip(geo_point["lng"].as_f64()) { | ||||||
|  |             let distance = milli::distance_between_two_points(&base, &[lat, lng]); | ||||||
|  |             document.insert("_geoDistance".to_string(), json!(distance.round() as usize)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn compute_formatted_options( | ||||||
|  |     attr_to_highlight: &HashSet<String>, | ||||||
|  |     attr_to_crop: &[String], | ||||||
|  |     query_crop_length: usize, | ||||||
|  |     to_retrieve_ids: &BTreeSet<FieldId>, | ||||||
|  |     fields_ids_map: &FieldsIdsMap, | ||||||
|  |     displayed_ids: &BTreeSet<FieldId>, | ||||||
|  | ) -> BTreeMap<FieldId, FormatOptions> { | ||||||
|  |     let mut formatted_options = BTreeMap::new(); | ||||||
|  |  | ||||||
|  |     add_highlight_to_formatted_options( | ||||||
|  |         &mut formatted_options, | ||||||
|  |         attr_to_highlight, | ||||||
|  |         fields_ids_map, | ||||||
|  |         displayed_ids, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     add_crop_to_formatted_options( | ||||||
|  |         &mut formatted_options, | ||||||
|  |         attr_to_crop, | ||||||
|  |         query_crop_length, | ||||||
|  |         fields_ids_map, | ||||||
|  |         displayed_ids, | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     // Should not return `_formatted` if no valid attributes to highlight/crop | ||||||
|  |     if !formatted_options.is_empty() { | ||||||
|  |         add_non_formatted_ids_to_formatted_options(&mut formatted_options, to_retrieve_ids); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     formatted_options | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn add_highlight_to_formatted_options( | ||||||
|  |     formatted_options: &mut BTreeMap<FieldId, FormatOptions>, | ||||||
|  |     attr_to_highlight: &HashSet<String>, | ||||||
|  |     fields_ids_map: &FieldsIdsMap, | ||||||
|  |     displayed_ids: &BTreeSet<FieldId>, | ||||||
|  | ) { | ||||||
|  |     for attr in attr_to_highlight { | ||||||
|  |         let new_format = FormatOptions { | ||||||
|  |             highlight: true, | ||||||
|  |             crop: None, | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         if attr == "*" { | ||||||
|  |             for id in displayed_ids { | ||||||
|  |                 formatted_options.insert(*id, new_format); | ||||||
|  |             } | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if let Some(id) = fields_ids_map.id(attr) { | ||||||
|  |             if displayed_ids.contains(&id) { | ||||||
|  |                 formatted_options.insert(id, new_format); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn add_crop_to_formatted_options( | ||||||
|  |     formatted_options: &mut BTreeMap<FieldId, FormatOptions>, | ||||||
|  |     attr_to_crop: &[String], | ||||||
|  |     crop_length: usize, | ||||||
|  |     fields_ids_map: &FieldsIdsMap, | ||||||
|  |     displayed_ids: &BTreeSet<FieldId>, | ||||||
|  | ) { | ||||||
|  |     for attr in attr_to_crop { | ||||||
|  |         let mut split = attr.rsplitn(2, ':'); | ||||||
|  |         let (attr_name, attr_len) = match split.next().zip(split.next()) { | ||||||
|  |             Some((len, name)) => { | ||||||
|  |                 let crop_len = len.parse::<usize>().unwrap_or(crop_length); | ||||||
|  |                 (name, crop_len) | ||||||
|  |             } | ||||||
|  |             None => (attr.as_str(), crop_length), | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         if attr_name == "*" { | ||||||
|  |             for id in displayed_ids { | ||||||
|  |                 formatted_options | ||||||
|  |                     .entry(*id) | ||||||
|  |                     .and_modify(|f| f.crop = Some(attr_len)) | ||||||
|  |                     .or_insert(FormatOptions { | ||||||
|  |                         highlight: false, | ||||||
|  |                         crop: Some(attr_len), | ||||||
|  |                     }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if let Some(id) = fields_ids_map.id(attr_name) { | ||||||
|  |             if displayed_ids.contains(&id) { | ||||||
|  |                 formatted_options | ||||||
|  |                     .entry(id) | ||||||
|  |                     .and_modify(|f| f.crop = Some(attr_len)) | ||||||
|  |                     .or_insert(FormatOptions { | ||||||
|  |                         highlight: false, | ||||||
|  |                         crop: Some(attr_len), | ||||||
|  |                     }); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn add_non_formatted_ids_to_formatted_options( | ||||||
|  |     formatted_options: &mut BTreeMap<FieldId, FormatOptions>, | ||||||
|  |     to_retrieve_ids: &BTreeSet<FieldId>, | ||||||
|  | ) { | ||||||
|  |     for id in to_retrieve_ids { | ||||||
|  |         formatted_options.entry(*id).or_insert(FormatOptions { | ||||||
|  |             highlight: false, | ||||||
|  |             crop: None, | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn make_document( | ||||||
|  |     displayed_attributes: &BTreeSet<FieldId>, | ||||||
|  |     field_ids_map: &FieldsIdsMap, | ||||||
|  |     obkv: obkv::KvReaderU16, | ||||||
|  | ) -> Result<Document, MeilisearchHttpError> { | ||||||
|  |     let mut document = serde_json::Map::new(); | ||||||
|  |  | ||||||
|  |     // recreate the original json | ||||||
|  |     for (key, value) in obkv.iter() { | ||||||
|  |         let value = serde_json::from_slice(value)?; | ||||||
|  |         let key = field_ids_map | ||||||
|  |             .name(key) | ||||||
|  |             .expect("Missing field name") | ||||||
|  |             .to_string(); | ||||||
|  |  | ||||||
|  |         document.insert(key, value); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // select the attributes to retrieve | ||||||
|  |     let displayed_attributes = displayed_attributes | ||||||
|  |         .iter() | ||||||
|  |         .map(|&fid| field_ids_map.name(fid).expect("Missing field name")); | ||||||
|  |  | ||||||
|  |     let document = permissive_json_pointer::select_values(&document, displayed_attributes); | ||||||
|  |     Ok(document) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn format_fields<'a, A: AsRef<[u8]>>( | ||||||
|  |     document: &Document, | ||||||
|  |     field_ids_map: &FieldsIdsMap, | ||||||
|  |     builder: &MatcherBuilder<'a, A>, | ||||||
|  |     formatted_options: &BTreeMap<FieldId, FormatOptions>, | ||||||
|  |     compute_matches: bool, | ||||||
|  |     displayable_ids: &BTreeSet<FieldId>, | ||||||
|  | ) -> Result<(Option<MatchesPosition>, Document), MeilisearchHttpError> { | ||||||
|  |     let mut matches_position = compute_matches.then(BTreeMap::new); | ||||||
|  |     let mut document = document.clone(); | ||||||
|  |  | ||||||
|  |     // select the attributes to retrieve | ||||||
|  |     let displayable_names = displayable_ids | ||||||
|  |         .iter() | ||||||
|  |         .map(|&fid| field_ids_map.name(fid).expect("Missing field name")); | ||||||
|  |     permissive_json_pointer::map_leaf_values(&mut document, displayable_names, |key, value| { | ||||||
|  |         // To get the formatting option of each key we need to see all the rules that applies | ||||||
|  |         // to the value and merge them together. eg. If a user said he wanted to highlight `doggo` | ||||||
|  |         // and crop `doggo.name`. `doggo.name` needs to be highlighted + cropped while `doggo.age` is only | ||||||
|  |         // highlighted. | ||||||
|  |         let format = formatted_options | ||||||
|  |             .iter() | ||||||
|  |             .filter(|(field, _option)| { | ||||||
|  |                 let name = field_ids_map.name(**field).unwrap(); | ||||||
|  |                 milli::is_faceted_by(name, key) || milli::is_faceted_by(key, name) | ||||||
|  |             }) | ||||||
|  |             .map(|(_, option)| *option) | ||||||
|  |             .reduce(|acc, option| acc.merge(option)); | ||||||
|  |         let mut infos = Vec::new(); | ||||||
|  |  | ||||||
|  |         *value = format_value( | ||||||
|  |             std::mem::take(value), | ||||||
|  |             builder, | ||||||
|  |             format, | ||||||
|  |             &mut infos, | ||||||
|  |             compute_matches, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         if let Some(matches) = matches_position.as_mut() { | ||||||
|  |             if !infos.is_empty() { | ||||||
|  |                 matches.insert(key.to_owned(), infos); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     let selectors = formatted_options | ||||||
|  |         .keys() | ||||||
|  |         // This unwrap must be safe since we got the ids from the fields_ids_map just | ||||||
|  |         // before. | ||||||
|  |         .map(|&fid| field_ids_map.name(fid).unwrap()); | ||||||
|  |     let document = permissive_json_pointer::select_values(&document, selectors); | ||||||
|  |  | ||||||
|  |     Ok((matches_position, document)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn format_value<'a, A: AsRef<[u8]>>( | ||||||
|  |     value: Value, | ||||||
|  |     builder: &MatcherBuilder<'a, A>, | ||||||
|  |     format_options: Option<FormatOptions>, | ||||||
|  |     infos: &mut Vec<MatchBounds>, | ||||||
|  |     compute_matches: bool, | ||||||
|  | ) -> Value { | ||||||
|  |     match value { | ||||||
|  |         Value::String(old_string) => { | ||||||
|  |             let mut matcher = builder.build(&old_string); | ||||||
|  |             if compute_matches { | ||||||
|  |                 let matches = matcher.matches(); | ||||||
|  |                 infos.extend_from_slice(&matches[..]); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             match format_options { | ||||||
|  |                 Some(format_options) => { | ||||||
|  |                     let value = matcher.format(format_options); | ||||||
|  |                     Value::String(value.into_owned()) | ||||||
|  |                 } | ||||||
|  |                 None => Value::String(old_string), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         Value::Array(values) => Value::Array( | ||||||
|  |             values | ||||||
|  |                 .into_iter() | ||||||
|  |                 .map(|v| { | ||||||
|  |                     format_value( | ||||||
|  |                         v, | ||||||
|  |                         builder, | ||||||
|  |                         format_options.map(|format_options| FormatOptions { | ||||||
|  |                             highlight: format_options.highlight, | ||||||
|  |                             crop: None, | ||||||
|  |                         }), | ||||||
|  |                         infos, | ||||||
|  |                         compute_matches, | ||||||
|  |                     ) | ||||||
|  |                 }) | ||||||
|  |                 .collect(), | ||||||
|  |         ), | ||||||
|  |         Value::Object(object) => Value::Object( | ||||||
|  |             object | ||||||
|  |                 .into_iter() | ||||||
|  |                 .map(|(k, v)| { | ||||||
|  |                     ( | ||||||
|  |                         k, | ||||||
|  |                         format_value( | ||||||
|  |                             v, | ||||||
|  |                             builder, | ||||||
|  |                             format_options.map(|format_options| FormatOptions { | ||||||
|  |                                 highlight: format_options.highlight, | ||||||
|  |                                 crop: None, | ||||||
|  |                             }), | ||||||
|  |                             infos, | ||||||
|  |                             compute_matches, | ||||||
|  |                         ), | ||||||
|  |                     ) | ||||||
|  |                 }) | ||||||
|  |                 .collect(), | ||||||
|  |         ), | ||||||
|  |         Value::Number(number) => { | ||||||
|  |             let s = number.to_string(); | ||||||
|  |  | ||||||
|  |             let mut matcher = builder.build(&s); | ||||||
|  |             if compute_matches { | ||||||
|  |                 let matches = matcher.matches(); | ||||||
|  |                 infos.extend_from_slice(&matches[..]); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             match format_options { | ||||||
|  |                 Some(format_options) => { | ||||||
|  |                     let value = matcher.format(format_options); | ||||||
|  |                     Value::String(value.into_owned()) | ||||||
|  |                 } | ||||||
|  |                 None => Value::Number(number), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |         value => value, | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn parse_filter(facets: &Value) -> Result<Option<Filter>, MeilisearchHttpError> { | ||||||
|  |     match facets { | ||||||
|  |         Value::String(expr) => { | ||||||
|  |             let condition = Filter::from_str(expr)?; | ||||||
|  |             Ok(condition) | ||||||
|  |         } | ||||||
|  |         Value::Array(arr) => parse_filter_array(arr), | ||||||
|  |         v => Err(MeilisearchHttpError::InvalidExpression(&["Array"], v.clone()).into()), | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn parse_filter_array(arr: &[Value]) -> Result<Option<Filter>, MeilisearchHttpError> { | ||||||
|  |     let mut ands = Vec::new(); | ||||||
|  |     for value in arr { | ||||||
|  |         match value { | ||||||
|  |             Value::String(s) => ands.push(Either::Right(s.as_str())), | ||||||
|  |             Value::Array(arr) => { | ||||||
|  |                 let mut ors = Vec::new(); | ||||||
|  |                 for value in arr { | ||||||
|  |                     match value { | ||||||
|  |                         Value::String(s) => ors.push(s.as_str()), | ||||||
|  |                         v => { | ||||||
|  |                             return Err(MeilisearchHttpError::InvalidExpression( | ||||||
|  |                                 &["String"], | ||||||
|  |                                 v.clone(), | ||||||
|  |                             ) | ||||||
|  |                             .into()) | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 ands.push(Either::Left(ors)); | ||||||
|  |             } | ||||||
|  |             v => { | ||||||
|  |                 return Err(MeilisearchHttpError::InvalidExpression( | ||||||
|  |                     &["String", "[String]"], | ||||||
|  |                     v.clone(), | ||||||
|  |                 ) | ||||||
|  |                 .into()) | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     Ok(Filter::from_array(ands)?) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[cfg(test)] | ||||||
|  | mod test { | ||||||
|  |     use super::*; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn test_insert_geo_distance() { | ||||||
|  |         let value: Document = serde_json::from_str( | ||||||
|  |             r#"{ | ||||||
|  |               "_geo": { | ||||||
|  |                 "lat": 50.629973371633746, | ||||||
|  |                 "lng": 3.0569447399419567 | ||||||
|  |               }, | ||||||
|  |               "city": "Lille", | ||||||
|  |               "id": "1" | ||||||
|  |             }"#, | ||||||
|  |         ) | ||||||
|  |         .unwrap(); | ||||||
|  |  | ||||||
|  |         let sorters = &["_geoPoint(50.629973371633746,3.0569447399419567):desc".to_string()]; | ||||||
|  |         let mut document = value.clone(); | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), Some(&json!(0))); | ||||||
|  |  | ||||||
|  |         let sorters = &["_geoPoint(50.629973371633746, 3.0569447399419567):asc".to_string()]; | ||||||
|  |         let mut document = value.clone(); | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), Some(&json!(0))); | ||||||
|  |  | ||||||
|  |         let sorters = | ||||||
|  |             &["_geoPoint(   50.629973371633746   ,  3.0569447399419567   ):desc".to_string()]; | ||||||
|  |         let mut document = value.clone(); | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), Some(&json!(0))); | ||||||
|  |  | ||||||
|  |         let sorters = &[ | ||||||
|  |             "prix:asc", | ||||||
|  |             "villeneuve:desc", | ||||||
|  |             "_geoPoint(50.629973371633746, 3.0569447399419567):asc", | ||||||
|  |             "ubu:asc", | ||||||
|  |         ] | ||||||
|  |         .map(|s| s.to_string()); | ||||||
|  |         let mut document = value.clone(); | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), Some(&json!(0))); | ||||||
|  |  | ||||||
|  |         // only the first geoPoint is used to compute the distance | ||||||
|  |         let sorters = &[ | ||||||
|  |             "chien:desc", | ||||||
|  |             "_geoPoint(50.629973371633746, 3.0569447399419567):asc", | ||||||
|  |             "pangolin:desc", | ||||||
|  |             "_geoPoint(100.0, -80.0):asc", | ||||||
|  |             "chat:asc", | ||||||
|  |         ] | ||||||
|  |         .map(|s| s.to_string()); | ||||||
|  |         let mut document = value.clone(); | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), Some(&json!(0))); | ||||||
|  |  | ||||||
|  |         // there was no _geoPoint so nothing is inserted in the document | ||||||
|  |         let sorters = &["chien:asc".to_string()]; | ||||||
|  |         let mut document = value; | ||||||
|  |         insert_geo_distance(sorters, &mut document); | ||||||
|  |         assert_eq!(document.get("_geoDistance"), None); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -15,6 +15,7 @@ serde = { version = "1.0.145", features = ["derive"] } | |||||||
| serde_json = "1.0.85" | serde_json = "1.0.85" | ||||||
| time = { version = "0.3.7", features = ["serde-well-known", "formatting", "parsing", "macros"] } | time = { version = "0.3.7", features = ["serde-well-known", "formatting", "parsing", "macros"] } | ||||||
| tokio = "1.0" | tokio = "1.0" | ||||||
|  | uuid = { version = "1.1.2", features = ["serde", "v4"] } | ||||||
|  |  | ||||||
| [dev-dependencies] | [dev-dependencies] | ||||||
| proptest = "1.0.0" | proptest = "1.0.0" | ||||||
|   | |||||||
| @@ -2,9 +2,11 @@ use milli::update::IndexDocumentsMethod; | |||||||
| use serde::{Deserialize, Serialize, Serializer}; | use serde::{Deserialize, Serialize, Serializer}; | ||||||
| use std::{ | use std::{ | ||||||
|     fmt::{Display, Write}, |     fmt::{Display, Write}, | ||||||
|  |     path::PathBuf, | ||||||
|     str::FromStr, |     str::FromStr, | ||||||
| }; | }; | ||||||
| use time::{Duration, OffsetDateTime}; | use time::{Duration, OffsetDateTime}; | ||||||
|  | use uuid::Uuid; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|     error::{Code, ResponseError}, |     error::{Code, ResponseError}, | ||||||
| @@ -13,74 +15,192 @@ use crate::{ | |||||||
|  |  | ||||||
| pub type TaskId = u32; | pub type TaskId = u32; | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, PartialEq, Serialize)] | #[derive(Debug, PartialEq, Serialize, Deserialize)] | ||||||
| #[serde(rename_all = "camelCase")] | #[serde(rename_all = "camelCase")] | ||||||
| pub struct TaskView { | pub struct Task { | ||||||
|     pub uid: TaskId, |     pub uid: TaskId, | ||||||
|     #[serde(default)] |  | ||||||
|     pub index_uid: Option<String>, |  | ||||||
|     pub status: Status, |  | ||||||
|     // TODO use our own Kind for the user |  | ||||||
|     #[serde(rename = "type")] |  | ||||||
|     pub kind: Kind, |  | ||||||
|  |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub details: Option<DetailsView>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub error: Option<ResponseError>, |  | ||||||
|  |  | ||||||
|     #[serde( |  | ||||||
|         serialize_with = "serialize_duration", |  | ||||||
|         skip_serializing_if = "Option::is_none", |  | ||||||
|         default |  | ||||||
|     )] |  | ||||||
|     pub duration: Option<Duration>, |  | ||||||
|     #[serde(with = "time::serde::rfc3339")] |     #[serde(with = "time::serde::rfc3339")] | ||||||
|     pub enqueued_at: OffsetDateTime, |     pub enqueued_at: OffsetDateTime, | ||||||
|     #[serde( |     #[serde(with = "time::serde::rfc3339::option")] | ||||||
|         with = "time::serde::rfc3339::option", |  | ||||||
|         skip_serializing_if = "Option::is_none", |  | ||||||
|         default |  | ||||||
|     )] |  | ||||||
|     pub started_at: Option<OffsetDateTime>, |     pub started_at: Option<OffsetDateTime>, | ||||||
|     #[serde( |     #[serde(with = "time::serde::rfc3339::option")] | ||||||
|         with = "time::serde::rfc3339::option", |  | ||||||
|         skip_serializing_if = "Option::is_none", |  | ||||||
|         default |  | ||||||
|     )] |  | ||||||
|     pub finished_at: Option<OffsetDateTime>, |     pub finished_at: Option<OffsetDateTime>, | ||||||
|  |  | ||||||
|  |     pub error: Option<ResponseError>, | ||||||
|  |     pub details: Option<Details>, | ||||||
|  |  | ||||||
|  |     pub status: Status, | ||||||
|  |     pub kind: KindWithContent, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] | impl Task { | ||||||
|  |     pub fn index_uid(&self) -> Option<&str> { | ||||||
|  |         use KindWithContent::*; | ||||||
|  |  | ||||||
|  |         match &self.kind { | ||||||
|  |             DumpExport { .. } | ||||||
|  |             | Snapshot | ||||||
|  |             | CancelTask { .. } | ||||||
|  |             | DeleteTasks { .. } | ||||||
|  |             | IndexSwap { .. } => None, | ||||||
|  |             DocumentImport { index_uid, .. } | ||||||
|  |             | DocumentDeletion { index_uid, .. } | ||||||
|  |             | DocumentClear { index_uid } | ||||||
|  |             | Settings { index_uid, .. } | ||||||
|  |             | IndexCreation { index_uid, .. } | ||||||
|  |             | IndexUpdate { index_uid, .. } | ||||||
|  |             | IndexDeletion { index_uid } => Some(index_uid), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Return the list of indexes updated by this tasks. | ||||||
|  |     pub fn indexes(&self) -> Option<Vec<&str>> { | ||||||
|  |         use KindWithContent::*; | ||||||
|  |  | ||||||
|  |         match &self.kind { | ||||||
|  |             DumpExport { .. } | Snapshot | CancelTask { .. } | DeleteTasks { .. } => None, | ||||||
|  |             DocumentImport { index_uid, .. } | ||||||
|  |             | DocumentDeletion { index_uid, .. } | ||||||
|  |             | DocumentClear { index_uid } | ||||||
|  |             | Settings { index_uid, .. } | ||||||
|  |             | IndexCreation { index_uid, .. } | ||||||
|  |             | IndexUpdate { index_uid, .. } | ||||||
|  |             | IndexDeletion { index_uid } => Some(vec![index_uid]), | ||||||
|  |             IndexSwap { lhs, rhs } => Some(vec![lhs, rhs]), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #[derive(Debug, PartialEq, Serialize, Deserialize)] | ||||||
| #[serde(rename_all = "camelCase")] | #[serde(rename_all = "camelCase")] | ||||||
| pub struct TaskDump { | pub enum KindWithContent { | ||||||
|     pub uid: TaskId, |     DocumentImport { | ||||||
|     #[serde(default)] |         index_uid: String, | ||||||
|     pub index_uid: Option<String>, |         primary_key: Option<String>, | ||||||
|     pub status: Status, |         method: IndexDocumentsMethod, | ||||||
|     // TODO use our own Kind for the user |         content_file: Uuid, | ||||||
|     #[serde(rename = "type")] |         documents_count: u64, | ||||||
|     pub kind: Kind, |         allow_index_creation: bool, | ||||||
|  |     }, | ||||||
|  |     DocumentDeletion { | ||||||
|  |         index_uid: String, | ||||||
|  |         documents_ids: Vec<String>, | ||||||
|  |     }, | ||||||
|  |     DocumentClear { | ||||||
|  |         index_uid: String, | ||||||
|  |     }, | ||||||
|  |     Settings { | ||||||
|  |         index_uid: String, | ||||||
|  |         new_settings: Settings<Unchecked>, | ||||||
|  |         is_deletion: bool, | ||||||
|  |         allow_index_creation: bool, | ||||||
|  |     }, | ||||||
|  |     IndexDeletion { | ||||||
|  |         index_uid: String, | ||||||
|  |     }, | ||||||
|  |     IndexCreation { | ||||||
|  |         index_uid: String, | ||||||
|  |         primary_key: Option<String>, | ||||||
|  |     }, | ||||||
|  |     IndexUpdate { | ||||||
|  |         index_uid: String, | ||||||
|  |         primary_key: Option<String>, | ||||||
|  |     }, | ||||||
|  |     IndexSwap { | ||||||
|  |         lhs: String, | ||||||
|  |         rhs: String, | ||||||
|  |     }, | ||||||
|  |     CancelTask { | ||||||
|  |         tasks: Vec<TaskId>, | ||||||
|  |     }, | ||||||
|  |     DeleteTasks { | ||||||
|  |         query: String, | ||||||
|  |         tasks: Vec<TaskId>, | ||||||
|  |     }, | ||||||
|  |     DumpExport { | ||||||
|  |         output: PathBuf, | ||||||
|  |     }, | ||||||
|  |     Snapshot, | ||||||
|  | } | ||||||
|  |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] | impl KindWithContent { | ||||||
|     pub details: Option<Details>, |     pub fn as_kind(&self) -> Kind { | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |         match self { | ||||||
|     pub error: Option<ResponseError>, |             KindWithContent::DocumentImport { .. } => Kind::DocumentImport, | ||||||
|  |             KindWithContent::DocumentDeletion { .. } => Kind::DocumentDeletion, | ||||||
|  |             KindWithContent::DocumentClear { .. } => Kind::DocumentClear, | ||||||
|  |             KindWithContent::Settings { .. } => Kind::Settings, | ||||||
|  |             KindWithContent::IndexCreation { .. } => Kind::IndexCreation, | ||||||
|  |             KindWithContent::IndexDeletion { .. } => Kind::IndexDeletion, | ||||||
|  |             KindWithContent::IndexUpdate { .. } => Kind::IndexUpdate, | ||||||
|  |             KindWithContent::IndexSwap { .. } => Kind::IndexSwap, | ||||||
|  |             KindWithContent::CancelTask { .. } => Kind::CancelTask, | ||||||
|  |             KindWithContent::DeleteTasks { .. } => Kind::DeleteTasks, | ||||||
|  |             KindWithContent::DumpExport { .. } => Kind::DumpExport, | ||||||
|  |             KindWithContent::Snapshot => Kind::Snapshot, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     #[serde(with = "time::serde::rfc3339")] |     pub fn indexes(&self) -> Option<Vec<&str>> { | ||||||
|     pub enqueued_at: OffsetDateTime, |         use KindWithContent::*; | ||||||
|     #[serde( |  | ||||||
|         with = "time::serde::rfc3339::option", |         match self { | ||||||
|         skip_serializing_if = "Option::is_none", |             DumpExport { .. } | Snapshot | CancelTask { .. } | DeleteTasks { .. } => None, | ||||||
|         default |             DocumentImport { index_uid, .. } | ||||||
|     )] |             | DocumentDeletion { index_uid, .. } | ||||||
|     pub started_at: Option<OffsetDateTime>, |             | DocumentClear { index_uid } | ||||||
|     #[serde( |             | Settings { index_uid, .. } | ||||||
|         with = "time::serde::rfc3339::option", |             | IndexCreation { index_uid, .. } | ||||||
|         skip_serializing_if = "Option::is_none", |             | IndexUpdate { index_uid, .. } | ||||||
|         default |             | IndexDeletion { index_uid } => Some(vec![index_uid]), | ||||||
|     )] |             IndexSwap { lhs, rhs } => Some(vec![lhs, rhs]), | ||||||
|     pub finished_at: Option<OffsetDateTime>, |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Returns the default `Details` that correspond to this `KindWithContent`, | ||||||
|  |     /// `None` if it cannot be generated. | ||||||
|  |     pub fn default_details(&self) -> Option<Details> { | ||||||
|  |         match self { | ||||||
|  |             KindWithContent::DocumentImport { | ||||||
|  |                 documents_count, .. | ||||||
|  |             } => Some(Details::DocumentAddition { | ||||||
|  |                 received_documents: *documents_count, | ||||||
|  |                 indexed_documents: 0, | ||||||
|  |             }), | ||||||
|  |             KindWithContent::DocumentDeletion { | ||||||
|  |                 index_uid: _, | ||||||
|  |                 documents_ids, | ||||||
|  |             } => Some(Details::DocumentDeletion { | ||||||
|  |                 received_document_ids: documents_ids.len(), | ||||||
|  |                 deleted_documents: None, | ||||||
|  |             }), | ||||||
|  |             KindWithContent::DocumentClear { .. } => Some(Details::ClearAll { | ||||||
|  |                 deleted_documents: None, | ||||||
|  |             }), | ||||||
|  |             KindWithContent::Settings { new_settings, .. } => Some(Details::Settings { | ||||||
|  |                 settings: new_settings.clone(), | ||||||
|  |             }), | ||||||
|  |             KindWithContent::IndexDeletion { .. } => None, | ||||||
|  |             KindWithContent::IndexCreation { primary_key, .. } | ||||||
|  |             | KindWithContent::IndexUpdate { primary_key, .. } => Some(Details::IndexInfo { | ||||||
|  |                 primary_key: primary_key.clone(), | ||||||
|  |             }), | ||||||
|  |             KindWithContent::IndexSwap { .. } => { | ||||||
|  |                 todo!() | ||||||
|  |             } | ||||||
|  |             KindWithContent::CancelTask { .. } => { | ||||||
|  |                 None // TODO: check correctness of this return value | ||||||
|  |             } | ||||||
|  |             KindWithContent::DeleteTasks { query, tasks } => Some(Details::DeleteTasks { | ||||||
|  |                 matched_tasks: tasks.len(), | ||||||
|  |                 deleted_tasks: None, | ||||||
|  |                 original_query: query.clone(), | ||||||
|  |             }), | ||||||
|  |             KindWithContent::DumpExport { .. } => None, | ||||||
|  |             KindWithContent::Snapshot => None, | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)] | #[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)] | ||||||
| @@ -123,15 +243,10 @@ impl FromStr for Status { | |||||||
| #[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)] | #[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)] | ||||||
| #[serde(rename_all = "camelCase")] | #[serde(rename_all = "camelCase")] | ||||||
| pub enum Kind { | pub enum Kind { | ||||||
|     DocumentImport { |     DocumentImport, | ||||||
|         method: IndexDocumentsMethod, |  | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     DocumentDeletion, |     DocumentDeletion, | ||||||
|     DocumentClear, |     DocumentClear, | ||||||
|     Settings { |     Settings, | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     IndexCreation, |     IndexCreation, | ||||||
|     IndexDeletion, |     IndexDeletion, | ||||||
|     IndexUpdate, |     IndexUpdate, | ||||||
| @@ -147,22 +262,11 @@ impl FromStr for Kind { | |||||||
|  |  | ||||||
|     fn from_str(s: &str) -> Result<Self, Self::Err> { |     fn from_str(s: &str) -> Result<Self, Self::Err> { | ||||||
|         match s { |         match s { | ||||||
|             "document_addition" => Ok(Kind::DocumentImport { |             "document_addition" => Ok(Kind::DocumentImport), | ||||||
|                 method: IndexDocumentsMethod::ReplaceDocuments, |             "document_update" => Ok(Kind::DocumentImport), | ||||||
|                 // TODO this doesn't make sense |  | ||||||
|                 allow_index_creation: false, |  | ||||||
|             }), |  | ||||||
|             "document_update" => Ok(Kind::DocumentImport { |  | ||||||
|                 method: IndexDocumentsMethod::UpdateDocuments, |  | ||||||
|                 // TODO this doesn't make sense |  | ||||||
|                 allow_index_creation: false, |  | ||||||
|             }), |  | ||||||
|             "document_deletion" => Ok(Kind::DocumentDeletion), |             "document_deletion" => Ok(Kind::DocumentDeletion), | ||||||
|             "document_clear" => Ok(Kind::DocumentClear), |             "document_clear" => Ok(Kind::DocumentClear), | ||||||
|             "settings" => Ok(Kind::Settings { |             "settings" => Ok(Kind::Settings), | ||||||
|                 // TODO this doesn't make sense |  | ||||||
|                 allow_index_creation: false, |  | ||||||
|             }), |  | ||||||
|             "index_creation" => Ok(Kind::IndexCreation), |             "index_creation" => Ok(Kind::IndexCreation), | ||||||
|             "index_deletion" => Ok(Kind::IndexDeletion), |             "index_deletion" => Ok(Kind::IndexDeletion), | ||||||
|             "index_update" => Ok(Kind::IndexUpdate), |             "index_update" => Ok(Kind::IndexUpdate), | ||||||
| @@ -179,73 +283,6 @@ impl FromStr for Kind { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Default, Debug, PartialEq, Clone, Serialize)] |  | ||||||
| #[serde(rename_all = "camelCase")] |  | ||||||
| pub struct DetailsView { |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub received_documents: Option<u64>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub indexed_documents: Option<u64>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub primary_key: Option<Option<String>>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub received_document_ids: Option<usize>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub deleted_documents: Option<Option<u64>>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub matched_tasks: Option<usize>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub deleted_tasks: Option<Option<usize>>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub original_query: Option<String>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     pub dump_uid: Option<String>, |  | ||||||
|     #[serde(skip_serializing_if = "Option::is_none")] |  | ||||||
|     #[serde(flatten)] |  | ||||||
|     pub settings: Option<Settings<Unchecked>>, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // A `Kind` specific version made for the dump. If modified you may break the dump. |  | ||||||
| #[derive(Debug, PartialEq, Serialize, Deserialize)] |  | ||||||
| #[serde(rename_all = "camelCase")] |  | ||||||
| pub enum KindDump { |  | ||||||
|     DocumentImport { |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|         method: IndexDocumentsMethod, |  | ||||||
|         documents_count: u64, |  | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     DocumentDeletion { |  | ||||||
|         documents_ids: Vec<String>, |  | ||||||
|     }, |  | ||||||
|     DocumentClear, |  | ||||||
|     Settings { |  | ||||||
|         new_settings: Settings<Unchecked>, |  | ||||||
|         is_deletion: bool, |  | ||||||
|         allow_index_creation: bool, |  | ||||||
|     }, |  | ||||||
|     IndexDeletion, |  | ||||||
|     IndexCreation { |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|     }, |  | ||||||
|     IndexUpdate { |  | ||||||
|         primary_key: Option<String>, |  | ||||||
|     }, |  | ||||||
|     IndexSwap { |  | ||||||
|         lhs: String, |  | ||||||
|         rhs: String, |  | ||||||
|     }, |  | ||||||
|     CancelTask { |  | ||||||
|         tasks: Vec<TaskId>, |  | ||||||
|     }, |  | ||||||
|     DeleteTasks { |  | ||||||
|         query: String, |  | ||||||
|         tasks: Vec<TaskId>, |  | ||||||
|     }, |  | ||||||
|     DumpExport, |  | ||||||
|     Snapshot, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] | #[derive(Debug, PartialEq, Clone, Serialize, Deserialize)] | ||||||
| #[allow(clippy::large_enum_variant)] | #[allow(clippy::large_enum_variant)] | ||||||
| pub enum Details { | pub enum Details { | ||||||
| @@ -277,59 +314,10 @@ pub enum Details { | |||||||
|     }, |     }, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Details { |  | ||||||
|     pub fn as_details_view(&self) -> DetailsView { |  | ||||||
|         match self.clone() { |  | ||||||
|             Details::DocumentAddition { |  | ||||||
|                 received_documents, |  | ||||||
|                 indexed_documents, |  | ||||||
|             } => DetailsView { |  | ||||||
|                 received_documents: Some(received_documents), |  | ||||||
|                 indexed_documents: Some(indexed_documents), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::Settings { settings } => DetailsView { |  | ||||||
|                 settings: Some(settings), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::IndexInfo { primary_key } => DetailsView { |  | ||||||
|                 primary_key: Some(primary_key), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::DocumentDeletion { |  | ||||||
|                 received_document_ids, |  | ||||||
|                 deleted_documents, |  | ||||||
|             } => DetailsView { |  | ||||||
|                 received_document_ids: Some(received_document_ids), |  | ||||||
|                 deleted_documents: Some(deleted_documents), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::ClearAll { deleted_documents } => DetailsView { |  | ||||||
|                 deleted_documents: Some(deleted_documents), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::DeleteTasks { |  | ||||||
|                 matched_tasks, |  | ||||||
|                 deleted_tasks, |  | ||||||
|                 original_query, |  | ||||||
|             } => DetailsView { |  | ||||||
|                 matched_tasks: Some(matched_tasks), |  | ||||||
|                 deleted_tasks: Some(deleted_tasks), |  | ||||||
|                 original_query: Some(original_query), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|             Details::Dump { dump_uid } => DetailsView { |  | ||||||
|                 dump_uid: Some(dump_uid), |  | ||||||
|                 ..DetailsView::default() |  | ||||||
|             }, |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /// Serialize a `time::Duration` as a best effort ISO 8601 while waiting for | /// Serialize a `time::Duration` as a best effort ISO 8601 while waiting for | ||||||
| /// https://github.com/time-rs/time/issues/378. | /// https://github.com/time-rs/time/issues/378. | ||||||
| /// This code is a port of the old code of time that was removed in 0.2. | /// This code is a port of the old code of time that was removed in 0.2. | ||||||
| fn serialize_duration<S: Serializer>( | pub fn serialize_duration<S: Serializer>( | ||||||
|     duration: &Option<Duration>, |     duration: &Option<Duration>, | ||||||
|     serializer: S, |     serializer: S, | ||||||
| ) -> Result<S::Ok, S::Error> { | ) -> Result<S::Ok, S::Error> { | ||||||
| @@ -376,3 +364,26 @@ fn serialize_duration<S: Serializer>( | |||||||
|         None => serializer.serialize_none(), |         None => serializer.serialize_none(), | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | #[cfg(test)] | ||||||
|  | mod tests { | ||||||
|  |     use crate::assert_smol_debug_snapshot; | ||||||
|  |     use crate::heed::{types::SerdeJson, BytesDecode, BytesEncode}; | ||||||
|  |  | ||||||
|  |     use super::Details; | ||||||
|  |  | ||||||
|  |     #[test] | ||||||
|  |     fn bad_deser() { | ||||||
|  |         let details = Details::DeleteTasks { | ||||||
|  |             matched_tasks: 1, | ||||||
|  |             deleted_tasks: None, | ||||||
|  |             original_query: "hello".to_owned(), | ||||||
|  |         }; | ||||||
|  |         let serialised = SerdeJson::<Details>::bytes_encode(&details).unwrap(); | ||||||
|  |         let deserialised = SerdeJson::<Details>::bytes_decode(&serialised).unwrap(); | ||||||
|  |         assert_smol_debug_snapshot!(details, @r###"DeleteTasks { matched_tasks: 1, deleted_tasks: None, original_query: "hello" }"###); | ||||||
|  |         assert_smol_debug_snapshot!(deserialised, @r###"DeleteTasks { matched_tasks: 1, deleted_tasks: None, original_query: "hello" }"###); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user