bacon 3.23.0

background rust compiler
Documentation
use {
    crate::*,
    rustc_hash::FxHashMap,
};

pub fn build_report<L: LineAnalyzer>(
    cmd_lines: &[CommandOutputLine],
    mut line_analyzer: L,
) -> Report {
    #[derive(Debug, Default)]
    struct Test {
        passed: bool,
        has_title: bool,
    }
    // we first accumulate warnings, test fails and errors in separate vectors
    let mut warnings = Vec::new();
    let mut errors = Vec::new();
    let mut fails = Vec::new();
    let mut tests: FxHashMap<String, Test> = Default::default();
    let mut passed_tests = 0;
    let mut cur_err_kind = None; // the current kind among stderr lines
    let mut is_in_out_fail = false;
    let mut suggest_backtrace = false;
    for cmd_line in cmd_lines {
        let line_analysis = line_analyzer.analyze_line(cmd_line);
        let line_type = line_analysis.line_type;
        let mut line = Line {
            item_idx: 0, // will be filled later
            line_type,
            content: cmd_line.content.clone(),
        };
        match (line_type, line_analysis.key) {
            (LineType::TestResult(r), Some(key)) => {
                if r {
                    passed_tests += 1;
                    tests.insert(
                        key,
                        Test {
                            passed: true,
                            has_title: false,
                        },
                    );
                } else if !tests.contains_key(&key) {
                    // we should receive the test test section later,
                    // right now we just whitelist it
                    tests.insert(key, Test::default());
                }
            }
            (LineType::Title(Kind::TestOutput | Kind::TestFail), Some(key)) => {
                let test = tests.entry(key.clone()).or_default();
                is_in_out_fail = true;
                cur_err_kind = Some(Kind::TestFail);
                if test.has_title {
                    // we already have a title for this test
                    // (for nextest, we have a title for stdout and one for stderr)
                    continue;
                }
                test.has_title = true;
                line.content = if test.passed {
                    TLine::passed(&key)
                } else {
                    TLine::failed(&key)
                };
                fails.push(line);
            }
            (LineType::Normal, None) => {
                if line.content.is_blank() {
                    if cur_err_kind == Some(Kind::TestFail) {
                        // beautification: we remove some blank lines
                        if let Some(last) = fails.last() {
                            if last.line_type != LineType::Normal || last.content.is_blank() {
                                continue;
                            }
                        }
                    } else {
                        is_in_out_fail = false;
                    }
                }
                if is_in_out_fail {
                    fails.push(line);
                } else {
                    match cur_err_kind {
                        Some(Kind::Warning) => warnings.push(line),
                        Some(Kind::Error) => errors.push(line),
                        _ => {}
                    }
                }
            }
            (LineType::Title(Kind::Sum), None) => {
                // we're not interested in this section
                cur_err_kind = None;
                is_in_out_fail = false;
            }
            (LineType::SectionEnd, None) => {
                cur_err_kind = None;
                is_in_out_fail = false;
            }
            (LineType::Title(kind), _) => {
                cur_err_kind = Some(kind);
                match cur_err_kind {
                    Some(Kind::Warning) => warnings.push(line),
                    Some(Kind::Error) => errors.push(line),
                    _ => {} // before warnings and errors, or in a sum
                }
            }
            (LineType::BacktraceSuggestion, _) => {
                suggest_backtrace = true;
            }
            (LineType::Location, _) => {
                match cur_err_kind {
                    Some(Kind::Warning) => warnings.push(line),
                    Some(Kind::Error) => errors.push(line),
                    Some(Kind::TestFail) => fails.push(line),
                    _ => {} // before warnings and errors, or in a sum
                }
            }
            _ => {}
        }
    }
    for (key, test) in &tests {
        // if we know of a test but there was no content, we add some
        if test.has_title || test.passed {
            continue;
        }
        fails.push(Line {
            item_idx: 0, // will be filled later
            line_type: LineType::Title(Kind::TestFail),
            content: TLine::failed(key),
        });
        fails.push(Line {
            item_idx: 0,
            line_type: LineType::Normal,
            content: TLine::italic("no output".to_string()),
        });
    }
    // we now build a common vector, with errors first
    let mut lines = errors;
    lines.append(&mut fails);
    lines.append(&mut warnings);
    // and we assign the indexes
    let mut item_idx = 0;
    for line in &mut lines {
        if matches!(line.line_type, LineType::Title(_)) {
            item_idx += 1;
        }
        line.item_idx = item_idx;
    }

    let mut report = Report::new(lines);
    report.suggest_backtrace = suggest_backtrace;
    report.has_passed_tests = passed_tests > 0;
    report.failure_keys = tests
        .drain()
        .filter_map(|(key, test)| if !test.passed { Some(key) } else { None })
        .collect::<Vec<_>>();
    report
}