mirror of
https://github.com/meilisearch/MeiliSearch
synced 2025-01-10 21:44:34 +01:00
Processing time without autobatching implementation
This commit is contained in:
parent
05cc463fbc
commit
732c52093d
@ -101,6 +101,9 @@ pub enum KindDump {
|
|||||||
documents_ids: Vec<String>,
|
documents_ids: Vec<String>,
|
||||||
},
|
},
|
||||||
DocumentClear,
|
DocumentClear,
|
||||||
|
DocumentDeletionByFilter {
|
||||||
|
filter: String,
|
||||||
|
},
|
||||||
Settings {
|
Settings {
|
||||||
settings: Box<meilisearch_types::settings::Settings<Unchecked>>,
|
settings: Box<meilisearch_types::settings::Settings<Unchecked>>,
|
||||||
is_deletion: bool,
|
is_deletion: bool,
|
||||||
@ -166,6 +169,9 @@ impl From<KindWithContent> for KindDump {
|
|||||||
KindWithContent::DocumentDeletion { documents_ids, .. } => {
|
KindWithContent::DocumentDeletion { documents_ids, .. } => {
|
||||||
KindDump::DocumentDeletion { documents_ids }
|
KindDump::DocumentDeletion { documents_ids }
|
||||||
}
|
}
|
||||||
|
KindWithContent::DocumentDeletionByFilter { filter_expr, .. } => {
|
||||||
|
KindDump::DocumentDeletionByFilter { filter: filter_expr.to_string() }
|
||||||
|
}
|
||||||
KindWithContent::DocumentClear { .. } => KindDump::DocumentClear,
|
KindWithContent::DocumentClear { .. } => KindDump::DocumentClear,
|
||||||
KindWithContent::SettingsUpdate {
|
KindWithContent::SettingsUpdate {
|
||||||
new_settings,
|
new_settings,
|
||||||
|
@ -25,6 +25,7 @@ enum AutobatchKind {
|
|||||||
primary_key: Option<String>,
|
primary_key: Option<String>,
|
||||||
},
|
},
|
||||||
DocumentDeletion,
|
DocumentDeletion,
|
||||||
|
DocumentDeletionByFilter,
|
||||||
DocumentClear,
|
DocumentClear,
|
||||||
Settings {
|
Settings {
|
||||||
allow_index_creation: bool,
|
allow_index_creation: bool,
|
||||||
@ -64,6 +65,9 @@ impl From<KindWithContent> for AutobatchKind {
|
|||||||
} => AutobatchKind::DocumentImport { method, allow_index_creation, primary_key },
|
} => AutobatchKind::DocumentImport { method, allow_index_creation, primary_key },
|
||||||
KindWithContent::DocumentDeletion { .. } => AutobatchKind::DocumentDeletion,
|
KindWithContent::DocumentDeletion { .. } => AutobatchKind::DocumentDeletion,
|
||||||
KindWithContent::DocumentClear { .. } => AutobatchKind::DocumentClear,
|
KindWithContent::DocumentClear { .. } => AutobatchKind::DocumentClear,
|
||||||
|
KindWithContent::DocumentDeletionByFilter { .. } => {
|
||||||
|
AutobatchKind::DocumentDeletionByFilter
|
||||||
|
}
|
||||||
KindWithContent::SettingsUpdate { allow_index_creation, is_deletion, .. } => {
|
KindWithContent::SettingsUpdate { allow_index_creation, is_deletion, .. } => {
|
||||||
AutobatchKind::Settings {
|
AutobatchKind::Settings {
|
||||||
allow_index_creation: allow_index_creation && !is_deletion,
|
allow_index_creation: allow_index_creation && !is_deletion,
|
||||||
@ -97,6 +101,9 @@ pub enum BatchKind {
|
|||||||
DocumentDeletion {
|
DocumentDeletion {
|
||||||
deletion_ids: Vec<TaskId>,
|
deletion_ids: Vec<TaskId>,
|
||||||
},
|
},
|
||||||
|
DocumentDeletionByFilter {
|
||||||
|
id: TaskId,
|
||||||
|
},
|
||||||
ClearAndSettings {
|
ClearAndSettings {
|
||||||
other: Vec<TaskId>,
|
other: Vec<TaskId>,
|
||||||
allow_index_creation: bool,
|
allow_index_creation: bool,
|
||||||
@ -195,6 +202,9 @@ impl BatchKind {
|
|||||||
K::DocumentDeletion => {
|
K::DocumentDeletion => {
|
||||||
(Continue(BatchKind::DocumentDeletion { deletion_ids: vec![task_id] }), false)
|
(Continue(BatchKind::DocumentDeletion { deletion_ids: vec![task_id] }), false)
|
||||||
}
|
}
|
||||||
|
K::DocumentDeletionByFilter => {
|
||||||
|
(Break(BatchKind::DocumentDeletionByFilter { id: task_id }), false)
|
||||||
|
}
|
||||||
K::Settings { allow_index_creation } => (
|
K::Settings { allow_index_creation } => (
|
||||||
Continue(BatchKind::Settings { allow_index_creation, settings_ids: vec![task_id] }),
|
Continue(BatchKind::Settings { allow_index_creation, settings_ids: vec![task_id] }),
|
||||||
allow_index_creation,
|
allow_index_creation,
|
||||||
@ -212,7 +222,7 @@ impl BatchKind {
|
|||||||
|
|
||||||
match (self, kind) {
|
match (self, kind) {
|
||||||
// We don't batch any of these operations
|
// We don't batch any of these operations
|
||||||
(this, K::IndexCreation | K::IndexUpdate | K::IndexSwap) => Break(this),
|
(this, K::IndexCreation | K::IndexUpdate | K::IndexSwap | K::DocumentDeletionByFilter) => Break(this),
|
||||||
// We must not batch tasks that don't have the same index creation rights if the index doesn't already exists.
|
// We must not batch tasks that don't have the same index creation rights if the index doesn't already exists.
|
||||||
(this, kind) if !index_already_exists && this.allow_index_creation() == Some(false) && kind.allow_index_creation() == Some(true) => {
|
(this, kind) if !index_already_exists && this.allow_index_creation() == Some(false) && kind.allow_index_creation() == Some(true) => {
|
||||||
Break(this)
|
Break(this)
|
||||||
@ -471,7 +481,8 @@ impl BatchKind {
|
|||||||
BatchKind::IndexCreation { .. }
|
BatchKind::IndexCreation { .. }
|
||||||
| BatchKind::IndexDeletion { .. }
|
| BatchKind::IndexDeletion { .. }
|
||||||
| BatchKind::IndexUpdate { .. }
|
| BatchKind::IndexUpdate { .. }
|
||||||
| BatchKind::IndexSwap { .. },
|
| BatchKind::IndexSwap { .. }
|
||||||
|
| BatchKind::DocumentDeletionByFilter { .. },
|
||||||
_,
|
_,
|
||||||
) => {
|
) => {
|
||||||
unreachable!()
|
unreachable!()
|
||||||
|
@ -28,9 +28,10 @@ use meilisearch_types::heed::{RoTxn, RwTxn};
|
|||||||
use meilisearch_types::milli::documents::{obkv_to_object, DocumentsBatchReader};
|
use meilisearch_types::milli::documents::{obkv_to_object, DocumentsBatchReader};
|
||||||
use meilisearch_types::milli::heed::CompactionOption;
|
use meilisearch_types::milli::heed::CompactionOption;
|
||||||
use meilisearch_types::milli::update::{
|
use meilisearch_types::milli::update::{
|
||||||
DocumentDeletionResult, IndexDocumentsConfig, IndexDocumentsMethod, Settings as MilliSettings,
|
DeleteDocuments, DocumentDeletionResult, IndexDocumentsConfig, IndexDocumentsMethod,
|
||||||
|
Settings as MilliSettings,
|
||||||
};
|
};
|
||||||
use meilisearch_types::milli::{self, BEU32};
|
use meilisearch_types::milli::{self, Filter, BEU32};
|
||||||
use meilisearch_types::settings::{apply_settings_to_builder, Settings, Unchecked};
|
use meilisearch_types::settings::{apply_settings_to_builder, Settings, Unchecked};
|
||||||
use meilisearch_types::tasks::{Details, IndexSwap, Kind, KindWithContent, Status, Task};
|
use meilisearch_types::tasks::{Details, IndexSwap, Kind, KindWithContent, Status, Task};
|
||||||
use meilisearch_types::{compression, Index, VERSION_FILE_NAME};
|
use meilisearch_types::{compression, Index, VERSION_FILE_NAME};
|
||||||
@ -65,6 +66,10 @@ pub(crate) enum Batch {
|
|||||||
op: IndexOperation,
|
op: IndexOperation,
|
||||||
must_create_index: bool,
|
must_create_index: bool,
|
||||||
},
|
},
|
||||||
|
IndexDocumentDeletionByFilter {
|
||||||
|
index_uid: String,
|
||||||
|
task: Task,
|
||||||
|
},
|
||||||
IndexCreation {
|
IndexCreation {
|
||||||
index_uid: String,
|
index_uid: String,
|
||||||
primary_key: Option<String>,
|
primary_key: Option<String>,
|
||||||
@ -149,6 +154,7 @@ impl Batch {
|
|||||||
| Batch::TaskDeletion(task)
|
| Batch::TaskDeletion(task)
|
||||||
| Batch::Dump(task)
|
| Batch::Dump(task)
|
||||||
| Batch::IndexCreation { task, .. }
|
| Batch::IndexCreation { task, .. }
|
||||||
|
| Batch::IndexDocumentDeletionByFilter { task, .. }
|
||||||
| Batch::IndexUpdate { task, .. } => vec![task.uid],
|
| Batch::IndexUpdate { task, .. } => vec![task.uid],
|
||||||
Batch::SnapshotCreation(tasks) | Batch::IndexDeletion { tasks, .. } => {
|
Batch::SnapshotCreation(tasks) | Batch::IndexDeletion { tasks, .. } => {
|
||||||
tasks.iter().map(|task| task.uid).collect()
|
tasks.iter().map(|task| task.uid).collect()
|
||||||
@ -187,7 +193,8 @@ impl Batch {
|
|||||||
IndexOperation { op, .. } => Some(op.index_uid()),
|
IndexOperation { op, .. } => Some(op.index_uid()),
|
||||||
IndexCreation { index_uid, .. }
|
IndexCreation { index_uid, .. }
|
||||||
| IndexUpdate { index_uid, .. }
|
| IndexUpdate { index_uid, .. }
|
||||||
| IndexDeletion { index_uid, .. } => Some(index_uid),
|
| IndexDeletion { index_uid, .. }
|
||||||
|
| IndexDocumentDeletionByFilter { index_uid, .. } => Some(index_uid),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -227,6 +234,18 @@ impl IndexScheduler {
|
|||||||
},
|
},
|
||||||
must_create_index,
|
must_create_index,
|
||||||
})),
|
})),
|
||||||
|
BatchKind::DocumentDeletionByFilter { id } => {
|
||||||
|
let task = self.get_task(rtxn, id)?.ok_or(Error::CorruptedTaskQueue)?;
|
||||||
|
match &task.kind {
|
||||||
|
KindWithContent::DocumentDeletionByFilter { index_uid, .. } => {
|
||||||
|
Ok(Some(Batch::IndexDocumentDeletionByFilter {
|
||||||
|
index_uid: index_uid.clone(),
|
||||||
|
task,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
_ => unreachable!(),
|
||||||
|
}
|
||||||
|
}
|
||||||
BatchKind::DocumentOperation { method, operation_ids, .. } => {
|
BatchKind::DocumentOperation { method, operation_ids, .. } => {
|
||||||
let tasks = self.get_existing_tasks(rtxn, operation_ids)?;
|
let tasks = self.get_existing_tasks(rtxn, operation_ids)?;
|
||||||
let primary_key = tasks
|
let primary_key = tasks
|
||||||
@ -867,6 +886,64 @@ impl IndexScheduler {
|
|||||||
|
|
||||||
Ok(tasks)
|
Ok(tasks)
|
||||||
}
|
}
|
||||||
|
Batch::IndexDocumentDeletionByFilter { mut task, index_uid: _ } => {
|
||||||
|
let (index_uid, filter) =
|
||||||
|
if let KindWithContent::DocumentDeletionByFilter { index_uid, filter_expr } =
|
||||||
|
&task.kind
|
||||||
|
{
|
||||||
|
(index_uid, filter_expr)
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
};
|
||||||
|
let index = {
|
||||||
|
let rtxn = self.env.read_txn()?;
|
||||||
|
self.index_mapper.index(&rtxn, index_uid)?
|
||||||
|
};
|
||||||
|
let filter = Filter::from_json(filter)?;
|
||||||
|
let deleted_documents = if let Some(filter) = filter {
|
||||||
|
let index_rtxn = index.read_txn()?;
|
||||||
|
|
||||||
|
let candidates = filter.evaluate(&index_rtxn, &index)?;
|
||||||
|
let mut wtxn = index.write_txn()?;
|
||||||
|
let mut delete_operation = DeleteDocuments::new(&mut wtxn, &index)?;
|
||||||
|
delete_operation.delete_documents(&candidates);
|
||||||
|
let result = delete_operation.execute().map(|result| result.deleted_documents);
|
||||||
|
wtxn.commit()?;
|
||||||
|
result
|
||||||
|
} else {
|
||||||
|
Ok(0)
|
||||||
|
};
|
||||||
|
let original_filter = if let Some(Details::DocumentDeletionByFilter {
|
||||||
|
original_filter,
|
||||||
|
deleted_documents: _,
|
||||||
|
}) = task.details
|
||||||
|
{
|
||||||
|
original_filter
|
||||||
|
} else {
|
||||||
|
// In the case of a `documentDeleteByFilter` the details MUST be set
|
||||||
|
unreachable!();
|
||||||
|
};
|
||||||
|
|
||||||
|
match deleted_documents {
|
||||||
|
Ok(deleted_documents) => {
|
||||||
|
task.status = Status::Succeeded;
|
||||||
|
task.details = Some(Details::DocumentDeletionByFilter {
|
||||||
|
original_filter,
|
||||||
|
deleted_documents: Some(deleted_documents),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
task.status = Status::Failed;
|
||||||
|
task.details = Some(Details::DocumentDeletionByFilter {
|
||||||
|
original_filter,
|
||||||
|
deleted_documents: Some(0),
|
||||||
|
});
|
||||||
|
task.error = Some(e.into());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(vec![task])
|
||||||
|
}
|
||||||
Batch::IndexCreation { index_uid, primary_key, task } => {
|
Batch::IndexCreation { index_uid, primary_key, task } => {
|
||||||
let wtxn = self.env.write_txn()?;
|
let wtxn = self.env.write_txn()?;
|
||||||
if self.index_mapper.exists(&wtxn, &index_uid)? {
|
if self.index_mapper.exists(&wtxn, &index_uid)? {
|
||||||
|
@ -183,6 +183,9 @@ fn snapshot_details(d: &Details) -> String {
|
|||||||
provided_ids: received_document_ids,
|
provided_ids: received_document_ids,
|
||||||
deleted_documents,
|
deleted_documents,
|
||||||
} => format!("{{ received_document_ids: {received_document_ids}, deleted_documents: {deleted_documents:?} }}"),
|
} => format!("{{ received_document_ids: {received_document_ids}, deleted_documents: {deleted_documents:?} }}"),
|
||||||
|
Details::DocumentDeletionByFilter { original_filter, deleted_documents } => format!(
|
||||||
|
"{{ original_filter: {original_filter}, deleted_documents: {deleted_documents:?} }}"
|
||||||
|
),
|
||||||
Details::ClearAll { deleted_documents } => {
|
Details::ClearAll { deleted_documents } => {
|
||||||
format!("{{ deleted_documents: {deleted_documents:?} }}")
|
format!("{{ deleted_documents: {deleted_documents:?} }}")
|
||||||
},
|
},
|
||||||
|
@ -1208,6 +1208,13 @@ impl<'a> Dump<'a> {
|
|||||||
documents_ids,
|
documents_ids,
|
||||||
index_uid: task.index_uid.ok_or(Error::CorruptedDump)?,
|
index_uid: task.index_uid.ok_or(Error::CorruptedDump)?,
|
||||||
},
|
},
|
||||||
|
KindDump::DocumentDeletionByFilter { filter } => {
|
||||||
|
KindWithContent::DocumentDeletionByFilter {
|
||||||
|
filter_expr: serde_json::from_str(&filter)
|
||||||
|
.map_err(|_| Error::CorruptedDump)?,
|
||||||
|
index_uid: task.index_uid.ok_or(Error::CorruptedDump)?,
|
||||||
|
}
|
||||||
|
}
|
||||||
KindDump::DocumentClear => KindWithContent::DocumentClear {
|
KindDump::DocumentClear => KindWithContent::DocumentClear {
|
||||||
index_uid: task.index_uid.ok_or(Error::CorruptedDump)?,
|
index_uid: task.index_uid.ok_or(Error::CorruptedDump)?,
|
||||||
},
|
},
|
||||||
|
@ -239,6 +239,7 @@ pub fn swap_index_uid_in_task(task: &mut Task, swap: (&str, &str)) {
|
|||||||
match &mut task.kind {
|
match &mut task.kind {
|
||||||
K::DocumentAdditionOrUpdate { index_uid, .. } => index_uids.push(index_uid),
|
K::DocumentAdditionOrUpdate { index_uid, .. } => index_uids.push(index_uid),
|
||||||
K::DocumentDeletion { index_uid, .. } => index_uids.push(index_uid),
|
K::DocumentDeletion { index_uid, .. } => index_uids.push(index_uid),
|
||||||
|
K::DocumentDeletionByFilter { index_uid, .. } => index_uids.push(index_uid),
|
||||||
K::DocumentClear { index_uid } => index_uids.push(index_uid),
|
K::DocumentClear { index_uid } => index_uids.push(index_uid),
|
||||||
K::SettingsUpdate { index_uid, .. } => index_uids.push(index_uid),
|
K::SettingsUpdate { index_uid, .. } => index_uids.push(index_uid),
|
||||||
K::IndexDeletion { index_uid } => index_uids.push(index_uid),
|
K::IndexDeletion { index_uid } => index_uids.push(index_uid),
|
||||||
@ -464,6 +465,29 @@ impl IndexScheduler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Details::DocumentDeletionByFilter { deleted_documents, original_filter: _ } => {
|
||||||
|
assert_eq!(kind.as_kind(), Kind::DocumentDeletionByFilter);
|
||||||
|
let (index_uid, _) = if let KindWithContent::DocumentDeletionByFilter {
|
||||||
|
ref index_uid,
|
||||||
|
ref filter_expr,
|
||||||
|
} = kind
|
||||||
|
{
|
||||||
|
(index_uid, filter_expr)
|
||||||
|
} else {
|
||||||
|
unreachable!()
|
||||||
|
};
|
||||||
|
assert_eq!(&task_index_uid.unwrap(), index_uid);
|
||||||
|
|
||||||
|
match status {
|
||||||
|
Status::Enqueued | Status::Processing => (),
|
||||||
|
Status::Succeeded => {
|
||||||
|
assert!(deleted_documents.is_some());
|
||||||
|
}
|
||||||
|
Status::Failed | Status::Canceled => {
|
||||||
|
assert!(deleted_documents == Some(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Details::ClearAll { deleted_documents } => {
|
Details::ClearAll { deleted_documents } => {
|
||||||
assert!(matches!(
|
assert!(matches!(
|
||||||
kind.as_kind(),
|
kind.as_kind(),
|
||||||
|
@ -315,6 +315,7 @@ impl ErrorCode for milli::Error {
|
|||||||
UserError::MaxDatabaseSizeReached => Code::DatabaseSizeLimitReached,
|
UserError::MaxDatabaseSizeReached => Code::DatabaseSizeLimitReached,
|
||||||
UserError::AttributeLimitReached => Code::MaxFieldsLimitExceeded,
|
UserError::AttributeLimitReached => Code::MaxFieldsLimitExceeded,
|
||||||
UserError::InvalidFilter(_) => Code::InvalidSearchFilter,
|
UserError::InvalidFilter(_) => Code::InvalidSearchFilter,
|
||||||
|
UserError::InvalidFilterExpression(..) => Code::InvalidSearchFilter,
|
||||||
UserError::MissingDocumentId { .. } => Code::MissingDocumentId,
|
UserError::MissingDocumentId { .. } => Code::MissingDocumentId,
|
||||||
UserError::InvalidDocumentId { .. } | UserError::TooManyDocumentIds { .. } => {
|
UserError::InvalidDocumentId { .. } | UserError::TooManyDocumentIds { .. } => {
|
||||||
Code::InvalidDocumentId
|
Code::InvalidDocumentId
|
||||||
|
@ -49,6 +49,7 @@ impl Task {
|
|||||||
| IndexSwap { .. } => None,
|
| IndexSwap { .. } => None,
|
||||||
DocumentAdditionOrUpdate { index_uid, .. }
|
DocumentAdditionOrUpdate { index_uid, .. }
|
||||||
| DocumentDeletion { index_uid, .. }
|
| DocumentDeletion { index_uid, .. }
|
||||||
|
| DocumentDeletionByFilter { index_uid, .. }
|
||||||
| DocumentClear { index_uid }
|
| DocumentClear { index_uid }
|
||||||
| SettingsUpdate { index_uid, .. }
|
| SettingsUpdate { index_uid, .. }
|
||||||
| IndexCreation { index_uid, .. }
|
| IndexCreation { index_uid, .. }
|
||||||
@ -67,6 +68,7 @@ impl Task {
|
|||||||
match self.kind {
|
match self.kind {
|
||||||
KindWithContent::DocumentAdditionOrUpdate { content_file, .. } => Some(content_file),
|
KindWithContent::DocumentAdditionOrUpdate { content_file, .. } => Some(content_file),
|
||||||
KindWithContent::DocumentDeletion { .. }
|
KindWithContent::DocumentDeletion { .. }
|
||||||
|
| KindWithContent::DocumentDeletionByFilter { .. }
|
||||||
| KindWithContent::DocumentClear { .. }
|
| KindWithContent::DocumentClear { .. }
|
||||||
| KindWithContent::SettingsUpdate { .. }
|
| KindWithContent::SettingsUpdate { .. }
|
||||||
| KindWithContent::IndexDeletion { .. }
|
| KindWithContent::IndexDeletion { .. }
|
||||||
@ -81,6 +83,11 @@ impl Task {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub enum DocumentDeletionContent {
|
||||||
|
ByDocumentIds(Vec<String>),
|
||||||
|
ByFilter(serde_json::Value),
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
||||||
#[serde(rename_all = "camelCase")]
|
#[serde(rename_all = "camelCase")]
|
||||||
pub enum KindWithContent {
|
pub enum KindWithContent {
|
||||||
@ -96,6 +103,10 @@ pub enum KindWithContent {
|
|||||||
index_uid: String,
|
index_uid: String,
|
||||||
documents_ids: Vec<String>,
|
documents_ids: Vec<String>,
|
||||||
},
|
},
|
||||||
|
DocumentDeletionByFilter {
|
||||||
|
index_uid: String,
|
||||||
|
filter_expr: serde_json::Value,
|
||||||
|
},
|
||||||
DocumentClear {
|
DocumentClear {
|
||||||
index_uid: String,
|
index_uid: String,
|
||||||
},
|
},
|
||||||
@ -145,6 +156,7 @@ impl KindWithContent {
|
|||||||
match self {
|
match self {
|
||||||
KindWithContent::DocumentAdditionOrUpdate { .. } => Kind::DocumentAdditionOrUpdate,
|
KindWithContent::DocumentAdditionOrUpdate { .. } => Kind::DocumentAdditionOrUpdate,
|
||||||
KindWithContent::DocumentDeletion { .. } => Kind::DocumentDeletion,
|
KindWithContent::DocumentDeletion { .. } => Kind::DocumentDeletion,
|
||||||
|
KindWithContent::DocumentDeletionByFilter { .. } => Kind::DocumentDeletion,
|
||||||
KindWithContent::DocumentClear { .. } => Kind::DocumentDeletion,
|
KindWithContent::DocumentClear { .. } => Kind::DocumentDeletion,
|
||||||
KindWithContent::SettingsUpdate { .. } => Kind::SettingsUpdate,
|
KindWithContent::SettingsUpdate { .. } => Kind::SettingsUpdate,
|
||||||
KindWithContent::IndexCreation { .. } => Kind::IndexCreation,
|
KindWithContent::IndexCreation { .. } => Kind::IndexCreation,
|
||||||
@ -168,6 +180,7 @@ impl KindWithContent {
|
|||||||
| TaskDeletion { .. } => vec![],
|
| TaskDeletion { .. } => vec![],
|
||||||
DocumentAdditionOrUpdate { index_uid, .. }
|
DocumentAdditionOrUpdate { index_uid, .. }
|
||||||
| DocumentDeletion { index_uid, .. }
|
| DocumentDeletion { index_uid, .. }
|
||||||
|
| DocumentDeletionByFilter { index_uid, .. }
|
||||||
| DocumentClear { index_uid }
|
| DocumentClear { index_uid }
|
||||||
| SettingsUpdate { index_uid, .. }
|
| SettingsUpdate { index_uid, .. }
|
||||||
| IndexCreation { index_uid, .. }
|
| IndexCreation { index_uid, .. }
|
||||||
@ -200,6 +213,12 @@ impl KindWithContent {
|
|||||||
deleted_documents: None,
|
deleted_documents: None,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
KindWithContent::DocumentDeletionByFilter { index_uid: _, filter_expr } => {
|
||||||
|
Some(Details::DocumentDeletionByFilter {
|
||||||
|
original_filter: filter_expr.to_string(),
|
||||||
|
deleted_documents: None,
|
||||||
|
})
|
||||||
|
}
|
||||||
KindWithContent::DocumentClear { .. } | KindWithContent::IndexDeletion { .. } => {
|
KindWithContent::DocumentClear { .. } | KindWithContent::IndexDeletion { .. } => {
|
||||||
Some(Details::ClearAll { deleted_documents: None })
|
Some(Details::ClearAll { deleted_documents: None })
|
||||||
}
|
}
|
||||||
@ -242,6 +261,12 @@ impl KindWithContent {
|
|||||||
deleted_documents: Some(0),
|
deleted_documents: Some(0),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
KindWithContent::DocumentDeletionByFilter { index_uid: _, filter_expr } => {
|
||||||
|
Some(Details::DocumentDeletionByFilter {
|
||||||
|
original_filter: filter_expr.to_string(),
|
||||||
|
deleted_documents: Some(0),
|
||||||
|
})
|
||||||
|
}
|
||||||
KindWithContent::DocumentClear { .. } => {
|
KindWithContent::DocumentClear { .. } => {
|
||||||
Some(Details::ClearAll { deleted_documents: None })
|
Some(Details::ClearAll { deleted_documents: None })
|
||||||
}
|
}
|
||||||
@ -282,6 +307,7 @@ impl From<&KindWithContent> for Option<Details> {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
KindWithContent::DocumentDeletion { .. } => None,
|
KindWithContent::DocumentDeletion { .. } => None,
|
||||||
|
KindWithContent::DocumentDeletionByFilter { .. } => None,
|
||||||
KindWithContent::DocumentClear { .. } => None,
|
KindWithContent::DocumentClear { .. } => None,
|
||||||
KindWithContent::SettingsUpdate { new_settings, .. } => {
|
KindWithContent::SettingsUpdate { new_settings, .. } => {
|
||||||
Some(Details::SettingsUpdate { settings: new_settings.clone() })
|
Some(Details::SettingsUpdate { settings: new_settings.clone() })
|
||||||
@ -374,6 +400,7 @@ impl std::error::Error for ParseTaskStatusError {}
|
|||||||
pub enum Kind {
|
pub enum Kind {
|
||||||
DocumentAdditionOrUpdate,
|
DocumentAdditionOrUpdate,
|
||||||
DocumentDeletion,
|
DocumentDeletion,
|
||||||
|
DocumentDeletionByFilter,
|
||||||
SettingsUpdate,
|
SettingsUpdate,
|
||||||
IndexCreation,
|
IndexCreation,
|
||||||
IndexDeletion,
|
IndexDeletion,
|
||||||
@ -390,6 +417,7 @@ impl Kind {
|
|||||||
match self {
|
match self {
|
||||||
Kind::DocumentAdditionOrUpdate
|
Kind::DocumentAdditionOrUpdate
|
||||||
| Kind::DocumentDeletion
|
| Kind::DocumentDeletion
|
||||||
|
| Kind::DocumentDeletionByFilter
|
||||||
| Kind::SettingsUpdate
|
| Kind::SettingsUpdate
|
||||||
| Kind::IndexCreation
|
| Kind::IndexCreation
|
||||||
| Kind::IndexDeletion
|
| Kind::IndexDeletion
|
||||||
@ -407,6 +435,7 @@ impl Display for Kind {
|
|||||||
match self {
|
match self {
|
||||||
Kind::DocumentAdditionOrUpdate => write!(f, "documentAdditionOrUpdate"),
|
Kind::DocumentAdditionOrUpdate => write!(f, "documentAdditionOrUpdate"),
|
||||||
Kind::DocumentDeletion => write!(f, "documentDeletion"),
|
Kind::DocumentDeletion => write!(f, "documentDeletion"),
|
||||||
|
Kind::DocumentDeletionByFilter => write!(f, "documentDeletionByFilter"),
|
||||||
Kind::SettingsUpdate => write!(f, "settingsUpdate"),
|
Kind::SettingsUpdate => write!(f, "settingsUpdate"),
|
||||||
Kind::IndexCreation => write!(f, "indexCreation"),
|
Kind::IndexCreation => write!(f, "indexCreation"),
|
||||||
Kind::IndexDeletion => write!(f, "indexDeletion"),
|
Kind::IndexDeletion => write!(f, "indexDeletion"),
|
||||||
@ -478,6 +507,7 @@ pub enum Details {
|
|||||||
SettingsUpdate { settings: Box<Settings<Unchecked>> },
|
SettingsUpdate { settings: Box<Settings<Unchecked>> },
|
||||||
IndexInfo { primary_key: Option<String> },
|
IndexInfo { primary_key: Option<String> },
|
||||||
DocumentDeletion { provided_ids: usize, deleted_documents: Option<u64> },
|
DocumentDeletion { provided_ids: usize, deleted_documents: Option<u64> },
|
||||||
|
DocumentDeletionByFilter { original_filter: String, deleted_documents: Option<u64> },
|
||||||
ClearAll { deleted_documents: Option<u64> },
|
ClearAll { deleted_documents: Option<u64> },
|
||||||
TaskCancelation { matched_tasks: u64, canceled_tasks: Option<u64>, original_filter: String },
|
TaskCancelation { matched_tasks: u64, canceled_tasks: Option<u64>, original_filter: String },
|
||||||
TaskDeletion { matched_tasks: u64, deleted_tasks: Option<u64>, original_filter: String },
|
TaskDeletion { matched_tasks: u64, deleted_tasks: Option<u64>, original_filter: String },
|
||||||
@ -493,6 +523,9 @@ impl Details {
|
|||||||
*indexed_documents = Some(0)
|
*indexed_documents = Some(0)
|
||||||
}
|
}
|
||||||
Self::DocumentDeletion { deleted_documents, .. } => *deleted_documents = Some(0),
|
Self::DocumentDeletion { deleted_documents, .. } => *deleted_documents = Some(0),
|
||||||
|
Self::DocumentDeletionByFilter { deleted_documents, .. } => {
|
||||||
|
*deleted_documents = Some(0)
|
||||||
|
}
|
||||||
Self::ClearAll { deleted_documents } => *deleted_documents = Some(0),
|
Self::ClearAll { deleted_documents } => *deleted_documents = Some(0),
|
||||||
Self::TaskCancelation { canceled_tasks, .. } => *canceled_tasks = Some(0),
|
Self::TaskCancelation { canceled_tasks, .. } => *canceled_tasks = Some(0),
|
||||||
Self::TaskDeletion { deleted_tasks, .. } => *deleted_tasks = Some(0),
|
Self::TaskDeletion { deleted_tasks, .. } => *deleted_tasks = Some(0),
|
||||||
|
@ -17,7 +17,6 @@ use meilisearch_types::error::{Code, ResponseError};
|
|||||||
use meilisearch_types::heed::RoTxn;
|
use meilisearch_types::heed::RoTxn;
|
||||||
use meilisearch_types::index_uid::IndexUid;
|
use meilisearch_types::index_uid::IndexUid;
|
||||||
use meilisearch_types::milli::update::IndexDocumentsMethod;
|
use meilisearch_types::milli::update::IndexDocumentsMethod;
|
||||||
use meilisearch_types::milli::InternalError;
|
|
||||||
use meilisearch_types::star_or::OptionStarOrList;
|
use meilisearch_types::star_or::OptionStarOrList;
|
||||||
use meilisearch_types::tasks::KindWithContent;
|
use meilisearch_types::tasks::KindWithContent;
|
||||||
use meilisearch_types::{milli, Document, Index};
|
use meilisearch_types::{milli, Document, Index};
|
||||||
@ -381,31 +380,6 @@ pub enum DocumentDeletionQuery {
|
|||||||
Object { filter: Option<Value> },
|
Object { filter: Option<Value> },
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Parses a Json encoded document id and validate it, returning a user error when it is one.
|
|
||||||
/// FIXME: stolen from milli
|
|
||||||
fn validate_document_id_value(document_id: Value) -> String {
|
|
||||||
match document_id {
|
|
||||||
Value::String(string) => match validate_document_id(&string) {
|
|
||||||
Some(s) if s.len() == string.len() => string,
|
|
||||||
Some(s) => s.to_string(),
|
|
||||||
None => panic!(),
|
|
||||||
},
|
|
||||||
Value::Number(number) if number.is_i64() => number.to_string(),
|
|
||||||
_content => panic!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// FIXME: stolen from milli
|
|
||||||
fn validate_document_id(document_id: &str) -> Option<&str> {
|
|
||||||
if !document_id.is_empty()
|
|
||||||
&& document_id.chars().all(|c| matches!(c, 'a'..='z' | 'A'..='Z' | '0'..='9' | '-' | '_'))
|
|
||||||
{
|
|
||||||
Some(document_id)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn delete_documents(
|
pub async fn delete_documents(
|
||||||
index_scheduler: GuardedData<ActionPolicy<{ actions::DOCUMENTS_DELETE }>, Data<IndexScheduler>>,
|
index_scheduler: GuardedData<ActionPolicy<{ actions::DOCUMENTS_DELETE }>, Data<IndexScheduler>>,
|
||||||
index_uid: web::Path<String>,
|
index_uid: web::Path<String>,
|
||||||
@ -428,34 +402,22 @@ pub async fn delete_documents(
|
|||||||
debug!("filter: {:?}", filter);
|
debug!("filter: {:?}", filter);
|
||||||
|
|
||||||
// FIXME: spawn_blocking
|
// FIXME: spawn_blocking
|
||||||
if let Some(ref filter) = filter {
|
if let Some(mut filter) = filter {
|
||||||
if let Some(facets) = crate::search::parse_filter(filter)? {
|
if let Some(facets) = crate::search::parse_filter(&filter)? {
|
||||||
debug!("facets: {:?}", facets);
|
debug!("facets: {:?}", facets);
|
||||||
|
|
||||||
let index = index_scheduler.index(&index_uid)?;
|
let task = KindWithContent::DocumentDeletionByFilter {
|
||||||
let rtxn = index.read_txn()?;
|
index_uid: index_uid.to_string(),
|
||||||
let filtered_candidates = facets.evaluate(&rtxn, &index)?;
|
filter_expr: filter.take(),
|
||||||
debug!("filtered_candidates.len(): {:?}", filtered_candidates.len());
|
};
|
||||||
|
|
||||||
// FIXME: unwraps
|
let task: SummarizedTaskView =
|
||||||
let primary_key = index.primary_key(&rtxn)?.unwrap();
|
tokio::task::spawn_blocking(move || index_scheduler.register(task))
|
||||||
let primary_key = index.fields_ids_map(&rtxn)?.id(primary_key).unwrap();
|
.await??
|
||||||
|
.into();
|
||||||
|
|
||||||
let documents = index.documents(&rtxn, filtered_candidates.into_iter())?;
|
debug!("returns: {:?}", task);
|
||||||
debug!("documents.len(): {:?}", documents.len());
|
return Ok(HttpResponse::Accepted().json(task));
|
||||||
let documents: Vec<String> = documents
|
|
||||||
.into_iter()
|
|
||||||
.map(|(_, document)| {
|
|
||||||
let value = document.get(primary_key).unwrap();
|
|
||||||
let value: Value = serde_json::from_slice(value)
|
|
||||||
.map_err(InternalError::SerdeJson)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
validate_document_id_value(value)
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
debug!("documents: {:?}", documents);
|
|
||||||
documents
|
|
||||||
} else {
|
} else {
|
||||||
vec![]
|
vec![]
|
||||||
}
|
}
|
||||||
|
@ -133,6 +133,13 @@ impl From<Details> for DetailsView {
|
|||||||
deleted_documents: Some(deleted_documents),
|
deleted_documents: Some(deleted_documents),
|
||||||
..DetailsView::default()
|
..DetailsView::default()
|
||||||
},
|
},
|
||||||
|
Details::DocumentDeletionByFilter { original_filter, deleted_documents } => {
|
||||||
|
DetailsView {
|
||||||
|
original_filter: Some(original_filter),
|
||||||
|
deleted_documents: Some(deleted_documents),
|
||||||
|
..DetailsView::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
Details::ClearAll { deleted_documents } => {
|
Details::ClearAll { deleted_documents } => {
|
||||||
DetailsView { deleted_documents: Some(deleted_documents), ..DetailsView::default() }
|
DetailsView { deleted_documents: Some(deleted_documents), ..DetailsView::default() }
|
||||||
}
|
}
|
||||||
|
@ -112,6 +112,8 @@ only composed of alphanumeric characters (a-z A-Z 0-9), hyphens (-) and undersco
|
|||||||
InvalidGeoField(#[from] GeoError),
|
InvalidGeoField(#[from] GeoError),
|
||||||
#[error("{0}")]
|
#[error("{0}")]
|
||||||
InvalidFilter(String),
|
InvalidFilter(String),
|
||||||
|
#[error("Invalid type for filter subexpression: `expected {}, found: {1}`.", .0.join(", "))]
|
||||||
|
InvalidFilterExpression(&'static [&'static str], Value),
|
||||||
#[error("Attribute `{}` is not sortable. {}",
|
#[error("Attribute `{}` is not sortable. {}",
|
||||||
.field,
|
.field,
|
||||||
match .valid_fields.is_empty() {
|
match .valid_fields.is_empty() {
|
||||||
|
@ -5,6 +5,7 @@ use std::ops::Bound::{self, Excluded, Included};
|
|||||||
use either::Either;
|
use either::Either;
|
||||||
pub use filter_parser::{Condition, Error as FPError, FilterCondition, Span, Token};
|
pub use filter_parser::{Condition, Error as FPError, FilterCondition, Span, Token};
|
||||||
use roaring::RoaringBitmap;
|
use roaring::RoaringBitmap;
|
||||||
|
use serde_json::Value;
|
||||||
|
|
||||||
use super::facet_range_search;
|
use super::facet_range_search;
|
||||||
use crate::error::{Error, UserError};
|
use crate::error::{Error, UserError};
|
||||||
@ -112,6 +113,52 @@ impl<'a> From<Filter<'a>> for FilterCondition<'a> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Filter<'a> {
|
impl<'a> Filter<'a> {
|
||||||
|
pub fn from_json(facets: &'a Value) -> Result<Option<Self>> {
|
||||||
|
match facets {
|
||||||
|
Value::String(expr) => {
|
||||||
|
let condition = Filter::from_str(expr)?;
|
||||||
|
Ok(condition)
|
||||||
|
}
|
||||||
|
Value::Array(arr) => Self::parse_filter_array(arr),
|
||||||
|
v => Err(Error::UserError(UserError::InvalidFilterExpression(
|
||||||
|
&["String", "Array"],
|
||||||
|
v.clone(),
|
||||||
|
))),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_filter_array(arr: &'a [Value]) -> Result<Option<Self>> {
|
||||||
|
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(Error::UserError(UserError::InvalidFilterExpression(
|
||||||
|
&["String"],
|
||||||
|
v.clone(),
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ands.push(Either::Left(ors));
|
||||||
|
}
|
||||||
|
v => {
|
||||||
|
return Err(Error::UserError(UserError::InvalidFilterExpression(
|
||||||
|
&["String", "[String]"],
|
||||||
|
v.clone(),
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Filter::from_array(ands)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn from_array<I, J>(array: I) -> Result<Option<Self>>
|
pub fn from_array<I, J>(array: I) -> Result<Option<Self>>
|
||||||
where
|
where
|
||||||
I: IntoIterator<Item = Either<J, &'a str>>,
|
I: IntoIterator<Item = Either<J, &'a str>>,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user