refactor comment extraction from tasks

This commit is contained in:
Aleksey Kladov 2019-10-24 19:29:38 +03:00
parent a409a12f1b
commit a40d02c9eb
2 changed files with 44 additions and 39 deletions

View file

@ -8,7 +8,7 @@
mod gen_syntax; mod gen_syntax;
mod gen_parser_tests; mod gen_parser_tests;
use std::{fs, path::Path}; use std::{fs, mem, path::Path};
use crate::Result; use crate::Result;
@ -44,3 +44,26 @@ pub fn update(path: &Path, contents: &str, mode: Mode) -> Result<()> {
fs::write(path, contents)?; fs::write(path, contents)?;
Ok(()) Ok(())
} }
fn extract_comment_blocks(text: &str) -> Vec<Vec<String>> {
let mut res = Vec::new();
let prefix = "// ";
let lines = text.lines().map(str::trim_start);
let mut block = vec![];
for line in lines {
let is_comment = line.starts_with(prefix);
if is_comment {
block.push(line[prefix.len()..].to_string());
} else {
if !block.is_empty() {
res.push(mem::replace(&mut block, Vec::new()))
}
}
}
if !block.is_empty() {
res.push(mem::replace(&mut block, Vec::new()))
}
res
}

View file

@ -3,12 +3,12 @@
use std::{ use std::{
collections::HashMap, collections::HashMap,
fs, fs, iter,
path::{Path, PathBuf}, path::{Path, PathBuf},
}; };
use crate::{ use crate::{
codegen::{self, update, Mode}, codegen::{self, extract_comment_blocks, update, Mode},
project_root, Result, project_root, Result,
}; };
@ -57,46 +57,28 @@ struct Tests {
} }
fn collect_tests(s: &str) -> Vec<Test> { fn collect_tests(s: &str) -> Vec<Test> {
let mut res = vec![]; let mut res = Vec::new();
let prefix = "// "; for comment_block in extract_comment_blocks(s) {
let lines = s.lines().map(str::trim_start); let first_line = &comment_block[0];
let (name, ok) = if first_line.starts_with("test ") {
let mut block = vec![]; let name = first_line["test ".len()..].to_string();
for line in lines { (name, true)
let is_comment = line.starts_with(prefix); } else if first_line.starts_with("test_err ") {
if is_comment { let name = first_line["test_err ".len()..].to_string();
block.push(&line[prefix.len()..]); (name, false)
} else { } else {
process_block(&mut res, &block); continue;
block.clear();
}
}
process_block(&mut res, &block);
return res;
fn process_block(acc: &mut Vec<Test>, block: &[&str]) {
if block.is_empty() {
return;
}
let mut ok = true;
let mut block = block.iter();
let name = loop {
match block.next() {
Some(line) if line.starts_with("test ") => {
break line["test ".len()..].to_string();
}
Some(line) if line.starts_with("test_err ") => {
ok = false;
break line["test_err ".len()..].to_string();
}
Some(_) => (),
None => return,
}
}; };
let text: String = block.copied().chain(std::iter::once("")).collect::<Vec<_>>().join("\n"); let text: String = comment_block[1..]
.iter()
.cloned()
.chain(iter::once(String::new()))
.collect::<Vec<_>>()
.join("\n");
assert!(!text.trim().is_empty() && text.ends_with('\n')); assert!(!text.trim().is_empty() && text.ends_with('\n'));
acc.push(Test { name, text, ok }) res.push(Test { name, text, ok })
} }
res
} }
fn tests_from_dir(dir: &Path) -> Result<Tests> { fn tests_from_dir(dir: &Path) -> Result<Tests> {