MeiliSearch/meilisearch-http/src/routes/index.rs

330 lines
9.2 KiB
Rust
Raw Normal View History

use chrono::{DateTime, Utc};
2019-11-20 17:28:46 +01:00
use log::error;
2019-11-26 11:06:55 +01:00
use meilisearch_core::ProcessedUpdateResult;
use rand::seq::SliceRandom;
use serde::{Deserialize, Serialize};
2019-10-31 15:00:36 +01:00
use serde_json::json;
2020-01-15 17:10:33 +01:00
use tide::{Request, Response};
2019-10-31 15:00:36 +01:00
use crate::error::{ResponseError, SResult};
2020-01-15 17:10:33 +01:00
use crate::helpers::tide::RequestExt;
2019-10-31 15:00:36 +01:00
use crate::models::token::ACL::*;
use crate::Data;
fn generate_uid() -> String {
let mut rng = rand::thread_rng();
let sample = b"abcdefghijklmnopqrstuvwxyz0123456789";
sample
.choose_multiple(&mut rng, 8)
.map(|c| *c as char)
.collect()
}
2020-01-15 17:10:33 +01:00
pub async fn list_indexes(ctx: Request<Data>) -> SResult<Response> {
2019-10-31 15:00:36 +01:00
ctx.is_allowed(IndexesRead)?;
2019-11-20 11:24:08 +01:00
let indexes_uids = ctx.state().db.indexes_uids();
let db = &ctx.state().db;
let reader = db.main_read_txn().map_err(ResponseError::internal)?;
let mut response_body = Vec::new();
for index_uid in indexes_uids {
2019-11-20 17:28:46 +01:00
let index = ctx.state().db.open_index(&index_uid);
match index {
Some(index) => {
let name = index
.main
.name(&reader)
.map_err(ResponseError::internal)?
.ok_or(ResponseError::internal("'name' not found"))?;
let created_at = index
.main
.created_at(&reader)
.map_err(ResponseError::internal)?
.ok_or(ResponseError::internal("'created_at' date not found"))?;
let updated_at = index
.main
.updated_at(&reader)
.map_err(ResponseError::internal)?
.ok_or(ResponseError::internal("'updated_at' date not found"))?;
let index_response = IndexResponse {
2019-11-20 17:28:46 +01:00
name,
uid: index_uid,
created_at,
updated_at,
};
response_body.push(index_response);
2019-11-20 17:28:46 +01:00
}
None => error!(
"Index {} is referenced in the indexes list but cannot be found",
index_uid
),
}
}
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(200).body_json(&response_body).unwrap())
2019-10-31 15:00:36 +01:00
}
2019-11-20 17:28:46 +01:00
#[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")]
struct IndexResponse {
2019-11-19 17:38:02 +01:00
name: String,
uid: String,
created_at: DateTime<Utc>,
updated_at: DateTime<Utc>,
}
2020-01-15 17:10:33 +01:00
pub async fn get_index(ctx: Request<Data>) -> SResult<Response> {
2019-11-19 17:38:02 +01:00
ctx.is_allowed(IndexesRead)?;
let index = ctx.index()?;
let db = &ctx.state().db;
let reader = db.main_read_txn().map_err(ResponseError::internal)?;
2019-11-19 17:38:02 +01:00
2019-11-20 17:28:46 +01:00
let uid = ctx.url_param("index")?;
2019-11-20 11:24:08 +01:00
let name = index
.main
2019-11-20 17:28:46 +01:00
.name(&reader)
2019-11-19 17:38:02 +01:00
.map_err(ResponseError::internal)?
2019-11-20 17:28:46 +01:00
.ok_or(ResponseError::internal("'name' not found"))?;
2019-11-20 11:24:08 +01:00
let created_at = index
.main
2019-11-20 17:28:46 +01:00
.created_at(&reader)
2019-11-19 17:38:02 +01:00
.map_err(ResponseError::internal)?
2019-11-20 17:28:46 +01:00
.ok_or(ResponseError::internal("'created_at' date not found"))?;
2019-11-20 11:24:08 +01:00
let updated_at = index
.main
2019-11-20 17:28:46 +01:00
.updated_at(&reader)
2019-11-19 17:38:02 +01:00
.map_err(ResponseError::internal)?
2019-11-20 17:28:46 +01:00
.ok_or(ResponseError::internal("'updated_at' date not found"))?;
2019-11-19 17:38:02 +01:00
let response_body = IndexResponse {
2019-11-19 17:38:02 +01:00
name,
uid,
created_at,
updated_at,
};
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(200).body_json(&response_body).unwrap())
2019-11-19 17:38:02 +01:00
}
2019-11-20 17:28:46 +01:00
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase", deny_unknown_fields)]
struct IndexCreateRequest {
name: Option<String>,
uid: Option<String>,
}
2019-11-20 17:28:46 +01:00
#[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")]
struct IndexCreateResponse {
name: String,
uid: String,
created_at: DateTime<Utc>,
updated_at: DateTime<Utc>,
}
2020-01-15 17:10:33 +01:00
pub async fn create_index(mut ctx: Request<Data>) -> SResult<Response> {
2019-10-31 15:00:36 +01:00
ctx.is_allowed(IndexesWrite)?;
2019-11-20 11:24:08 +01:00
let body = ctx
.body_json::<IndexCreateRequest>()
.await
.map_err(ResponseError::bad_request)?;
2019-10-31 15:00:36 +01:00
if let (None, None) = (body.name.clone(), body.uid.clone()) {
return Err(ResponseError::bad_request("Index creation must have an uid"));
}
2019-10-31 15:00:36 +01:00
let db = &ctx.state().db;
let uid = match body.uid {
Some(uid) => uid,
None => loop {
let uid = generate_uid();
if db.open_index(&uid).is_none() {
break uid;
}
},
};
let created_index = match db.create_index(&uid) {
2019-10-31 15:00:36 +01:00
Ok(index) => index,
Err(e) => return Err(ResponseError::create_index(e)),
};
let mut writer = db.main_write_txn().map_err(ResponseError::internal)?;
2019-10-31 15:00:36 +01:00
let name = body.name.unwrap_or(uid.clone());
created_index.main
.put_name(&mut writer, &name)
.map_err(ResponseError::internal)?;
2019-10-31 15:00:36 +01:00
let created_at = created_index.main
.created_at(&writer)
.map_err(ResponseError::internal)?
.unwrap_or(Utc::now());
let updated_at = created_index.main
.updated_at(&writer)
.map_err(ResponseError::internal)?
.unwrap_or(Utc::now());
2020-01-15 17:10:33 +01:00
writer.commit().map_err(ResponseError::internal)?;
2019-10-31 15:00:36 +01:00
let response_body = IndexCreateResponse {
2020-01-15 17:10:33 +01:00
name: name,
uid,
created_at,
updated_at,
};
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(201).body_json(&response_body).unwrap())
2019-10-31 15:00:36 +01:00
}
2019-11-20 17:28:46 +01:00
#[derive(Debug, Deserialize)]
2019-11-20 15:00:06 +01:00
#[serde(rename_all = "camelCase", deny_unknown_fields)]
struct UpdateIndexRequest {
name: String,
}
2019-11-20 17:28:46 +01:00
#[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")]
2019-11-20 15:00:06 +01:00
struct UpdateIndexResponse {
name: String,
uid: String,
created_at: DateTime<Utc>,
updated_at: DateTime<Utc>,
}
2020-01-15 17:10:33 +01:00
pub async fn update_index(mut ctx: Request<Data>) -> SResult<Response> {
2019-11-20 15:00:06 +01:00
ctx.is_allowed(IndexesWrite)?;
let body = ctx
.body_json::<UpdateIndexRequest>()
.await
.map_err(ResponseError::bad_request)?;
let index_uid = ctx.url_param("index")?;
let index = ctx.index()?;
let db = &ctx.state().db;
let mut writer = db.main_write_txn().map_err(ResponseError::internal)?;
2019-11-20 15:00:06 +01:00
index
.main
.put_name(&mut writer, &body.name)
.map_err(ResponseError::internal)?;
index
.main
.put_updated_at(&mut writer)
.map_err(ResponseError::internal)?;
writer.commit().map_err(ResponseError::internal)?;
let reader = db.main_read_txn().map_err(ResponseError::internal)?;
2019-11-20 15:00:06 +01:00
let created_at = index
.main
.created_at(&reader)
.map_err(ResponseError::internal)?
2019-11-20 17:28:46 +01:00
.ok_or(ResponseError::internal("'created_at' date not found"))?;
2019-11-20 15:00:06 +01:00
let updated_at = index
.main
.updated_at(&reader)
.map_err(ResponseError::internal)?
2019-11-20 17:28:46 +01:00
.ok_or(ResponseError::internal("'updated_at' date not found"))?;
2019-11-20 15:00:06 +01:00
let response_body = UpdateIndexResponse {
name: body.name,
uid: index_uid,
created_at,
updated_at,
};
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(200).body_json(&response_body).unwrap())
2019-11-20 15:00:06 +01:00
}
2020-01-15 17:10:33 +01:00
pub async fn get_update_status(ctx: Request<Data>) -> SResult<Response> {
2019-10-31 15:00:36 +01:00
ctx.is_allowed(IndexesRead)?;
let db = &ctx.state().db;
let reader = db.update_read_txn().map_err(ResponseError::internal)?;
2019-10-31 15:00:36 +01:00
let update_id = ctx
.param::<u64>("update_id")
.map_err(|e| ResponseError::bad_parameter("update_id", e))?;
let index = ctx.index()?;
let status = index
.update_status(&reader, update_id)
.map_err(ResponseError::internal)?;
let response = match status {
2020-01-15 17:10:33 +01:00
Some(status) => tide::Response::new(200).body_json(&status).unwrap(),
None => tide::Response::new(404).body_json(&json!({ "message": "unknown update id" })).unwrap(),
2019-10-31 15:00:36 +01:00
};
Ok(response)
}
2020-01-15 17:10:33 +01:00
pub async fn get_all_updates_status(ctx: Request<Data>) -> SResult<Response> {
2019-10-31 15:00:36 +01:00
ctx.is_allowed(IndexesRead)?;
let db = &ctx.state().db;
let reader = db.update_read_txn().map_err(ResponseError::internal)?;
2019-10-31 15:00:36 +01:00
let index = ctx.index()?;
2020-01-15 17:10:33 +01:00
let response = index
2019-10-31 15:00:36 +01:00
.all_updates_status(&reader)
.map_err(ResponseError::internal)?;
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(200).body_json(&response).unwrap())
2019-10-31 15:00:36 +01:00
}
2020-01-15 17:10:33 +01:00
pub async fn delete_index(ctx: Request<Data>) -> SResult<Response> {
2019-10-31 15:00:36 +01:00
ctx.is_allowed(IndexesWrite)?;
let _ = ctx.index()?;
2019-11-19 16:15:49 +01:00
let index_uid = ctx.url_param("index")?;
ctx.state().db.delete_index(&index_uid).map_err(ResponseError::internal)?;
2020-01-15 17:10:33 +01:00
Ok(tide::Response::new(204))
2019-10-31 15:00:36 +01:00
}
2019-11-20 17:28:46 +01:00
pub fn index_update_callback(index_uid: &str, data: &Data, status: ProcessedUpdateResult) {
if status.error.is_some() {
return;
}
2019-10-31 15:00:36 +01:00
if let Some(index) = data.db.open_index(&index_uid) {
let db = &data.db;
let mut writer = match db.main_write_txn() {
2019-11-20 17:28:46 +01:00
Ok(writer) => writer,
Err(e) => {
error!("Impossible to get write_txn; {}", e);
return;
}
};
if let Err(e) = data.compute_stats(&mut writer, &index_uid) {
error!("Impossible to compute stats; {}", e)
}
if let Err(e) = data.set_last_update(&mut writer) {
error!("Impossible to update last_update; {}", e)
}
if let Err(e) = index.main.put_updated_at(&mut writer) {
error!("Impossible to update updated_at; {}", e)
}
2019-11-20 17:28:46 +01:00
if let Err(e) = writer.commit() {
error!("Impossible to get write_txn; {}", e);
}
}
2019-10-31 15:00:36 +01:00
}