mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	2745: Config file support r=curquiza a=mlemesle # Pull Request ## What does this PR do? Fixes #2558 ## PR checklist Please check if your PR fulfills the following requirements: - [x] Does this PR fix an existing issue? - [x] Have you read the contributing guidelines? - [x] Have you made sure that the title is accurate and descriptive of the changes? Thank you so much for contributing to Meilisearch! 2789: Fix typos r=Kerollmops a=kianmeng # Pull Request ## What does this PR do? Found via `codespell -L crate,nam,hart`. ## PR checklist Please check if your PR fulfills the following requirements: - [ ] Does this PR fix an existing issue? - [x] Have you read the contributing guidelines? - [x] Have you made sure that the title is accurate and descriptive of the changes? Thank you so much for contributing to Meilisearch! 2814: Skip dashboard test if mini-dashboard feature is disabled r=Kerollmops a=jirutka Fixes #2813 Fixes the following error: cargo test --no-default-features ... error: couldn't read target/debug/build/meilisearch-http-ec029d8c902cf2cb/out/generated.rs: No such file or directory (os error 2) --> meilisearch-http/tests/dashboard/mod.rs:8:9 | 8 | include!(concat!(env!("OUT_DIR"), "/generated.rs")); | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | = note: this error originates in the macro `include` (in Nightly builds, run with -Z macro-backtrace for more info) error: could not compile `meilisearch-http` due to previous error 2826: Rename receivedDocumentIds into matchedDocuments r=Kerollmops a=Ugzuzg # Pull Request ## What does this PR do? Fixes #2799 Changes DocumentDeletion task details response. ## PR checklist Please check if your PR fulfills the following requirements: - [x] Does this PR fix an existing issue? - [x] Have you read the contributing guidelines? - [x] Have you made sure that the title is accurate and descriptive of the changes? Tested with curl: ``` curl \ -X POST 'http://localhost:7700/indexes/movies/documents/delete-batch' \ -H 'Content-Type: application/json' \ --data-binary '[ 23488, 153738, 437035, 363869 ]' {"taskUid":1,"indexUid":"movies","status":"enqueued","type":"documentDeletion","enqueuedAt":"2022-10-01T20:06:37.105416054Z"}% curl \ -X GET 'http://localhost:7700/tasks/1' {"uid":1,"indexUid":"movies","status":"succeeded","type":"documentDeletion","details":{"matchedDocuments":4,"deletedDocuments":2},"duration":"PT0.005708322S","enqueuedAt":"2022-10-01T20:06:37.105416054Z","startedAt":"2022-10-01T20:06:37.115562733Z","finishedAt":"2022-10-01T20:06:37.121271055Z"} ``` Co-authored-by: mlemesle <lemesle.martin@hotmail.fr> Co-authored-by: Kian-Meng Ang <kianmeng@cpan.org> Co-authored-by: Jakub Jirutka <jakub@jirutka.cz> Co-authored-by: Jarasłaŭ Viktorčyk <ugzuzg@gmail.com>
This commit is contained in:
		
							
								
								
									
										2
									
								
								.github/scripts/is-latest-release.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/scripts/is-latest-release.sh
									
									
									
									
										vendored
									
									
								
							| @@ -85,7 +85,7 @@ get_latest() { | |||||||
|     latest="" |     latest="" | ||||||
|     current_tag="" |     current_tag="" | ||||||
|     for release_info in $releases; do |     for release_info in $releases; do | ||||||
|         if [ $i -eq 0 ]; then # Cheking tag_name |         if [ $i -eq 0 ]; then # Checking tag_name | ||||||
|             if echo "$release_info" | grep -q "$GREP_SEMVER_REGEXP"; then # If it's not an alpha or beta release |             if echo "$release_info" | grep -q "$GREP_SEMVER_REGEXP"; then # If it's not an alpha or beta release | ||||||
|                 current_tag=$release_info |                 current_tag=$release_info | ||||||
|             else |             else | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								.github/workflows/milestone-workflow.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/milestone-workflow.yml
									
									
									
									
										vendored
									
									
								
							| @@ -62,12 +62,12 @@ jobs: | |||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|       - name: Download the issue template |       - name: Download the issue template | ||||||
|         run: curl -s https://raw.githubusercontent.com/meilisearch/core-team/main/issue-templates/roadmap-issue.md > $ISSUE_TEMPLATE |         run: curl -s https://raw.githubusercontent.com/meilisearch/core-team/main/issue-templates/roadmap-issue.md > $ISSUE_TEMPLATE | ||||||
|       - name: Replace all empty occurences in the templates |       - name: Replace all empty occurrences in the templates | ||||||
|         run: | |         run: | | ||||||
|           # Replace all <<version>> occurences |           # Replace all <<version>> occurrences | ||||||
|           sed -i "s/<<version>>/$MILESTONE_VERSION/g" $ISSUE_TEMPLATE |           sed -i "s/<<version>>/$MILESTONE_VERSION/g" $ISSUE_TEMPLATE | ||||||
|  |  | ||||||
|           # Replace all <<milestone_id>> occurences |           # Replace all <<milestone_id>> occurrences | ||||||
|           milestone_id=$(echo $MILESTONE_URL | cut -d '/' -f 7) |           milestone_id=$(echo $MILESTONE_URL | cut -d '/' -f 7) | ||||||
|           sed -i "s/<<milestone_id>>/$milestone_id/g" $ISSUE_TEMPLATE |           sed -i "s/<<milestone_id>>/$milestone_id/g" $ISSUE_TEMPLATE | ||||||
|  |  | ||||||
| @@ -95,12 +95,12 @@ jobs: | |||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|       - name: Download the issue template |       - name: Download the issue template | ||||||
|         run: curl -s https://raw.githubusercontent.com/meilisearch/core-team/main/issue-templates/changelog-issue.md > $ISSUE_TEMPLATE |         run: curl -s https://raw.githubusercontent.com/meilisearch/core-team/main/issue-templates/changelog-issue.md > $ISSUE_TEMPLATE | ||||||
|       - name: Replace all empty occurences in the templates |       - name: Replace all empty occurrences in the templates | ||||||
|         run: | |         run: | | ||||||
|           # Replace all <<version>> occurences |           # Replace all <<version>> occurrences | ||||||
|           sed -i "s/<<version>>/$MILESTONE_VERSION/g" $ISSUE_TEMPLATE |           sed -i "s/<<version>>/$MILESTONE_VERSION/g" $ISSUE_TEMPLATE | ||||||
|  |  | ||||||
|           # Replace all <<milestone_id>> occurences |           # Replace all <<milestone_id>> occurrences | ||||||
|           milestone_id=$(echo $MILESTONE_URL | cut -d '/' -f 7) |           milestone_id=$(echo $MILESTONE_URL | cut -d '/' -f 7) | ||||||
|           sed -i "s/<<milestone_id>>/$milestone_id/g" $ISSUE_TEMPLATE |           sed -i "s/<<milestone_id>>/$milestone_id/g" $ISSUE_TEMPLATE | ||||||
|       - name: Create the issue |       - name: Create the issue | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/publish-docker-images.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/publish-docker-images.yml
									
									
									
									
										vendored
									
									
								
							| @@ -53,7 +53,7 @@ jobs: | |||||||
|         uses: docker/metadata-action@v4 |         uses: docker/metadata-action@v4 | ||||||
|         with: |         with: | ||||||
|           images: getmeili/meilisearch |           images: getmeili/meilisearch | ||||||
|           # The lastest and `vX.Y` tags are only pushed for the official Meilisearch releases |           # The latest and `vX.Y` tags are only pushed for the official Meilisearch releases | ||||||
|           # See https://github.com/docker/metadata-action#latest-tag |           # See https://github.com/docker/metadata-action#latest-tag | ||||||
|           flavor: latest=false |           flavor: latest=false | ||||||
|           tags: | |           tags: | | ||||||
|   | |||||||
| @@ -112,7 +112,7 @@ The full Meilisearch release process is described in [this guide](https://github | |||||||
| ### Release assets | ### Release assets | ||||||
|  |  | ||||||
| For each release, the following assets are created: | For each release, the following assets are created: | ||||||
| - Binaries for differents platforms (Linux, MacOS, Windows and ARM architectures) are attached to the GitHub release | - Binaries for different platforms (Linux, MacOS, Windows and ARM architectures) are attached to the GitHub release | ||||||
| - Binaries are pushed to HomeBrew and APT (not published for RC) | - Binaries are pushed to HomeBrew and APT (not published for RC) | ||||||
| - Docker tags are created/updated: | - Docker tags are created/updated: | ||||||
|   - `vX.Y.Z` |   - `vX.Y.Z` | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -2087,6 +2087,7 @@ dependencies = [ | |||||||
|  "time 0.3.9", |  "time 0.3.9", | ||||||
|  "tokio", |  "tokio", | ||||||
|  "tokio-stream", |  "tokio-stream", | ||||||
|  |  "toml", | ||||||
|  "urlencoding", |  "urlencoding", | ||||||
|  "uuid", |  "uuid", | ||||||
|  "vergen", |  "vergen", | ||||||
|   | |||||||
							
								
								
									
										129
									
								
								config.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								config.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | |||||||
|  | # This file shows the default configuration of Meilisearch. | ||||||
|  | # All variables are defined here https://docs.meilisearch.com/learn/configuration/instance_options.html#environment-variables | ||||||
|  |  | ||||||
|  | db_path = "./data.ms" | ||||||
|  | # The destination where the database must be created. | ||||||
|  |  | ||||||
|  | env = "development" # Possible values: [development, production] | ||||||
|  | # This environment variable must be set to `production` if you are running in production. | ||||||
|  | # More logs wiil be displayed if the server is running in development mode. Setting the master | ||||||
|  | # key is optional; hence no security on the updates routes. This | ||||||
|  | # is useful to debug when integrating the engine with another service. | ||||||
|  |  | ||||||
|  | http_addr = "127.0.0.1:7700" | ||||||
|  | # The address on which the HTTP server will listen. | ||||||
|  |  | ||||||
|  | # master_key = "MASTER_KEY" | ||||||
|  | # Sets the instance's master key, automatically protecting all routes except GET /health. | ||||||
|  |  | ||||||
|  | # no_analytics = false | ||||||
|  | # Do not send analytics to Meilisearch. | ||||||
|  |  | ||||||
|  | disable_auto_batching = false | ||||||
|  | # The engine will disable task auto-batching, and will sequencialy compute each task one by one. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### DUMP | ||||||
|  |  | ||||||
|  | dumps_dir = "dumps/" | ||||||
|  | # Folder where dumps are created when the dump route is called. | ||||||
|  |  | ||||||
|  | # import_dump = "./path/to/my/file.dump" | ||||||
|  | # Import a dump from the specified path, must be a `.dump` file. | ||||||
|  |  | ||||||
|  | ignore_missing_dump = false | ||||||
|  | # If the dump doesn't exist, load or create the database specified by `db_path` instead. | ||||||
|  |  | ||||||
|  | ignore_dump_if_db_exists = false | ||||||
|  | # Ignore the dump if a database already exists, and load that database instead. | ||||||
|  |  | ||||||
|  | ### | ||||||
|  |  | ||||||
|  |  | ||||||
|  | log_level = "INFO" # Possible values: [ERROR, WARN, INFO, DEBUG, TRACE] | ||||||
|  | # Set the log level. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### INDEX | ||||||
|  |  | ||||||
|  | max_index_size = "100 GiB" | ||||||
|  | # The maximum size, in bytes, of the main LMDB database directory. | ||||||
|  |  | ||||||
|  | # max_indexing_memory = "2 GiB" | ||||||
|  | # The maximum amount of memory the indexer will use.  | ||||||
|  | # | ||||||
|  | # In case the engine is unable to retrieve the available memory the engine will try to use | ||||||
|  | # the memory it needs but without real limit, this can lead to Out-Of-Memory issues and it | ||||||
|  | # is recommended to specify the amount of memory to use. | ||||||
|  | # | ||||||
|  | # /!\ The default value is system dependant /!\ | ||||||
|  |  | ||||||
|  | # max_indexing_threads = 4 | ||||||
|  | # The maximum number of threads the indexer will use. If the number set is higher than the | ||||||
|  | # real number of cores available in the machine, it will use the maximum number of | ||||||
|  | # available cores. | ||||||
|  | # | ||||||
|  | # It defaults to half of the available threads. | ||||||
|  |  | ||||||
|  | ### | ||||||
|  |  | ||||||
|  |  | ||||||
|  | max_task_db_size = "100 GiB" | ||||||
|  | # The maximum size, in bytes, of the update LMDB database directory. | ||||||
|  |  | ||||||
|  | http_payload_size_limit = "100 MB" | ||||||
|  | # The maximum size, in bytes, of accepted JSON payloads. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### SNAPSHOT | ||||||
|  |  | ||||||
|  | schedule_snapshot = false | ||||||
|  | # Activate snapshot scheduling. | ||||||
|  |  | ||||||
|  | snapshot_dir = "snapshots/" | ||||||
|  | # Defines the directory path where Meilisearch will create a snapshot each snapshot_interval_sec. | ||||||
|  |  | ||||||
|  | snapshot_interval_sec = 86400 | ||||||
|  | # Defines time interval, in seconds, between each snapshot creation. | ||||||
|  |  | ||||||
|  | # import_snapshot = "./path/to/my/snapshot" | ||||||
|  | # Defines the path of the snapshot file to import. This option will, by default, stop the | ||||||
|  | # process if a database already exists, or if no snapshot exists at the given path. If this | ||||||
|  | # option is not specified, no snapshot is imported. | ||||||
|  |  | ||||||
|  | ignore_missing_snapshot = false | ||||||
|  | # The engine will ignore a missing snapshot and not return an error in such a case. | ||||||
|  |  | ||||||
|  | ignore_snapshot_if_db_exists = false | ||||||
|  | # The engine will skip snapshot importation and not return an error in such a case. | ||||||
|  |  | ||||||
|  | ### | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### SSL | ||||||
|  |  | ||||||
|  | # ssl_auth_path = "./path/to/root" | ||||||
|  | # Enable client authentication, and accept certificates signed by those roots provided in CERTFILE. | ||||||
|  |  | ||||||
|  | # ssl_cert_path = "./path/to/CERTFILE" | ||||||
|  | # Read server certificates from CERTFILE. This should contain PEM-format certificates in | ||||||
|  | # the right order (the first certificate should certify KEYFILE, the last should be a root | ||||||
|  | # CA). | ||||||
|  |  | ||||||
|  | # ssl_key_path = "./path/to/private-key" | ||||||
|  | # Read the private key from KEYFILE.  This should be an RSA private key or PKCS8-encoded | ||||||
|  | # private key, in PEM format. | ||||||
|  |  | ||||||
|  | # ssl_ocsp_path = "./path/to/OCSPFILE" | ||||||
|  | # Read DER-encoded OCSP response from OCSPFILE and staple to certificate. Optional. | ||||||
|  |  | ||||||
|  | ssl_require_auth = false | ||||||
|  | # Send a fatal alert if the client does not complete client authentication. | ||||||
|  |  | ||||||
|  | ssl_resumption = false | ||||||
|  | # SSL support session resumption. | ||||||
|  |   | ||||||
|  | ssl_tickets = false | ||||||
|  | # SSL support tickets. | ||||||
|  |  | ||||||
|  | ### | ||||||
| @@ -76,6 +76,7 @@ thiserror = "1.0.30" | |||||||
| time = { version = "0.3.7", features = ["serde-well-known", "formatting", "parsing", "macros"] } | time = { version = "0.3.7", features = ["serde-well-known", "formatting", "parsing", "macros"] } | ||||||
| tokio = { version = "1.17.0", features = ["full"] } | tokio = { version = "1.17.0", features = ["full"] } | ||||||
| tokio-stream = "0.1.8" | tokio-stream = "0.1.8" | ||||||
|  | toml = "0.5.9" | ||||||
| uuid = { version = "1.1.2", features = ["serde", "v4"] } | uuid = { version = "1.1.2", features = ["serde", "v4"] } | ||||||
| walkdir = "2.3.2" | walkdir = "2.3.2" | ||||||
| prometheus = { version = "0.13.0", features = ["process"], optional = true } | prometheus = { version = "0.13.0", features = ["process"], optional = true } | ||||||
|   | |||||||
| @@ -349,16 +349,16 @@ pub struct SearchAggregator { | |||||||
|  |  | ||||||
|     // sort |     // sort | ||||||
|     sort_with_geo_point: bool, |     sort_with_geo_point: bool, | ||||||
|     // everytime a request has a filter, this field must be incremented by the number of terms it contains |     // every time a request has a filter, this field must be incremented by the number of terms it contains | ||||||
|     sort_sum_of_criteria_terms: usize, |     sort_sum_of_criteria_terms: usize, | ||||||
|     // everytime a request has a filter, this field must be incremented by one |     // every time a request has a filter, this field must be incremented by one | ||||||
|     sort_total_number_of_criteria: usize, |     sort_total_number_of_criteria: usize, | ||||||
|  |  | ||||||
|     // filter |     // filter | ||||||
|     filter_with_geo_radius: bool, |     filter_with_geo_radius: bool, | ||||||
|     // everytime a request has a filter, this field must be incremented by the number of terms it contains |     // every time a request has a filter, this field must be incremented by the number of terms it contains | ||||||
|     filter_sum_of_criteria_terms: usize, |     filter_sum_of_criteria_terms: usize, | ||||||
|     // everytime a request has a filter, this field must be incremented by one |     // every time a request has a filter, this field must be incremented by one | ||||||
|     filter_total_number_of_criteria: usize, |     filter_total_number_of_criteria: usize, | ||||||
|     used_syntax: HashMap<String, usize>, |     used_syntax: HashMap<String, usize>, | ||||||
|  |  | ||||||
| @@ -366,7 +366,7 @@ pub struct SearchAggregator { | |||||||
|     // The maximum number of terms in a q request |     // The maximum number of terms in a q request | ||||||
|     max_terms_number: usize, |     max_terms_number: usize, | ||||||
|  |  | ||||||
|     // everytime a search is done, we increment the counter linked to the used settings |     // every time a search is done, we increment the counter linked to the used settings | ||||||
|     matching_strategy: HashMap<String, usize>, |     matching_strategy: HashMap<String, usize>, | ||||||
|  |  | ||||||
|     // pagination |     // pagination | ||||||
|   | |||||||
| @@ -1,9 +1,9 @@ | |||||||
| use std::env; | use std::env; | ||||||
|  | use std::path::PathBuf; | ||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
|  |  | ||||||
| use actix_web::http::KeepAlive; | use actix_web::http::KeepAlive; | ||||||
| use actix_web::HttpServer; | use actix_web::HttpServer; | ||||||
| use clap::Parser; |  | ||||||
| use meilisearch_auth::AuthController; | use meilisearch_auth::AuthController; | ||||||
| use meilisearch_http::analytics; | use meilisearch_http::analytics; | ||||||
| use meilisearch_http::analytics::Analytics; | use meilisearch_http::analytics::Analytics; | ||||||
| @@ -29,7 +29,7 @@ fn setup(opt: &Opt) -> anyhow::Result<()> { | |||||||
|  |  | ||||||
| #[actix_web::main] | #[actix_web::main] | ||||||
| async fn main() -> anyhow::Result<()> { | async fn main() -> anyhow::Result<()> { | ||||||
|     let opt = Opt::parse(); |     let (opt, config_read_from) = Opt::try_build()?; | ||||||
|  |  | ||||||
|     setup(&opt)?; |     setup(&opt)?; | ||||||
|  |  | ||||||
| @@ -58,7 +58,7 @@ async fn main() -> anyhow::Result<()> { | |||||||
|     #[cfg(any(debug_assertions, not(feature = "analytics")))] |     #[cfg(any(debug_assertions, not(feature = "analytics")))] | ||||||
|     let (analytics, user) = analytics::MockAnalytics::new(&opt); |     let (analytics, user) = analytics::MockAnalytics::new(&opt); | ||||||
|  |  | ||||||
|     print_launch_resume(&opt, &user); |     print_launch_resume(&opt, &user, config_read_from); | ||||||
|  |  | ||||||
|     run_http(meilisearch, auth_controller, opt, analytics).await?; |     run_http(meilisearch, auth_controller, opt, analytics).await?; | ||||||
|  |  | ||||||
| @@ -97,7 +97,7 @@ async fn run_http( | |||||||
|     Ok(()) |     Ok(()) | ||||||
| } | } | ||||||
|  |  | ||||||
| pub fn print_launch_resume(opt: &Opt, user: &str) { | pub fn print_launch_resume(opt: &Opt, user: &str, config_read_from: Option<PathBuf>) { | ||||||
|     let commit_sha = option_env!("VERGEN_GIT_SHA").unwrap_or("unknown"); |     let commit_sha = option_env!("VERGEN_GIT_SHA").unwrap_or("unknown"); | ||||||
|     let commit_date = option_env!("VERGEN_GIT_COMMIT_TIMESTAMP").unwrap_or("unknown"); |     let commit_date = option_env!("VERGEN_GIT_COMMIT_TIMESTAMP").unwrap_or("unknown"); | ||||||
|     let protocol = if opt.ssl_cert_path.is_some() && opt.ssl_key_path.is_some() { |     let protocol = if opt.ssl_cert_path.is_some() && opt.ssl_key_path.is_some() { | ||||||
| @@ -118,6 +118,12 @@ pub fn print_launch_resume(opt: &Opt, user: &str) { | |||||||
|  |  | ||||||
|     eprintln!("{}", ascii_name); |     eprintln!("{}", ascii_name); | ||||||
|  |  | ||||||
|  |     eprintln!( | ||||||
|  |         "Config file path:\t{:?}", | ||||||
|  |         config_read_from | ||||||
|  |             .map(|config_file_path| config_file_path.display().to_string()) | ||||||
|  |             .unwrap_or_else(|| "none".to_string()) | ||||||
|  |     ); | ||||||
|     eprintln!("Database path:\t\t{:?}", opt.db_path); |     eprintln!("Database path:\t\t{:?}", opt.db_path); | ||||||
|     eprintln!("Server listening on:\t\"{}://{}\"", protocol, opt.http_addr); |     eprintln!("Server listening on:\t\"{}://{}\"", protocol, opt.http_addr); | ||||||
|     eprintln!("Environment:\t\t{:?}", opt.env); |     eprintln!("Environment:\t\t{:?}", opt.env); | ||||||
|   | |||||||
| @@ -5,7 +5,10 @@ use std::sync::Arc; | |||||||
|  |  | ||||||
| use byte_unit::Byte; | use byte_unit::Byte; | ||||||
| use clap::Parser; | use clap::Parser; | ||||||
| use meilisearch_lib::options::{IndexerOpts, SchedulerConfig}; | use meilisearch_lib::{ | ||||||
|  |     export_to_env_if_not_present, | ||||||
|  |     options::{IndexerOpts, SchedulerConfig}, | ||||||
|  | }; | ||||||
| use rustls::{ | use rustls::{ | ||||||
|     server::{ |     server::{ | ||||||
|         AllowAnyAnonymousOrAuthenticatedClient, AllowAnyAuthenticatedClient, |         AllowAnyAnonymousOrAuthenticatedClient, AllowAnyAuthenticatedClient, | ||||||
| @@ -14,149 +17,208 @@ use rustls::{ | |||||||
|     RootCertStore, |     RootCertStore, | ||||||
| }; | }; | ||||||
| use rustls_pemfile::{certs, pkcs8_private_keys, rsa_private_keys}; | use rustls_pemfile::{certs, pkcs8_private_keys, rsa_private_keys}; | ||||||
| use serde::Serialize; | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
| const POSSIBLE_ENV: [&str; 2] = ["development", "production"]; | const POSSIBLE_ENV: [&str; 2] = ["development", "production"]; | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Parser, Serialize)] | const MEILI_DB_PATH: &str = "MEILI_DB_PATH"; | ||||||
|  | const MEILI_HTTP_ADDR: &str = "MEILI_HTTP_ADDR"; | ||||||
|  | const MEILI_MASTER_KEY: &str = "MEILI_MASTER_KEY"; | ||||||
|  | const MEILI_ENV: &str = "MEILI_ENV"; | ||||||
|  | #[cfg(all(not(debug_assertions), feature = "analytics"))] | ||||||
|  | const MEILI_NO_ANALYTICS: &str = "MEILI_NO_ANALYTICS"; | ||||||
|  | const MEILI_MAX_INDEX_SIZE: &str = "MEILI_MAX_INDEX_SIZE"; | ||||||
|  | const MEILI_MAX_TASK_DB_SIZE: &str = "MEILI_MAX_TASK_DB_SIZE"; | ||||||
|  | const MEILI_HTTP_PAYLOAD_SIZE_LIMIT: &str = "MEILI_HTTP_PAYLOAD_SIZE_LIMIT"; | ||||||
|  | const MEILI_SSL_CERT_PATH: &str = "MEILI_SSL_CERT_PATH"; | ||||||
|  | const MEILI_SSL_KEY_PATH: &str = "MEILI_SSL_KEY_PATH"; | ||||||
|  | const MEILI_SSL_AUTH_PATH: &str = "MEILI_SSL_AUTH_PATH"; | ||||||
|  | const MEILI_SSL_OCSP_PATH: &str = "MEILI_SSL_OCSP_PATH"; | ||||||
|  | const MEILI_SSL_REQUIRE_AUTH: &str = "MEILI_SSL_REQUIRE_AUTH"; | ||||||
|  | const MEILI_SSL_RESUMPTION: &str = "MEILI_SSL_RESUMPTION"; | ||||||
|  | const MEILI_SSL_TICKETS: &str = "MEILI_SSL_TICKETS"; | ||||||
|  | const MEILI_IMPORT_SNAPSHOT: &str = "MEILI_IMPORT_SNAPSHOT"; | ||||||
|  | const MEILI_IGNORE_MISSING_SNAPSHOT: &str = "MEILI_IGNORE_MISSING_SNAPSHOT"; | ||||||
|  | const MEILI_IGNORE_SNAPSHOT_IF_DB_EXISTS: &str = "MEILI_IGNORE_SNAPSHOT_IF_DB_EXISTS"; | ||||||
|  | const MEILI_SNAPSHOT_DIR: &str = "MEILI_SNAPSHOT_DIR"; | ||||||
|  | const MEILI_SCHEDULE_SNAPSHOT: &str = "MEILI_SCHEDULE_SNAPSHOT"; | ||||||
|  | const MEILI_SNAPSHOT_INTERVAL_SEC: &str = "MEILI_SNAPSHOT_INTERVAL_SEC"; | ||||||
|  | const MEILI_IMPORT_DUMP: &str = "MEILI_IMPORT_DUMP"; | ||||||
|  | const MEILI_IGNORE_MISSING_DUMP: &str = "MEILI_IGNORE_MISSING_DUMP"; | ||||||
|  | const MEILI_IGNORE_DUMP_IF_DB_EXISTS: &str = "MEILI_IGNORE_DUMP_IF_DB_EXISTS"; | ||||||
|  | const MEILI_DUMPS_DIR: &str = "MEILI_DUMPS_DIR"; | ||||||
|  | const MEILI_LOG_LEVEL: &str = "MEILI_LOG_LEVEL"; | ||||||
|  | #[cfg(feature = "metrics")] | ||||||
|  | const MEILI_ENABLE_METRICS_ROUTE: &str = "MEILI_ENABLE_METRICS_ROUTE"; | ||||||
|  |  | ||||||
|  | const DEFAULT_DB_PATH: &str = "./data.ms"; | ||||||
|  | const DEFAULT_HTTP_ADDR: &str = "127.0.0.1:7700"; | ||||||
|  | const DEFAULT_ENV: &str = "development"; | ||||||
|  | const DEFAULT_MAX_INDEX_SIZE: &str = "100 GiB"; | ||||||
|  | const DEFAULT_MAX_TASK_DB_SIZE: &str = "100 GiB"; | ||||||
|  | const DEFAULT_HTTP_PAYLOAD_SIZE_LIMIT: &str = "100 MB"; | ||||||
|  | const DEFAULT_SNAPSHOT_DIR: &str = "snapshots/"; | ||||||
|  | const DEFAULT_SNAPSHOT_INTERVAL_SEC: u64 = 86400; | ||||||
|  | const DEFAULT_DUMPS_DIR: &str = "dumps/"; | ||||||
|  | const DEFAULT_LOG_LEVEL: &str = "INFO"; | ||||||
|  |  | ||||||
|  | #[derive(Debug, Clone, Parser, Serialize, Deserialize)] | ||||||
| #[clap(version)] | #[clap(version)] | ||||||
|  | #[serde(rename_all = "snake_case", deny_unknown_fields)] | ||||||
| pub struct Opt { | pub struct Opt { | ||||||
|     /// The destination where the database must be created. |     /// The destination where the database must be created. | ||||||
|     #[clap(long, env = "MEILI_DB_PATH", default_value = "./data.ms")] |     #[clap(long, env = MEILI_DB_PATH, default_value_os_t = default_db_path())] | ||||||
|  |     #[serde(default = "default_db_path")] | ||||||
|     pub db_path: PathBuf, |     pub db_path: PathBuf, | ||||||
|  |  | ||||||
|     /// The address on which the http server will listen. |     /// The address on which the http server will listen. | ||||||
|     #[clap(long, env = "MEILI_HTTP_ADDR", default_value = "127.0.0.1:7700")] |     #[clap(long, env = MEILI_HTTP_ADDR, default_value_t = default_http_addr())] | ||||||
|  |     #[serde(default = "default_http_addr")] | ||||||
|     pub http_addr: String, |     pub http_addr: String, | ||||||
|  |  | ||||||
|     /// The master key allowing you to do everything on the server. |     /// Sets the instance's master key, automatically protecting all routes except GET /health | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing)] | ||||||
|     #[clap(long, env = "MEILI_MASTER_KEY")] |     #[clap(long, env = MEILI_MASTER_KEY)] | ||||||
|     pub master_key: Option<String>, |     pub master_key: Option<String>, | ||||||
|  |  | ||||||
|     /// This environment variable must be set to `production` if you are running in production. |     /// This environment variable must be set to `production` if you are running in production. | ||||||
|     /// If the server is running in development mode more logs will be displayed, |     /// More logs wiil be displayed if the server is running in development mode. Setting the master | ||||||
|     /// and the master key can be avoided which implies that there is no security on the updates routes. |     /// key is optional; hence no security on the updates routes. This | ||||||
|     /// This is useful to debug when integrating the engine with another service. |     /// is useful to debug when integrating the engine with another service | ||||||
|     #[clap(long, env = "MEILI_ENV", default_value = "development", possible_values = &POSSIBLE_ENV)] |     #[clap(long, env = MEILI_ENV, default_value_t = default_env(), possible_values = &POSSIBLE_ENV)] | ||||||
|  |     #[serde(default = "default_env")] | ||||||
|     pub env: String, |     pub env: String, | ||||||
|  |  | ||||||
|     /// Do not send analytics to Meili. |     /// Do not send analytics to Meili. | ||||||
|     #[cfg(all(not(debug_assertions), feature = "analytics"))] |     #[cfg(all(not(debug_assertions), feature = "analytics"))] | ||||||
|     #[serde(skip)] // we can't send true |     #[serde(skip_serializing, default)] // we can't send true | ||||||
|     #[clap(long, env = "MEILI_NO_ANALYTICS")] |     #[clap(long, env = MEILI_NO_ANALYTICS)] | ||||||
|     pub no_analytics: bool, |     pub no_analytics: bool, | ||||||
|  |  | ||||||
|     /// The maximum size, in bytes, of the main lmdb database directory |     /// The maximum size, in bytes, of the main LMDB database directory | ||||||
|     #[clap(long, env = "MEILI_MAX_INDEX_SIZE", default_value = "100 GiB")] |     #[clap(long, env = MEILI_MAX_INDEX_SIZE, default_value_t = default_max_index_size())] | ||||||
|  |     #[serde(default = "default_max_index_size")] | ||||||
|     pub max_index_size: Byte, |     pub max_index_size: Byte, | ||||||
|  |  | ||||||
|     /// The maximum size, in bytes, of the update lmdb database directory |     /// The maximum size, in bytes, of the update LMDB database directory | ||||||
|     #[clap(long, env = "MEILI_MAX_TASK_DB_SIZE", default_value = "100 GiB")] |     #[clap(long, env = MEILI_MAX_TASK_DB_SIZE, default_value_t = default_max_task_db_size())] | ||||||
|  |     #[serde(default = "default_max_task_db_size")] | ||||||
|     pub max_task_db_size: Byte, |     pub max_task_db_size: Byte, | ||||||
|  |  | ||||||
|     /// The maximum size, in bytes, of accepted JSON payloads |     /// The maximum size, in bytes, of accepted JSON payloads | ||||||
|     #[clap(long, env = "MEILI_HTTP_PAYLOAD_SIZE_LIMIT", default_value = "100 MB")] |     #[clap(long, env = MEILI_HTTP_PAYLOAD_SIZE_LIMIT, default_value_t = default_http_payload_size_limit())] | ||||||
|  |     #[serde(default = "default_http_payload_size_limit")] | ||||||
|     pub http_payload_size_limit: Byte, |     pub http_payload_size_limit: Byte, | ||||||
|  |  | ||||||
|     /// Read server certificates from CERTFILE. |     /// Read server certificates from CERTFILE. | ||||||
|     /// This should contain PEM-format certificates |     /// This should contain PEM-format certificates | ||||||
|     /// in the right order (the first certificate should |     /// in the right order (the first certificate should | ||||||
|     /// certify KEYFILE, the last should be a root CA). |     /// certify KEYFILE, the last should be a root CA). | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing)] | ||||||
|     #[clap(long, env = "MEILI_SSL_CERT_PATH", parse(from_os_str))] |     #[clap(long, env = MEILI_SSL_CERT_PATH, parse(from_os_str))] | ||||||
|     pub ssl_cert_path: Option<PathBuf>, |     pub ssl_cert_path: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// Read private key from KEYFILE.  This should be a RSA |     /// Read the private key from KEYFILE.  This should be an RSA | ||||||
|     /// private key or PKCS8-encoded private key, in PEM format. |     /// private key or PKCS8-encoded private key, in PEM format. | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing)] | ||||||
|     #[clap(long, env = "MEILI_SSL_KEY_PATH", parse(from_os_str))] |     #[clap(long, env = MEILI_SSL_KEY_PATH, parse(from_os_str))] | ||||||
|     pub ssl_key_path: Option<PathBuf>, |     pub ssl_key_path: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// Enable client authentication, and accept certificates |     /// Enable client authentication, and accept certificates | ||||||
|     /// signed by those roots provided in CERTFILE. |     /// signed by those roots provided in CERTFILE. | ||||||
|     #[clap(long, env = "MEILI_SSL_AUTH_PATH", parse(from_os_str))] |     #[serde(skip_serializing)] | ||||||
|     #[serde(skip)] |     #[clap(long, env = MEILI_SSL_AUTH_PATH, parse(from_os_str))] | ||||||
|     pub ssl_auth_path: Option<PathBuf>, |     pub ssl_auth_path: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// Read DER-encoded OCSP response from OCSPFILE and staple to certificate. |     /// Read DER-encoded OCSP response from OCSPFILE and staple to certificate. | ||||||
|     /// Optional |     /// Optional | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing)] | ||||||
|     #[clap(long, env = "MEILI_SSL_OCSP_PATH", parse(from_os_str))] |     #[clap(long, env = MEILI_SSL_OCSP_PATH, parse(from_os_str))] | ||||||
|     pub ssl_ocsp_path: Option<PathBuf>, |     pub ssl_ocsp_path: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// Send a fatal alert if the client does not complete client authentication. |     /// Send a fatal alert if the client does not complete client authentication. | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing, default)] | ||||||
|     #[clap(long, env = "MEILI_SSL_REQUIRE_AUTH")] |     #[clap(long, env = MEILI_SSL_REQUIRE_AUTH)] | ||||||
|     pub ssl_require_auth: bool, |     pub ssl_require_auth: bool, | ||||||
|  |  | ||||||
|     /// SSL support session resumption |     /// SSL support session resumption | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing, default)] | ||||||
|     #[clap(long, env = "MEILI_SSL_RESUMPTION")] |     #[clap(long, env = MEILI_SSL_RESUMPTION)] | ||||||
|     pub ssl_resumption: bool, |     pub ssl_resumption: bool, | ||||||
|  |  | ||||||
|     /// SSL support tickets. |     /// SSL support tickets. | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing, default)] | ||||||
|     #[clap(long, env = "MEILI_SSL_TICKETS")] |     #[clap(long, env = MEILI_SSL_TICKETS)] | ||||||
|     pub ssl_tickets: bool, |     pub ssl_tickets: bool, | ||||||
|  |  | ||||||
|     /// Defines the path of the snapshot file to import. |     /// Defines the path of the snapshot file to import. | ||||||
|     /// This option will, by default, stop the process if a database already exist or if no snapshot exists at |     /// This option will, by default, stop the process if a database already exists, or if no snapshot exists at | ||||||
|     /// the given path. If this option is not specified no snapshot is imported. |     /// the given path. If this option is not specified, no snapshot is imported. | ||||||
|     #[clap(long, env = "MEILI_IMPORT_SNAPSHOT")] |     #[clap(long, env = MEILI_IMPORT_SNAPSHOT)] | ||||||
|     pub import_snapshot: Option<PathBuf>, |     pub import_snapshot: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// The engine will ignore a missing snapshot and not return an error in such case. |     /// The engine will ignore a missing snapshot and not return an error in such a case. | ||||||
|     #[clap( |     #[clap( | ||||||
|         long, |         long, | ||||||
|         env = "MEILI_IGNORE_MISSING_SNAPSHOT", |         env = MEILI_IGNORE_MISSING_SNAPSHOT, | ||||||
|         requires = "import-snapshot" |         requires = "import-snapshot" | ||||||
|     )] |     )] | ||||||
|  |     #[serde(default)] | ||||||
|     pub ignore_missing_snapshot: bool, |     pub ignore_missing_snapshot: bool, | ||||||
|  |  | ||||||
|     /// The engine will skip snapshot importation and not return an error in such case. |     /// The engine will skip snapshot importation and not return an error in such case. | ||||||
|     #[clap( |     #[clap( | ||||||
|         long, |         long, | ||||||
|         env = "MEILI_IGNORE_SNAPSHOT_IF_DB_EXISTS", |         env = MEILI_IGNORE_SNAPSHOT_IF_DB_EXISTS, | ||||||
|         requires = "import-snapshot" |         requires = "import-snapshot" | ||||||
|     )] |     )] | ||||||
|  |     #[serde(default)] | ||||||
|     pub ignore_snapshot_if_db_exists: bool, |     pub ignore_snapshot_if_db_exists: bool, | ||||||
|  |  | ||||||
|     /// Defines the directory path where meilisearch will create snapshot each snapshot_time_gap. |     /// Defines the directory path where Meilisearch will create a snapshot each snapshot-interval-sec. | ||||||
|     #[clap(long, env = "MEILI_SNAPSHOT_DIR", default_value = "snapshots/")] |     #[clap(long, env = MEILI_SNAPSHOT_DIR, default_value_os_t = default_snapshot_dir())] | ||||||
|  |     #[serde(default = "default_snapshot_dir")] | ||||||
|     pub snapshot_dir: PathBuf, |     pub snapshot_dir: PathBuf, | ||||||
|  |  | ||||||
|     /// Activate snapshot scheduling. |     /// Activate snapshot scheduling. | ||||||
|     #[clap(long, env = "MEILI_SCHEDULE_SNAPSHOT")] |     #[clap(long, env = MEILI_SCHEDULE_SNAPSHOT)] | ||||||
|  |     #[serde(default)] | ||||||
|     pub schedule_snapshot: bool, |     pub schedule_snapshot: bool, | ||||||
|  |  | ||||||
|     /// Defines time interval, in seconds, between each snapshot creation. |     /// Defines time interval, in seconds, between each snapshot creation. | ||||||
|     #[clap(long, env = "MEILI_SNAPSHOT_INTERVAL_SEC", default_value = "86400")] // 24h |     #[clap(long, env = MEILI_SNAPSHOT_INTERVAL_SEC, default_value_t = default_snapshot_interval_sec())] | ||||||
|  |     #[serde(default = "default_snapshot_interval_sec")] | ||||||
|  |     // 24h | ||||||
|     pub snapshot_interval_sec: u64, |     pub snapshot_interval_sec: u64, | ||||||
|  |  | ||||||
|     /// Import a dump from the specified path, must be a `.dump` file. |     /// Import a dump from the specified path, must be a `.dump` file. | ||||||
|     #[clap(long, env = "MEILI_IMPORT_DUMP", conflicts_with = "import-snapshot")] |     #[clap(long, env = MEILI_IMPORT_DUMP, conflicts_with = "import-snapshot")] | ||||||
|     pub import_dump: Option<PathBuf>, |     pub import_dump: Option<PathBuf>, | ||||||
|  |  | ||||||
|     /// If the dump doesn't exists, load or create the database specified by `db-path` instead. |     /// If the dump doesn't exist, load or create the database specified by `db-path` instead. | ||||||
|     #[clap(long, env = "MEILI_IGNORE_MISSING_DUMP", requires = "import-dump")] |     #[clap(long, env = MEILI_IGNORE_MISSING_DUMP, requires = "import-dump")] | ||||||
|  |     #[serde(default)] | ||||||
|     pub ignore_missing_dump: bool, |     pub ignore_missing_dump: bool, | ||||||
|  |  | ||||||
|     /// Ignore the dump if a database already exists, and load that database instead. |     /// Ignore the dump if a database already exists, and load that database instead. | ||||||
|     #[clap(long, env = "MEILI_IGNORE_DUMP_IF_DB_EXISTS", requires = "import-dump")] |     #[clap(long, env = MEILI_IGNORE_DUMP_IF_DB_EXISTS, requires = "import-dump")] | ||||||
|  |     #[serde(default)] | ||||||
|     pub ignore_dump_if_db_exists: bool, |     pub ignore_dump_if_db_exists: bool, | ||||||
|  |  | ||||||
|     /// Folder where dumps are created when the dump route is called. |     /// Folder where dumps are created when the dump route is called. | ||||||
|     #[clap(long, env = "MEILI_DUMPS_DIR", default_value = "dumps/")] |     #[clap(long, env = MEILI_DUMPS_DIR, default_value_os_t = default_dumps_dir())] | ||||||
|  |     #[serde(default = "default_dumps_dir")] | ||||||
|     pub dumps_dir: PathBuf, |     pub dumps_dir: PathBuf, | ||||||
|  |  | ||||||
|     /// Set the log level |     /// Set the log level. # Possible values: [ERROR, WARN, INFO, DEBUG, TRACE] | ||||||
|     #[clap(long, env = "MEILI_LOG_LEVEL", default_value = "info")] |     #[clap(long, env = MEILI_LOG_LEVEL, default_value_t = default_log_level())] | ||||||
|  |     #[serde(default = "default_log_level")] | ||||||
|     pub log_level: String, |     pub log_level: String, | ||||||
|  |  | ||||||
|     /// Enables Prometheus metrics and /metrics route. |     /// Enables Prometheus metrics and /metrics route. | ||||||
|     #[cfg(feature = "metrics")] |     #[cfg(feature = "metrics")] | ||||||
|     #[clap(long, env = "MEILI_ENABLE_METRICS_ROUTE")] |     #[clap(long, env = MEILI_ENABLE_METRICS_ROUTE)] | ||||||
|  |     #[serde(default)] | ||||||
|     pub enable_metrics_route: bool, |     pub enable_metrics_route: bool, | ||||||
|  |  | ||||||
|     #[serde(flatten)] |     #[serde(flatten)] | ||||||
| @@ -166,15 +228,139 @@ pub struct Opt { | |||||||
|     #[serde(flatten)] |     #[serde(flatten)] | ||||||
|     #[clap(flatten)] |     #[clap(flatten)] | ||||||
|     pub scheduler_options: SchedulerConfig, |     pub scheduler_options: SchedulerConfig, | ||||||
|  |  | ||||||
|  |     /// The path to a configuration file that should be used to setup the engine. | ||||||
|  |     /// Format must be TOML. | ||||||
|  |     #[serde(skip_serializing)] | ||||||
|  |     #[clap(long)] | ||||||
|  |     pub config_file_path: Option<PathBuf>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl Opt { | impl Opt { | ||||||
|     /// Wether analytics should be enabled or not. |     /// Whether analytics should be enabled or not. | ||||||
|     #[cfg(all(not(debug_assertions), feature = "analytics"))] |     #[cfg(all(not(debug_assertions), feature = "analytics"))] | ||||||
|     pub fn analytics(&self) -> bool { |     pub fn analytics(&self) -> bool { | ||||||
|         !self.no_analytics |         !self.no_analytics | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Build a new Opt from config file, env vars and cli args. | ||||||
|  |     pub fn try_build() -> anyhow::Result<(Self, Option<PathBuf>)> { | ||||||
|  |         // Parse the args to get the config_file_path. | ||||||
|  |         let mut opts = Opt::parse(); | ||||||
|  |         let mut config_read_from = None; | ||||||
|  |         if let Some(config_file_path) = opts | ||||||
|  |             .config_file_path | ||||||
|  |             .clone() | ||||||
|  |             .or_else(|| Some(PathBuf::from("./config.toml"))) | ||||||
|  |         { | ||||||
|  |             match std::fs::read(&config_file_path) { | ||||||
|  |                 Ok(config) => { | ||||||
|  |                     // If the file is successfully read, we deserialize it with `toml`. | ||||||
|  |                     let opt_from_config = toml::from_slice::<Opt>(&config)?; | ||||||
|  |                     // We inject the values from the toml in the corresponding env vars if needs be. Doing so, we respect the priority toml < env vars < cli args. | ||||||
|  |                     opt_from_config.export_to_env(); | ||||||
|  |                     // Once injected we parse the cli args once again to take the new env vars into scope. | ||||||
|  |                     opts = Opt::parse(); | ||||||
|  |                     config_read_from = Some(config_file_path); | ||||||
|  |                 } | ||||||
|  |                 // If we have an error while reading the file defined by the user. | ||||||
|  |                 Err(_) if opts.config_file_path.is_some() => anyhow::bail!( | ||||||
|  |                     "unable to open or read the {:?} configuration file.", | ||||||
|  |                     opts.config_file_path.unwrap().display().to_string() | ||||||
|  |                 ), | ||||||
|  |                 _ => (), | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         Ok((opts, config_read_from)) | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /// Exports the opts values to their corresponding env vars if they are not set. | ||||||
|  |     fn export_to_env(self) { | ||||||
|  |         let Opt { | ||||||
|  |             db_path, | ||||||
|  |             http_addr, | ||||||
|  |             master_key, | ||||||
|  |             env, | ||||||
|  |             max_index_size, | ||||||
|  |             max_task_db_size, | ||||||
|  |             http_payload_size_limit, | ||||||
|  |             ssl_cert_path, | ||||||
|  |             ssl_key_path, | ||||||
|  |             ssl_auth_path, | ||||||
|  |             ssl_ocsp_path, | ||||||
|  |             ssl_require_auth, | ||||||
|  |             ssl_resumption, | ||||||
|  |             ssl_tickets, | ||||||
|  |             snapshot_dir, | ||||||
|  |             schedule_snapshot, | ||||||
|  |             snapshot_interval_sec, | ||||||
|  |             dumps_dir, | ||||||
|  |             log_level, | ||||||
|  |             indexer_options, | ||||||
|  |             scheduler_options, | ||||||
|  |             import_snapshot: _, | ||||||
|  |             ignore_missing_snapshot: _, | ||||||
|  |             ignore_snapshot_if_db_exists: _, | ||||||
|  |             import_dump: _, | ||||||
|  |             ignore_missing_dump: _, | ||||||
|  |             ignore_dump_if_db_exists: _, | ||||||
|  |             config_file_path: _, | ||||||
|  |             #[cfg(all(not(debug_assertions), feature = "analytics"))] | ||||||
|  |             no_analytics, | ||||||
|  |             #[cfg(feature = "metrics")] | ||||||
|  |             enable_metrics_route, | ||||||
|  |         } = self; | ||||||
|  |         export_to_env_if_not_present(MEILI_DB_PATH, db_path); | ||||||
|  |         export_to_env_if_not_present(MEILI_HTTP_ADDR, http_addr); | ||||||
|  |         if let Some(master_key) = master_key { | ||||||
|  |             export_to_env_if_not_present(MEILI_MASTER_KEY, master_key); | ||||||
|  |         } | ||||||
|  |         export_to_env_if_not_present(MEILI_ENV, env); | ||||||
|  |         #[cfg(all(not(debug_assertions), feature = "analytics"))] | ||||||
|  |         { | ||||||
|  |             export_to_env_if_not_present(MEILI_NO_ANALYTICS, no_analytics.to_string()); | ||||||
|  |         } | ||||||
|  |         export_to_env_if_not_present(MEILI_MAX_INDEX_SIZE, max_index_size.to_string()); | ||||||
|  |         export_to_env_if_not_present(MEILI_MAX_TASK_DB_SIZE, max_task_db_size.to_string()); | ||||||
|  |         export_to_env_if_not_present( | ||||||
|  |             MEILI_HTTP_PAYLOAD_SIZE_LIMIT, | ||||||
|  |             http_payload_size_limit.to_string(), | ||||||
|  |         ); | ||||||
|  |         if let Some(ssl_cert_path) = ssl_cert_path { | ||||||
|  |             export_to_env_if_not_present(MEILI_SSL_CERT_PATH, ssl_cert_path); | ||||||
|  |         } | ||||||
|  |         if let Some(ssl_key_path) = ssl_key_path { | ||||||
|  |             export_to_env_if_not_present(MEILI_SSL_KEY_PATH, ssl_key_path); | ||||||
|  |         } | ||||||
|  |         if let Some(ssl_auth_path) = ssl_auth_path { | ||||||
|  |             export_to_env_if_not_present(MEILI_SSL_AUTH_PATH, ssl_auth_path); | ||||||
|  |         } | ||||||
|  |         if let Some(ssl_ocsp_path) = ssl_ocsp_path { | ||||||
|  |             export_to_env_if_not_present(MEILI_SSL_OCSP_PATH, ssl_ocsp_path); | ||||||
|  |         } | ||||||
|  |         export_to_env_if_not_present(MEILI_SSL_REQUIRE_AUTH, ssl_require_auth.to_string()); | ||||||
|  |         export_to_env_if_not_present(MEILI_SSL_RESUMPTION, ssl_resumption.to_string()); | ||||||
|  |         export_to_env_if_not_present(MEILI_SSL_TICKETS, ssl_tickets.to_string()); | ||||||
|  |         export_to_env_if_not_present(MEILI_SNAPSHOT_DIR, snapshot_dir); | ||||||
|  |         export_to_env_if_not_present(MEILI_SCHEDULE_SNAPSHOT, schedule_snapshot.to_string()); | ||||||
|  |         export_to_env_if_not_present( | ||||||
|  |             MEILI_SNAPSHOT_INTERVAL_SEC, | ||||||
|  |             snapshot_interval_sec.to_string(), | ||||||
|  |         ); | ||||||
|  |         export_to_env_if_not_present(MEILI_DUMPS_DIR, dumps_dir); | ||||||
|  |         export_to_env_if_not_present(MEILI_LOG_LEVEL, log_level); | ||||||
|  |         #[cfg(feature = "metrics")] | ||||||
|  |         { | ||||||
|  |             export_to_env_if_not_present( | ||||||
|  |                 MEILI_ENABLE_METRICS_ROUTE, | ||||||
|  |                 enable_metrics_route.to_string(), | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         indexer_options.export_to_env(); | ||||||
|  |         scheduler_options.export_to_env(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     pub fn get_ssl_config(&self) -> anyhow::Result<Option<rustls::ServerConfig>> { |     pub fn get_ssl_config(&self) -> anyhow::Result<Option<rustls::ServerConfig>> { | ||||||
|         if let (Some(cert_path), Some(key_path)) = (&self.ssl_cert_path, &self.ssl_key_path) { |         if let (Some(cert_path), Some(key_path)) = (&self.ssl_cert_path, &self.ssl_key_path) { | ||||||
|             let config = rustls::ServerConfig::builder().with_safe_defaults(); |             let config = rustls::ServerConfig::builder().with_safe_defaults(); | ||||||
| @@ -273,6 +459,48 @@ fn load_ocsp(filename: &Option<PathBuf>) -> anyhow::Result<Vec<u8>> { | |||||||
|     Ok(ret) |     Ok(ret) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Functions used to get default value for `Opt` fields, needs to be function because of serde's default attribute. | ||||||
|  |  | ||||||
|  | fn default_db_path() -> PathBuf { | ||||||
|  |     PathBuf::from(DEFAULT_DB_PATH) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_http_addr() -> String { | ||||||
|  |     DEFAULT_HTTP_ADDR.to_string() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_env() -> String { | ||||||
|  |     DEFAULT_ENV.to_string() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_max_index_size() -> Byte { | ||||||
|  |     Byte::from_str(DEFAULT_MAX_INDEX_SIZE).unwrap() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_max_task_db_size() -> Byte { | ||||||
|  |     Byte::from_str(DEFAULT_MAX_TASK_DB_SIZE).unwrap() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_http_payload_size_limit() -> Byte { | ||||||
|  |     Byte::from_str(DEFAULT_HTTP_PAYLOAD_SIZE_LIMIT).unwrap() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_snapshot_dir() -> PathBuf { | ||||||
|  |     PathBuf::from(DEFAULT_SNAPSHOT_DIR) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_snapshot_interval_sec() -> u64 { | ||||||
|  |     DEFAULT_SNAPSHOT_INTERVAL_SEC | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_dumps_dir() -> PathBuf { | ||||||
|  |     PathBuf::from(DEFAULT_DUMPS_DIR) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | fn default_log_level() -> String { | ||||||
|  |     DEFAULT_LOG_LEVEL.to_string() | ||||||
|  | } | ||||||
|  |  | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod test { | mod test { | ||||||
|     use super::*; |     use super::*; | ||||||
|   | |||||||
| @@ -147,7 +147,7 @@ enum TaskDetails { | |||||||
|     IndexInfo { primary_key: Option<String> }, |     IndexInfo { primary_key: Option<String> }, | ||||||
|     #[serde(rename_all = "camelCase")] |     #[serde(rename_all = "camelCase")] | ||||||
|     DocumentDeletion { |     DocumentDeletion { | ||||||
|         received_document_ids: usize, |         matched_documents: usize, | ||||||
|         deleted_documents: Option<u64>, |         deleted_documents: Option<u64>, | ||||||
|     }, |     }, | ||||||
|     #[serde(rename_all = "camelCase")] |     #[serde(rename_all = "camelCase")] | ||||||
| @@ -255,7 +255,7 @@ impl From<Task> for TaskView { | |||||||
|             } => ( |             } => ( | ||||||
|                 TaskType::DocumentDeletion, |                 TaskType::DocumentDeletion, | ||||||
|                 Some(TaskDetails::DocumentDeletion { |                 Some(TaskDetails::DocumentDeletion { | ||||||
|                     received_document_ids: ids.len(), |                     matched_documents: ids.len(), | ||||||
|                     deleted_documents: None, |                     deleted_documents: None, | ||||||
|                 }), |                 }), | ||||||
|             ), |             ), | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| use crate::common::Server; | use crate::common::Server; | ||||||
|  |  | ||||||
|  | #[cfg(feature = "mini-dashboard")] | ||||||
| #[actix_rt::test] | #[actix_rt::test] | ||||||
| async fn dashboard_assets_load() { | async fn dashboard_assets_load() { | ||||||
|     let server = Server::new().await; |     let server = Server::new().await; | ||||||
|   | |||||||
| @@ -145,7 +145,7 @@ pub fn error_code_from_str(s: &str) -> anyhow::Result<Code> { | |||||||
|         "unsupported_media_type" => Code::UnsupportedMediaType, |         "unsupported_media_type" => Code::UnsupportedMediaType, | ||||||
|         "dump_already_in_progress" => Code::DumpAlreadyInProgress, |         "dump_already_in_progress" => Code::DumpAlreadyInProgress, | ||||||
|         "dump_process_failed" => Code::DumpProcessFailed, |         "dump_process_failed" => Code::DumpProcessFailed, | ||||||
|         _ => bail!("unknow error code."), |         _ => bail!("unknown error code."), | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     Ok(code) |     Ok(code) | ||||||
|   | |||||||
| @@ -57,10 +57,10 @@ fn patch_updates(src: impl AsRef<Path>, dst: impl AsRef<Path>) -> anyhow::Result | |||||||
|     let updates_path = src.as_ref().join("updates/data.jsonl"); |     let updates_path = src.as_ref().join("updates/data.jsonl"); | ||||||
|     let output_updates_path = dst.as_ref().join("updates/data.jsonl"); |     let output_updates_path = dst.as_ref().join("updates/data.jsonl"); | ||||||
|     create_dir_all(output_updates_path.parent().unwrap())?; |     create_dir_all(output_updates_path.parent().unwrap())?; | ||||||
|     let udpates_file = File::open(updates_path)?; |     let updates_file = File::open(updates_path)?; | ||||||
|     let mut output_update_file = File::create(output_updates_path)?; |     let mut output_update_file = File::create(output_updates_path)?; | ||||||
|  |  | ||||||
|     serde_json::Deserializer::from_reader(udpates_file) |     serde_json::Deserializer::from_reader(updates_file) | ||||||
|         .into_iter::<compat::v4::Task>() |         .into_iter::<compat::v4::Task>() | ||||||
|         .try_for_each(|task| -> anyhow::Result<()> { |         .try_for_each(|task| -> anyhow::Result<()> { | ||||||
|             let task: Task = task?.into(); |             let task: Task = task?.into(); | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ pub const DEFAULT_CROP_MARKER: fn() -> String = || "…".to_string(); | |||||||
| pub const DEFAULT_HIGHLIGHT_PRE_TAG: fn() -> String = || "<em>".to_string(); | pub const DEFAULT_HIGHLIGHT_PRE_TAG: fn() -> String = || "<em>".to_string(); | ||||||
| pub const DEFAULT_HIGHLIGHT_POST_TAG: fn() -> String = || "</em>".to_string(); | pub const DEFAULT_HIGHLIGHT_POST_TAG: fn() -> String = || "</em>".to_string(); | ||||||
|  |  | ||||||
| /// The maximimum number of results that the engine | /// The maximum number of results that the engine | ||||||
| /// will be able to return in one search call. | /// will be able to return in one search call. | ||||||
| pub const DEFAULT_PAGINATION_MAX_TOTAL_HITS: usize = 1000; | pub const DEFAULT_PAGINATION_MAX_TOTAL_HITS: usize = 1000; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -51,7 +51,7 @@ impl MapIndexStore { | |||||||
| #[async_trait::async_trait] | #[async_trait::async_trait] | ||||||
| impl IndexStore for MapIndexStore { | impl IndexStore for MapIndexStore { | ||||||
|     async fn create(&self, uuid: Uuid) -> Result<Index> { |     async fn create(&self, uuid: Uuid) -> Result<Index> { | ||||||
|         // We need to keep the lock until we are sure the db file has been opened correclty, to |         // We need to keep the lock until we are sure the db file has been opened correctly, to | ||||||
|         // ensure that another db is not created at the same time. |         // ensure that another db is not created at the same time. | ||||||
|         let mut lock = self.index_store.write().await; |         let mut lock = self.index_store.write().await; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -11,6 +11,8 @@ mod snapshot; | |||||||
| pub mod tasks; | pub mod tasks; | ||||||
| mod update_file_store; | mod update_file_store; | ||||||
|  |  | ||||||
|  | use std::env::VarError; | ||||||
|  | use std::ffi::OsStr; | ||||||
| use std::path::Path; | use std::path::Path; | ||||||
|  |  | ||||||
| pub use index_controller::MeiliSearch; | pub use index_controller::MeiliSearch; | ||||||
| @@ -35,3 +37,14 @@ pub fn is_empty_db(db_path: impl AsRef<Path>) -> bool { | |||||||
|         true |         true | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /// Checks if the key is defined in the environment variables. | ||||||
|  | /// If not, inserts it with the given value. | ||||||
|  | pub fn export_to_env_if_not_present<T>(key: &str, value: T) | ||||||
|  | where | ||||||
|  |     T: AsRef<OsStr>, | ||||||
|  | { | ||||||
|  |     if let Err(VarError::NotPresent) = std::env::var(key) { | ||||||
|  |         std::env::set_var(key, value); | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,33 +1,40 @@ | |||||||
|  | use crate::export_to_env_if_not_present; | ||||||
|  |  | ||||||
| use core::fmt; | use core::fmt; | ||||||
| use std::{convert::TryFrom, num::ParseIntError, ops::Deref, str::FromStr}; | use std::{convert::TryFrom, num::ParseIntError, ops::Deref, str::FromStr}; | ||||||
|  |  | ||||||
| use byte_unit::{Byte, ByteError}; | use byte_unit::{Byte, ByteError}; | ||||||
| use clap::Parser; | use clap::Parser; | ||||||
| use milli::update::IndexerConfig; | use milli::update::IndexerConfig; | ||||||
| use serde::Serialize; | use serde::{Deserialize, Serialize}; | ||||||
| use sysinfo::{RefreshKind, System, SystemExt}; | use sysinfo::{RefreshKind, System, SystemExt}; | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Parser, Serialize)] | const MEILI_MAX_INDEXING_MEMORY: &str = "MEILI_MAX_INDEXING_MEMORY"; | ||||||
|  | const MEILI_MAX_INDEXING_THREADS: &str = "MEILI_MAX_INDEXING_THREADS"; | ||||||
|  | const DISABLE_AUTO_BATCHING: &str = "DISABLE_AUTO_BATCHING"; | ||||||
|  | const DEFAULT_LOG_EVERY_N: usize = 100000; | ||||||
|  |  | ||||||
|  | #[derive(Debug, Clone, Parser, Serialize, Deserialize)] | ||||||
|  | #[serde(rename_all = "snake_case", deny_unknown_fields)] | ||||||
| pub struct IndexerOpts { | pub struct IndexerOpts { | ||||||
|     /// The amount of documents to skip before printing |     /// The amount of documents to skip before printing | ||||||
|     /// a log regarding the indexing advancement. |     /// a log regarding the indexing advancement. | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing, default = "default_log_every_n")] | ||||||
|     #[clap(long, default_value = "100000", hide = true)] // 100k |     #[clap(long, default_value_t = default_log_every_n(), hide = true)] // 100k | ||||||
|     pub log_every_n: usize, |     pub log_every_n: usize, | ||||||
|  |  | ||||||
|     /// Grenad max number of chunks in bytes. |     /// Grenad max number of chunks in bytes. | ||||||
|     #[serde(skip)] |     #[serde(skip_serializing)] | ||||||
|     #[clap(long, hide = true)] |     #[clap(long, hide = true)] | ||||||
|     pub max_nb_chunks: Option<usize>, |     pub max_nb_chunks: Option<usize>, | ||||||
|  |  | ||||||
|     /// The maximum amount of memory the indexer will use. It defaults to 2/3 |     /// The maximum amount of memory the indexer will use. | ||||||
|     /// of the available memory. It is recommended to use something like 80%-90% |  | ||||||
|     /// of the available memory, no more. |  | ||||||
|     /// |     /// | ||||||
|     /// In case the engine is unable to retrieve the available memory the engine will |     /// In case the engine is unable to retrieve the available memory the engine will | ||||||
|     /// try to use the memory it needs but without real limit, this can lead to |     /// try to use the memory it needs but without real limit, this can lead to | ||||||
|     /// Out-Of-Memory issues and it is recommended to specify the amount of memory to use. |     /// Out-Of-Memory issues and it is recommended to specify the amount of memory to use. | ||||||
|     #[clap(long, env = "MEILI_MAX_INDEXING_MEMORY", default_value_t)] |     #[clap(long, env = MEILI_MAX_INDEXING_MEMORY, default_value_t)] | ||||||
|  |     #[serde(default)] | ||||||
|     pub max_indexing_memory: MaxMemory, |     pub max_indexing_memory: MaxMemory, | ||||||
|  |  | ||||||
|     /// The maximum number of threads the indexer will use. |     /// The maximum number of threads the indexer will use. | ||||||
| @@ -35,18 +42,43 @@ pub struct IndexerOpts { | |||||||
|     /// it will use the maximum number of available cores. |     /// it will use the maximum number of available cores. | ||||||
|     /// |     /// | ||||||
|     /// It defaults to half of the available threads. |     /// It defaults to half of the available threads. | ||||||
|     #[clap(long, env = "MEILI_MAX_INDEXING_THREADS", default_value_t)] |     #[clap(long, env = MEILI_MAX_INDEXING_THREADS, default_value_t)] | ||||||
|  |     #[serde(default)] | ||||||
|     pub max_indexing_threads: MaxThreads, |     pub max_indexing_threads: MaxThreads, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Parser, Default, Serialize)] | #[derive(Debug, Clone, Parser, Default, Serialize, Deserialize)] | ||||||
|  | #[serde(rename_all = "snake_case", deny_unknown_fields)] | ||||||
| pub struct SchedulerConfig { | pub struct SchedulerConfig { | ||||||
|     /// The engine will disable task auto-batching, |     /// The engine will disable task auto-batching, | ||||||
|     /// and will sequencialy compute each task one by one. |     /// and will sequencialy compute each task one by one. | ||||||
|     #[clap(long, env = "DISABLE_AUTO_BATCHING")] |     #[clap(long, env = DISABLE_AUTO_BATCHING)] | ||||||
|  |     #[serde(default)] | ||||||
|     pub disable_auto_batching: bool, |     pub disable_auto_batching: bool, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | impl IndexerOpts { | ||||||
|  |     /// Exports the values to their corresponding env vars if they are not set. | ||||||
|  |     pub fn export_to_env(self) { | ||||||
|  |         let IndexerOpts { | ||||||
|  |             max_indexing_memory, | ||||||
|  |             max_indexing_threads, | ||||||
|  |             log_every_n: _, | ||||||
|  |             max_nb_chunks: _, | ||||||
|  |         } = self; | ||||||
|  |         if let Some(max_indexing_memory) = max_indexing_memory.0 { | ||||||
|  |             export_to_env_if_not_present( | ||||||
|  |                 MEILI_MAX_INDEXING_MEMORY, | ||||||
|  |                 max_indexing_memory.to_string(), | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |         export_to_env_if_not_present( | ||||||
|  |             MEILI_MAX_INDEXING_THREADS, | ||||||
|  |             max_indexing_threads.0.to_string(), | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| impl TryFrom<&IndexerOpts> for IndexerConfig { | impl TryFrom<&IndexerOpts> for IndexerConfig { | ||||||
|     type Error = anyhow::Error; |     type Error = anyhow::Error; | ||||||
|  |  | ||||||
| @@ -77,8 +109,17 @@ impl Default for IndexerOpts { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | impl SchedulerConfig { | ||||||
|  |     pub fn export_to_env(self) { | ||||||
|  |         let SchedulerConfig { | ||||||
|  |             disable_auto_batching, | ||||||
|  |         } = self; | ||||||
|  |         export_to_env_if_not_present(DISABLE_AUTO_BATCHING, disable_auto_batching.to_string()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| /// A type used to detect the max memory available and use 2/3 of it. | /// A type used to detect the max memory available and use 2/3 of it. | ||||||
| #[derive(Debug, Clone, Copy, Serialize)] | #[derive(Debug, Clone, Copy, Serialize, Deserialize)] | ||||||
| pub struct MaxMemory(Option<Byte>); | pub struct MaxMemory(Option<Byte>); | ||||||
|  |  | ||||||
| impl FromStr for MaxMemory { | impl FromStr for MaxMemory { | ||||||
| @@ -134,7 +175,7 @@ fn total_memory_bytes() -> Option<u64> { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Copy, Serialize)] | #[derive(Debug, Clone, Copy, Serialize, Deserialize)] | ||||||
| pub struct MaxThreads(usize); | pub struct MaxThreads(usize); | ||||||
|  |  | ||||||
| impl FromStr for MaxThreads { | impl FromStr for MaxThreads { | ||||||
| @@ -164,3 +205,7 @@ impl Deref for MaxThreads { | |||||||
|         &self.0 |         &self.0 | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | fn default_log_every_n() -> usize { | ||||||
|  |     DEFAULT_LOG_EVERY_N | ||||||
|  | } | ||||||
|   | |||||||
| @@ -63,7 +63,7 @@ impl Store { | |||||||
|     /// Returns the id for the next task. |     /// Returns the id for the next task. | ||||||
|     /// |     /// | ||||||
|     /// The required `mut txn` acts as a reservation system. It guarantees that as long as you commit |     /// The required `mut txn` acts as a reservation system. It guarantees that as long as you commit | ||||||
|     /// the task to the store in the same transaction, no one else will hav this task id. |     /// the task to the store in the same transaction, no one else will have this task id. | ||||||
|     pub fn next_task_id(&self, txn: &mut RwTxn) -> Result<TaskId> { |     pub fn next_task_id(&self, txn: &mut RwTxn) -> Result<TaskId> { | ||||||
|         let id = self |         let id = self | ||||||
|             .tasks |             .tasks | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user