mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 13:06:27 +00:00 
			
		
		
		
	let you specify your task id
This commit is contained in:
		| @@ -251,7 +251,7 @@ pub fn setup_meilisearch(opt: &Opt) -> anyhow::Result<(Arc<IndexScheduler>, Arc< | ||||
|             .name(String::from("register-snapshot-tasks")) | ||||
|             .spawn(move || loop { | ||||
|                 thread::sleep(snapshot_delay); | ||||
|                 if let Err(e) = index_scheduler.register(KindWithContent::SnapshotCreation) { | ||||
|                 if let Err(e) = index_scheduler.register(KindWithContent::SnapshotCreation, None) { | ||||
|                     error!("Error while registering snapshot: {}", e); | ||||
|                 } | ||||
|             }) | ||||
|   | ||||
| @@ -11,7 +11,7 @@ use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| use crate::extractors::sequential_extractor::SeqHandler; | ||||
| use crate::routes::SummarizedTaskView; | ||||
| use crate::routes::{get_task_id, SummarizedTaskView}; | ||||
|  | ||||
| pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|     cfg.service(web::resource("").route(web::post().to(SeqHandler(create_dump)))); | ||||
| @@ -29,8 +29,9 @@ pub async fn create_dump( | ||||
|         keys: auth_controller.list_keys()?, | ||||
|         instance_uid: analytics.instance_uid().cloned(), | ||||
|     }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Create dump"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
|   | ||||
| @@ -7,7 +7,7 @@ use bstr::ByteSlice as _; | ||||
| use deserr::actix_web::{AwebJson, AwebQueryParameter}; | ||||
| use deserr::Deserr; | ||||
| use futures::StreamExt; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use index_scheduler::{IndexScheduler, TaskId}; | ||||
| use meilisearch_types::deserr::query_params::Param; | ||||
| use meilisearch_types::deserr::{DeserrJsonError, DeserrQueryParamError}; | ||||
| use meilisearch_types::document_formats::{read_csv, read_json, read_ndjson, PayloadType}; | ||||
| @@ -36,7 +36,7 @@ use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| use crate::extractors::payload::Payload; | ||||
| use crate::extractors::sequential_extractor::SeqHandler; | ||||
| use crate::routes::{PaginationView, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
| use crate::routes::{get_task_id, PaginationView, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
| use crate::search::parse_filter; | ||||
|  | ||||
| static ACCEPTED_CONTENT_TYPE: Lazy<Vec<String>> = Lazy::new(|| { | ||||
| @@ -130,9 +130,10 @@ pub async fn delete_document( | ||||
|         index_uid: index_uid.to_string(), | ||||
|         documents_ids: vec![document_id], | ||||
|     }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|     debug!(returns = ?task, "Delete document"); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|     debug!("returns: {:?}", task); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| } | ||||
|  | ||||
| @@ -277,6 +278,7 @@ pub async fn replace_documents( | ||||
|     analytics.add_documents(¶ms, index_scheduler.index(&index_uid).is_err(), &req); | ||||
|  | ||||
|     let allow_index_creation = index_scheduler.filters().allow_index_creation(&index_uid); | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task = document_addition( | ||||
|         extract_mime_type(&req)?, | ||||
|         index_scheduler, | ||||
| @@ -285,6 +287,7 @@ pub async fn replace_documents( | ||||
|         params.csv_delimiter, | ||||
|         body, | ||||
|         IndexDocumentsMethod::ReplaceDocuments, | ||||
|         uid, | ||||
|         allow_index_creation, | ||||
|     ) | ||||
|     .await?; | ||||
| @@ -309,6 +312,7 @@ pub async fn update_documents( | ||||
|     analytics.update_documents(¶ms, index_scheduler.index(&index_uid).is_err(), &req); | ||||
|  | ||||
|     let allow_index_creation = index_scheduler.filters().allow_index_creation(&index_uid); | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task = document_addition( | ||||
|         extract_mime_type(&req)?, | ||||
|         index_scheduler, | ||||
| @@ -317,6 +321,7 @@ pub async fn update_documents( | ||||
|         params.csv_delimiter, | ||||
|         body, | ||||
|         IndexDocumentsMethod::UpdateDocuments, | ||||
|         uid, | ||||
|         allow_index_creation, | ||||
|     ) | ||||
|     .await?; | ||||
| @@ -334,6 +339,7 @@ async fn document_addition( | ||||
|     csv_delimiter: Option<u8>, | ||||
|     mut body: Payload, | ||||
|     method: IndexDocumentsMethod, | ||||
|     task_id: Option<TaskId>, | ||||
|     allow_index_creation: bool, | ||||
| ) -> Result<SummarizedTaskView, MeilisearchHttpError> { | ||||
|     let format = match ( | ||||
| @@ -450,7 +456,7 @@ async fn document_addition( | ||||
|     }; | ||||
|  | ||||
|     let scheduler = index_scheduler.clone(); | ||||
|     let task = match tokio::task::spawn_blocking(move || scheduler.register(task)).await? { | ||||
|     let task = match tokio::task::spawn_blocking(move || scheduler.register(task, task_id)).await? { | ||||
|         Ok(task) => task, | ||||
|         Err(e) => { | ||||
|             index_scheduler.delete_update_file(uuid)?; | ||||
| @@ -480,8 +486,9 @@ pub async fn delete_documents_batch( | ||||
|  | ||||
|     let task = | ||||
|         KindWithContent::DocumentDeletion { index_uid: index_uid.to_string(), documents_ids: ids }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Delete documents by batch"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| @@ -516,8 +523,9 @@ pub async fn delete_documents_by_filter( | ||||
|     .map_err(|err| ResponseError::from_msg(err.message, Code::InvalidDocumentFilter))?; | ||||
|     let task = KindWithContent::DocumentDeletionByFilter { index_uid, filter_expr: filter }; | ||||
|  | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Delete documents by filter"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| @@ -533,8 +541,9 @@ pub async fn clear_all_documents( | ||||
|     analytics.delete_documents(DocumentDeletionKind::ClearAll, &req); | ||||
|  | ||||
|     let task = KindWithContent::DocumentClear { index_uid: index_uid.to_string() }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Delete all documents"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
|   | ||||
| @@ -17,7 +17,7 @@ use serde_json::json; | ||||
| use time::OffsetDateTime; | ||||
| use tracing::debug; | ||||
|  | ||||
| use super::{Pagination, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
| use super::{get_task_id, Pagination, SummarizedTaskView, PAGINATION_DEFAULT_LIMIT}; | ||||
| use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::{AuthenticationError, GuardedData}; | ||||
| @@ -137,8 +137,9 @@ pub async fn create_index( | ||||
|         ); | ||||
|  | ||||
|         let task = KindWithContent::IndexCreation { index_uid: uid.to_string(), primary_key }; | ||||
|         let uid = get_task_id(&req)?; | ||||
|         let task: SummarizedTaskView = | ||||
|             tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|             tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|         debug!(returns = ?task, "Create index"); | ||||
|  | ||||
|         Ok(HttpResponse::Accepted().json(task)) | ||||
| @@ -206,8 +207,9 @@ pub async fn update_index( | ||||
|         primary_key: body.primary_key, | ||||
|     }; | ||||
|  | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Update index"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| @@ -216,11 +218,13 @@ pub async fn update_index( | ||||
| pub async fn delete_index( | ||||
|     index_scheduler: GuardedData<ActionPolicy<{ actions::INDEXES_DELETE }>, Data<IndexScheduler>>, | ||||
|     index_uid: web::Path<String>, | ||||
|     req: HttpRequest, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let index_uid = IndexUid::try_from(index_uid.into_inner())?; | ||||
|     let task = KindWithContent::IndexDeletion { index_uid: index_uid.into_inner() }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|     debug!(returns = ?task, "Delete index"); | ||||
|  | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
|   | ||||
| @@ -15,7 +15,7 @@ use tracing::debug; | ||||
| use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| use crate::routes::SummarizedTaskView; | ||||
| use crate::routes::{get_task_id, SummarizedTaskView}; | ||||
|  | ||||
| #[macro_export] | ||||
| macro_rules! make_setting_route { | ||||
| @@ -34,7 +34,7 @@ macro_rules! make_setting_route { | ||||
|             use $crate::extractors::authentication::policies::*; | ||||
|             use $crate::extractors::authentication::GuardedData; | ||||
|             use $crate::extractors::sequential_extractor::SeqHandler; | ||||
|             use $crate::routes::SummarizedTaskView; | ||||
|             use $crate::routes::{get_task_id, SummarizedTaskView}; | ||||
|  | ||||
|             pub async fn delete( | ||||
|                 index_scheduler: GuardedData< | ||||
| @@ -42,6 +42,7 @@ macro_rules! make_setting_route { | ||||
|                     Data<IndexScheduler>, | ||||
|                 >, | ||||
|                 index_uid: web::Path<String>, | ||||
|                 req: HttpRequest, | ||||
|             ) -> Result<HttpResponse, ResponseError> { | ||||
|                 let index_uid = IndexUid::try_from(index_uid.into_inner())?; | ||||
|  | ||||
| @@ -56,8 +57,9 @@ macro_rules! make_setting_route { | ||||
|                     is_deletion: true, | ||||
|                     allow_index_creation, | ||||
|                 }; | ||||
|                 let uid = get_task_id(&req)?; | ||||
|                 let task: SummarizedTaskView = | ||||
|                     tokio::task::spawn_blocking(move || index_scheduler.register(task)) | ||||
|                     tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)) | ||||
|                         .await?? | ||||
|                         .into(); | ||||
|  | ||||
| @@ -105,8 +107,9 @@ macro_rules! make_setting_route { | ||||
|                     is_deletion: false, | ||||
|                     allow_index_creation, | ||||
|                 }; | ||||
|                 let uid = get_task_id(&req)?; | ||||
|                 let task: SummarizedTaskView = | ||||
|                     tokio::task::spawn_blocking(move || index_scheduler.register(task)) | ||||
|                     tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)) | ||||
|                         .await?? | ||||
|                         .into(); | ||||
|  | ||||
| @@ -767,8 +770,9 @@ pub async fn update_all( | ||||
|         is_deletion: false, | ||||
|         allow_index_creation, | ||||
|     }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Update all settings"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| @@ -790,6 +794,7 @@ pub async fn get_all( | ||||
| pub async fn delete_all( | ||||
|     index_scheduler: GuardedData<ActionPolicy<{ actions::SETTINGS_UPDATE }>, Data<IndexScheduler>>, | ||||
|     index_uid: web::Path<String>, | ||||
|     req: HttpRequest, | ||||
| ) -> Result<HttpResponse, ResponseError> { | ||||
|     let index_uid = IndexUid::try_from(index_uid.into_inner())?; | ||||
|  | ||||
| @@ -803,8 +808,9 @@ pub async fn delete_all( | ||||
|         is_deletion: true, | ||||
|         allow_index_creation, | ||||
|     }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Delete all settings"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
|   | ||||
| @@ -4,7 +4,7 @@ use actix_web::web::Data; | ||||
| use actix_web::{web, HttpRequest, HttpResponse}; | ||||
| use index_scheduler::IndexScheduler; | ||||
| use meilisearch_auth::AuthController; | ||||
| use meilisearch_types::error::ResponseError; | ||||
| use meilisearch_types::error::{Code, ResponseError}; | ||||
| use meilisearch_types::settings::{Settings, Unchecked}; | ||||
| use meilisearch_types::tasks::{Kind, Status, Task, TaskId}; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| @@ -45,6 +45,34 @@ pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|         .service(web::scope("/experimental-features").configure(features::configure)); | ||||
| } | ||||
|  | ||||
| pub fn get_task_id(req: &HttpRequest) -> Result<Option<TaskId>, ResponseError> { | ||||
|     let task_id = req | ||||
|         .headers() | ||||
|         .get("TaskId") | ||||
|         .map(|header| { | ||||
|             header.to_str().map_err(|e| { | ||||
|                 ResponseError::from_msg( | ||||
|                     format!("TaskId is not a valid utf-8 string: {e}"), | ||||
|                     Code::BadRequest, | ||||
|                 ) | ||||
|             }) | ||||
|         }) | ||||
|         .transpose()? | ||||
|         .map(|s| { | ||||
|             s.parse::<TaskId>().map_err(|e| { | ||||
|                 ResponseError::from_msg( | ||||
|                     format!( | ||||
|                         "Could not parse the TaskId as a {}: {e}", | ||||
|                         std::any::type_name::<TaskId>(), | ||||
|                     ), | ||||
|                     Code::BadRequest, | ||||
|                 ) | ||||
|             }) | ||||
|         }) | ||||
|         .transpose()?; | ||||
|     Ok(task_id) | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Serialize)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| pub struct SummarizedTaskView { | ||||
|   | ||||
| @@ -10,7 +10,7 @@ use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| use crate::extractors::sequential_extractor::SeqHandler; | ||||
| use crate::routes::SummarizedTaskView; | ||||
| use crate::routes::{get_task_id, SummarizedTaskView}; | ||||
|  | ||||
| pub fn configure(cfg: &mut web::ServiceConfig) { | ||||
|     cfg.service(web::resource("").route(web::post().to(SeqHandler(create_snapshot)))); | ||||
| @@ -24,8 +24,9 @@ pub async fn create_snapshot( | ||||
|     analytics.publish("Snapshot Created".to_string(), json!({}), Some(&req)); | ||||
|  | ||||
|     let task = KindWithContent::SnapshotCreation; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|  | ||||
|     debug!(returns = ?task, "Create snapshot"); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
|   | ||||
| @@ -10,7 +10,7 @@ use meilisearch_types::index_uid::IndexUid; | ||||
| use meilisearch_types::tasks::{IndexSwap, KindWithContent}; | ||||
| use serde_json::json; | ||||
|  | ||||
| use super::SummarizedTaskView; | ||||
| use super::{get_task_id, SummarizedTaskView}; | ||||
| use crate::analytics::Analytics; | ||||
| use crate::error::MeilisearchHttpError; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| @@ -60,7 +60,8 @@ pub async fn swap_indexes( | ||||
|     } | ||||
|  | ||||
|     let task = KindWithContent::IndexSwap { swaps }; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task: SummarizedTaskView = | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into(); | ||||
|         tokio::task::spawn_blocking(move || index_scheduler.register(task, uid)).await??.into(); | ||||
|     Ok(HttpResponse::Accepted().json(task)) | ||||
| } | ||||
|   | ||||
| @@ -18,7 +18,7 @@ use time::macros::format_description; | ||||
| use time::{Date, Duration, OffsetDateTime, Time}; | ||||
| use tokio::task; | ||||
|  | ||||
| use super::SummarizedTaskView; | ||||
| use super::{get_task_id, SummarizedTaskView}; | ||||
| use crate::analytics::Analytics; | ||||
| use crate::extractors::authentication::policies::*; | ||||
| use crate::extractors::authentication::GuardedData; | ||||
| @@ -197,7 +197,9 @@ async fn cancel_tasks( | ||||
|     let task_cancelation = | ||||
|         KindWithContent::TaskCancelation { query: format!("?{}", req.query_string()), tasks }; | ||||
|  | ||||
|     let task = task::spawn_blocking(move || index_scheduler.register(task_cancelation)).await??; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task = | ||||
|         task::spawn_blocking(move || index_scheduler.register(task_cancelation, uid)).await??; | ||||
|     let task: SummarizedTaskView = task.into(); | ||||
|  | ||||
|     Ok(HttpResponse::Ok().json(task)) | ||||
| @@ -242,7 +244,8 @@ async fn delete_tasks( | ||||
|     let task_deletion = | ||||
|         KindWithContent::TaskDeletion { query: format!("?{}", req.query_string()), tasks }; | ||||
|  | ||||
|     let task = task::spawn_blocking(move || index_scheduler.register(task_deletion)).await??; | ||||
|     let uid = get_task_id(&req)?; | ||||
|     let task = task::spawn_blocking(move || index_scheduler.register(task_deletion, uid)).await??; | ||||
|     let task: SummarizedTaskView = task.into(); | ||||
|  | ||||
|     Ok(HttpResponse::Ok().json(task)) | ||||
|   | ||||
| @@ -199,3 +199,74 @@ async fn error_create_with_invalid_index_uid() { | ||||
|     } | ||||
|     "###); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn send_task_id() { | ||||
|     let server = Server::new().await; | ||||
|     let app = server.init_web_app().await; | ||||
|     let index = server.index("catto"); | ||||
|     let (response, code) = index.create(None).await; | ||||
|     snapshot!(code, @"202 Accepted"); | ||||
|     snapshot!(json_string!(response, { ".enqueuedAt" => "[date]" }), @r###" | ||||
|     { | ||||
|       "taskUid": 0, | ||||
|       "indexUid": "catto", | ||||
|       "status": "enqueued", | ||||
|       "type": "indexCreation", | ||||
|       "enqueuedAt": "[date]" | ||||
|     } | ||||
|     "###); | ||||
|  | ||||
|     let body = serde_json::to_string(&json!({ | ||||
|         "uid": "doggo", | ||||
|         "primaryKey": None::<&str>, | ||||
|     })) | ||||
|     .unwrap(); | ||||
|     let req = test::TestRequest::post() | ||||
|         .uri("/indexes") | ||||
|         .insert_header(("TaskId", "25")) | ||||
|         .insert_header(ContentType::json()) | ||||
|         .set_payload(body) | ||||
|         .to_request(); | ||||
|  | ||||
|     let res = test::call_service(&app, req).await; | ||||
|     snapshot!(res.status(), @"202 Accepted"); | ||||
|  | ||||
|     let bytes = test::read_body(res).await; | ||||
|     let response = serde_json::from_slice::<Value>(&bytes).expect("Expecting valid json"); | ||||
|     snapshot!(json_string!(response, { ".enqueuedAt" => "[date]" }), @r###" | ||||
|     { | ||||
|       "taskUid": 25, | ||||
|       "indexUid": "doggo", | ||||
|       "status": "enqueued", | ||||
|       "type": "indexCreation", | ||||
|       "enqueuedAt": "[date]" | ||||
|     } | ||||
|     "###); | ||||
|  | ||||
|     let body = serde_json::to_string(&json!({ | ||||
|         "uid": "girafo", | ||||
|         "primaryKey": None::<&str>, | ||||
|     })) | ||||
|     .unwrap(); | ||||
|     let req = test::TestRequest::post() | ||||
|         .uri("/indexes") | ||||
|         .insert_header(("TaskId", "12")) | ||||
|         .insert_header(ContentType::json()) | ||||
|         .set_payload(body) | ||||
|         .to_request(); | ||||
|  | ||||
|     let res = test::call_service(&app, req).await; | ||||
|     snapshot!(res.status(), @"400 Bad Request"); | ||||
|  | ||||
|     let bytes = test::read_body(res).await; | ||||
|     let response = serde_json::from_slice::<Value>(&bytes).expect("Expecting valid json"); | ||||
|     snapshot!(json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Received bad task id: 12 should be >= to 26.", | ||||
|       "code": "bad_request", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#bad_request" | ||||
|     } | ||||
|     "###); | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user