mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-25 21:16:28 +00:00 
			
		
		
		
	Merge #4389
4389: Stabilize scoreDetails r=dureuill a=dureuill # Pull Request ## Related issue Fixes #4359 ## What does this PR do? ### User standpoint - Users no longer need to enable the `scoreDetails` experimental feature to use `showRankingScoreDetails` in search queries. - ⚠️ **Breaking change**: sending an object containing the key `"scoreDetails"` to the `/experimental-features` route is now an error. However, importing a dump of a database where that feature was enabled completes successfully. ### Implementation standpoint - remove `scoreDetails` from the experimental features - remove check on the experimental feature `scoreDetails` before accepting `showRankingScoreDetails` - remove `scoreDetails` from the accepted fields in the `/experimental-features` route - fix tests accordingly ## Manual tests 1. exported a dump with the `scoreDetails` feature enabled on `main` - tried to import the dump after the changes in this PR - the dump imported successfully 2. tried to make a search with `showRankingScoreDetails: true` - the ranking score details are displayed - an automated test case also exists and passes 3. tried to enable the `scoreDetails` in `/experimental-features` - get error message ``` Unknown field `scoreDetails`: expected one of `vectorStore`, `metrics`, `exportPuffinReports` ``` Co-authored-by: Louis Dureuil <louis@meilisearch.com>
This commit is contained in:
		| @@ -30,19 +30,6 @@ impl RoFeatures { | ||||
|         self.runtime | ||||
|     } | ||||
|  | ||||
|     pub fn check_score_details(&self) -> Result<()> { | ||||
|         if self.runtime.score_details { | ||||
|             Ok(()) | ||||
|         } else { | ||||
|             Err(FeatureNotEnabledError { | ||||
|                 disabled_action: "Computing score details", | ||||
|                 feature: "score details", | ||||
|                 issue_link: "https://github.com/meilisearch/product/discussions/674", | ||||
|             } | ||||
|             .into()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn check_metrics(&self) -> Result<()> { | ||||
|         if self.runtime.metrics { | ||||
|             Ok(()) | ||||
|   | ||||
| @@ -3,7 +3,6 @@ use serde::{Deserialize, Serialize}; | ||||
| #[derive(Serialize, Deserialize, Debug, Clone, Copy, Default, PartialEq, Eq)] | ||||
| #[serde(rename_all = "camelCase", default)] | ||||
| pub struct RuntimeTogglableFeatures { | ||||
|     pub score_details: bool, | ||||
|     pub vector_store: bool, | ||||
|     pub metrics: bool, | ||||
|     pub export_puffin_reports: bool, | ||||
|   | ||||
| @@ -40,8 +40,6 @@ async fn get_features( | ||||
| #[derive(Debug, Deserr)] | ||||
| #[deserr(error = DeserrJsonError, rename_all = camelCase, deny_unknown_fields)] | ||||
| pub struct RuntimeTogglableFeatures { | ||||
|     #[deserr(default)] | ||||
|     pub score_details: Option<bool>, | ||||
|     #[deserr(default)] | ||||
|     pub vector_store: Option<bool>, | ||||
|     #[deserr(default)] | ||||
| @@ -63,7 +61,6 @@ async fn patch_features( | ||||
|  | ||||
|     let old_features = features.runtime_features(); | ||||
|     let new_features = meilisearch_types::features::RuntimeTogglableFeatures { | ||||
|         score_details: new_features.0.score_details.unwrap_or(old_features.score_details), | ||||
|         vector_store: new_features.0.vector_store.unwrap_or(old_features.vector_store), | ||||
|         metrics: new_features.0.metrics.unwrap_or(old_features.metrics), | ||||
|         export_puffin_reports: new_features | ||||
| @@ -76,7 +73,6 @@ async fn patch_features( | ||||
|     // the it renames to camelCase, which we don't want for analytics. | ||||
|     // **Do not** ignore fields with `..` or `_` here, because we want to add them in the future. | ||||
|     let meilisearch_types::features::RuntimeTogglableFeatures { | ||||
|         score_details, | ||||
|         vector_store, | ||||
|         metrics, | ||||
|         export_puffin_reports, | ||||
| @@ -85,7 +81,6 @@ async fn patch_features( | ||||
|     analytics.publish( | ||||
|         "Experimental features Updated".to_string(), | ||||
|         json!({ | ||||
|             "score_details": score_details, | ||||
|             "vector_store": vector_store, | ||||
|             "metrics": metrics, | ||||
|             "export_puffin_reports": export_puffin_reports, | ||||
|   | ||||
| @@ -441,10 +441,6 @@ fn prepare_search<'t>( | ||||
|         ScoringStrategy::Skip | ||||
|     }); | ||||
|  | ||||
|     if query.show_ranking_score_details { | ||||
|         features.check_score_details()?; | ||||
|     } | ||||
|  | ||||
|     if let Some(HybridQuery { embedder: Some(embedder), .. }) = &query.hybrid { | ||||
|         search.embedder_name(embedder); | ||||
|     } | ||||
|   | ||||
| @@ -1845,7 +1845,6 @@ async fn import_dump_v6_containing_experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": false, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
|   | ||||
| @@ -18,7 +18,6 @@ async fn experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": false, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
| @@ -30,7 +29,6 @@ async fn experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": true, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
| @@ -42,7 +40,6 @@ async fn experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": true, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
| @@ -55,7 +52,6 @@ async fn experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": true, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
| @@ -68,7 +64,6 @@ async fn experimental_features() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": true, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
| @@ -88,7 +83,6 @@ async fn experimental_feature_metrics() { | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": false, | ||||
|       "metrics": true, | ||||
|       "exportPuffinReports": false | ||||
| @@ -146,7 +140,7 @@ async fn errors() { | ||||
|     meili_snap::snapshot!(code, @"400 Bad Request"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "message": "Unknown field `NotAFeature`: expected one of `scoreDetails`, `vectorStore`, `metrics`, `exportPuffinReports`", | ||||
|       "message": "Unknown field `NotAFeature`: expected one of `vectorStore`, `metrics`, `exportPuffinReports`", | ||||
|       "code": "bad_request", | ||||
|       "type": "invalid_request", | ||||
|       "link": "https://docs.meilisearch.com/errors#bad_request" | ||||
|   | ||||
| @@ -13,7 +13,6 @@ async fn index_with_documents<'a>(server: &'a Server, documents: &Value) -> Inde | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|     { | ||||
|       "scoreDetails": false, | ||||
|       "vectorStore": true, | ||||
|       "metrics": false, | ||||
|       "exportPuffinReports": false | ||||
|   | ||||
| @@ -766,38 +766,14 @@ async fn faceting_max_values_per_facet() { | ||||
| } | ||||
|  | ||||
| #[actix_rt::test] | ||||
| async fn experimental_feature_score_details() { | ||||
| async fn test_score_details() { | ||||
|     let server = Server::new().await; | ||||
|     let index = server.index("test"); | ||||
|  | ||||
|     let documents = DOCUMENTS.clone(); | ||||
|  | ||||
|     index.add_documents(json!(documents), None).await; | ||||
|     index.wait_task(0).await; | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|             json!({ | ||||
|                 "q": "train dragon", | ||||
|                 "showRankingScoreDetails": true, | ||||
|             }), | ||||
|             |response, code| { | ||||
|                 meili_snap::snapshot!(code, @"400 Bad Request"); | ||||
|                 meili_snap::snapshot!(meili_snap::json_string!(response), @r###" | ||||
|                 { | ||||
|                   "message": "Computing score details requires enabling the `score details` experimental feature. See https://github.com/meilisearch/product/discussions/674", | ||||
|                   "code": "feature_not_enabled", | ||||
|                   "type": "invalid_request", | ||||
|                   "link": "https://docs.meilisearch.com/errors#feature_not_enabled" | ||||
|                 } | ||||
|                 "###); | ||||
|             }, | ||||
|         ) | ||||
|         .await; | ||||
|  | ||||
|     let (response, code) = server.set_features(json!({"scoreDetails": true})).await; | ||||
|     meili_snap::snapshot!(code, @"200 OK"); | ||||
|     meili_snap::snapshot!(response["scoreDetails"], @"true"); | ||||
|     let res = index.add_documents(json!(documents), None).await; | ||||
|     index.wait_task(res.0.uid()).await; | ||||
|  | ||||
|     index | ||||
|         .search( | ||||
|   | ||||
		Reference in New Issue
	
	Block a user