mirror of
				https://github.com/meilisearch/meilisearch.git
				synced 2025-10-24 20:46:27 +00:00 
			
		
		
		
	Merge #4651
4651: Allow to comment with the results of benchmark invocation r=Kerollmops a=dureuill Co-authored-by: Louis Dureuil <louis@meilisearch.com>
This commit is contained in:
		
							
								
								
									
										9
									
								
								.github/workflows/bench-pr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										9
									
								
								.github/workflows/bench-pr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -43,4 +43,11 @@ jobs: | |||||||
|  |  | ||||||
|         - name: Run benchmarks on PR ${{ github.event.issue.id }} |         - name: Run benchmarks on PR ${{ github.event.issue.id }} | ||||||
|           run: | |           run: | | ||||||
|             cargo xtask bench --api-key "${{ secrets.BENCHMARK_API_KEY }}" --dashboard-url "${{ vars.BENCHMARK_DASHBOARD_URL }}" --reason "[Comment](${{ github.event.comment.html_url }}) on [#${{ github.event.issue.number }}](${{ github.event.issue.html_url }})" -- ${{ steps.command.outputs.command-arguments }} |             cargo xtask bench --api-key "${{ secrets.BENCHMARK_API_KEY }}" \ | ||||||
|  |                --dashboard-url "${{ vars.BENCHMARK_DASHBOARD_URL }}" \ | ||||||
|  |                --reason "[Comment](${{ github.event.comment.html_url }}) on [#${{ github.event.issue.number }}](${{ github.event.issue.html_url }})" \ | ||||||
|  |                -- ${{ steps.command.outputs.command-arguments }} > benchlinks.txt | ||||||
|  |  | ||||||
|  |         - name: Send comment in PR | ||||||
|  |           run: | | ||||||
|  |             gh pr comment ${{github.event.issue.number}} --body-file benchlinks.txt | ||||||
|   | |||||||
| @@ -55,6 +55,10 @@ impl Client { | |||||||
|     pub fn delete(&self, route: &str) -> reqwest::RequestBuilder { |     pub fn delete(&self, route: &str) -> reqwest::RequestBuilder { | ||||||
|         self.request(reqwest::Method::DELETE, route) |         self.request(reqwest::Method::DELETE, route) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     pub fn base_url(&self) -> Option<&str> { | ||||||
|  |         self.base_url.as_deref() | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Debug, Clone, Copy, Deserialize)] | #[derive(Debug, Clone, Copy, Deserialize)] | ||||||
|   | |||||||
| @@ -18,12 +18,9 @@ pub enum DashboardClient { | |||||||
| } | } | ||||||
|  |  | ||||||
| impl DashboardClient { | impl DashboardClient { | ||||||
|     pub fn new(dashboard_url: &str, api_key: Option<&str>) -> anyhow::Result<Self> { |     pub fn new(dashboard_url: String, api_key: Option<&str>) -> anyhow::Result<Self> { | ||||||
|         let dashboard_client = Client::new( |         let dashboard_client = | ||||||
|             Some(format!("{}/api/v1", dashboard_url)), |             Client::new(Some(dashboard_url), api_key, Some(std::time::Duration::from_secs(60)))?; | ||||||
|             api_key, |  | ||||||
|             Some(std::time::Duration::from_secs(60)), |  | ||||||
|         )?; |  | ||||||
|  |  | ||||||
|         Ok(Self::Client(dashboard_client)) |         Ok(Self::Client(dashboard_client)) | ||||||
|     } |     } | ||||||
| @@ -36,7 +33,7 @@ impl DashboardClient { | |||||||
|         let Self::Client(dashboard_client) = self else { return Ok(()) }; |         let Self::Client(dashboard_client) = self else { return Ok(()) }; | ||||||
|  |  | ||||||
|         let response = dashboard_client |         let response = dashboard_client | ||||||
|             .put("machine") |             .put("/api/v1/machine") | ||||||
|             .json(&json!({"hostname": env.hostname})) |             .json(&json!({"hostname": env.hostname})) | ||||||
|             .send() |             .send() | ||||||
|             .await |             .await | ||||||
| @@ -62,7 +59,7 @@ impl DashboardClient { | |||||||
|         let Self::Client(dashboard_client) = self else { return Ok(Uuid::now_v7()) }; |         let Self::Client(dashboard_client) = self else { return Ok(Uuid::now_v7()) }; | ||||||
|  |  | ||||||
|         let response = dashboard_client |         let response = dashboard_client | ||||||
|             .put("invocation") |             .put("/api/v1/invocation") | ||||||
|             .json(&json!({ |             .json(&json!({ | ||||||
|                 "commit": { |                 "commit": { | ||||||
|                     "sha1": build_info.commit_sha1, |                     "sha1": build_info.commit_sha1, | ||||||
| @@ -97,7 +94,7 @@ impl DashboardClient { | |||||||
|         let Self::Client(dashboard_client) = self else { return Ok(Uuid::now_v7()) }; |         let Self::Client(dashboard_client) = self else { return Ok(Uuid::now_v7()) }; | ||||||
|  |  | ||||||
|         let response = dashboard_client |         let response = dashboard_client | ||||||
|             .put("workload") |             .put("/api/v1/workload") | ||||||
|             .json(&json!({ |             .json(&json!({ | ||||||
|                 "invocation_uuid": invocation_uuid, |                 "invocation_uuid": invocation_uuid, | ||||||
|                 "name": &workload.name, |                 "name": &workload.name, | ||||||
| @@ -124,7 +121,7 @@ impl DashboardClient { | |||||||
|         let Self::Client(dashboard_client) = self else { return Ok(()) }; |         let Self::Client(dashboard_client) = self else { return Ok(()) }; | ||||||
|  |  | ||||||
|         let response = dashboard_client |         let response = dashboard_client | ||||||
|             .put("run") |             .put("/api/v1/run") | ||||||
|             .json(&json!({ |             .json(&json!({ | ||||||
|                 "workload_uuid": workload_uuid, |                 "workload_uuid": workload_uuid, | ||||||
|                 "data": report |                 "data": report | ||||||
| @@ -159,7 +156,7 @@ impl DashboardClient { | |||||||
|     pub async fn mark_as_failed(&self, invocation_uuid: Uuid, failure_reason: Option<String>) { |     pub async fn mark_as_failed(&self, invocation_uuid: Uuid, failure_reason: Option<String>) { | ||||||
|         if let DashboardClient::Client(client) = self { |         if let DashboardClient::Client(client) = self { | ||||||
|             let response = client |             let response = client | ||||||
|                 .post("cancel-invocation") |                 .post("/api/v1/cancel-invocation") | ||||||
|                 .json(&json!({ |                 .json(&json!({ | ||||||
|                     "invocation_uuid": invocation_uuid, |                     "invocation_uuid": invocation_uuid, | ||||||
|                     "failure_reason": failure_reason, |                     "failure_reason": failure_reason, | ||||||
| @@ -186,4 +183,28 @@ impl DashboardClient { | |||||||
|  |  | ||||||
|         tracing::warn!(%invocation_uuid, "marked invocation as failed or canceled"); |         tracing::warn!(%invocation_uuid, "marked invocation as failed or canceled"); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /// Result URL in markdown | ||||||
|  |     pub(crate) fn result_url( | ||||||
|  |         &self, | ||||||
|  |         workload_name: &str, | ||||||
|  |         build_info: &build_info::BuildInfo, | ||||||
|  |         baseline_branch: &str, | ||||||
|  |     ) -> String { | ||||||
|  |         let Self::Client(client) = self else { return Default::default() }; | ||||||
|  |         let Some(base_url) = client.base_url() else { return Default::default() }; | ||||||
|  |  | ||||||
|  |         let Some(commit_sha1) = build_info.commit_sha1 else { return Default::default() }; | ||||||
|  |  | ||||||
|  |         // https://bench.meilisearch.dev/view_spans?commit_sha1=500ddc76b549fb9f1af54b2dd6abfa15960381bb&workload_name=settings-add-remove-filters.json&target_branch=reduce-transform-disk-usage&baseline_branch=main | ||||||
|  |         let mut url = format!( | ||||||
|  |             "{base_url}/view_spans?commit_sha1={commit_sha1}&workload_name={workload_name}" | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         if let Some(target_branch) = build_info.branch { | ||||||
|  |             url += &format!("&target_branch={target_branch}&baseline_branch={baseline_branch}"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         format!("[{workload_name} compared with {baseline_branch}]({url})") | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ mod env_info; | |||||||
| mod meili_process; | mod meili_process; | ||||||
| mod workload; | mod workload; | ||||||
|  |  | ||||||
|  | use std::io::LineWriter; | ||||||
| use std::path::PathBuf; | use std::path::PathBuf; | ||||||
|  |  | ||||||
| use anyhow::Context; | use anyhow::Context; | ||||||
| @@ -90,6 +91,7 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|  |  | ||||||
|     let subscriber = tracing_subscriber::registry().with( |     let subscriber = tracing_subscriber::registry().with( | ||||||
|         tracing_subscriber::fmt::layer() |         tracing_subscriber::fmt::layer() | ||||||
|  |             .with_writer(|| LineWriter::new(std::io::stderr())) | ||||||
|             .with_span_events(FmtSpan::NEW | FmtSpan::CLOSE) |             .with_span_events(FmtSpan::NEW | FmtSpan::CLOSE) | ||||||
|             .with_filter(filter), |             .with_filter(filter), | ||||||
|     ); |     ); | ||||||
| @@ -110,7 +112,7 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|     let dashboard_client = if args.no_dashboard { |     let dashboard_client = if args.no_dashboard { | ||||||
|         dashboard::DashboardClient::new_dry() |         dashboard::DashboardClient::new_dry() | ||||||
|     } else { |     } else { | ||||||
|         dashboard::DashboardClient::new(&args.dashboard_url, args.api_key.as_deref())? |         dashboard::DashboardClient::new(args.dashboard_url.clone(), args.api_key.as_deref())? | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     // reporting uses its own client because keeping the stream open to wait for entries |     // reporting uses its own client because keeping the stream open to wait for entries | ||||||
| @@ -136,7 +138,7 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|         let commit_message = build_info.commit_msg.context("missing commit message")?.split('\n').next().unwrap(); |         let commit_message = build_info.commit_msg.context("missing commit message")?.split('\n').next().unwrap(); | ||||||
|         let max_workloads = args.workload_file.len(); |         let max_workloads = args.workload_file.len(); | ||||||
|         let reason: Option<&str> = args.reason.as_deref(); |         let reason: Option<&str> = args.reason.as_deref(); | ||||||
|         let invocation_uuid = dashboard_client.create_invocation( build_info, commit_message, env, max_workloads, reason).await?; |         let invocation_uuid = dashboard_client.create_invocation(build_info.clone(), commit_message, env, max_workloads, reason).await?; | ||||||
|  |  | ||||||
|         tracing::info!(workload_count = args.workload_file.len(), "handling workload files"); |         tracing::info!(workload_count = args.workload_file.len(), "handling workload files"); | ||||||
|  |  | ||||||
| @@ -144,6 +146,7 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|         let workload_runs = tokio::spawn( |         let workload_runs = tokio::spawn( | ||||||
|             { |             { | ||||||
|                 let dashboard_client = dashboard_client.clone(); |                 let dashboard_client = dashboard_client.clone(); | ||||||
|  |                 let mut dashboard_urls = Vec::new(); | ||||||
|                 async move { |                 async move { | ||||||
|             for workload_file in args.workload_file.iter() { |             for workload_file in args.workload_file.iter() { | ||||||
|                 let workload: Workload = serde_json::from_reader( |                 let workload: Workload = serde_json::from_reader( | ||||||
| @@ -152,6 +155,8 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|                 ) |                 ) | ||||||
|                 .with_context(|| format!("error parsing {} as JSON", workload_file.display()))?; |                 .with_context(|| format!("error parsing {} as JSON", workload_file.display()))?; | ||||||
|  |  | ||||||
|  |                 let workload_name = workload.name.clone(); | ||||||
|  |  | ||||||
|                 workload::execute( |                 workload::execute( | ||||||
|                     &assets_client, |                     &assets_client, | ||||||
|                     &dashboard_client, |                     &dashboard_client, | ||||||
| @@ -163,8 +168,23 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|                     &args, |                     &args, | ||||||
|                 ) |                 ) | ||||||
|                 .await?; |                 .await?; | ||||||
|  |  | ||||||
|  |                 let result_url = dashboard_client.result_url(&workload_name, &build_info, "main"); | ||||||
|  |  | ||||||
|  |                 if !result_url.is_empty() { | ||||||
|  |                 dashboard_urls.push(result_url); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 if let Some(branch) = build_info.branch { | ||||||
|  |                     let result_url = dashboard_client.result_url(&workload_name, &build_info, branch); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |                     if !result_url.is_empty() { | ||||||
|  |                     dashboard_urls.push(result_url); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|             } |             } | ||||||
|             Ok::<(), anyhow::Error>(()) |             Ok::<_, anyhow::Error>(dashboard_urls) | ||||||
|         }}); |         }}); | ||||||
|  |  | ||||||
|         // handle ctrl-c |         // handle ctrl-c | ||||||
| @@ -176,13 +196,19 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|  |  | ||||||
|         // wait for the end of the main task, handle result |         // wait for the end of the main task, handle result | ||||||
|         match workload_runs.await { |         match workload_runs.await { | ||||||
|             Ok(Ok(_)) => { |             Ok(Ok(urls)) => { | ||||||
|                 tracing::info!("Success"); |                 tracing::info!("Success"); | ||||||
|  |                 println!("☀️ Benchmark invocation completed, please find the results for your workloads below:"); | ||||||
|  |                 for url in urls { | ||||||
|  |                     println!("- {url}"); | ||||||
|  |                 } | ||||||
|                 Ok::<(), anyhow::Error>(()) |                 Ok::<(), anyhow::Error>(()) | ||||||
|             } |             } | ||||||
|             Ok(Err(error)) => { |             Ok(Err(error)) => { | ||||||
|                 tracing::error!(%invocation_uuid, error = %error, "invocation failed, attempting to report the failure to dashboard"); |                 tracing::error!(%invocation_uuid, error = %error, "invocation failed, attempting to report the failure to dashboard"); | ||||||
|                 dashboard_client.mark_as_failed(invocation_uuid, Some(error.to_string())).await; |                 dashboard_client.mark_as_failed(invocation_uuid, Some(error.to_string())).await; | ||||||
|  |                 println!("☔️ Benchmark invocation failed..."); | ||||||
|  |                 println!("{error}"); | ||||||
|                 tracing::warn!(%invocation_uuid, "invocation marked as failed following error"); |                 tracing::warn!(%invocation_uuid, "invocation marked as failed following error"); | ||||||
|                 Err(error) |                 Err(error) | ||||||
|             }, |             }, | ||||||
| @@ -191,10 +217,20 @@ pub fn run(args: BenchDeriveArgs) -> anyhow::Result<()> { | |||||||
|                     Ok(panic) => { |                     Ok(panic) => { | ||||||
|                         tracing::error!("invocation panicked, attempting to report the failure to dashboard"); |                         tracing::error!("invocation panicked, attempting to report the failure to dashboard"); | ||||||
|                         dashboard_client.mark_as_failed( invocation_uuid, Some("Panicked".into())).await; |                         dashboard_client.mark_as_failed( invocation_uuid, Some("Panicked".into())).await; | ||||||
|  |                         println!("‼️ Benchmark invocation panicked 😱"); | ||||||
|  |                         let msg = match panic.downcast_ref::<&'static str>() { | ||||||
|  |                             Some(s) => *s, | ||||||
|  |                             None => match panic.downcast_ref::<String>() { | ||||||
|  |                                 Some(s) => &s[..], | ||||||
|  |                                 None => "Box<dyn Any>", | ||||||
|  |                             }, | ||||||
|  |                         }; | ||||||
|  |                         println!("panicked at {msg}"); | ||||||
|                         std::panic::resume_unwind(panic) |                         std::panic::resume_unwind(panic) | ||||||
|                     } |                     } | ||||||
|                     Err(_) => { |                     Err(_) => { | ||||||
|                         tracing::warn!("task was canceled"); |                         tracing::warn!("task was canceled"); | ||||||
|  |                         println!("🚫 Benchmark invocation was canceled"); | ||||||
|                         Ok(()) |                         Ok(()) | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user