use std::fs; use std::path::PathBuf; struct KdlCodeBlock { filename: String, code: String, line_number: usize, must_fail: bool, } fn extract_kdl_from_file(file_contents: &str, filename: &str) -> Vec { let mut lines = file_contents .lines() .map(|line| { // Removes the > from callouts that might contain ```kdl``` let line = line.trim(); if line.starts_with('>') { if line.len() == 1 { "" } else { &line[2..] } } else { line } }) .enumerate(); let mut kdl_code_blocks = vec![]; while let Some((line_number, line)) = lines.next() { if !line.starts_with("```kdl") { continue; } let mut snippet = String::new(); for (_, line) in lines .by_ref() .take_while(|(_, line)| !line.starts_with("```")) { snippet.push_str(line); snippet.push('\n'); } kdl_code_blocks.push(KdlCodeBlock { code: snippet, line_number, filename: filename.to_string(), must_fail: line.contains("must-fail"), }); } kdl_code_blocks } #[test] fn wiki_docs_parses() { let wiki_dir = PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("../wiki"); let code_blocks = fs::read_dir(wiki_dir) .unwrap() .filter_map(|entry| entry.ok()) .filter(|entry| entry.file_type().is_ok_and(|ft| ft.is_file())) .filter(|file| { file.path() .extension() .map(|ext| ext == "md") .unwrap_or(false) }) .flat_map(|file| { let file_contents = fs::read_to_string(file.path()).unwrap(); let file_path = file.path(); let filename = file_path.to_str().unwrap(); extract_kdl_from_file(&file_contents, filename) }); let mut errors = vec![]; for KdlCodeBlock { code, line_number, filename, must_fail, } in code_blocks { if let Err(error) = niri_config::Config::parse(&filename, &code) { if !must_fail { errors.push(format!( "Error parsing wiki KDL code block at {}:{}: {:?}", filename, line_number, miette::Report::new(error) )); } } else if must_fail { errors.push(format!( "Expected error parsing wiki KDL code block at {}:{}", filename, line_number )); } } if !errors.is_empty() { panic!( "Errors parsing {} wiki KDL code blocks:\n{}", errors.len(), errors.join("\n") ); } }