mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	Merge branch 'main' into request-fragments-test
This commit is contained in:
		
							
								
								
									
										2
									
								
								.github/workflows/publish-apt-brew-pkg.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/publish-apt-brew-pkg.yml
									
									
									
									
										vendored
									
									
								
							| @@ -32,7 +32,7 @@ jobs: | ||||
|       - name: Build deb package | ||||
|         run: cargo deb -p meilisearch -o target/debian/meilisearch.deb | ||||
|       - name: Upload debian pkg to release | ||||
|         uses: svenstaro/upload-release-action@2.7.0 | ||||
|         uses: svenstaro/upload-release-action@2.11.1 | ||||
|         with: | ||||
|           repo_token: ${{ secrets.MEILI_BOT_GH_PAT }} | ||||
|           file: target/debian/meilisearch.deb | ||||
|   | ||||
							
								
								
									
										8
									
								
								.github/workflows/publish-binaries.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/publish-binaries.yml
									
									
									
									
										vendored
									
									
								
							| @@ -51,7 +51,7 @@ jobs: | ||||
|       # No need to upload binaries for dry run (cron) | ||||
|       - name: Upload binaries to release | ||||
|         if: github.event_name == 'release' | ||||
|         uses: svenstaro/upload-release-action@2.7.0 | ||||
|         uses: svenstaro/upload-release-action@2.11.1 | ||||
|         with: | ||||
|           repo_token: ${{ secrets.MEILI_BOT_GH_PAT }} | ||||
|           file: target/release/meilisearch | ||||
| @@ -81,7 +81,7 @@ jobs: | ||||
|       # No need to upload binaries for dry run (cron) | ||||
|       - name: Upload binaries to release | ||||
|         if: github.event_name == 'release' | ||||
|         uses: svenstaro/upload-release-action@2.7.0 | ||||
|         uses: svenstaro/upload-release-action@2.11.1 | ||||
|         with: | ||||
|           repo_token: ${{ secrets.MEILI_BOT_GH_PAT }} | ||||
|           file: target/release/${{ matrix.artifact_name }} | ||||
| @@ -113,7 +113,7 @@ jobs: | ||||
|       - name: Upload the binary to release | ||||
|         # No need to upload binaries for dry run (cron) | ||||
|         if: github.event_name == 'release' | ||||
|         uses: svenstaro/upload-release-action@2.7.0 | ||||
|         uses: svenstaro/upload-release-action@2.11.1 | ||||
|         with: | ||||
|           repo_token: ${{ secrets.MEILI_BOT_GH_PAT }} | ||||
|           file: target/${{ matrix.target }}/release/meilisearch | ||||
| @@ -178,7 +178,7 @@ jobs: | ||||
|       - name: Upload the binary to release | ||||
|         # No need to upload binaries for dry run (cron) | ||||
|         if: github.event_name == 'release' | ||||
|         uses: svenstaro/upload-release-action@2.7.0 | ||||
|         uses: svenstaro/upload-release-action@2.11.1 | ||||
|         with: | ||||
|           repo_token: ${{ secrets.MEILI_BOT_GH_PAT }} | ||||
|           file: target/${{ matrix.target }}/release/meilisearch | ||||
|   | ||||
							
								
								
									
										10
									
								
								.github/workflows/test-suite.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/test-suite.yml
									
									
									
									
										vendored
									
									
								
							| @@ -29,7 +29,7 @@ jobs: | ||||
|       - name: Setup test with Rust stable | ||||
|         uses: dtolnay/rust-toolchain@1.85 | ||||
|       - name: Cache dependencies | ||||
|         uses: Swatinem/rust-cache@v2.7.8 | ||||
|         uses: Swatinem/rust-cache@v2.8.0 | ||||
|       - name: Run cargo check without any default features | ||||
|         uses: actions-rs/cargo@v1 | ||||
|         with: | ||||
| @@ -51,7 +51,7 @@ jobs: | ||||
|     steps: | ||||
|       - uses: actions/checkout@v3 | ||||
|       - name: Cache dependencies | ||||
|         uses: Swatinem/rust-cache@v2.7.8 | ||||
|         uses: Swatinem/rust-cache@v2.8.0 | ||||
|       - uses: dtolnay/rust-toolchain@1.85 | ||||
|       - name: Run cargo check without any default features | ||||
|         uses: actions-rs/cargo@v1 | ||||
| @@ -155,7 +155,7 @@ jobs: | ||||
|           apt-get install build-essential -y | ||||
|       - uses: dtolnay/rust-toolchain@1.85 | ||||
|       - name: Cache dependencies | ||||
|         uses: Swatinem/rust-cache@v2.7.8 | ||||
|         uses: Swatinem/rust-cache@v2.8.0 | ||||
|       - name: Run tests in debug | ||||
|         uses: actions-rs/cargo@v1 | ||||
|         with: | ||||
| @@ -172,7 +172,7 @@ jobs: | ||||
|           profile: minimal | ||||
|           components: clippy | ||||
|       - name: Cache dependencies | ||||
|         uses: Swatinem/rust-cache@v2.7.8 | ||||
|         uses: Swatinem/rust-cache@v2.8.0 | ||||
|       - name: Run cargo clippy | ||||
|         uses: actions-rs/cargo@v1 | ||||
|         with: | ||||
| @@ -191,7 +191,7 @@ jobs: | ||||
|           override: true | ||||
|           components: rustfmt | ||||
|       - name: Cache dependencies | ||||
|         uses: Swatinem/rust-cache@v2.7.8 | ||||
|         uses: Swatinem/rust-cache@v2.8.0 | ||||
|       - name: Run cargo fmt | ||||
|         # Since we never ran the `build.rs` script in the benchmark directory we are missing one auto-generated import file. | ||||
|         # Since we want to trigger (and fail) this action as fast as possible, instead of building the benchmark crate | ||||
|   | ||||
							
								
								
									
										34
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										34
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -580,7 +580,7 @@ source = "git+https://github.com/meilisearch/bbqueue#cbb87cc707b5af415ef203bdaf2 | ||||
|  | ||||
| [[package]] | ||||
| name = "benchmarks" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "bumpalo", | ||||
| @@ -770,7 +770,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "build-info" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "time", | ||||
| @@ -1774,7 +1774,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "dump" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "big_s", | ||||
| @@ -2006,7 +2006,7 @@ checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be" | ||||
|  | ||||
| [[package]] | ||||
| name = "file-store" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "tempfile", | ||||
|  "thiserror 2.0.12", | ||||
| @@ -2028,7 +2028,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "filter-parser" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "insta", | ||||
|  "nom", | ||||
| @@ -2049,7 +2049,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "flatten-serde-json" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "criterion", | ||||
|  "serde_json", | ||||
| @@ -2194,7 +2194,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "fuzzers" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "arbitrary", | ||||
|  "bumpalo", | ||||
| @@ -2994,7 +2994,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "index-scheduler" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "backoff", | ||||
| @@ -3230,7 +3230,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "json-depth-checker" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "criterion", | ||||
|  "serde_json", | ||||
| @@ -3724,7 +3724,7 @@ checksum = "490cc448043f947bae3cbee9c203358d62dbee0db12107a74be5c30ccfd09771" | ||||
|  | ||||
| [[package]] | ||||
| name = "meili-snap" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "insta", | ||||
|  "md5", | ||||
| @@ -3735,7 +3735,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "meilisearch" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "actix-cors", | ||||
|  "actix-http", | ||||
| @@ -3830,7 +3830,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "meilisearch-auth" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "base64 0.22.1", | ||||
|  "enum-iterator", | ||||
| @@ -3849,7 +3849,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "meilisearch-types" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "actix-web", | ||||
|  "anyhow", | ||||
| @@ -3884,7 +3884,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "meilitool" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "clap", | ||||
| @@ -3918,7 +3918,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "milli" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "allocator-api2 0.3.0", | ||||
|  "arroy", | ||||
| @@ -4470,7 +4470,7 @@ checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" | ||||
|  | ||||
| [[package]] | ||||
| name = "permissive-json-pointer" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "big_s", | ||||
|  "serde_json", | ||||
| @@ -7258,7 +7258,7 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "xtask" | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| dependencies = [ | ||||
|  "anyhow", | ||||
|  "build-info", | ||||
|   | ||||
| @@ -22,7 +22,7 @@ members = [ | ||||
| ] | ||||
|  | ||||
| [workspace.package] | ||||
| version = "1.15.2" | ||||
| version = "1.16.0" | ||||
| authors = [ | ||||
|     "Quentin de Quelen <quentin@dequelen.me>", | ||||
|     "Clément Renault <clement@meilisearch.com>", | ||||
|   | ||||
| @@ -116,6 +116,15 @@ impl DumpReader { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn chat_completions_settings( | ||||
|         &mut self, | ||||
|     ) -> Result<Box<dyn Iterator<Item = Result<(String, v6::ChatCompletionSettings)>> + '_>> { | ||||
|         match self { | ||||
|             DumpReader::Current(current) => current.chat_completions_settings(), | ||||
|             DumpReader::Compat(_compat) => Ok(Box::new(std::iter::empty())), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn features(&self) -> Result<Option<v6::RuntimeTogglableFeatures>> { | ||||
|         match self { | ||||
|             DumpReader::Current(current) => Ok(current.features()), | ||||
|   | ||||
| @@ -1,3 +1,4 @@ | ||||
| use std::ffi::OsStr; | ||||
| use std::fs::{self, File}; | ||||
| use std::io::{BufRead, BufReader, ErrorKind}; | ||||
| use std::path::Path; | ||||
| @@ -21,6 +22,7 @@ pub type Unchecked = meilisearch_types::settings::Unchecked; | ||||
| pub type Task = crate::TaskDump; | ||||
| pub type Batch = meilisearch_types::batches::Batch; | ||||
| pub type Key = meilisearch_types::keys::Key; | ||||
| pub type ChatCompletionSettings = meilisearch_types::features::ChatCompletionSettings; | ||||
| pub type RuntimeTogglableFeatures = meilisearch_types::features::RuntimeTogglableFeatures; | ||||
| pub type Network = meilisearch_types::features::Network; | ||||
|  | ||||
| @@ -192,6 +194,34 @@ impl V6Reader { | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     pub fn chat_completions_settings( | ||||
|         &mut self, | ||||
|     ) -> Result<Box<dyn Iterator<Item = Result<(String, ChatCompletionSettings)>> + '_>> { | ||||
|         let entries = match fs::read_dir(self.dump.path().join("chat-completions-settings")) { | ||||
|             Ok(entries) => entries, | ||||
|             Err(e) if e.kind() == ErrorKind::NotFound => return Ok(Box::new(std::iter::empty())), | ||||
|             Err(e) => return Err(e.into()), | ||||
|         }; | ||||
|         Ok(Box::new( | ||||
|             entries | ||||
|                 .map(|entry| -> Result<Option<_>> { | ||||
|                     let entry = entry?; | ||||
|                     let file_name = entry.file_name(); | ||||
|                     let path = Path::new(&file_name); | ||||
|                     if entry.file_type()?.is_file() && path.extension() == Some(OsStr::new("json")) | ||||
|                     { | ||||
|                         let name = path.file_stem().unwrap().to_str().unwrap().to_string(); | ||||
|                         let file = File::open(entry.path())?; | ||||
|                         let settings = serde_json::from_reader(file)?; | ||||
|                         Ok(Some((name, settings))) | ||||
|                     } else { | ||||
|                         Ok(None) | ||||
|                     } | ||||
|                 }) | ||||
|                 .filter_map(|entry| entry.transpose()), | ||||
|         )) | ||||
|     } | ||||
|  | ||||
|     pub fn features(&self) -> Option<RuntimeTogglableFeatures> { | ||||
|         self.features | ||||
|     } | ||||
|   | ||||
| @@ -5,7 +5,7 @@ use std::path::PathBuf; | ||||
| use flate2::write::GzEncoder; | ||||
| use flate2::Compression; | ||||
| use meilisearch_types::batches::Batch; | ||||
| use meilisearch_types::features::{Network, RuntimeTogglableFeatures}; | ||||
| use meilisearch_types::features::{ChatCompletionSettings, Network, RuntimeTogglableFeatures}; | ||||
| use meilisearch_types::keys::Key; | ||||
| use meilisearch_types::settings::{Checked, Settings}; | ||||
| use serde_json::{Map, Value}; | ||||
| @@ -51,6 +51,10 @@ impl DumpWriter { | ||||
|         KeyWriter::new(self.dir.path().to_path_buf()) | ||||
|     } | ||||
|  | ||||
|     pub fn create_chat_completions_settings(&self) -> Result<ChatCompletionsSettingsWriter> { | ||||
|         ChatCompletionsSettingsWriter::new(self.dir.path().join("chat-completions-settings")) | ||||
|     } | ||||
|  | ||||
|     pub fn create_tasks_queue(&self) -> Result<TaskWriter> { | ||||
|         TaskWriter::new(self.dir.path().join("tasks")) | ||||
|     } | ||||
| @@ -104,6 +108,24 @@ impl KeyWriter { | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct ChatCompletionsSettingsWriter { | ||||
|     path: PathBuf, | ||||
| } | ||||
|  | ||||
| impl ChatCompletionsSettingsWriter { | ||||
|     pub(crate) fn new(path: PathBuf) -> Result<Self> { | ||||
|         std::fs::create_dir(&path)?; | ||||
|         Ok(ChatCompletionsSettingsWriter { path }) | ||||
|     } | ||||
|  | ||||
|     pub fn push_settings(&mut self, name: &str, settings: &ChatCompletionSettings) -> Result<()> { | ||||
|         let mut settings_file = File::create(self.path.join(name).with_extension("json"))?; | ||||
|         serde_json::to_writer(&mut settings_file, &settings)?; | ||||
|         settings_file.flush()?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct TaskWriter { | ||||
|     queue: BufWriter<File>, | ||||
|     update_files: PathBuf, | ||||
|   | ||||
| @@ -103,6 +103,7 @@ make_enum_progress! { | ||||
|     pub enum DumpCreationProgress { | ||||
|         StartTheDumpCreation, | ||||
|         DumpTheApiKeys, | ||||
|         DumpTheChatCompletionSettings, | ||||
|         DumpTheTasks, | ||||
|         DumpTheBatches, | ||||
|         DumpTheIndexes, | ||||
|   | ||||
| @@ -43,7 +43,16 @@ impl IndexScheduler { | ||||
|  | ||||
|         let rtxn = self.env.read_txn()?; | ||||
|  | ||||
|         // 2. dump the tasks | ||||
|         // 2. dump the chat completion settings | ||||
|         // TODO should I skip the export if the chat completion has been disabled? | ||||
|         progress.update_progress(DumpCreationProgress::DumpTheChatCompletionSettings); | ||||
|         let mut dump_chat_completion_settings = dump.create_chat_completions_settings()?; | ||||
|         for result in self.chat_settings.iter(&rtxn)? { | ||||
|             let (name, chat_settings) = result?; | ||||
|             dump_chat_completion_settings.push_settings(name, &chat_settings)?; | ||||
|         } | ||||
|  | ||||
|         // 3. dump the tasks | ||||
|         progress.update_progress(DumpCreationProgress::DumpTheTasks); | ||||
|         let mut dump_tasks = dump.create_tasks_queue()?; | ||||
|  | ||||
| @@ -81,7 +90,7 @@ impl IndexScheduler { | ||||
|  | ||||
|             let mut dump_content_file = dump_tasks.push_task(&t.into())?; | ||||
|  | ||||
|             // 2.1. Dump the `content_file` associated with the task if there is one and the task is not finished yet. | ||||
|             // 3.1. Dump the `content_file` associated with the task if there is one and the task is not finished yet. | ||||
|             if let Some(content_file) = content_file { | ||||
|                 if self.scheduler.must_stop_processing.get() { | ||||
|                     return Err(Error::AbortedTask); | ||||
| @@ -105,7 +114,7 @@ impl IndexScheduler { | ||||
|         } | ||||
|         dump_tasks.flush()?; | ||||
|  | ||||
|         // 3. dump the batches | ||||
|         // 4. dump the batches | ||||
|         progress.update_progress(DumpCreationProgress::DumpTheBatches); | ||||
|         let mut dump_batches = dump.create_batches_queue()?; | ||||
|  | ||||
| @@ -138,7 +147,7 @@ impl IndexScheduler { | ||||
|         } | ||||
|         dump_batches.flush()?; | ||||
|  | ||||
|         // 4. Dump the indexes | ||||
|         // 5. Dump the indexes | ||||
|         progress.update_progress(DumpCreationProgress::DumpTheIndexes); | ||||
|         let nb_indexes = self.index_mapper.index_mapping.len(&rtxn)? as u32; | ||||
|         let mut count = 0; | ||||
| @@ -175,7 +184,7 @@ impl IndexScheduler { | ||||
|             let documents = index | ||||
|                 .all_documents(&rtxn) | ||||
|                 .map_err(|e| Error::from_milli(e, Some(uid.to_string())))?; | ||||
|             // 4.1. Dump the documents | ||||
|             // 5.1. Dump the documents | ||||
|             for ret in documents { | ||||
|                 if self.scheduler.must_stop_processing.get() { | ||||
|                     return Err(Error::AbortedTask); | ||||
| @@ -233,7 +242,7 @@ impl IndexScheduler { | ||||
|                 atomic.fetch_add(1, Ordering::Relaxed); | ||||
|             } | ||||
|  | ||||
|             // 4.2. Dump the settings | ||||
|             // 5.2. Dump the settings | ||||
|             let settings = meilisearch_types::settings::settings( | ||||
|                 index, | ||||
|                 &rtxn, | ||||
| @@ -244,7 +253,7 @@ impl IndexScheduler { | ||||
|             Ok(()) | ||||
|         })?; | ||||
|  | ||||
|         // 5. Dump experimental feature settings | ||||
|         // 6. Dump experimental feature settings | ||||
|         progress.update_progress(DumpCreationProgress::DumpTheExperimentalFeatures); | ||||
|         let features = self.features().runtime_features(); | ||||
|         dump.create_experimental_features(features)?; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, batch_uid: 0, status: succeeded, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, batch_uid: 0, status: succeeded, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 1 {uid: 1, batch_uid: 1, status: succeeded, details: { primary_key: Some("mouse") }, kind: IndexCreation { index_uid: "catto", primary_key: Some("mouse") }} | ||||
| 2 {uid: 2, batch_uid: 2, status: succeeded, details: { primary_key: Some("bone") }, kind: IndexCreation { index_uid: "doggo", primary_key: Some("bone") }} | ||||
| 3 {uid: 3, batch_uid: 3, status: failed, error: ResponseError { code: 200, message: "Index `doggo` already exists.", error_code: "index_already_exists", error_type: "invalid_request", error_link: "https://docs.meilisearch.com/errors#index_already_exists" }, details: { primary_key: Some("bone") }, kind: IndexCreation { index_uid: "doggo", primary_key: Some("bone") }} | ||||
| @@ -57,7 +57,7 @@ girafo: { number_of_documents: 0, field_distribution: {} } | ||||
| [timestamp] [4,] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Batches: | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.15.2"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.16.0"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| 1 {uid: 1, details: {"primaryKey":"mouse"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"indexCreation":1},"indexUids":{"catto":1}}, stop reason: "created batch containing only task with id 1 of type `indexCreation` that cannot be batched with any other task.", } | ||||
| 2 {uid: 2, details: {"primaryKey":"bone"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"indexCreation":1},"indexUids":{"doggo":1}}, stop reason: "created batch containing only task with id 2 of type `indexCreation` that cannot be batched with any other task.", } | ||||
| 3 {uid: 3, details: {"primaryKey":"bone"}, stats: {"totalNbTasks":1,"status":{"failed":1},"types":{"indexCreation":1},"indexUids":{"doggo":1}}, stop reason: "created batch containing only task with id 3 of type `indexCreation` that cannot be batched with any other task.", } | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, status: enqueued, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, status: enqueued, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| ---------------------------------------------------------------------- | ||||
| ### Status: | ||||
| enqueued [0,] | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, status: enqueued, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, status: enqueued, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 1 {uid: 1, status: enqueued, details: { primary_key: Some("mouse") }, kind: IndexCreation { index_uid: "catto", primary_key: Some("mouse") }} | ||||
| ---------------------------------------------------------------------- | ||||
| ### Status: | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, batch_uid: 0, status: failed, error: ResponseError { code: 200, message: "Planned failure for tests.", error_code: "internal", error_type: "internal", error_link: "https://docs.meilisearch.com/errors#internal" }, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, batch_uid: 0, status: failed, error: ResponseError { code: 200, message: "Planned failure for tests.", error_code: "internal", error_type: "internal", error_link: "https://docs.meilisearch.com/errors#internal" }, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 1 {uid: 1, status: enqueued, details: { primary_key: Some("mouse") }, kind: IndexCreation { index_uid: "catto", primary_key: Some("mouse") }} | ||||
| ---------------------------------------------------------------------- | ||||
| ### Status: | ||||
| @@ -37,7 +37,7 @@ catto [1,] | ||||
| [timestamp] [0,] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Batches: | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.15.2"}, stats: {"totalNbTasks":1,"status":{"failed":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.16.0"}, stats: {"totalNbTasks":1,"status":{"failed":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| ---------------------------------------------------------------------- | ||||
| ### Batch to tasks mapping: | ||||
| 0 [0,] | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, batch_uid: 0, status: failed, error: ResponseError { code: 200, message: "Planned failure for tests.", error_code: "internal", error_type: "internal", error_link: "https://docs.meilisearch.com/errors#internal" }, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, batch_uid: 0, status: failed, error: ResponseError { code: 200, message: "Planned failure for tests.", error_code: "internal", error_type: "internal", error_link: "https://docs.meilisearch.com/errors#internal" }, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 1 {uid: 1, status: enqueued, details: { primary_key: Some("mouse") }, kind: IndexCreation { index_uid: "catto", primary_key: Some("mouse") }} | ||||
| 2 {uid: 2, status: enqueued, details: { primary_key: Some("bone") }, kind: IndexCreation { index_uid: "doggo", primary_key: Some("bone") }} | ||||
| ---------------------------------------------------------------------- | ||||
| @@ -40,7 +40,7 @@ doggo [2,] | ||||
| [timestamp] [0,] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Batches: | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.15.2"}, stats: {"totalNbTasks":1,"status":{"failed":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.16.0"}, stats: {"totalNbTasks":1,"status":{"failed":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| ---------------------------------------------------------------------- | ||||
| ### Batch to tasks mapping: | ||||
| 0 [0,] | ||||
|   | ||||
| @@ -6,7 +6,7 @@ source: crates/index-scheduler/src/scheduler/test_failure.rs | ||||
| [] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Tasks: | ||||
| 0 {uid: 0, batch_uid: 0, status: succeeded, details: { from: (1, 12, 0), to: (1, 15, 2) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 0 {uid: 0, batch_uid: 0, status: succeeded, details: { from: (1, 12, 0), to: (1, 16, 0) }, kind: UpgradeDatabase { from: (1, 12, 0) }} | ||||
| 1 {uid: 1, status: enqueued, details: { primary_key: Some("mouse") }, kind: IndexCreation { index_uid: "catto", primary_key: Some("mouse") }} | ||||
| 2 {uid: 2, status: enqueued, details: { primary_key: Some("bone") }, kind: IndexCreation { index_uid: "doggo", primary_key: Some("bone") }} | ||||
| 3 {uid: 3, status: enqueued, details: { primary_key: Some("bone") }, kind: IndexCreation { index_uid: "doggo", primary_key: Some("bone") }} | ||||
| @@ -43,7 +43,7 @@ doggo [2,3,] | ||||
| [timestamp] [0,] | ||||
| ---------------------------------------------------------------------- | ||||
| ### All Batches: | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.15.2"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| 0 {uid: 0, details: {"upgradeFrom":"v1.12.0","upgradeTo":"v1.16.0"}, stats: {"totalNbTasks":1,"status":{"succeeded":1},"types":{"upgradeDatabase":1},"indexUids":{}}, stop reason: "stopped after the last task of type `upgradeDatabase` because they cannot be batched with tasks of any other type.", } | ||||
| ---------------------------------------------------------------------- | ||||
| ### Batch to tasks mapping: | ||||
| 0 [0,] | ||||
|   | ||||
| @@ -169,5 +169,5 @@ german = ["meilisearch-types/german"] | ||||
| turkish = ["meilisearch-types/turkish"] | ||||
|  | ||||
| [package.metadata.mini-dashboard] | ||||
| assets-url = "https://github.com/meilisearch/mini-dashboard/releases/download/v0.2.19/build.zip" | ||||
| sha1 = "7974430d5277c97f67cf6e95eec6faaac2788834" | ||||
| assets-url = "https://github.com/meilisearch/mini-dashboard/releases/download/v0.2.20/build.zip" | ||||
| sha1 = "82a7ddd7bf14bb5323c3d235d2b62892a98b6a59" | ||||
|   | ||||
| @@ -498,14 +498,20 @@ fn import_dump( | ||||
|         keys.push(key); | ||||
|     } | ||||
|  | ||||
|     // 3. Import the runtime features and network | ||||
|     // 3. Import the `ChatCompletionSettings`s. | ||||
|     for result in dump_reader.chat_completions_settings()? { | ||||
|         let (name, settings) = result?; | ||||
|         index_scheduler.put_chat_settings(&name, &settings)?; | ||||
|     } | ||||
|  | ||||
|     // 4. Import the runtime features and network | ||||
|     let features = dump_reader.features()?.unwrap_or_default(); | ||||
|     index_scheduler.put_runtime_features(features)?; | ||||
|  | ||||
|     let network = dump_reader.network()?.cloned().unwrap_or_default(); | ||||
|     index_scheduler.put_network(network)?; | ||||
|  | ||||
|     // 3.1 Use all cpus to process dump if `max_indexing_threads` not configured | ||||
|     // 4.1 Use all cpus to process dump if `max_indexing_threads` not configured | ||||
|     let backup_config; | ||||
|     let base_config = index_scheduler.indexer_config(); | ||||
|  | ||||
| @@ -522,7 +528,7 @@ fn import_dump( | ||||
|     // /!\ The tasks must be imported AFTER importing the indexes or else the scheduler might | ||||
|     // try to process tasks while we're trying to import the indexes. | ||||
|  | ||||
|     // 4. Import the indexes. | ||||
|     // 5. Import the indexes. | ||||
|     for index_reader in dump_reader.indexes()? { | ||||
|         let mut index_reader = index_reader?; | ||||
|         let metadata = index_reader.metadata(); | ||||
| @@ -535,20 +541,20 @@ fn import_dump( | ||||
|         let mut wtxn = index.write_txn()?; | ||||
|  | ||||
|         let mut builder = milli::update::Settings::new(&mut wtxn, &index, indexer_config); | ||||
|         // 4.1 Import the primary key if there is one. | ||||
|         // 5.1 Import the primary key if there is one. | ||||
|         if let Some(ref primary_key) = metadata.primary_key { | ||||
|             builder.set_primary_key(primary_key.to_string()); | ||||
|         } | ||||
|  | ||||
|         // 4.2 Import the settings. | ||||
|         // 5.2 Import the settings. | ||||
|         tracing::info!("Importing the settings."); | ||||
|         let settings = index_reader.settings()?; | ||||
|         apply_settings_to_builder(&settings, &mut builder); | ||||
|         let embedder_stats: Arc<EmbedderStats> = Default::default(); | ||||
|         builder.execute(&|| false, &progress, embedder_stats.clone())?; | ||||
|  | ||||
|         // 4.3 Import the documents. | ||||
|         // 4.3.1 We need to recreate the grenad+obkv format accepted by the index. | ||||
|         // 5.3 Import the documents. | ||||
|         // 5.3.1 We need to recreate the grenad+obkv format accepted by the index. | ||||
|         tracing::info!("Importing the documents."); | ||||
|         let file = tempfile::tempfile()?; | ||||
|         let mut builder = DocumentsBatchBuilder::new(BufWriter::new(file)); | ||||
| @@ -559,7 +565,7 @@ fn import_dump( | ||||
|         // This flush the content of the batch builder. | ||||
|         let file = builder.into_inner()?.into_inner()?; | ||||
|  | ||||
|         // 4.3.2 We feed it to the milli index. | ||||
|         // 5.3.2 We feed it to the milli index. | ||||
|         let reader = BufReader::new(file); | ||||
|         let reader = DocumentsBatchReader::from_reader(reader)?; | ||||
|  | ||||
| @@ -591,15 +597,15 @@ fn import_dump( | ||||
|         index_scheduler.refresh_index_stats(&uid)?; | ||||
|     } | ||||
|  | ||||
|     // 5. Import the queue | ||||
|     // 6. Import the queue | ||||
|     let mut index_scheduler_dump = index_scheduler.register_dumped_task()?; | ||||
|     // 5.1. Import the batches | ||||
|     // 6.1. Import the batches | ||||
|     for ret in dump_reader.batches()? { | ||||
|         let batch = ret?; | ||||
|         index_scheduler_dump.register_dumped_batch(batch)?; | ||||
|     } | ||||
|  | ||||
|     // 5.2. Import the tasks | ||||
|     // 6.2. Import the tasks | ||||
|     for ret in dump_reader.tasks()? { | ||||
|         let (task, file) = ret?; | ||||
|         index_scheduler_dump.register_dumped_task(task, file)?; | ||||
|   | ||||
| @@ -15,6 +15,33 @@ lazy_static! { | ||||
|         "Meilisearch number of degraded search requests" | ||||
|     )) | ||||
|     .expect("Can't create a metric"); | ||||
|     pub static ref MEILISEARCH_CHAT_SEARCH_REQUESTS: IntCounterVec = register_int_counter_vec!( | ||||
|         opts!( | ||||
|             "meilisearch_chat_search_requests", | ||||
|             "Meilisearch number of search requests performed by the chat route itself" | ||||
|         ), | ||||
|         &["type"] | ||||
|     ) | ||||
|     .expect("Can't create a metric"); | ||||
|     pub static ref MEILISEARCH_CHAT_PROMPT_TOKENS_USAGE: IntCounterVec = register_int_counter_vec!( | ||||
|         opts!("meilisearch_chat_prompt_tokens_usage", "Meilisearch Chat Prompt Tokens Usage"), | ||||
|         &["workspace", "model"] | ||||
|     ) | ||||
|     .expect("Can't create a metric"); | ||||
|     pub static ref MEILISEARCH_CHAT_COMPLETION_TOKENS_USAGE: IntCounterVec = | ||||
|         register_int_counter_vec!( | ||||
|             opts!( | ||||
|                 "meilisearch_chat_completion_tokens_usage", | ||||
|                 "Meilisearch Chat Completion Tokens Usage" | ||||
|             ), | ||||
|             &["workspace", "model"] | ||||
|         ) | ||||
|         .expect("Can't create a metric"); | ||||
|     pub static ref MEILISEARCH_CHAT_TOTAL_TOKENS_USAGE: IntCounterVec = register_int_counter_vec!( | ||||
|         opts!("meilisearch_chat_total_tokens_usage", "Meilisearch Chat Total Tokens Usage"), | ||||
|         &["workspace", "model"] | ||||
|     ) | ||||
|     .expect("Can't create a metric"); | ||||
|     pub static ref MEILISEARCH_DB_SIZE_BYTES: IntGauge = | ||||
|         register_int_gauge!(opts!("meilisearch_db_size_bytes", "Meilisearch DB Size In Bytes")) | ||||
|             .expect("Can't create a metric"); | ||||
|   | ||||
| @@ -13,9 +13,9 @@ use async_openai::types::{ | ||||
|     ChatCompletionRequestDeveloperMessageContent, ChatCompletionRequestMessage, | ||||
|     ChatCompletionRequestSystemMessage, ChatCompletionRequestSystemMessageContent, | ||||
|     ChatCompletionRequestToolMessage, ChatCompletionRequestToolMessageContent, | ||||
|     ChatCompletionStreamResponseDelta, ChatCompletionToolArgs, ChatCompletionToolType, | ||||
|     CreateChatCompletionRequest, CreateChatCompletionStreamResponse, FinishReason, FunctionCall, | ||||
|     FunctionCallStream, FunctionObjectArgs, | ||||
|     ChatCompletionStreamOptions, ChatCompletionStreamResponseDelta, ChatCompletionToolArgs, | ||||
|     ChatCompletionToolType, CreateChatCompletionRequest, CreateChatCompletionStreamResponse, | ||||
|     FinishReason, FunctionCall, FunctionCallStream, FunctionObjectArgs, | ||||
| }; | ||||
| use async_openai::Client; | ||||
| use bumpalo::Bump; | ||||
| @@ -48,7 +48,11 @@ use crate::analytics::Analytics; | ||||
| use crate::error::MeilisearchHttpError; | ||||
| use crate::extractors::authentication::policies::ActionPolicy; | ||||
| use crate::extractors::authentication::{extract_token_from_request, GuardedData, Policy as _}; | ||||
| use crate::metrics::MEILISEARCH_DEGRADED_SEARCH_REQUESTS; | ||||
| use crate::metrics::{ | ||||
|     MEILISEARCH_CHAT_COMPLETION_TOKENS_USAGE, MEILISEARCH_CHAT_PROMPT_TOKENS_USAGE, | ||||
|     MEILISEARCH_CHAT_SEARCH_REQUESTS, MEILISEARCH_CHAT_TOTAL_TOKENS_USAGE, | ||||
|     MEILISEARCH_DEGRADED_SEARCH_REQUESTS, | ||||
| }; | ||||
| use crate::routes::chats::utils::SseEventSender; | ||||
| use crate::routes::indexes::search::search_kind; | ||||
| use crate::search::{add_search_rules, prepare_search, search_from_kind, SearchQuery}; | ||||
| @@ -286,7 +290,7 @@ async fn process_search_request( | ||||
|     let output = output?; | ||||
|     let mut documents = Vec::new(); | ||||
|     if let Ok((ref rtxn, ref search_result)) = output { | ||||
|         // aggregate.succeed(search_result); | ||||
|         MEILISEARCH_CHAT_SEARCH_REQUESTS.with_label_values(&["internal"]).inc(); | ||||
|         if search_result.degraded { | ||||
|             MEILISEARCH_DEGRADED_SEARCH_REQUESTS.inc(); | ||||
|         } | ||||
| @@ -488,6 +492,7 @@ async fn streamed_chat( | ||||
|  | ||||
|     let (tx, rx) = tokio::sync::mpsc::channel(10); | ||||
|     let tx = SseEventSender::new(tx); | ||||
|     let workspace_uid = workspace_uid.to_string(); | ||||
|     let _join_handle = Handle::current().spawn(async move { | ||||
|         let client = Client::with_config(config.clone()); | ||||
|         let mut global_tool_calls = HashMap::<u32, Call>::new(); | ||||
| @@ -497,6 +502,7 @@ async fn streamed_chat( | ||||
|             let output = run_conversation( | ||||
|                 &index_scheduler, | ||||
|                 &auth_ctrl, | ||||
|                 &workspace_uid, | ||||
|                 &search_queue, | ||||
|                 &auth_token, | ||||
|                 &client, | ||||
| @@ -534,6 +540,7 @@ async fn run_conversation<C: async_openai::config::Config>( | ||||
|         Data<IndexScheduler>, | ||||
|     >, | ||||
|     auth_ctrl: &web::Data<AuthController>, | ||||
|     workspace_uid: &str, | ||||
|     search_queue: &web::Data<SearchQueue>, | ||||
|     auth_token: &str, | ||||
|     client: &Client<C>, | ||||
| @@ -543,13 +550,34 @@ async fn run_conversation<C: async_openai::config::Config>( | ||||
|     global_tool_calls: &mut HashMap<u32, Call>, | ||||
|     function_support: FunctionSupport, | ||||
| ) -> Result<ControlFlow<Option<FinishReason>, ()>, SendError<Event>> { | ||||
|     use DbChatCompletionSource::*; | ||||
|  | ||||
|     let mut finish_reason = None; | ||||
|     chat_completion.stream_options = match source { | ||||
|         OpenAi | AzureOpenAi => Some(ChatCompletionStreamOptions { include_usage: true }), | ||||
|         Mistral | VLlm => None, | ||||
|     }; | ||||
|  | ||||
|     // safety: unwrap: can only happens if `stream` was set to `false` | ||||
|     let mut response = client.chat().create_stream(chat_completion.clone()).await.unwrap(); | ||||
|     while let Some(result) = response.next().await { | ||||
|         match result { | ||||
|             Ok(resp) => { | ||||
|                 let choice = &resp.choices[0]; | ||||
|                 if let Some(usage) = resp.usage.as_ref() { | ||||
|                     MEILISEARCH_CHAT_PROMPT_TOKENS_USAGE | ||||
|                         .with_label_values(&[workspace_uid, &chat_completion.model]) | ||||
|                         .inc_by(usage.prompt_tokens as u64); | ||||
|                     MEILISEARCH_CHAT_COMPLETION_TOKENS_USAGE | ||||
|                         .with_label_values(&[workspace_uid, &chat_completion.model]) | ||||
|                         .inc_by(usage.completion_tokens as u64); | ||||
|                     MEILISEARCH_CHAT_TOTAL_TOKENS_USAGE | ||||
|                         .with_label_values(&[workspace_uid, &chat_completion.model]) | ||||
|                         .inc_by(usage.total_tokens as u64); | ||||
|                 } | ||||
|                 let choice = match resp.choices.first() { | ||||
|                     Some(choice) => choice, | ||||
|                     None => break, | ||||
|                 }; | ||||
|                 finish_reason = choice.finish_reason; | ||||
|  | ||||
|                 let ChatCompletionStreamResponseDelta { ref tool_calls, .. } = &choice.delta; | ||||
|   | ||||
| @@ -43,7 +43,7 @@ async fn version_too_old() { | ||||
|     std::fs::write(db_path.join("VERSION"), "1.11.9999").unwrap(); | ||||
|     let options = Opt { experimental_dumpless_upgrade: true, ..default_settings }; | ||||
|     let err = Server::new_with_options(options).await.map(|_| ()).unwrap_err(); | ||||
|     snapshot!(err, @"Database version 1.11.9999 is too old for the experimental dumpless upgrade feature. Please generate a dump using the v1.11.9999 and import it in the v1.15.2"); | ||||
|     snapshot!(err, @"Database version 1.11.9999 is too old for the experimental dumpless upgrade feature. Please generate a dump using the v1.11.9999 and import it in the v1.16.0"); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| @@ -58,7 +58,7 @@ async fn version_requires_downgrade() { | ||||
|     std::fs::write(db_path.join("VERSION"), format!("{major}.{minor}.{patch}")).unwrap(); | ||||
|     let options = Opt { experimental_dumpless_upgrade: true, ..default_settings }; | ||||
|     let err = Server::new_with_options(options).await.map(|_| ()).unwrap_err(); | ||||
|     snapshot!(err, @"Database version 1.15.3 is higher than the Meilisearch version 1.15.2. Downgrade is not supported"); | ||||
|     snapshot!(err, @"Database version 1.16.1 is higher than the Meilisearch version 1.16.0. Downgrade is not supported"); | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
|   | ||||
| @@ -8,7 +8,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "progress": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "stats": { | ||||
|         "totalNbTasks": 1, | ||||
|   | ||||
| @@ -8,7 +8,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "progress": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "stats": { | ||||
|         "totalNbTasks": 1, | ||||
|   | ||||
| @@ -8,7 +8,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "progress": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "stats": { | ||||
|         "totalNbTasks": 1, | ||||
|   | ||||
| @@ -12,7 +12,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "canceledBy": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "error": null, | ||||
|       "duration": "[duration]", | ||||
|   | ||||
| @@ -12,7 +12,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "canceledBy": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "error": null, | ||||
|       "duration": "[duration]", | ||||
|   | ||||
| @@ -12,7 +12,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "canceledBy": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "error": null, | ||||
|       "duration": "[duration]", | ||||
|   | ||||
| @@ -8,7 +8,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "progress": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "stats": { | ||||
|         "totalNbTasks": 1, | ||||
|   | ||||
| @@ -12,7 +12,7 @@ source: crates/meilisearch/tests/upgrade/v1_12/v1_12_0.rs | ||||
|       "canceledBy": null, | ||||
|       "details": { | ||||
|         "upgradeFrom": "v1.12.0", | ||||
|         "upgradeTo": "v1.15.2" | ||||
|         "upgradeTo": "v1.16.0" | ||||
|       }, | ||||
|       "error": null, | ||||
|       "duration": "[duration]", | ||||
|   | ||||
		Reference in New Issue
	
	Block a user