| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  | use std::{
 | 
					
						
							|  |  |  |     fs,
 | 
					
						
							|  |  |  |     path::{Path, PathBuf},
 | 
					
						
							|  |  |  | };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use criterion::{black_box, criterion_group, criterion_main, Criterion};
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const IGNORE_DIRS: [&str; 2] = ["step", "screenshots"];
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn discover_benchmark_dirs(base_path: &Path) -> Vec<PathBuf> {
 | 
					
						
							|  |  |  |     let mut benchmark_dirs = Vec::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if let Ok(entries) = fs::read_dir(base_path) {
 | 
					
						
							|  |  |  |         for entry in entries.filter_map(Result::ok) {
 | 
					
						
							|  |  |  |             let path = entry.path();
 | 
					
						
							|  |  |  |             if path.is_dir() {
 | 
					
						
							|  |  |  |                 let dir_name = path.file_name().unwrap().to_string_lossy();
 | 
					
						
							|  |  |  |                 if !IGNORE_DIRS.iter().any(|&x| x == dir_name) {
 | 
					
						
							|  |  |  |                     benchmark_dirs.push(path);
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  |         }
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     benchmark_dirs
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn find_main_kcl_file(dir_path: &Path) -> PathBuf {
 | 
					
						
							|  |  |  |     let file_path = dir_path.join("main.kcl");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if !file_path.exists() || !file_path.is_file() {
 | 
					
						
							|  |  |  |         panic!("Required main.kcl file not found in directory: {}", dir_path.display());
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     file_path
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | fn run_benchmarks(c: &mut Criterion) {
 | 
					
						
							|  |  |  |     // Specify the base directory containing benchmark subdirectories
 | 
					
						
							|  |  |  |     let base_dir = std::path::Path::new(env!("CARGO_MANIFEST_DIR")).join("../../public/kcl-samples");
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if !base_dir.exists() || !base_dir.is_dir() {
 | 
					
						
							|  |  |  |         panic!("Invalid base directory: {}", base_dir.display());
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let benchmark_dirs = discover_benchmark_dirs(&base_dir);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-16 21:23:30 -07:00
										 |  |  |     #[cfg(feature = "benchmark-execution")]
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |     let rt = tokio::runtime::Runtime::new().unwrap();
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for dir in benchmark_dirs {
 | 
					
						
							|  |  |  |         let dir_name = dir.file_name().unwrap().to_string_lossy().to_string();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Change the current directory to the benchmark directory.
 | 
					
						
							|  |  |  |         // This is necessary for the kcl-lib to correctly resolve relative paths.
 | 
					
						
							|  |  |  |         std::env::set_current_dir(&dir).unwrap();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Find main.kcl file (will panic if not found)
 | 
					
						
							|  |  |  |         let input_file = find_main_kcl_file(&dir);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Read the file content (panic on failure)
 | 
					
						
							|  |  |  |         let input_content = fs::read_to_string(&input_file)
 | 
					
						
							|  |  |  |             .unwrap_or_else(|e| panic!("Failed to read main.kcl in directory {}: {}", dir_name, e));
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Create a benchmark group for this directory
 | 
					
						
							|  |  |  |         let mut group = c.benchmark_group(&dir_name);
 | 
					
						
							|  |  |  |         group
 | 
					
						
							|  |  |  |             .sample_size(10)
 | 
					
						
							|  |  |  |             .measurement_time(std::time::Duration::from_secs(1)); // Short measurement time to keep it from running in parallel
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-16 21:23:30 -07:00
										 |  |  |         #[cfg(feature = "benchmark-execution")]
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |         let program = kcl_lib::Program::parse_no_errs(&input_content).unwrap();
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 16:29:26 -07:00
										 |  |  |         group.bench_function(format!("parse_{}", dir_name), |b| {
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  |             b.iter(|| kcl_lib::Program::parse_no_errs(black_box(&input_content)).unwrap())
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-16 21:23:30 -07:00
										 |  |  |         #[cfg(feature = "benchmark-execution")]
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |         group.bench_function(format!("execute_{}", dir_name), |b| {
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  |             b.iter(|| {
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |                 if let Err(err) = rt.block_on(async {
 | 
					
						
							| 
									
										
										
										
											2025-03-31 10:56:03 -04:00
										 |  |  |                     let ctx = kcl_lib::ExecutorContext::new_with_default_client().await?;
 | 
					
						
							| 
									
										
										
										
											2025-03-15 10:08:39 -07:00
										 |  |  |                     let mut exec_state = kcl_lib::ExecState::new(&ctx);
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |                     ctx.run(black_box(&program), &mut exec_state).await?;
 | 
					
						
							|  |  |  |                     ctx.close().await;
 | 
					
						
							|  |  |  |                     Ok::<(), anyhow::Error>(())
 | 
					
						
							|  |  |  |                 }) {
 | 
					
						
							|  |  |  |                     panic!("Failed to execute program: {}", err);
 | 
					
						
							| 
									
										
										
										
											2025-03-11 15:21:28 -07:00
										 |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  |             })
 | 
					
						
							| 
									
										
										
										
											2025-03-13 10:33:00 -07:00
										 |  |  |         });
 | 
					
						
							| 
									
										
										
										
											2025-03-10 18:04:16 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         group.finish();
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | criterion_group!(benches, run_benchmarks);
 | 
					
						
							|  |  |  | criterion_main!(benches);
 |