use std::collections::BTreeMap; use std::fs::File; use std::io::{Seek as _, Write as _}; use anyhow::{bail, Context as _}; use futures_util::TryStreamExt as _; use serde::Deserialize; use serde_json::json; use tokio::task::JoinHandle; use uuid::Uuid; use super::assets::Asset; use super::client::Client; use super::command::SyncMode; use super::dashboard::DashboardClient; use super::BenchDeriveArgs; use crate::bench::{assets, meili_process}; #[derive(Deserialize)] pub struct Workload { pub name: String, pub run_count: u16, pub extra_cli_args: Vec, pub assets: BTreeMap, #[serde(default)] pub target: String, #[serde(default)] pub precommands: Vec, pub commands: Vec, } async fn run_commands( dashboard_client: &DashboardClient, logs_client: &Client, meili_client: &Client, workload_uuid: Uuid, workload: &Workload, args: &BenchDeriveArgs, run_number: u16, ) -> anyhow::Result>> { let report_folder = &args.report_folder; let workload_name = &workload.name; for batch in workload .precommands .as_slice() .split_inclusive(|command| !matches!(command.synchronous, SyncMode::DontWait)) { super::command::run_batch(meili_client, batch, &workload.assets, &args.asset_folder) .await?; } std::fs::create_dir_all(report_folder) .with_context(|| format!("could not create report directory at {report_folder}"))?; let trace_filename = format!("{report_folder}/{workload_name}-{run_number}-trace.json"); let report_filename = format!("{report_folder}/{workload_name}-{run_number}-report.json"); let report_handle = start_report(logs_client, trace_filename, &workload.target).await?; for batch in workload .commands .as_slice() .split_inclusive(|command| !matches!(command.synchronous, SyncMode::DontWait)) { super::command::run_batch(meili_client, batch, &workload.assets, &args.asset_folder) .await?; } let processor = stop_report(dashboard_client, logs_client, workload_uuid, report_filename, report_handle) .await?; Ok(processor) } #[allow(clippy::too_many_arguments)] // not best code quality, but this is a benchmark runner #[tracing::instrument(skip(assets_client, dashboard_client, logs_client, meili_client, workload, master_key, args), fields(workload = workload.name))] pub async fn execute( assets_client: &Client, dashboard_client: &DashboardClient, logs_client: &Client, meili_client: &Client, invocation_uuid: Uuid, master_key: Option<&str>, workload: Workload, args: &BenchDeriveArgs, ) -> anyhow::Result<()> { assets::fetch_assets(assets_client, &workload.assets, &args.asset_folder).await?; let workload_uuid = dashboard_client.create_workload(invocation_uuid, &workload).await?; let mut tasks = Vec::new(); for i in 0..workload.run_count { tasks.push( execute_run( dashboard_client, logs_client, meili_client, workload_uuid, master_key, &workload, args, i, ) .await?, ); } let mut reports = Vec::with_capacity(workload.run_count as usize); for task in tasks { reports.push( task.await .context("task panicked while processing report")? .context("task failed while processing report")?, ); } tracing::info!(workload = workload.name, "Successful workload"); Ok(()) } #[allow(clippy::too_many_arguments)] // not best code quality, but this is a benchmark runner #[tracing::instrument(skip(dashboard_client, logs_client, meili_client, workload, master_key, args), fields(workload = %workload.name))] async fn execute_run( dashboard_client: &DashboardClient, logs_client: &Client, meili_client: &Client, workload_uuid: Uuid, master_key: Option<&str>, workload: &Workload, args: &BenchDeriveArgs, run_number: u16, ) -> anyhow::Result>> { meili_process::delete_db(); meili_process::build().await?; let meilisearch = meili_process::start(meili_client, master_key, workload, &args.asset_folder).await?; let processor = run_commands( dashboard_client, logs_client, meili_client, workload_uuid, workload, args, run_number, ) .await?; meili_process::kill(meilisearch).await; tracing::info!(run_number, "Successful run"); Ok(processor) } async fn start_report( logs_client: &Client, filename: String, target: &str, ) -> anyhow::Result>> { const DEFAULT_TARGET: &str = "indexing::=trace"; let target = if target.is_empty() { DEFAULT_TARGET } else { target }; let report_file = std::fs::File::options() .create(true) .truncate(true) .write(true) .read(true) .open(&filename) .with_context(|| format!("could not create file at {filename}"))?; let mut report_file = std::io::BufWriter::new(report_file); let response = logs_client .post("") .json(&json!({ "mode": "profile", "target": target, })) .send() .await .context("failed to start report")?; let code = response.status(); if code.is_client_error() { tracing::error!(%code, "request error when trying to start report"); let response: serde_json::Value = response .json() .await .context("could not deserialize response as JSON") .context("response error when trying to start report")?; bail!( "request error when trying to start report: server responded with error code {code} and '{response}'" ) } else if code.is_server_error() { tracing::error!(%code, "server error when trying to start report"); let response: serde_json::Value = response .json() .await .context("could not deserialize response as JSON") .context("response error trying to start report")?; bail!("server error when trying to start report: server responded with error code {code} and '{response}'") } Ok(tokio::task::spawn(async move { let mut stream = response.bytes_stream(); while let Some(bytes) = stream.try_next().await.context("while waiting for report")? { report_file .write_all(&bytes) .with_context(|| format!("while writing report to {filename}"))?; } report_file.into_inner().with_context(|| format!("while writing report to {filename}")) })) } async fn stop_report( dashboard_client: &DashboardClient, logs_client: &Client, workload_uuid: Uuid, filename: String, report_handle: tokio::task::JoinHandle>, ) -> anyhow::Result>> { let response = logs_client.delete("").send().await.context("while stopping report")?; if !response.status().is_success() { bail!("received HTTP {} while stopping report", response.status()) } let mut file = tokio::time::timeout(std::time::Duration::from_secs(1000), report_handle) .await .context("while waiting for the end of the report")? .context("report writing task panicked")? .context("while writing report")?; file.rewind().context("while rewinding report file")?; let process_handle = tokio::task::spawn({ let dashboard_client = dashboard_client.clone(); async move { let span = tracing::info_span!("processing trace to report", filename); let _guard = span.enter(); let report = tracing_trace::processor::span_stats::to_call_stats( tracing_trace::TraceReader::new(std::io::BufReader::new(file)), ) .context("could not convert trace to report")?; let context = || format!("writing report to {filename}"); dashboard_client.create_run(workload_uuid, &report).await?; let mut output_file = std::io::BufWriter::new( std::fs::File::options() .create(true) .truncate(true) .write(true) .read(true) .open(&filename) .with_context(context)?, ); for (key, value) in report { serde_json::to_writer(&mut output_file, &json!({key: value})) .context("serializing span stat")?; writeln!(&mut output_file).with_context(context)?; } output_file.flush().with_context(context)?; let mut output_file = output_file.into_inner().with_context(context)?; output_file.rewind().context("could not rewind output_file").with_context(context)?; Ok(output_file) } }); Ok(process_handle) }