mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 13:06:27 +00:00 
			
		
		
		
	Add the tasks cancel route to cancel tasks
This commit is contained in:
		
				
					committed by
					
						 Clément Renault
						Clément Renault
					
				
			
			
				
	
			
			
			
						parent
						
							290945e258
						
					
				
				
					commit
					751e9bac3b
				
			| @@ -17,9 +17,10 @@ pub enum Error { | |||||||
|     CorruptedDump, |     CorruptedDump, | ||||||
|     #[error("Task `{0}` not found")] |     #[error("Task `{0}` not found")] | ||||||
|     TaskNotFound(TaskId), |     TaskNotFound(TaskId), | ||||||
|     // TODO: Lo: proper error message for this |     #[error("Query parameters to filter the tasks to delete are missing. Available query parameters are: `uid`, `indexUid`, `status`, `type`")] | ||||||
|     #[error("Cannot delete all tasks")] |  | ||||||
|     TaskDeletionWithEmptyQuery, |     TaskDeletionWithEmptyQuery, | ||||||
|  |     #[error("Query parameters to filter the tasks to cancel are missing. Available query parameters are: `uid`, `indexUid`, `status`, `type`")] | ||||||
|  |     TaskCancelationWithEmptyQuery, | ||||||
|     // maybe the two next errors are going to move to the frontend |     // maybe the two next errors are going to move to the frontend | ||||||
|     #[error("`{0}` is not a status. Available status are")] |     #[error("`{0}` is not a status. Available status are")] | ||||||
|     InvalidStatus(String), |     InvalidStatus(String), | ||||||
| @@ -48,6 +49,7 @@ impl ErrorCode for Error { | |||||||
|             Error::IndexAlreadyExists(_) => Code::IndexAlreadyExists, |             Error::IndexAlreadyExists(_) => Code::IndexAlreadyExists, | ||||||
|             Error::TaskNotFound(_) => Code::TaskNotFound, |             Error::TaskNotFound(_) => Code::TaskNotFound, | ||||||
|             Error::TaskDeletionWithEmptyQuery => Code::TaskDeletionWithEmptyQuery, |             Error::TaskDeletionWithEmptyQuery => Code::TaskDeletionWithEmptyQuery, | ||||||
|  |             Error::TaskCancelationWithEmptyQuery => Code::TaskCancelationWithEmptyQuery, | ||||||
|             Error::InvalidStatus(_) => Code::BadRequest, |             Error::InvalidStatus(_) => Code::BadRequest, | ||||||
|             Error::InvalidKind(_) => Code::BadRequest, |             Error::InvalidKind(_) => Code::BadRequest, | ||||||
|  |  | ||||||
|   | |||||||
| @@ -829,7 +829,7 @@ mod tests { | |||||||
|             replace_document_import_task("catto", None, 0, 12), |             replace_document_import_task("catto", None, 0, 12), | ||||||
|             KindWithContent::TaskCancelation { |             KindWithContent::TaskCancelation { | ||||||
|                 query: format!("uid=0,1"), |                 query: format!("uid=0,1"), | ||||||
|                 tasks: vec![0, 1], |                 tasks: RoaringBitmap::from_iter([0, 1]), | ||||||
|             }, |             }, | ||||||
|             replace_document_import_task("catto", None, 1, 50), |             replace_document_import_task("catto", None, 1, 50), | ||||||
|             replace_document_import_task("doggo", Some("bone"), 2, 5000), |             replace_document_import_task("doggo", Some("bone"), 2, 5000), | ||||||
|   | |||||||
| @@ -104,6 +104,7 @@ fn snapshot_task(task: &Task) -> String { | |||||||
|         started_at: _, |         started_at: _, | ||||||
|         finished_at: _, |         finished_at: _, | ||||||
|         error, |         error, | ||||||
|  |         canceled_by: _, | ||||||
|         details, |         details, | ||||||
|         status, |         status, | ||||||
|         kind, |         kind, | ||||||
|   | |||||||
| @@ -25,7 +25,8 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | |||||||
|             .route(web::get().to(SeqHandler(get_tasks))) |             .route(web::get().to(SeqHandler(get_tasks))) | ||||||
|             .route(web::delete().to(SeqHandler(delete_tasks))), |             .route(web::delete().to(SeqHandler(delete_tasks))), | ||||||
|     ) |     ) | ||||||
|     .service(web::resource("/{task_id}").route(web::get().to(SeqHandler(get_task)))); |     .service(web::resource("/{task_id}").route(web::get().to(SeqHandler(get_task)))) | ||||||
|  |     .service(web::resource("/cancel").route(web::post().to(SeqHandler(cancel_tasks)))); | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, PartialEq, Serialize)] | #[derive(Debug, Clone, PartialEq, Serialize)] | ||||||
| @@ -200,6 +201,60 @@ pub struct TaskDeletionQuery { | |||||||
|     index_uid: Option<CS<IndexUid>>, |     index_uid: Option<CS<IndexUid>>, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[derive(Deserialize, Debug)] | ||||||
|  | #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||||
|  | pub struct TaskCancelationQuery { | ||||||
|  |     #[serde(rename = "type")] | ||||||
|  |     type_: Option<CS<Kind>>, | ||||||
|  |     uid: Option<CS<u32>>, | ||||||
|  |     status: Option<CS<Status>>, | ||||||
|  |     index_uid: Option<CS<IndexUid>>, | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async fn cancel_tasks( | ||||||
|  |     index_scheduler: GuardedData<ActionPolicy<{ actions::TASKS_CANCEL }>, Data<IndexScheduler>>, | ||||||
|  |     params: web::Query<TaskCancelationQuery>, | ||||||
|  | ) -> Result<HttpResponse, ResponseError> { | ||||||
|  |     let TaskCancelationQuery { | ||||||
|  |         type_, | ||||||
|  |         uid, | ||||||
|  |         status, | ||||||
|  |         index_uid, | ||||||
|  |     } = params.into_inner(); | ||||||
|  |  | ||||||
|  |     let kind: Option<Vec<_>> = type_.map(|x| x.into_iter().collect()); | ||||||
|  |     let uid: Option<Vec<_>> = uid.map(|x| x.into_iter().collect()); | ||||||
|  |     let status: Option<Vec<_>> = status.map(|x| x.into_iter().collect()); | ||||||
|  |     let index_uid: Option<Vec<_>> = | ||||||
|  |         index_uid.map(|x| x.into_iter().map(|x| x.to_string()).collect()); | ||||||
|  |  | ||||||
|  |     let query = Query { | ||||||
|  |         limit: None, | ||||||
|  |         from: None, | ||||||
|  |         status, | ||||||
|  |         kind, | ||||||
|  |         index_uid, | ||||||
|  |         uid, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if query.is_empty() { | ||||||
|  |         return Err(index_scheduler::Error::TaskCancelationWithEmptyQuery.into()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let filtered_query = filter_out_inaccessible_indexes_from_query(&index_scheduler, &query); | ||||||
|  |     let tasks = index_scheduler.get_task_ids(&filtered_query)?; | ||||||
|  |     let filtered_query_string = yaup::to_string(&filtered_query).unwrap(); | ||||||
|  |     let task_cancelation = KindWithContent::TaskCancelation { | ||||||
|  |         query: filtered_query_string, | ||||||
|  |         tasks, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     let task = index_scheduler.register(task_cancelation)?; | ||||||
|  |     let task_view = TaskView::from_task(&task); | ||||||
|  |  | ||||||
|  |     Ok(HttpResponse::Ok().json(task_view)) | ||||||
|  | } | ||||||
|  |  | ||||||
| async fn delete_tasks( | async fn delete_tasks( | ||||||
|     index_scheduler: GuardedData<ActionPolicy<{ actions::TASKS_DELETE }>, Data<IndexScheduler>>, |     index_scheduler: GuardedData<ActionPolicy<{ actions::TASKS_DELETE }>, Data<IndexScheduler>>, | ||||||
|     params: web::Query<TaskDeletionQuery>, |     params: web::Query<TaskDeletionQuery>, | ||||||
| @@ -225,12 +280,12 @@ async fn delete_tasks( | |||||||
|         index_uid, |         index_uid, | ||||||
|         uid, |         uid, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if query.is_empty() { |     if query.is_empty() { | ||||||
|         return Err(index_scheduler::Error::TaskDeletionWithEmptyQuery.into()); |         return Err(index_scheduler::Error::TaskDeletionWithEmptyQuery.into()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     let filtered_query = filter_out_inaccessible_indexes_from_query(&index_scheduler, &query); |     let filtered_query = filter_out_inaccessible_indexes_from_query(&index_scheduler, &query); | ||||||
|  |  | ||||||
|     let tasks = index_scheduler.get_task_ids(&filtered_query)?; |     let tasks = index_scheduler.get_task_ids(&filtered_query)?; | ||||||
|     let filtered_query_string = yaup::to_string(&filtered_query).unwrap(); |     let filtered_query_string = yaup::to_string(&filtered_query).unwrap(); | ||||||
|     let task_deletion = KindWithContent::TaskDeletion { |     let task_deletion = KindWithContent::TaskDeletion { | ||||||
| @@ -239,7 +294,6 @@ async fn delete_tasks( | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     let task = index_scheduler.register(task_deletion)?; |     let task = index_scheduler.register(task_deletion)?; | ||||||
|  |  | ||||||
|     let task_view = TaskView::from_task(&task); |     let task_view = TaskView::from_task(&task); | ||||||
|  |  | ||||||
|     Ok(HttpResponse::Ok().json(task_view)) |     Ok(HttpResponse::Ok().json(task_view)) | ||||||
|   | |||||||
| @@ -151,6 +151,7 @@ pub enum Code { | |||||||
|     DumpNotFound, |     DumpNotFound, | ||||||
|     TaskNotFound, |     TaskNotFound, | ||||||
|     TaskDeletionWithEmptyQuery, |     TaskDeletionWithEmptyQuery, | ||||||
|  |     TaskCancelationWithEmptyQuery, | ||||||
|     PayloadTooLarge, |     PayloadTooLarge, | ||||||
|     RetrieveDocument, |     RetrieveDocument, | ||||||
|     SearchDocuments, |     SearchDocuments, | ||||||
| @@ -241,9 +242,11 @@ impl Code { | |||||||
|                 ErrCode::authentication("missing_master_key", StatusCode::UNAUTHORIZED) |                 ErrCode::authentication("missing_master_key", StatusCode::UNAUTHORIZED) | ||||||
|             } |             } | ||||||
|             TaskNotFound => ErrCode::invalid("task_not_found", StatusCode::NOT_FOUND), |             TaskNotFound => ErrCode::invalid("task_not_found", StatusCode::NOT_FOUND), | ||||||
|             // TODO: Lo: find the proper error name & message for this one |  | ||||||
|             TaskDeletionWithEmptyQuery => { |             TaskDeletionWithEmptyQuery => { | ||||||
|                 ErrCode::invalid("task_deletion_with_empty_query", StatusCode::BAD_REQUEST) |                 ErrCode::invalid("missing_filters", StatusCode::BAD_REQUEST) | ||||||
|  |             } | ||||||
|  |             TaskCancelationWithEmptyQuery => { | ||||||
|  |                 ErrCode::invalid("missing_filters", StatusCode::BAD_REQUEST) | ||||||
|             } |             } | ||||||
|             DumpNotFound => ErrCode::invalid("dump_not_found", StatusCode::NOT_FOUND), |             DumpNotFound => ErrCode::invalid("dump_not_found", StatusCode::NOT_FOUND), | ||||||
|             NoSpaceLeftOnDevice => { |             NoSpaceLeftOnDevice => { | ||||||
|   | |||||||
| @@ -224,6 +224,8 @@ pub enum Action { | |||||||
|     IndexesDelete, |     IndexesDelete, | ||||||
|     #[serde(rename = "tasks.*")] |     #[serde(rename = "tasks.*")] | ||||||
|     TasksAll, |     TasksAll, | ||||||
|  |     #[serde(rename = "tasks.cancel")] | ||||||
|  |     TasksCancel, | ||||||
|     #[serde(rename = "tasks.delete")] |     #[serde(rename = "tasks.delete")] | ||||||
|     TasksDelete, |     TasksDelete, | ||||||
|     #[serde(rename = "tasks.get")] |     #[serde(rename = "tasks.get")] | ||||||
| @@ -274,6 +276,8 @@ impl Action { | |||||||
|             INDEXES_UPDATE => Some(Self::IndexesUpdate), |             INDEXES_UPDATE => Some(Self::IndexesUpdate), | ||||||
|             INDEXES_DELETE => Some(Self::IndexesDelete), |             INDEXES_DELETE => Some(Self::IndexesDelete), | ||||||
|             TASKS_ALL => Some(Self::TasksAll), |             TASKS_ALL => Some(Self::TasksAll), | ||||||
|  |             TASKS_CANCEL => Some(Self::TasksCancel), | ||||||
|  |             TASKS_DELETE => Some(Self::TasksDelete), | ||||||
|             TASKS_GET => Some(Self::TasksGet), |             TASKS_GET => Some(Self::TasksGet), | ||||||
|             SETTINGS_ALL => Some(Self::SettingsAll), |             SETTINGS_ALL => Some(Self::SettingsAll), | ||||||
|             SETTINGS_GET => Some(Self::SettingsGet), |             SETTINGS_GET => Some(Self::SettingsGet), | ||||||
| @@ -313,6 +317,7 @@ pub mod actions { | |||||||
|     pub const INDEXES_UPDATE: u8 = IndexesUpdate.repr(); |     pub const INDEXES_UPDATE: u8 = IndexesUpdate.repr(); | ||||||
|     pub const INDEXES_DELETE: u8 = IndexesDelete.repr(); |     pub const INDEXES_DELETE: u8 = IndexesDelete.repr(); | ||||||
|     pub const TASKS_ALL: u8 = TasksAll.repr(); |     pub const TASKS_ALL: u8 = TasksAll.repr(); | ||||||
|  |     pub const TASKS_CANCEL: u8 = TasksCancel.repr(); | ||||||
|     pub const TASKS_DELETE: u8 = TasksDelete.repr(); |     pub const TASKS_DELETE: u8 = TasksDelete.repr(); | ||||||
|     pub const TASKS_GET: u8 = TasksGet.repr(); |     pub const TASKS_GET: u8 = TasksGet.repr(); | ||||||
|     pub const SETTINGS_ALL: u8 = SettingsAll.repr(); |     pub const SETTINGS_ALL: u8 = SettingsAll.repr(); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user