mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-26 13:36:27 +00:00 
			
		
		
		
	Start change http server; finish document endpoint
This commit is contained in:
		
				
					committed by
					
						 qdequele
						qdequele
					
				
			
			
				
	
			
			
			
						parent
						
							6cc80d2565
						
					
				
				
					commit
					6d6c8e8fb2
				
			
							
								
								
									
										1167
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1167
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -756,16 +756,16 @@ mod tests { | ||||
|         update_reader.abort(); | ||||
|  | ||||
|         let reader = db.main_read_txn().unwrap(); | ||||
|         let document: Option<IgnoredAny> = index.document(&reader, None, DocumentId(25)).unwrap(); | ||||
|         let document: Option<IgnoredAny> = index.document::<_, String>(&reader, None, DocumentId(25)).unwrap(); | ||||
|         assert!(document.is_none()); | ||||
|  | ||||
|         let document: Option<IgnoredAny> = index | ||||
|             .document(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .unwrap(); | ||||
|         assert!(document.is_some()); | ||||
|  | ||||
|         let document: Option<IgnoredAny> = index | ||||
|             .document(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .unwrap(); | ||||
|         assert!(document.is_some()); | ||||
|     } | ||||
| @@ -836,16 +836,16 @@ mod tests { | ||||
|         update_reader.abort(); | ||||
|  | ||||
|         let reader = db.main_read_txn().unwrap(); | ||||
|         let document: Option<IgnoredAny> = index.document(&reader, None, DocumentId(25)).unwrap(); | ||||
|         let document: Option<IgnoredAny> = index.document::<_, String>(&reader, None, DocumentId(25)).unwrap(); | ||||
|         assert!(document.is_none()); | ||||
|  | ||||
|         let document: Option<IgnoredAny> = index | ||||
|             .document(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .unwrap(); | ||||
|         assert!(document.is_some()); | ||||
|  | ||||
|         let document: Option<IgnoredAny> = index | ||||
|             .document(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .unwrap(); | ||||
|         assert!(document.is_some()); | ||||
|  | ||||
| @@ -882,7 +882,7 @@ mod tests { | ||||
|  | ||||
|         let reader = db.main_read_txn().unwrap(); | ||||
|         let document: Option<serde_json::Value> = index | ||||
|             .document(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(7_900_334_843_754_999_545)) | ||||
|             .unwrap(); | ||||
|  | ||||
|         let new_doc1 = serde_json::json!({ | ||||
| @@ -893,7 +893,7 @@ mod tests { | ||||
|         assert_eq!(document, Some(new_doc1)); | ||||
|  | ||||
|         let document: Option<serde_json::Value> = index | ||||
|             .document(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .document::<_, String>(&reader, None, DocumentId(8_367_468_610_878_465_872)) | ||||
|             .unwrap(); | ||||
|  | ||||
|         let new_doc2 = serde_json::json!({ | ||||
|   | ||||
| @@ -214,17 +214,17 @@ pub struct Index { | ||||
| } | ||||
|  | ||||
| impl Index { | ||||
|     pub fn document<T: de::DeserializeOwned>( | ||||
|     pub fn document<T: de::DeserializeOwned, R: AsRef<str>>( | ||||
|         &self, | ||||
|         reader: &heed::RoTxn<MainT>, | ||||
|         attributes: Option<&HashSet<&str>>, | ||||
|         attributes: Option<HashSet<R>>, | ||||
|         document_id: DocumentId, | ||||
|     ) -> MResult<Option<T>> { | ||||
|         let schema = self.main.schema(reader)?; | ||||
|         let schema = schema.ok_or(Error::SchemaMissing)?; | ||||
|  | ||||
|         let attributes = match attributes { | ||||
|             Some(attributes) => Some(attributes.iter().filter_map(|name| schema.id(*name)).collect()), | ||||
|             Some(attributes) => Some(attributes.iter().filter_map(|name| schema.id(name.as_ref())).collect()), | ||||
|             None => None, | ||||
|         }; | ||||
|  | ||||
|   | ||||
| @@ -14,7 +14,7 @@ name = "meilisearch" | ||||
| path = "src/main.rs" | ||||
|  | ||||
| [dependencies] | ||||
| async-std = { version = "1.5.0", features = ["attributes"] } | ||||
| # async-std = { version = "1.5.0", features = ["attributes"] } | ||||
| chrono = { version = "0.4.11", features = ["serde"] } | ||||
| crossbeam-channel = "0.4.2" | ||||
| env_logger = "0.7.1" | ||||
| @@ -38,11 +38,15 @@ sha2 = "0.8.1" | ||||
| siphasher = "0.3.2" | ||||
| structopt = "0.3.12" | ||||
| sysinfo = "0.12.0" | ||||
| tide = "0.6.0" | ||||
| # tide = "0.6.0" | ||||
| ureq = { version = "0.12.0", features = ["tls"], default-features = false } | ||||
| walkdir = "2.3.1" | ||||
| whoami = "0.8.1" | ||||
| slice-group-by = "0.2.6" | ||||
| actix-rt = "1" | ||||
| actix-web = "2" | ||||
| actix-http = "1" | ||||
| tokio = { version = "0.2.0", features = ["macros"] } | ||||
|  | ||||
| [dev-dependencies] | ||||
| http-service = "0.4.0" | ||||
|   | ||||
| @@ -10,7 +10,7 @@ use sha2::Digest; | ||||
| use sysinfo::Pid; | ||||
|  | ||||
| use crate::option::Opt; | ||||
| use crate::routes::index::index_update_callback; | ||||
| // use crate::routes::index::index_update_callback; | ||||
|  | ||||
| const LAST_UPDATE_KEY: &str = "last-update"; | ||||
|  | ||||
| @@ -155,7 +155,7 @@ impl Data { | ||||
|  | ||||
|         let callback_context = data.clone(); | ||||
|         db.set_update_callback(Box::new(move |index_uid, status| { | ||||
|             index_update_callback(&index_uid, &callback_context, status); | ||||
|             // index_update_callback(&index_uid, &callback_context, status); | ||||
|         })); | ||||
|  | ||||
|         data | ||||
|   | ||||
| @@ -1,16 +1,14 @@ | ||||
| use std::fmt::Display; | ||||
|  | ||||
| use http::status::StatusCode; | ||||
| use log::{error, warn}; | ||||
| use std::fmt; | ||||
| use meilisearch_core::{FstError, HeedError}; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use tide::IntoResponse; | ||||
| use tide::Response; | ||||
| use serde_json::json; | ||||
| use actix_http::{ResponseBuilder, Response}; | ||||
| use actix_web::http::StatusCode; | ||||
| use actix_web::*; | ||||
| use futures::future::{ok, Ready}; | ||||
|  | ||||
| use crate::helpers::meilisearch::Error as SearchError; | ||||
|  | ||||
| pub type SResult<T> = Result<T, ResponseError>; | ||||
| // use crate::helpers::meilisearch::Error as SearchError; | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub enum ResponseError { | ||||
|     Internal(String), | ||||
|     BadRequest(String), | ||||
| @@ -23,169 +21,113 @@ pub enum ResponseError { | ||||
|     BadParameter(String, String), | ||||
|     OpenIndex(String), | ||||
|     CreateIndex(String), | ||||
|     CreateTransaction, | ||||
|     CommitTransaction, | ||||
|     Schema, | ||||
|     InferPrimaryKey, | ||||
|     InvalidIndexUid, | ||||
|     Maintenance, | ||||
| } | ||||
|  | ||||
| impl ResponseError { | ||||
|     pub fn internal(message: impl Display) -> ResponseError { | ||||
|         ResponseError::Internal(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn bad_request(message: impl Display) -> ResponseError { | ||||
|         ResponseError::BadRequest(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn invalid_token(message: impl Display) -> ResponseError { | ||||
|         ResponseError::InvalidToken(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn not_found(message: impl Display) -> ResponseError { | ||||
|         ResponseError::NotFound(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn index_not_found(message: impl Display) -> ResponseError { | ||||
|         ResponseError::IndexNotFound(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn document_not_found(message: impl Display) -> ResponseError { | ||||
|         ResponseError::DocumentNotFound(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn missing_header(message: impl Display) -> ResponseError { | ||||
|         ResponseError::MissingHeader(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn bad_parameter(name: impl Display, message: impl Display) -> ResponseError { | ||||
|         ResponseError::BadParameter(name.to_string(), message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn open_index(message: impl Display) -> ResponseError { | ||||
|         ResponseError::OpenIndex(message.to_string()) | ||||
|     } | ||||
|  | ||||
|     pub fn create_index(message: impl Display) -> ResponseError { | ||||
|         ResponseError::CreateIndex(message.to_string()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl IntoResponse for ResponseError { | ||||
|     fn into_response(self) -> Response { | ||||
| impl fmt::Display for ResponseError { | ||||
|     fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||||
|         match self { | ||||
|             ResponseError::Internal(err) => { | ||||
|                 error!("internal server error: {}", err); | ||||
|                 error("Internal server error".to_string(), | ||||
|                     StatusCode::INTERNAL_SERVER_ERROR, | ||||
|                 ) | ||||
|             } | ||||
|             ResponseError::FilterParsing(err) => { | ||||
|                 warn!("error paring filter: {}", err); | ||||
|                 error(format!("parsing error: {}", err), | ||||
|                 StatusCode::BAD_REQUEST) | ||||
|             } | ||||
|             ResponseError::BadRequest(err) => { | ||||
|                 warn!("bad request: {}", err); | ||||
|                 error(err, StatusCode::BAD_REQUEST) | ||||
|             } | ||||
|             ResponseError::InvalidToken(err) => { | ||||
|                 error(format!("Invalid API key: {}", err), StatusCode::FORBIDDEN) | ||||
|             } | ||||
|             ResponseError::NotFound(err) => error(err, StatusCode::NOT_FOUND), | ||||
|             ResponseError::IndexNotFound(index) => { | ||||
|                 error(format!("Index {} not found", index), StatusCode::NOT_FOUND) | ||||
|             } | ||||
|             ResponseError::DocumentNotFound(id) => error( | ||||
|                 format!("Document with id {} not found", id), | ||||
|                 StatusCode::NOT_FOUND, | ||||
|             ), | ||||
|             ResponseError::MissingHeader(header) => error( | ||||
|                 format!("Header {} is missing", header), | ||||
|                 StatusCode::UNAUTHORIZED, | ||||
|             ), | ||||
|             ResponseError::BadParameter(param, e) => error( | ||||
|                 format!("Url parameter {} error: {}", param, e), | ||||
|                 StatusCode::BAD_REQUEST, | ||||
|             ), | ||||
|             ResponseError::CreateIndex(err) => error( | ||||
|                 format!("Impossible to create index; {}", err), | ||||
|                 StatusCode::BAD_REQUEST, | ||||
|             ), | ||||
|             ResponseError::OpenIndex(err) => error( | ||||
|                 format!("Impossible to open index; {}", err), | ||||
|                 StatusCode::BAD_REQUEST, | ||||
|             ), | ||||
|             ResponseError::InvalidIndexUid => error( | ||||
|                 "Index must have a valid uid; Index uid can be of type integer or string only composed of alphanumeric characters, hyphens (-) and underscores (_).".to_string(), | ||||
|                 StatusCode::BAD_REQUEST, | ||||
|             ), | ||||
|             ResponseError::Maintenance => error( | ||||
|                 String::from("Server is in maintenance, please try again later"), | ||||
|                 StatusCode::SERVICE_UNAVAILABLE, | ||||
|             ), | ||||
|             Self::Internal(err) => write!(f, "Internal server error: {}", err), | ||||
|             Self::BadRequest(err) => write!(f, "Bad request: {}", err), | ||||
|             Self::InvalidToken(err) => write!(f, "Invalid API key: {}", err), | ||||
|             Self::NotFound(err) => write!(f, "{} not found", err), | ||||
|             Self::IndexNotFound(index_uid) => write!(f, "Index {} not found", index_uid), | ||||
|             Self::DocumentNotFound(document_id) => write!(f, "Document with id {} not found", document_id), | ||||
|             Self::MissingHeader(header) => write!(f, "Header {} is missing", header), | ||||
|             Self::BadParameter(param, err) => write!(f, "Url parameter {} error: {}", param, err), | ||||
|             Self::OpenIndex(err) => write!(f, "Impossible to open index; {}", err), | ||||
|             Self::CreateIndex(err) => write!(f, "Impossible to create index; {}", err), | ||||
|             Self::CreateTransaction => write!(f, "Impossible to create transaction"), | ||||
|             Self::CommitTransaction => write!(f, "Impossible to commit transaction"), | ||||
|             Self::Schema => write!(f, "Internal schema is innaccessible"), | ||||
|             Self::InferPrimaryKey => write!(f, "Could not infer primary key"), | ||||
|             Self::InvalidIndexUid => write!(f, "Index must have a valid uid; Index uid can be of type integer or string only composed of alphanumeric characters, hyphens (-) and underscores (_)."), | ||||
|             Self::Maintenance => write!(f, "Server is in maintenance, please try again later"), | ||||
|             Self::FilterParsing(err) => write!(f, "parsing error: {}", err), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Serialize, Deserialize)] | ||||
| struct ErrorMessage { | ||||
|     message: String, | ||||
| impl error::ResponseError for ResponseError { | ||||
|     fn error_response(&self) -> HttpResponse { | ||||
|         ResponseBuilder::new(self.status_code()).json(json!({ | ||||
|             "message": self.to_string(), | ||||
|         })) | ||||
|     } | ||||
|  | ||||
|     fn status_code(&self) -> StatusCode { | ||||
|         match *self { | ||||
|             Self::Internal(_) => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|             Self::BadRequest(_) => StatusCode::BAD_REQUEST, | ||||
|             Self::InvalidToken(_) => StatusCode::FORBIDDEN, | ||||
|             Self::NotFound(_) => StatusCode::NOT_FOUND, | ||||
|             Self::IndexNotFound(_) => StatusCode::NOT_FOUND, | ||||
|             Self::DocumentNotFound(_) => StatusCode::NOT_FOUND, | ||||
|             Self::MissingHeader(_) => StatusCode::UNAUTHORIZED, | ||||
|             Self::BadParameter(_, _) => StatusCode::BAD_REQUEST, | ||||
|             Self::OpenIndex(_) => StatusCode::BAD_REQUEST, | ||||
|             Self::CreateIndex(_) => StatusCode::BAD_REQUEST, | ||||
|             Self::CreateTransaction => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|             Self::CommitTransaction => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|             Self::Schema => StatusCode::INTERNAL_SERVER_ERROR, | ||||
|             Self::InferPrimaryKey => StatusCode::BAD_REQUEST, | ||||
|             Self::InvalidIndexUid => StatusCode::BAD_REQUEST, | ||||
|             Self::Maintenance => StatusCode::SERVICE_UNAVAILABLE, | ||||
|             Self::FilterParsing(_) => StatusCode::BAD_REQUEST, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn error(message: String, status: StatusCode) -> Response { | ||||
|     let message = ErrorMessage { message }; | ||||
|     tide::Response::new(status.as_u16()) | ||||
|         .body_json(&message) | ||||
|         .unwrap() | ||||
| } | ||||
| // impl Responder for ResponseError { | ||||
| //     type Error = Error; | ||||
| //     type Future = Ready<Result<Response, Error>>; | ||||
|  | ||||
| //     #[inline] | ||||
| //     fn respond_to(self, req: &HttpRequest) -> Self::Future { | ||||
| //         ok(self.error_response()) | ||||
| //     } | ||||
| // } | ||||
|  | ||||
| impl From<serde_json::Error> for ResponseError { | ||||
|     fn from(err: serde_json::Error) -> ResponseError { | ||||
|         ResponseError::internal(err) | ||||
|         ResponseError::Internal(err.to_string()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<meilisearch_core::Error> for ResponseError { | ||||
|     fn from(err: meilisearch_core::Error) -> ResponseError { | ||||
|         ResponseError::internal(err) | ||||
|         ResponseError::Internal(err.to_string()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<HeedError> for ResponseError { | ||||
|     fn from(err: HeedError) -> ResponseError { | ||||
|         ResponseError::internal(err) | ||||
|         ResponseError::Internal(err.to_string()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<FstError> for ResponseError { | ||||
|     fn from(err: FstError) -> ResponseError { | ||||
|         ResponseError::internal(err) | ||||
|         ResponseError::Internal(err.to_string()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<SearchError> for ResponseError { | ||||
|     fn from(err: SearchError) -> ResponseError { | ||||
|         match err { | ||||
|             SearchError::FilterParsing(s) => ResponseError::FilterParsing(s), | ||||
|             _ => ResponseError::internal(err), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| // impl From<SearchError> for ResponseError { | ||||
| //     fn from(err: SearchError) -> ResponseError { | ||||
| //         match err { | ||||
| //             SearchError::FilterParsing(s) => ResponseError::FilterParsing(s), | ||||
| //             _ => ResponseError::Internal(err), | ||||
| //         } | ||||
| //     } | ||||
| // } | ||||
|  | ||||
| impl From<meilisearch_core::settings::RankingRuleConversionError> for ResponseError { | ||||
|     fn from(err: meilisearch_core::settings::RankingRuleConversionError) -> ResponseError { | ||||
|         ResponseError::internal(err) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub trait IntoInternalError<T> { | ||||
|     fn into_internal_error(self) -> SResult<T>; | ||||
| } | ||||
|  | ||||
| impl<T> IntoInternalError<T> for Option<T> { | ||||
|     fn into_internal_error(self) -> SResult<T> { | ||||
|         match self { | ||||
|             Some(value) => Ok(value), | ||||
|             None => Err(ResponseError::internal("Heed cannot find requested value")), | ||||
|         } | ||||
|         ResponseError::Internal(err.to_string()) | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,2 +1,2 @@ | ||||
| pub mod meilisearch; | ||||
| pub mod tide; | ||||
| // pub mod meilisearch; | ||||
| // pub mod tide; | ||||
|   | ||||
| @@ -1,16 +1,14 @@ | ||||
| use std::{env, thread}; | ||||
|  | ||||
| use async_std::task; | ||||
| use log::info; | ||||
| use main_error::MainError; | ||||
| use structopt::StructOpt; | ||||
| use tide::middleware::{Cors, RequestLogger, Origin}; | ||||
| use http::header::HeaderValue; | ||||
|  | ||||
| use actix_web::middleware::Logger; | ||||
| use actix_web::{post, web, App, HttpServer, HttpResponse, Responder}; | ||||
| use meilisearch_http::data::Data; | ||||
| use meilisearch_http::option::Opt; | ||||
| use meilisearch_http::routes; | ||||
| use meilisearch_http::routes::index::index_update_callback; | ||||
| // use meilisearch_http::routes::index::index_update_callback; | ||||
|  | ||||
| mod analytics; | ||||
|  | ||||
| @@ -18,8 +16,11 @@ mod analytics; | ||||
| #[global_allocator] | ||||
| static ALLOC: jemallocator::Jemalloc = jemallocator::Jemalloc; | ||||
|  | ||||
| pub fn main() -> Result<(), MainError> { | ||||
| #[tokio::main] | ||||
| async fn main() -> Result<(), MainError> { | ||||
|     let opt = Opt::from_args(); | ||||
|     let local = tokio::task::LocalSet::new(); | ||||
|     let _sys = actix_rt::System::run_in_tokio("server", &local); | ||||
|  | ||||
|     match opt.env.as_ref() { | ||||
|         "production" => { | ||||
| @@ -29,8 +30,7 @@ pub fn main() -> Result<(), MainError> { | ||||
|                         .into(), | ||||
|                 ); | ||||
|             } | ||||
|             env_logger::init(); | ||||
|         } | ||||
|         }, | ||||
|         "development" => { | ||||
|             env_logger::from_env(env_logger::Env::default().default_filter_or("info")).init(); | ||||
|         } | ||||
| @@ -45,22 +45,27 @@ pub fn main() -> Result<(), MainError> { | ||||
|  | ||||
|     let data_cloned = data.clone(); | ||||
|     data.db.set_update_callback(Box::new(move |name, status| { | ||||
|         index_update_callback(name, &data_cloned, status); | ||||
|         // index_update_callback(name, &data_cloned, status); | ||||
|     })); | ||||
|  | ||||
|     print_launch_resume(&opt, &data); | ||||
|  | ||||
|     let mut app = tide::with_state(data); | ||||
|     HttpServer::new(move || | ||||
|         App::new() | ||||
|             .wrap(Logger::default()) | ||||
|             .app_data(web::Data::new(data.clone())) | ||||
|             .service(routes::document::get_document) | ||||
|             .service(routes::document::delete_document) | ||||
|             .service(routes::document::get_all_documents) | ||||
|             .service(routes::document::add_documents) | ||||
|             .service(routes::document::update_documents) | ||||
|             .service(routes::document::delete_documents) | ||||
|             .service(routes::document::clear_all_documents) | ||||
|         ) | ||||
|         .bind(opt.http_addr)? | ||||
|         .run() | ||||
|         .await?; | ||||
|  | ||||
|     app.middleware(Cors::new() | ||||
|         .allow_methods(HeaderValue::from_static("GET, POST, PUT, DELETE, OPTIONS")) | ||||
|         .allow_headers(HeaderValue::from_static("X-Meili-API-Key")) | ||||
|         .allow_origin(Origin::from("*"))); | ||||
|     app.middleware(RequestLogger::new()); | ||||
|  | ||||
|     routes::load_routes(&mut app); | ||||
|  | ||||
|     task::block_on(app.listen(opt.http_addr))?; | ||||
|     Ok(()) | ||||
| } | ||||
|  | ||||
| @@ -76,7 +81,7 @@ pub fn print_launch_resume(opt: &Opt, data: &Data) { | ||||
| 888       888  "Y8888  888 888 888  "Y8888P"   "Y8888  "Y888888 888     "Y8888P 888  888 | ||||
| "#; | ||||
|  | ||||
|     println!("{}", ascii_name); | ||||
|     info!("{}", ascii_name); | ||||
|  | ||||
|     info!("Database path: {:?}", opt.db_path); | ||||
|     info!("Start server on: {:?}", opt.http_addr); | ||||
|   | ||||
| @@ -1,111 +1,104 @@ | ||||
| use std::collections::{BTreeSet, HashSet}; | ||||
|  | ||||
| use std::collections::BTreeSet; | ||||
| use indexmap::IndexMap; | ||||
| use serde::{Deserialize, Serialize}; | ||||
| use serde::Deserialize; | ||||
| use serde_json::Value; | ||||
| use tide::{Request, Response}; | ||||
| use actix_web::*; | ||||
|  | ||||
| use crate::error::{ResponseError, SResult}; | ||||
| use crate::helpers::tide::RequestExt; | ||||
| use crate::helpers::tide::ACL::*; | ||||
| use crate::error::ResponseError; | ||||
| use crate::Data; | ||||
| use crate::routes::IndexUpdateResponse; | ||||
|  | ||||
| pub async fn get_document(ctx: Request<Data>) -> SResult<Response> { | ||||
|     ctx.is_allowed(Public)?; | ||||
| type Document = IndexMap<String, Value>; | ||||
|  | ||||
|     let index = ctx.index()?; | ||||
| #[get("/indexes/{index_uid}/documents/{document_id}")] | ||||
| pub async fn get_document( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<(String, String)>, | ||||
| ) -> Result<HttpResponse> { | ||||
|     let index = data.db.open_index(&path.0) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.0.clone()))?; | ||||
|     let document_id = meilisearch_core::serde::compute_document_id(path.1.clone()); | ||||
|  | ||||
|     let original_document_id = ctx.document_id()?; | ||||
|     let document_id = meilisearch_core::serde::compute_document_id(original_document_id.clone()); | ||||
|     let reader = data.db.main_read_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let db = &ctx.state().db; | ||||
|     let reader = db.main_read_txn()?; | ||||
|     let response = index.document::<Document, String>(&reader, None, document_id) | ||||
|         .map_err(|_| ResponseError::DocumentNotFound(path.1.clone()))? | ||||
|         .ok_or(ResponseError::DocumentNotFound(path.1.clone()))?; | ||||
|  | ||||
|     let response = index | ||||
|         .document::<IndexMap<String, Value>>(&reader, None, document_id)? | ||||
|         .ok_or(ResponseError::document_not_found(&original_document_id))?; | ||||
|  | ||||
|     if response.is_empty() { | ||||
|         return Err(ResponseError::document_not_found(&original_document_id)); | ||||
|     } | ||||
|  | ||||
|     Ok(tide::Response::new(200).body_json(&response)?) | ||||
|     Ok(HttpResponse::Ok().json(response)) | ||||
| } | ||||
|  | ||||
| #[derive(Default, Serialize)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| pub struct IndexUpdateResponse { | ||||
|     pub update_id: u64, | ||||
| } | ||||
| #[delete("/indexes/{index_uid}/documents/{document_id}")] | ||||
| pub async fn delete_document( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<(String, String)>, | ||||
| ) -> Result<HttpResponse> { | ||||
|     let index = data.db.open_index(&path.0) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.0.clone()))?; | ||||
|     let document_id = meilisearch_core::serde::compute_document_id(path.1.clone()); | ||||
|  | ||||
| pub async fn delete_document(ctx: Request<Data>) -> SResult<Response> { | ||||
|     ctx.is_allowed(Private)?; | ||||
|     let mut update_writer = data.db.update_write_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let index = ctx.index()?; | ||||
|     let document_id = ctx.document_id()?; | ||||
|     let document_id = meilisearch_core::serde::compute_document_id(document_id); | ||||
|     let db = &ctx.state().db; | ||||
|     let mut update_writer = db.update_write_txn()?; | ||||
|     let mut documents_deletion = index.documents_deletion(); | ||||
|     documents_deletion.delete_document_by_id(document_id); | ||||
|     let update_id = documents_deletion.finalize(&mut update_writer)?; | ||||
|  | ||||
|     update_writer.commit()?; | ||||
|     let update_id = documents_deletion.finalize(&mut update_writer) | ||||
|         .map_err(|_| ResponseError::Internal(path.1.clone()))?; | ||||
|  | ||||
|     let response_body = IndexUpdateResponse { update_id }; | ||||
|     Ok(tide::Response::new(202).body_json(&response_body)?) | ||||
|     update_writer.commit() | ||||
|         .map_err(|_| ResponseError::CommitTransaction)?; | ||||
|  | ||||
|     Ok(HttpResponse::Accepted().json(IndexUpdateResponse::with_id(update_id))) | ||||
| } | ||||
|  | ||||
|  | ||||
| #[derive(Default, Deserialize)] | ||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||
| struct BrowseQuery { | ||||
| pub struct BrowseQuery { | ||||
|     offset: Option<usize>, | ||||
|     limit: Option<usize>, | ||||
|     attributes_to_retrieve: Option<String>, | ||||
| } | ||||
|  | ||||
| pub async fn get_all_documents(ctx: Request<Data>) -> SResult<Response> { | ||||
|     ctx.is_allowed(Private)?; | ||||
| #[get("/indexes/{index_uid}/documents")] | ||||
| pub async fn get_all_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
|     params: web::Query<BrowseQuery>, | ||||
| ) -> Result<HttpResponse> { | ||||
|  | ||||
|     let index = ctx.index()?; | ||||
|     let query: BrowseQuery = ctx.query().unwrap_or_default(); | ||||
|     let index = data.db.open_index(path.clone()) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.clone()))?; | ||||
|  | ||||
|     let offset = query.offset.unwrap_or(0); | ||||
|     let limit = query.limit.unwrap_or(20); | ||||
|     let offset = params.offset.unwrap_or(0); | ||||
|     let limit = params.limit.unwrap_or(20); | ||||
|  | ||||
|     let db = &ctx.state().db; | ||||
|     let reader = db.main_read_txn()?; | ||||
|     let reader = data.db.main_read_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let documents_ids: Result<BTreeSet<_>, _> = index | ||||
|         .documents_fields_counts | ||||
|         .documents_ids(&reader)? | ||||
|         .documents_ids(&reader) | ||||
|         .map_err(|_| ResponseError::Internal(path.clone()))? | ||||
|         .skip(offset) | ||||
|         .take(limit) | ||||
|         .collect(); | ||||
|  | ||||
|     let documents_ids = match documents_ids { | ||||
|         Ok(documents_ids) => documents_ids, | ||||
|         Err(e) => return Err(ResponseError::internal(e)), | ||||
|     }; | ||||
|     let documents_ids = documents_ids.map_err(|err| ResponseError::Internal(err.to_string()))?; | ||||
|  | ||||
|     let attributes = params.attributes_to_retrieve.clone() | ||||
|         .map(|a| a.split(',').map(|a| a.to_string()).collect()); | ||||
|  | ||||
|     let mut response_body = Vec::<IndexMap<String, Value>>::new(); | ||||
|  | ||||
|     if let Some(attributes) = query.attributes_to_retrieve { | ||||
|         let attributes = attributes.split(',').collect::<HashSet<&str>>(); | ||||
|         for document_id in documents_ids { | ||||
|             if let Ok(Some(document)) = index.document(&reader, Some(&attributes), document_id) { | ||||
|                 response_body.push(document); | ||||
|             } | ||||
|         } | ||||
|     } else { | ||||
|         for document_id in documents_ids { | ||||
|             if let Ok(Some(document)) = index.document(&reader, None, document_id) { | ||||
|                 response_body.push(document); | ||||
|             } | ||||
|     for document_id in documents_ids { | ||||
|         if let Ok(Some(document)) = index.document(&reader, attributes.clone(), document_id) { | ||||
|             response_body.push(document); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     Ok(tide::Response::new(200).body_json(&response_body)?) | ||||
|     Ok(HttpResponse::Ok().json(response_body)) | ||||
| } | ||||
|  | ||||
| fn find_primary_key(document: &IndexMap<String, Value>) -> Option<String> { | ||||
| @@ -119,40 +112,49 @@ fn find_primary_key(document: &IndexMap<String, Value>) -> Option<String> { | ||||
|  | ||||
| #[derive(Default, Deserialize)] | ||||
| #[serde(rename_all = "camelCase", deny_unknown_fields)] | ||||
| struct UpdateDocumentsQuery { | ||||
| pub struct UpdateDocumentsQuery { | ||||
|     primary_key: Option<String>, | ||||
| } | ||||
|  | ||||
| async fn update_multiple_documents(mut ctx: Request<Data>, is_partial: bool) -> SResult<Response> { | ||||
|     ctx.is_allowed(Private)?; | ||||
| async fn update_multiple_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
|     params: web::Query<UpdateDocumentsQuery>, | ||||
|     body: web::Json<Vec<Document>>, | ||||
|     is_partial: bool | ||||
| ) -> Result<HttpResponse> { | ||||
|  | ||||
|     let index = ctx.index()?; | ||||
|     let index = data.db.open_index(path.clone()) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.clone()))?; | ||||
|  | ||||
|     let data: Vec<IndexMap<String, Value>> = | ||||
|         ctx.body_json().await.map_err(ResponseError::bad_request)?; | ||||
|     let query: UpdateDocumentsQuery = ctx.query().unwrap_or_default(); | ||||
|     let reader = data.db.main_read_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let db = &ctx.state().db; | ||||
|  | ||||
|     let reader = db.main_read_txn()?; | ||||
|     let mut schema = index | ||||
|         .main | ||||
|         .schema(&reader)? | ||||
|         .ok_or(ResponseError::internal("schema not found"))?; | ||||
|         .schema(&reader) | ||||
|         .map_err(|_| ResponseError::Schema)? | ||||
|         .ok_or(ResponseError::Schema)?; | ||||
|  | ||||
|     if schema.primary_key().is_none() { | ||||
|         let id = match query.primary_key { | ||||
|         let id = match params.primary_key.clone() { | ||||
|             Some(id) => id, | ||||
|             None => match data.first().and_then(|docs| find_primary_key(docs)) { | ||||
|                 Some(id) => id, | ||||
|                 None => return Err(ResponseError::bad_request("Could not infer a primary key")), | ||||
|             }, | ||||
|             None => { | ||||
|                 body.first() | ||||
|                     .and_then(|docs| find_primary_key(docs)) | ||||
|                     .ok_or(ResponseError::InferPrimaryKey)? | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         let mut writer = db.main_write_txn()?; | ||||
|         schema.set_primary_key(&id).map_err(ResponseError::bad_request)?; | ||||
|         index.main.put_schema(&mut writer, &schema)?; | ||||
|         writer.commit()?; | ||||
|         let mut writer = data.db.main_write_txn() | ||||
|             .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|         schema.set_primary_key(&id) | ||||
|             .map_err(|e| ResponseError::Internal(e.to_string()))?; | ||||
|         index.main.put_schema(&mut writer, &schema) | ||||
|             .map_err(|e| ResponseError::Internal(e.to_string()))?; | ||||
|         writer.commit() | ||||
|             .map_err(|_| ResponseError::CommitTransaction)?; | ||||
|     } | ||||
|  | ||||
|     let mut document_addition = if is_partial { | ||||
| @@ -161,63 +163,88 @@ async fn update_multiple_documents(mut ctx: Request<Data>, is_partial: bool) -> | ||||
|         index.documents_addition() | ||||
|     }; | ||||
|  | ||||
|     for document in data { | ||||
|     for document in body.into_inner() { | ||||
|         document_addition.update_document(document); | ||||
|     } | ||||
|  | ||||
|     let mut update_writer = db.update_write_txn()?; | ||||
|     let update_id = document_addition.finalize(&mut update_writer)?; | ||||
|     update_writer.commit()?; | ||||
|     let mut update_writer = data.db.update_write_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|     let update_id = document_addition.finalize(&mut update_writer) | ||||
|         .map_err(|e| ResponseError::Internal(e.to_string()))?; | ||||
|     update_writer.commit() | ||||
|         .map_err(|_| ResponseError::CommitTransaction)?; | ||||
|  | ||||
|     let response_body = IndexUpdateResponse { update_id }; | ||||
|     Ok(tide::Response::new(202).body_json(&response_body)?) | ||||
|     Ok(HttpResponse::Accepted().json(IndexUpdateResponse::with_id(update_id))) | ||||
| } | ||||
|  | ||||
| pub async fn add_or_replace_multiple_documents(ctx: Request<Data>) -> SResult<Response> { | ||||
|     update_multiple_documents(ctx, false).await | ||||
| #[post("/indexes/{index_uid}/documents")] | ||||
| pub async fn add_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
|     params: web::Query<UpdateDocumentsQuery>, | ||||
|     body: web::Json<Vec<Document>> | ||||
| ) -> Result<HttpResponse> { | ||||
|     update_multiple_documents(data, path, params, body, false).await | ||||
| } | ||||
|  | ||||
| pub async fn add_or_update_multiple_documents(ctx: Request<Data>) -> SResult<Response> { | ||||
|     update_multiple_documents(ctx, true).await | ||||
| #[put("/indexes/{index_uid}/documents")] | ||||
| pub async fn update_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
|     params: web::Query<UpdateDocumentsQuery>, | ||||
|     body: web::Json<Vec<Document>> | ||||
| ) -> Result<HttpResponse> { | ||||
|     update_multiple_documents(data, path, params, body, true).await | ||||
| } | ||||
|  | ||||
| pub async fn delete_multiple_documents(mut ctx: Request<Data>) -> SResult<Response> { | ||||
|     ctx.is_allowed(Private)?; | ||||
| #[post("/indexes/{index_uid}/documents/delete-batch")] | ||||
| pub async fn delete_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
|     body: web::Json<Vec<Value>> | ||||
| ) -> Result<HttpResponse> { | ||||
|  | ||||
|     let data: Vec<Value> = ctx.body_json().await.map_err(ResponseError::bad_request)?; | ||||
|     let index = ctx.index()?; | ||||
|     let index = data.db.open_index(path.clone()) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.clone()))?; | ||||
|  | ||||
|     let db = &ctx.state().db; | ||||
|     let mut writer = db.update_write_txn()?; | ||||
|     let mut writer = data.db.update_write_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let mut documents_deletion = index.documents_deletion(); | ||||
|  | ||||
|     for document_id in data { | ||||
|     for document_id in body.into_inner() { | ||||
|         if let Some(document_id) = meilisearch_core::serde::value_to_string(&document_id) { | ||||
|             documents_deletion | ||||
|                 .delete_document_by_id(meilisearch_core::serde::compute_document_id(document_id)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     let update_id = documents_deletion.finalize(&mut writer)?; | ||||
|     let update_id = documents_deletion.finalize(&mut writer) | ||||
|         .map_err(|e| ResponseError::Internal(e.to_string()))?; | ||||
|  | ||||
|     writer.commit()?; | ||||
|     writer.commit() | ||||
|         .map_err(|_| ResponseError::CommitTransaction)?; | ||||
|  | ||||
|     let response_body = IndexUpdateResponse { update_id }; | ||||
|     Ok(tide::Response::new(202).body_json(&response_body)?) | ||||
|     Ok(HttpResponse::Accepted().json(IndexUpdateResponse::with_id(update_id))) | ||||
| } | ||||
|  | ||||
| pub async fn clear_all_documents(ctx: Request<Data>) -> SResult<Response> { | ||||
|     ctx.is_allowed(Private)?; | ||||
| #[delete("/indexes/{index_uid}/documents")] | ||||
| pub async fn clear_all_documents( | ||||
|     data: web::Data<Data>, | ||||
|     path: web::Path<String>, | ||||
| ) -> Result<HttpResponse> { | ||||
|  | ||||
|     let index = ctx.index()?; | ||||
|     let index = data.db.open_index(path.clone()) | ||||
|         .ok_or(ResponseError::IndexNotFound(path.clone()))?; | ||||
|  | ||||
|     let db = &ctx.state().db; | ||||
|     let mut writer = db.update_write_txn()?; | ||||
|     let mut writer = data.db.update_write_txn() | ||||
|         .map_err(|_| ResponseError::CreateTransaction)?; | ||||
|  | ||||
|     let update_id = index.clear_all(&mut writer)?; | ||||
|     writer.commit()?; | ||||
|     let update_id = index.clear_all(&mut writer) | ||||
|         .map_err(|e| ResponseError::Internal(e.to_string()))?; | ||||
|  | ||||
|     let response_body = IndexUpdateResponse { update_id }; | ||||
|     Ok(tide::Response::new(202).body_json(&response_body)?) | ||||
|     writer.commit() | ||||
|         .map_err(|_| ResponseError::CommitTransaction)?; | ||||
|  | ||||
|     Ok(HttpResponse::Accepted().json(IndexUpdateResponse::with_id(update_id))) | ||||
| } | ||||
|   | ||||
| @@ -1,130 +1,135 @@ | ||||
| use crate::data::Data; | ||||
| use std::future::Future; | ||||
| use tide::IntoResponse; | ||||
| use tide::Response; | ||||
|  | ||||
| use serde::Serialize; | ||||
|  | ||||
| pub mod document; | ||||
| pub mod health; | ||||
| pub mod index; | ||||
| pub mod key; | ||||
| pub mod search; | ||||
| pub mod setting; | ||||
| pub mod stats; | ||||
| pub mod stop_words; | ||||
| pub mod synonym; | ||||
| // pub mod health; | ||||
| // pub mod index; | ||||
| // pub mod key; | ||||
| // pub mod search; | ||||
| // pub mod setting; | ||||
| // pub mod stats; | ||||
| // pub mod stop_words; | ||||
| // pub mod synonym; | ||||
|  | ||||
| async fn into_response<T: IntoResponse, U: IntoResponse>( | ||||
|     x: impl Future<Output = Result<T, U>>, | ||||
| ) -> Response { | ||||
|     match x.await { | ||||
|         Ok(resp) => resp.into_response(), | ||||
|         Err(resp) => resp.into_response(), | ||||
| #[derive(Default, Serialize)] | ||||
| #[serde(rename_all = "camelCase")] | ||||
| pub struct IndexUpdateResponse { | ||||
|     pub update_id: u64, | ||||
|     pub see_more: String, | ||||
| } | ||||
|  | ||||
| impl IndexUpdateResponse { | ||||
|     pub fn with_id(update_id: u64) -> Self { | ||||
|         Self { | ||||
|             update_id, | ||||
|             see_more: "https://docs.meilisearch.com/guides/advanced_guides/asynchronous_updates.html".to_string() | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn load_routes(app: &mut tide::Server<Data>) { | ||||
|     app.at("/").get(|_| async { | ||||
|         tide::Response::new(200) | ||||
|             .body_string(include_str!("../../public/interface.html").to_string()) | ||||
|             .set_mime(mime::TEXT_HTML_UTF_8) | ||||
|     }); | ||||
|     app.at("/bulma.min.css").get(|_| async { | ||||
|         tide::Response::new(200) | ||||
|             .body_string(include_str!("../../public/bulma.min.css").to_string()) | ||||
|             .set_mime(mime::TEXT_CSS_UTF_8) | ||||
|     }); | ||||
| // pub fn load_routes(app: &mut tide::Server<Data>) { | ||||
| //     app.at("/").get(|_| async { | ||||
| //         tide::Response::new(200) | ||||
| //             .body_string(include_str!("../../public/interface.html").to_string()) | ||||
| //             .set_mime(mime::TEXT_HTML_UTF_8) | ||||
| //     }); | ||||
| //     app.at("/bulma.min.css").get(|_| async { | ||||
| //         tide::Response::new(200) | ||||
| //             .body_string(include_str!("../../public/bulma.min.css").to_string()) | ||||
| //             .set_mime(mime::TEXT_CSS_UTF_8) | ||||
| //     }); | ||||
|  | ||||
|     app.at("/indexes") | ||||
|         .get(|ctx| into_response(index::list_indexes(ctx))) | ||||
|         .post(|ctx| into_response(index::create_index(ctx))); | ||||
| //     app.at("/indexes") | ||||
| //         .get(|ctx| into_response(index::list_indexes(ctx))) | ||||
| //         .post(|ctx| into_response(index::create_index(ctx))); | ||||
|  | ||||
|     app.at("/indexes/search") | ||||
|         .post(|ctx| into_response(search::search_multi_index(ctx))); | ||||
| //     app.at("/indexes/search") | ||||
| //         .post(|ctx| into_response(search::search_multi_index(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index") | ||||
|         .get(|ctx| into_response(index::get_index(ctx))) | ||||
|         .put(|ctx| into_response(index::update_index(ctx))) | ||||
|         .delete(|ctx| into_response(index::delete_index(ctx))); | ||||
| //     app.at("/indexes/:index") | ||||
| //         .get(|ctx| into_response(index::get_index(ctx))) | ||||
| //         .put(|ctx| into_response(index::update_index(ctx))) | ||||
| //         .delete(|ctx| into_response(index::delete_index(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/search") | ||||
|         .get(|ctx| into_response(search::search_with_url_query(ctx))); | ||||
| //     app.at("/indexes/:index/search") | ||||
| //         .get(|ctx| into_response(search::search_with_url_query(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/updates") | ||||
|         .get(|ctx| into_response(index::get_all_updates_status(ctx))); | ||||
| //     app.at("/indexes/:index/updates") | ||||
| //         .get(|ctx| into_response(index::get_all_updates_status(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/updates/:update_id") | ||||
|         .get(|ctx| into_response(index::get_update_status(ctx))); | ||||
| //     app.at("/indexes/:index/updates/:update_id") | ||||
| //         .get(|ctx| into_response(index::get_update_status(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/documents") | ||||
|         .get(|ctx| into_response(document::get_all_documents(ctx))) | ||||
|         .post(|ctx| into_response(document::add_or_replace_multiple_documents(ctx))) | ||||
|         .put(|ctx| into_response(document::add_or_update_multiple_documents(ctx))) | ||||
|         .delete(|ctx| into_response(document::clear_all_documents(ctx))); | ||||
| //     app.at("/indexes/:index/documents") | ||||
| //         .get(|ctx| into_response(document::get_all_documents(ctx))) | ||||
| //         .post(|ctx| into_response(document::add_or_replace_multiple_documents(ctx))) | ||||
| //         .put(|ctx| into_response(document::add_or_update_multiple_documents(ctx))) | ||||
| //         .delete(|ctx| into_response(document::clear_all_documents(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/documents/:document_id") | ||||
|         .get(|ctx| into_response(document::get_document(ctx))) | ||||
|         .delete(|ctx| into_response(document::delete_document(ctx))); | ||||
| //     app.at("/indexes/:index/documents/:document_id") | ||||
| //         .get(|ctx| into_response(document::get_document(ctx))) | ||||
| //         .delete(|ctx| into_response(document::delete_document(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/documents/delete-batch") | ||||
|         .post(|ctx| into_response(document::delete_multiple_documents(ctx))); | ||||
| //     app.at("/indexes/:index/documents/delete-batch") | ||||
| //         .post(|ctx| into_response(document::delete_multiple_documents(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings") | ||||
|         .get(|ctx| into_response(setting::get_all(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_all(ctx))) | ||||
|         .delete(|ctx| into_response(setting::delete_all(ctx))); | ||||
| //     app.at("/indexes/:index/settings") | ||||
| //         .get(|ctx| into_response(setting::get_all(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_all(ctx))) | ||||
| //         .delete(|ctx| into_response(setting::delete_all(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/ranking-rules") | ||||
|         .get(|ctx| into_response(setting::get_rules(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_rules(ctx))) | ||||
|         .delete(|ctx| into_response(setting::delete_rules(ctx))); | ||||
| //     app.at("/indexes/:index/settings/ranking-rules") | ||||
| //         .get(|ctx| into_response(setting::get_rules(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_rules(ctx))) | ||||
| //         .delete(|ctx| into_response(setting::delete_rules(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/distinct-attribute") | ||||
|         .get(|ctx| into_response(setting::get_distinct(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_distinct(ctx))) | ||||
|         .delete(|ctx| into_response(setting::delete_distinct(ctx))); | ||||
| //     app.at("/indexes/:index/settings/distinct-attribute") | ||||
| //         .get(|ctx| into_response(setting::get_distinct(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_distinct(ctx))) | ||||
| //         .delete(|ctx| into_response(setting::delete_distinct(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/searchable-attributes") | ||||
|         .get(|ctx| into_response(setting::get_searchable(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_searchable(ctx))) | ||||
|         .delete(|ctx| into_response(setting::delete_searchable(ctx))); | ||||
| //     app.at("/indexes/:index/settings/searchable-attributes") | ||||
| //         .get(|ctx| into_response(setting::get_searchable(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_searchable(ctx))) | ||||
| //         .delete(|ctx| into_response(setting::delete_searchable(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/displayed-attributes") | ||||
|         .get(|ctx| into_response(setting::displayed(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_displayed(ctx))) | ||||
|         .delete(|ctx| into_response(setting::delete_displayed(ctx))); | ||||
| //     app.at("/indexes/:index/settings/displayed-attributes") | ||||
| //         .get(|ctx| into_response(setting::displayed(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_displayed(ctx))) | ||||
| //         .delete(|ctx| into_response(setting::delete_displayed(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/accept-new-fields") | ||||
|         .get(|ctx| into_response(setting::get_accept_new_fields(ctx))) | ||||
|         .post(|ctx| into_response(setting::update_accept_new_fields(ctx))); | ||||
| //     app.at("/indexes/:index/settings/accept-new-fields") | ||||
| //         .get(|ctx| into_response(setting::get_accept_new_fields(ctx))) | ||||
| //         .post(|ctx| into_response(setting::update_accept_new_fields(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/synonyms") | ||||
|         .get(|ctx| into_response(synonym::get(ctx))) | ||||
|         .post(|ctx| into_response(synonym::update(ctx))) | ||||
|         .delete(|ctx| into_response(synonym::delete(ctx))); | ||||
| //     app.at("/indexes/:index/settings/synonyms") | ||||
| //         .get(|ctx| into_response(synonym::get(ctx))) | ||||
| //         .post(|ctx| into_response(synonym::update(ctx))) | ||||
| //         .delete(|ctx| into_response(synonym::delete(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/settings/stop-words") | ||||
|         .get(|ctx| into_response(stop_words::get(ctx))) | ||||
|         .post(|ctx| into_response(stop_words::update(ctx))) | ||||
|         .delete(|ctx| into_response(stop_words::delete(ctx))); | ||||
| //     app.at("/indexes/:index/settings/stop-words") | ||||
| //         .get(|ctx| into_response(stop_words::get(ctx))) | ||||
| //         .post(|ctx| into_response(stop_words::update(ctx))) | ||||
| //         .delete(|ctx| into_response(stop_words::delete(ctx))); | ||||
|  | ||||
|     app.at("/indexes/:index/stats") | ||||
|         .get(|ctx| into_response(stats::index_stats(ctx))); | ||||
| //     app.at("/indexes/:index/stats") | ||||
| //         .get(|ctx| into_response(stats::index_stats(ctx))); | ||||
|  | ||||
|     app.at("/keys").get(|ctx| into_response(key::list(ctx))); | ||||
| //     app.at("/keys").get(|ctx| into_response(key::list(ctx))); | ||||
|  | ||||
|     app.at("/health") | ||||
|         .get(|ctx| into_response(health::get_health(ctx))) | ||||
|         .put(|ctx| into_response(health::change_healthyness(ctx))); | ||||
| //     app.at("/health") | ||||
| //         .get(|ctx| into_response(health::get_health(ctx))) | ||||
| //         .put(|ctx| into_response(health::change_healthyness(ctx))); | ||||
|  | ||||
|     app.at("/stats") | ||||
|         .get(|ctx| into_response(stats::get_stats(ctx))); | ||||
| //     app.at("/stats") | ||||
| //         .get(|ctx| into_response(stats::get_stats(ctx))); | ||||
|  | ||||
|     app.at("/version") | ||||
|         .get(|ctx| into_response(stats::get_version(ctx))); | ||||
| //     app.at("/version") | ||||
| //         .get(|ctx| into_response(stats::get_version(ctx))); | ||||
|  | ||||
|     app.at("/sys-info") | ||||
|         .get(|ctx| into_response(stats::get_sys_info(ctx))); | ||||
| //     app.at("/sys-info") | ||||
| //         .get(|ctx| into_response(stats::get_sys_info(ctx))); | ||||
|  | ||||
|     app.at("/sys-info/pretty") | ||||
|         .get(|ctx| into_response(stats::get_sys_info_pretty(ctx))); | ||||
| } | ||||
| //     app.at("/sys-info/pretty") | ||||
| //         .get(|ctx| into_response(stats::get_sys_info_pretty(ctx))); | ||||
| // } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user