mirror of
https://github.com/meilisearch/MeiliSearch
synced 2024-11-23 21:34:27 +01:00
67 lines
2.5 KiB
Rust
67 lines
2.5 KiB
Rust
use actix_web::web::Data;
|
|
use actix_web::{web, HttpRequest, HttpResponse};
|
|
use deserr::actix_web::AwebJson;
|
|
use deserr::Deserr;
|
|
use index_scheduler::IndexScheduler;
|
|
use meilisearch_types::deserr::DeserrJsonError;
|
|
use meilisearch_types::error::deserr_codes::InvalidSwapIndexes;
|
|
use meilisearch_types::error::ResponseError;
|
|
use meilisearch_types::index_uid::IndexUid;
|
|
use meilisearch_types::tasks::{IndexSwap, KindWithContent};
|
|
use serde_json::json;
|
|
|
|
use super::SummarizedTaskView;
|
|
use crate::analytics::Analytics;
|
|
use crate::error::MeilisearchHttpError;
|
|
use crate::extractors::authentication::policies::*;
|
|
use crate::extractors::authentication::{AuthenticationError, GuardedData};
|
|
use crate::extractors::sequential_extractor::SeqHandler;
|
|
|
|
pub fn configure(cfg: &mut web::ServiceConfig) {
|
|
cfg.service(web::resource("").route(web::post().to(SeqHandler(swap_indexes))));
|
|
}
|
|
|
|
#[derive(Deserr, Debug, Clone, PartialEq, Eq)]
|
|
#[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields)]
|
|
pub struct SwapIndexesPayload {
|
|
#[deserr(error = DeserrJsonError<InvalidSwapIndexes>, missing_field_error = DeserrJsonError::missing_swap_indexes)]
|
|
indexes: Vec<IndexUid>,
|
|
}
|
|
|
|
pub async fn swap_indexes(
|
|
index_scheduler: GuardedData<ActionPolicy<{ actions::INDEXES_SWAP }>, Data<IndexScheduler>>,
|
|
params: AwebJson<Vec<SwapIndexesPayload>, DeserrJsonError>,
|
|
req: HttpRequest,
|
|
analytics: web::Data<dyn Analytics>,
|
|
) -> Result<HttpResponse, ResponseError> {
|
|
let params = params.into_inner();
|
|
analytics.publish(
|
|
"Indexes Swapped".to_string(),
|
|
json!({
|
|
"swap_operation_number": params.len(),
|
|
}),
|
|
Some(&req),
|
|
);
|
|
let filters = index_scheduler.filters();
|
|
|
|
let mut swaps = vec![];
|
|
for SwapIndexesPayload { indexes } in params.into_iter() {
|
|
// TODO: switch to deserr
|
|
let (lhs, rhs) = match indexes.as_slice() {
|
|
[lhs, rhs] => (lhs, rhs),
|
|
_ => {
|
|
return Err(MeilisearchHttpError::SwapIndexPayloadWrongLength(indexes).into());
|
|
}
|
|
};
|
|
if !filters.is_index_authorized(lhs) || !filters.is_index_authorized(rhs) {
|
|
return Err(AuthenticationError::InvalidToken.into());
|
|
}
|
|
swaps.push(IndexSwap { indexes: (lhs.to_string(), rhs.to_string()) });
|
|
}
|
|
|
|
let task = KindWithContent::IndexSwap { swaps };
|
|
let task: SummarizedTaskView =
|
|
tokio::task::spawn_blocking(move || index_scheduler.register(task)).await??.into();
|
|
Ok(HttpResponse::Accepted().json(task))
|
|
}
|