use meilisearch_types::error::{Code, ErrorCode}; use meilisearch_types::tasks::{Kind, Status}; use meilisearch_types::{heed, milli}; use thiserror::Error; use crate::TaskId; #[allow(clippy::large_enum_variant)] #[derive(Error, Debug)] pub enum Error { #[error("Index `{0}` not found.")] IndexNotFound(String), #[error( "Indexes {} not found.", .0.iter().map(|s| format!("`{}`", s)).collect::>().join(", ") )] IndexesNotFound(Vec), #[error("Index `{0}` already exists.")] IndexAlreadyExists(String), #[error( "Indexes must be declared only once during a swap. `{0}` was specified several times." )] SwapDuplicateIndexFound(String), #[error( "Indexes must be declared only once during a swap. {} were specified several times.", .0.iter().map(|s| format!("`{}`", s)).collect::>().join(", ") )] SwapDuplicateIndexesFound(Vec), #[error("Corrupted dump.")] CorruptedDump, #[error( "Task `{field}` `{date}` is invalid. It should follow the YYYY-MM-DD or RFC 3339 date-time format." )] InvalidTaskDate { field: String, date: String }, #[error("Task uid `{task_uid}` is invalid. It should only contain numeric characters.")] InvalidTaskUids { task_uid: String }, #[error( "Task status `{status}` is invalid. Available task statuses are {}.", enum_iterator::all::() .map(|s| format!("`{s}`")) .collect::>() .join(", ") )] InvalidTaskStatuses { status: String }, #[error( "Task type `{type_}` is invalid. Available task types are {}", enum_iterator::all::() .map(|s| format!("`{s}`")) .collect::>() .join(", ") )] InvalidTaskTypes { type_: String }, #[error( "Task canceledBy `{canceled_by}` is invalid. It should only contains numeric characters separated by `,` character." )] InvalidTaskCanceledBy { canceled_by: String }, #[error( "{index_uid} is not a valid index uid. Index uid can be an integer or a string containing only alphanumeric characters, hyphens (-) and underscores (_)." )] InvalidIndexUid { index_uid: String }, #[error("Task `{0}` not found.")] TaskNotFound(TaskId), #[error("Query parameters to filter the tasks to delete are missing. Available query parameters are: `uids`, `indexUids`, `statuses`, `types`, `beforeEnqueuedAt`, `afterEnqueuedAt`, `beforeStartedAt`, `afterStartedAt`, `beforeFinishedAt`, `afterFinishedAt`.")] TaskDeletionWithEmptyQuery, #[error("Query parameters to filter the tasks to cancel are missing. Available query parameters are: `uids`, `indexUids`, `statuses`, `types`, `beforeEnqueuedAt`, `afterEnqueuedAt`, `beforeStartedAt`, `afterStartedAt`, `beforeFinishedAt`, `afterFinishedAt`.")] TaskCancelationWithEmptyQuery, #[error(transparent)] Dump(#[from] dump::Error), #[error(transparent)] Heed(#[from] heed::Error), #[error(transparent)] Milli(#[from] milli::Error), #[error("An unexpected crash occurred when processing the task.")] ProcessBatchPanicked, #[error(transparent)] FileStore(#[from] file_store::Error), #[error(transparent)] IoError(#[from] std::io::Error), #[error(transparent)] Persist(#[from] tempfile::PersistError), #[error(transparent)] Anyhow(#[from] anyhow::Error), // Irrecoverable errors: #[error(transparent)] CreateBatch(Box), #[error("Corrupted task queue.")] CorruptedTaskQueue, #[error(transparent)] TaskDatabaseUpdate(Box), #[error(transparent)] HeedTransaction(heed::Error), } impl ErrorCode for Error { fn error_code(&self) -> Code { match self { Error::IndexNotFound(_) => Code::IndexNotFound, Error::IndexesNotFound(_) => Code::IndexNotFound, Error::IndexAlreadyExists(_) => Code::IndexAlreadyExists, Error::SwapDuplicateIndexesFound(_) => Code::DuplicateIndexFound, Error::SwapDuplicateIndexFound(_) => Code::DuplicateIndexFound, Error::InvalidTaskDate { .. } => Code::InvalidTaskDateFilter, Error::InvalidTaskUids { .. } => Code::InvalidTaskUidsFilter, Error::InvalidTaskStatuses { .. } => Code::InvalidTaskStatusesFilter, Error::InvalidTaskTypes { .. } => Code::InvalidTaskTypesFilter, Error::InvalidTaskCanceledBy { .. } => Code::InvalidTaskCanceledByFilter, Error::InvalidIndexUid { .. } => Code::InvalidIndexUid, Error::TaskNotFound(_) => Code::TaskNotFound, Error::TaskDeletionWithEmptyQuery => Code::TaskDeletionWithEmptyQuery, Error::TaskCancelationWithEmptyQuery => Code::TaskCancelationWithEmptyQuery, Error::Dump(e) => e.error_code(), Error::Milli(e) => e.error_code(), Error::ProcessBatchPanicked => Code::Internal, Error::Heed(e) => e.error_code(), Error::HeedTransaction(e) => e.error_code(), Error::FileStore(e) => e.error_code(), Error::IoError(e) => e.error_code(), Error::Persist(e) => e.error_code(), // Irrecoverable errors Error::Anyhow(_) => Code::Internal, Error::CorruptedTaskQueue => Code::Internal, Error::CorruptedDump => Code::Internal, Error::TaskDatabaseUpdate(_) => Code::Internal, Error::CreateBatch(_) => Code::Internal, } } }