Files
hangshuo652 7ac887c776 feat: complete INSPECT/SEARCH support, fix PERFORM/EVAL coverage marking
- Add INSPECT (TALLYING/REPLACING/CONVERTING) with BEFORE/AFTER INITIAL
- Add SEARCH/SEARCH ALL with element-assignment path enumeration
- Fix _mark_perform compound condition marking via evaluate_tree
- Fix EVALUATE TRUE prior_false to collect all MC/DC false sets
- Add impossible path filtering (Pass A.5) with trace-to-root conflict detection
- Fix multi-line PERFORM VARYING parsing (VARYING/FROM/BY/UNTIL on separate lines)
- Remove dead code: agents.py LLM parser (replaced by rule-based _BrParser)
- 59 unit tests passing, 5 integration programs verified
2026-06-10 22:56:22 +08:00

1208 lines
46 KiB
Python
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
"""覆盖率统计:决策点收集 + 路径标记 + HTML报告"""
import re
import logging
from dataclasses import dataclass, field
from pathlib import Path
logger = logging.getLogger(__name__)
from .models import BrSeq, BrIf, BrEval, BrPerform, BrSearch, CondLeaf
from .cond import parse_single_condition, parse_compound_condition, is_field, collect_leaves, evaluate_tree
# ── 数据模型 ──
@dataclass
class LeafStat:
field: str
op: str
value: str
covered_true: bool = False
covered_false: bool = False
@dataclass
class DecisionPoint:
id: int
kind: str # "IF" | "EVALUATE" | "PERFORM"
label: str
branch_names: list[str]
covered_branches: set = field(default_factory=set)
active_branches: set = field(default_factory=set)
implied_branches: set = field(default_factory=set)
leaves: list[LeafStat] = field(default_factory=list)
source_line: int = 0
when_list: list = field(default_factory=list)
cond_tree: object = None
cond_leaves: list = field(default_factory=list)
# ── 决策点收集 ──
def collect_decision_points(node, fields, counter=None):
if counter is None:
counter = [0]
points = []
all_leaves = []
if isinstance(node, BrIf):
counter[0] += 1
dp = DecisionPoint(id=counter[0], kind='IF', label=node.condition,
branch_names=['T', 'F'])
simple = parse_single_condition(node.condition)
if simple and is_field(simple[0], fields):
dp.parsed = simple
elif simple:
dp.parsed = simple
elif node.cond_tree:
leaves = collect_leaves(node.cond_tree)
if leaves:
dp.cond_tree = node.cond_tree
dp.cond_leaves = list(leaves)
for leaf in leaves:
ls = LeafStat(field=leaf.field, op=leaf.op, value=leaf.value)
dp.leaves.append(ls)
all_leaves.append(ls)
points.append(dp)
p, l = _walk_collect(node.true_seq, fields, counter)
points.extend(p); all_leaves.extend(l)
p, l = _walk_collect(node.false_seq, fields, counter)
points.extend(p); all_leaves.extend(l)
elif isinstance(node, BrEval):
counter[0] += 1
names = [f"WHEN {v}" for v, _ in node.when_list]
if node.has_other:
names.append("OTHER")
dp = DecisionPoint(id=counter[0], kind='EVALUATE', label=node.subject,
branch_names=names, when_list=node.when_list)
points.append(dp)
for _, seq in node.when_list:
p, l = _walk_collect(seq, fields, counter)
points.extend(p); all_leaves.extend(l)
p, l = _walk_collect(node.other_seq, fields, counter)
points.extend(p); all_leaves.extend(l)
elif isinstance(node, BrSearch):
counter[0] += 1
branch_names = []
for cond_text, seq in node.when_list:
branch_names.append(f'WHEN {cond_text[:40]}')
if node.has_at_end:
branch_names.append('AT END')
dp = DecisionPoint(id=counter[0], kind='SEARCH',
label=node.table_name, branch_names=branch_names)
dp.when_list = node.when_list
dp.cond_trees = node.cond_trees
dp.has_other = node.has_at_end
points.append(dp)
for cond_text, seq in node.when_list:
p, l = _walk_collect(seq, fields, counter)
points.extend(p); all_leaves.extend(l)
if node.has_at_end:
p, l = _walk_collect(node.at_end_seq, fields, counter)
points.extend(p); all_leaves.extend(l)
elif isinstance(node, BrPerform):
if node.perf_type in ('until', 'para_until', 'varying', 'para_varying'):
counter[0] += 1
dp = DecisionPoint(id=counter[0], kind='PERFORM',
label=node.condition or '',
branch_names=['Enter', 'Skip'])
simple = parse_single_condition(node.condition) if node.condition else None
if simple and is_field(simple[0], fields):
dp.parsed = simple
elif node.condition:
cond_tree = parse_compound_condition(node.condition, fields)
if cond_tree:
leaves = collect_leaves(cond_tree)
if leaves:
dp.cond_tree = cond_tree
dp.cond_leaves = list(leaves)
points.append(dp)
p, l = _walk_collect(node.body_seq, fields, counter)
points.extend(p); all_leaves.extend(l)
elif isinstance(node, BrSeq):
for child in node.children:
p, l = collect_decision_points(child, fields, counter)
points.extend(p); all_leaves.extend(l)
return points, all_leaves
def _walk_collect(node, fields, counter):
return collect_decision_points(node, fields, counter)
# ── 覆盖率标记 ──
def mark_coverage(decision_points, leaf_stats, branch_paths, fields):
for cons, _assign in branch_paths:
for dp in decision_points:
if dp.kind == 'IF':
_mark_if(dp, cons)
elif dp.kind == 'EVALUATE':
_mark_eval(dp, cons, fields)
elif dp.kind == 'PERFORM':
_mark_perform(dp, cons)
elif dp.kind == 'SEARCH':
_mark_search(dp, cons, fields)
for leaf in leaf_stats:
for c in cons:
if _match_leaf(c, leaf):
if c[3]:
leaf.covered_true = True
else:
leaf.covered_false = True
for dp in decision_points:
dp.implied_branches = set(dp.active_branches)
def _match_constraint(c, parsed):
if len(c) != 4:
return False
return (c[0] == parsed[0] and c[1] == parsed[1]
and str(c[2]) == str(parsed[2]))
def _match_leaf(c, leaf):
if len(c) != 4:
return False
return (c[0] == leaf.field and c[1] == leaf.op
and str(c[2]) == str(leaf.value))
def _mark_if(dp, cons):
simple = getattr(dp, 'parsed', None)
if simple:
for c in cons:
if _match_constraint(c, simple):
if c[3]:
dp.active_branches.add('T')
else:
dp.active_branches.add('F')
elif dp.cond_tree and dp.cond_leaves:
assignment = {}
for leaf in dp.cond_leaves:
for c in cons:
if _match_leaf(c, leaf):
assignment[leaf] = c[3]
break
if len(assignment) == len(dp.cond_leaves):
if evaluate_tree(dp.cond_tree, assignment):
dp.active_branches.add('T')
else:
dp.active_branches.add('F')
else:
matched = 0
for leaf in dp.leaves:
for c in cons:
if _match_leaf(c, leaf):
matched += 1
break
if matched <= 1:
for c in cons:
for leaf in dp.leaves:
if _match_leaf(c, leaf):
dp.active_branches.add('T' if c[3] else 'F')
def _mark_eval(dp, cons, fields=None):
if dp.label == 'TRUE':
matched = False
for when_val, _ in dp.when_list:
parsed = parse_single_condition(when_val, fields)
if parsed:
for c in cons:
if _match_constraint(c, parsed) and c[3]:
name = f"WHEN {when_val}"
if name in dp.branch_names:
dp.active_branches.add(name)
matched = True
else:
cond_tree = parse_compound_condition(when_val, fields)
if cond_tree and not isinstance(cond_tree, CondLeaf):
leaves = list(collect_leaves(cond_tree))
assignment = {}
for leaf in leaves:
for c in cons:
if _match_leaf(c, leaf):
assignment[leaf] = c[3]
break
if len(assignment) == len(leaves):
if evaluate_tree(cond_tree, assignment):
name = f"WHEN {when_val}"
if name in dp.branch_names:
dp.active_branches.add(name)
matched = True
if not matched and 'OTHER' in dp.branch_names:
when_fields = set()
for when_val, _ in dp.when_list:
for c in cons:
if c[0] in when_val:
when_fields.add(c[0])
if when_fields:
dp.active_branches.add('OTHER')
return
for c in cons:
if c[0] == dp.label and c[1] == '=':
name = f"WHEN {c[2]}"
if name in dp.branch_names:
dp.active_branches.add(name)
elif c[0] == dp.label and c[1] == 'not_in':
dp.active_branches.add('OTHER')
def _mark_search(dp, cons, fields=None):
branch_masks = [False] * len(dp.branch_names)
for i, (cond_text, body_seq) in enumerate(dp.when_list):
cond_tree = dp.cond_trees[i] if i < len(dp.cond_trees) else None
if not cond_tree:
continue
if isinstance(cond_tree, CondLeaf):
for c in cons:
if len(c) == 4:
base_c = re.sub(r'\s*\(.*?\)\s*$', '', c[0])
base_cond = re.sub(r'\s*\(.*?\)\s*$', '', cond_tree.field)
if base_c == base_cond and c[1] == cond_tree.op \
and str(c[2]) == str(cond_tree.value) and c[3]:
branch_masks[i] = True
break
else:
leaves = list(collect_leaves(cond_tree))
assignment = {}
for leaf in leaves:
for c in cons:
if len(c) == 4:
base_c = re.sub(r'\s*\(.*?\)\s*$', '', c[0])
base_l = re.sub(r'\s*\(.*?\)\s*$', '', leaf.field)
if base_c == base_l and c[1] == leaf.op and str(c[2]) == str(leaf.value):
assignment[leaf] = c[3]
break
if len(assignment) == len(leaves):
if evaluate_tree(cond_tree, assignment):
branch_masks[i] = True
if dp.has_other:
at_end_idx = len(dp.branch_names) - 1
if not any(branch_masks[:at_end_idx]):
branch_masks[at_end_idx] = True
for i, m in enumerate(branch_masks):
if m:
dp.active_branches.add(dp.branch_names[i])
def _mark_perform(dp, cons):
simple = getattr(dp, 'parsed', None)
if simple:
for c in cons:
if _match_constraint(c, simple):
if c[3]:
dp.active_branches.add('Skip')
else:
dp.active_branches.add('Enter')
elif dp.cond_tree and dp.cond_leaves:
assignment = {}
for leaf in dp.cond_leaves:
for c in cons:
if _match_leaf(c, leaf):
assignment[leaf] = c[3]
break
if len(assignment) == len(dp.cond_leaves):
if evaluate_tree(dp.cond_tree, assignment):
dp.active_branches.add('Skip')
else:
dp.active_branches.add('Enter')
else:
for c in cons:
if c[0] == dp.label or any(c[0] == f for f in _get_fields_in_cond(dp.label)):
if c[3]:
dp.active_branches.add('Skip')
else:
dp.active_branches.add('Enter')
def _get_fields_in_cond(cond_text):
return re.findall(r'[A-Z][A-Z0-9-]*', cond_text.upper())
# ── 行号定位(基于原始源文本)──
def locate_decision_lines(decision_points, raw_source):
"""在原始源文本中搜索每个决策点的近似行号"""
lines = raw_source.upper().splitlines()
for dp in decision_points:
patterns = _build_search_patterns(dp)
for i, line in enumerate(lines):
for pat in patterns:
if re.search(pat, line):
dp.source_line = i + 1
break
if dp.source_line:
break
def _normalize(text):
"""标准化条件文本用于比较:去多余空白、标准化引号"""
t = re.sub(r'\s+', ' ', text).strip()
t = t.replace('"', "'")
return t
def _build_search_patterns(dp):
texts = []
if dp.kind == 'IF':
texts.append((r'\bIF\b', dp.label))
elif dp.kind == 'EVALUATE':
texts.append((r'\bEVALUATE\b', dp.label))
elif dp.kind == 'PERFORM':
texts.append((r'\bUNTIL\b', dp.condition if hasattr(dp, 'condition') else dp.label
if dp.label else ''))
else:
return [r'$^'] # 永不匹配
patterns = []
for keyword, condition in texts:
if not condition:
continue
norm_cond = _normalize(condition)
# 转义正则特殊字符,但保留空格(替换为\s+)
esc = re.escape(norm_cond)
esc = esc.replace(r'\ ', r'\s+')
esc = esc.replace(r'\'', r"['\"]")
patterns.append(keyword + r'\s+' + esc)
if not patterns:
return [r'$^']
return patterns
# ── HTML 报告(详情页)──
_DETAIL_HTML = '''<!DOCTYPE html>
<html lang="zh">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>{title}</title>
<style>
*, *::before, *::after {{ box-sizing: border-box; margin: 0; padding: 0; }}
body {{
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Noto Sans SC", "PingFang SC", "Microsoft YaHei", sans-serif;
background: #f0f2f5; color: #37474f; font-size: 14px; line-height: 1.6;
}}
.topbar {{
background: linear-gradient(135deg, #1a237e, #283593);
color: #fff; padding: 14px 32px;
display: flex; align-items: center; gap: 16px;
box-shadow: 0 2px 8px rgba(0,0,0,0.15);
}}
.topbar a {{ color: rgba(255,255,255,0.8); text-decoration: none; font-size: 14px; }}
.topbar a:hover {{ color: #fff; text-decoration: underline; }}
.topbar .sep {{ color: rgba(255,255,255,0.4); }}
.topbar h1 {{ font-size: 18px; font-weight: 600; }}
.container {{ max-width: 1000px; margin: 0 auto; padding: 28px 24px; }}
.section {{
background: #fff; border-radius: 10px; padding: 20px 24px;
box-shadow: 0 1px 4px rgba(0,0,0,0.06); margin-bottom: 20px;
}}
.section h2 {{ font-size: 16px; font-weight: 600; color: #1a237e; margin-bottom: 16px; padding-bottom: 8px; border-bottom: 2px solid #e8eaf6; }}
/* 统计卡片行 */
.stats-row {{ display: flex; gap: 16px; flex-wrap: wrap; }}
.stat-card {{
flex: 1; min-width: 140px; background: #f5f7fa; border-radius: 8px; padding: 14px 18px;
text-align: center;
}}
.stat-card .val {{ font-size: 22px; font-weight: 700; font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; }}
.stat-card .lbl {{ font-size: 12px; color: #78909c; margin-top: 2px; }}
.val-green {{ color: #00c853; }}
.val-amber {{ color: #ff8f00; }}
.val-red {{ color: #ff1744; }}
.val-blue {{ color: #1a237e; }}
.legend {{ display: flex; gap: 20px; margin: 16px 0 0 0; font-size: 13px; color: #546e7a; }}
.legend .dot {{ display: inline-block; width: 10px; height: 10px; border-radius: 50%; margin-right: 5px; vertical-align: middle; }}
.dot-green {{ background: #c8e6c9; }}
.dot-red {{ background: #ffcdd2; }}
.dot-amber {{ background: #fff9c4; }}
/* 进度条 */
.prog-bar-detail {{
width: 100%; height: 12px; border-radius: 6px; background: #ffcdd2; overflow: hidden; margin: 10px 0 6px 0;
}}
.prog-fill-detail {{
height: 100%; border-radius: 6px; background: linear-gradient(90deg, #66bb6a, #00c853);
}}
.prog-fill-detail.amber {{ background: linear-gradient(90deg, #ffca28, #ff8f00); }}
.prog-fill-detail.red {{ background: linear-gradient(90deg, #ef5350, #ff1744); }}
/* 表格 */
table {{ width: 100%; border-collapse: collapse; table-layout: fixed; }}
th, td {{ padding: 10px 14px; text-align: left; border-bottom: 1px solid #eceff1; word-break: break-all; }}
th {{ background: #f5f7fa; font-weight: 600; font-size: 12px; color: #78909c; text-transform: uppercase; letter-spacing: 0.5px; }}
tbody tr:hover {{ background: #e8eaf6; }}
tbody tr:last-child td {{ border-bottom: none; }}
/* 决策表列宽 */
.dp-table th:nth-child(1), .dp-table td:nth-child(1) {{ width: 50px; }}
.dp-table th:nth-child(2), .dp-table td:nth-child(2) {{ width: 70px; }}
.dp-table th:nth-child(3), .dp-table td:nth-child(3) {{ width: 50px; }}
.dp-table th:nth-child(5), .dp-table td:nth-child(5) {{ width: 160px; }}
/* 叶条件表列宽 */
.leaf-table th:nth-child(1), .leaf-table td:nth-child(1) {{ width: 110px; }}
.leaf-table th:nth-child(2), .leaf-table td:nth-child(2) {{ width: 60px; }}
.leaf-table th:nth-child(4), .leaf-table td:nth-child(4),
.leaf-table th:nth-child(5), .leaf-table td:nth-child(5) {{ width: 50px; text-align: center; }}
.branch-cell {{ white-space: nowrap; }}
.branch-true {{ background: #c8e6c9; padding: 2px 8px; border-radius: 4px; font-size: 12px; margin: 0 2px; }}
.branch-false {{ background: #ffcdd2; padding: 2px 8px; border-radius: 4px; font-size: 12px; margin: 0 2px; }}
.branch-implied {{ background: #fff9c4; padding: 2px 8px; border-radius: 4px; font-size: 12px; margin: 0 2px; }}
.cond-cell {{ font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; text-align: center; }}
.cond-ok {{ color: #00c853; }}
.cond-miss {{ color: #ff5252; }}
/* 源码 */
.source-section {{ font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; font-size: 13px; }}
.source-line {{ display: flex; padding: 1px 0; }}
.source-line:hover {{ background: #f5f5f5; }}
.source-line .ln {{ width: 3.5em; color: #90a4ae; text-align: right; padding-right: 1em; user-select: none; flex-shrink: 0; }}
.source-line .code {{ white-space: pre; flex: 1; }}
.source-line.hl-green {{ background: #a5d6a7; }}
.source-line.hl-green .ln {{ color: #1b5e20; font-weight: 700; }}
.source-line.hl-red {{ background: #ef9a9a; }}
.source-line.hl-red .ln {{ color: #b71c1c; font-weight: 700; }}
.source-line.hl-amber {{ background: #ffe082; }}
.source-line.hl-amber .ln {{ color: #e65100; font-weight: 700; }}
@media (max-width: 680px) {{
.topbar {{ padding: 12px 16px; flex-wrap: wrap; }}
.container {{ padding: 16px 12px; }}
.section {{ padding: 14px 16px; }}
.stat-card {{ min-width: 100px; padding: 10px 12px; }}
.stat-card .val {{ font-size: 18px; }}
th, td {{ padding: 8px 10px; }}
}}
</style>
</head>
<body>
<div class="topbar">
<a href="{index_relpath}">&#8592; 覆盖率总览</a>
<span class="sep">|</span>
<h1>{title}</h1>
</div>
<div class="container">
<div class="section">
<h2>&#128200; 覆盖率概要</h2>
<div class="stats-row">
<div class="stat-card">
<div class="val {dec_val_cls}">{dec_frac}</div>
<div class="lbl">决策覆盖率</div>
</div>
<div class="stat-card">
<div class="val {cond_val_cls}">{cond_frac}</div>
<div class="lbl">条件覆盖率</div>
</div>
<div class="stat-card">
<div class="val val-blue">{dp_count_text}</div>
<div class="lbl">决策点</div>
</div>
</div>
<div class="prog-bar-detail">
<div class="prog-fill-detail{bar_cls}" style="width:{bar_pct}%"></div>
</div>
<div style="text-align:right;font-size:12px;color:#78909c;">{dec_pct_text}</div>
<div class="legend">
<span><span class="dot dot-green"></span>已覆盖</span>
<span><span class="dot dot-red"></span>未覆盖</span>
<span><span class="dot dot-amber"></span>推断覆盖</span>
</div>
</div>
{decision_table}
{leaf_table}
{source_section}
</div>
</body>
</html>'''
def generate_html_report(decision_points, leaf_stats, source_lines, outpath,
filename='', index_relpath=None, covered_lines=None):
title = f"覆盖率报告 — {filename}" if filename else "覆盖率报告"
total_branches = sum(len(dp.branch_names) for dp in decision_points)
covered_branches = sum(len(dp.active_branches) for dp in decision_points)
implied_branches = sum(len(dp.implied_branches) for dp in decision_points)
if covered_lines:
# 无分支程序:隐式 100%
total_branches = max(total_branches, 1)
covered_branches = max(covered_branches, 1)
total_leaves = len(leaf_stats) * 2
covered_leaves = (sum(1 for l in leaf_stats if l.covered_true) +
sum(1 for l in leaf_stats if l.covered_false))
# 计算数值
is_implicit = bool(covered_lines) # 无分支程序,隐式 100%
dec_pct_val = (covered_branches / total_branches * 100) if total_branches else 0
dec_pct_text = "100%" if is_implicit else (f"{dec_pct_val:.1f}%" if total_branches else "")
dec_frac = "全部覆盖" if is_implicit else (f"{covered_branches}/{total_branches}" if total_branches else "")
cond_frac = f"{covered_leaves}/{total_leaves}" if total_leaves else ""
implied_text = f'+{implied_branches - covered_branches} 推断)' if implied_branches > covered_branches else ''
# 颜色
if is_implicit or not total_branches or dec_pct_val >= 100:
dec_val_cls = 'val-green'
bar_cls = ''
elif dec_pct_val >= 80:
dec_val_cls = 'val-amber'
bar_cls = ' amber'
else:
dec_val_cls = 'val-red'
bar_cls = ' red'
if not total_leaves or covered_leaves == total_leaves:
cond_val_cls = 'val-green'
elif covered_leaves / total_leaves >= 0.8:
cond_val_cls = 'val-amber'
else:
cond_val_cls = 'val-red'
# 决策点表格
if decision_points:
dp_rows = []
for dp in decision_points:
ln = str(dp.source_line) if dp.source_line else '?'
branch_cells = []
for bn in dp.branch_names:
if bn in dp.active_branches:
branch_cells.append(f'<span class="branch-true">{bn} &#10003;</span>')
elif bn in dp.implied_branches:
branch_cells.append(f'<span class="branch-implied">{bn} &#9675;</span>')
else:
branch_cells.append(f'<span class="branch-false">{bn} &#10007;</span>')
dp_rows.append(f'<tr><td>#{dp.id}</td><td>{dp.kind}</td><td>{ln}</td>'
f'<td style="font-family:monospace">{dp.label}</td>'
f'<td class="branch-cell">{" ".join(branch_cells)}</td></tr>')
decision_table = f'''<div class="section">
<h2>&#128220; 决策点</h2>
<table class="dp-table">
<thead><tr><th>#</th><th>类型</th><th>行号</th><th>条件</th><th>分支</th></tr></thead>
<tbody>{"".join(dp_rows)}</tbody>
</table>
</div>'''
else:
decision_table = ''
# 叶条件表格
if leaf_stats:
leaf_rows = []
for leaf in leaf_stats:
t = '<span class="cond-ok cond-cell">&#10003;</span>' if leaf.covered_true else '<span class="cond-miss cond-cell">&#10007;</span>'
f = '<span class="cond-ok cond-cell">&#10003;</span>' if leaf.covered_false else '<span class="cond-miss cond-cell">&#10007;</span>'
leaf_rows.append(f'<tr><td>{leaf.field}</td><td>{leaf.op}</td>'
f'<td>{leaf.value}</td><td>{t}</td><td>{f}</td></tr>')
leaf_table = f'''<div class="section">
<h2>&#128290; 条件覆盖明细(叶条件)</h2>
<table class="leaf-table">
<thead><tr><th>字段</th><th>运算符</th><th>值</th><th>真</th><th>假</th></tr></thead>
<tbody>{"".join(leaf_rows)}</tbody>
</table>
</div>'''
else:
leaf_table = ''
# 源码标注
if source_lines:
line_cov = {}
for dp in decision_points:
if dp.source_line:
if dp.source_line not in line_cov:
line_cov[dp.source_line] = []
has_missed = any(bn not in dp.active_branches for bn in dp.branch_names)
has_active = any(bn in dp.active_branches for bn in dp.branch_names)
if has_active and not has_missed:
line_cov[dp.source_line].append('hl-green')
elif has_active:
line_cov[dp.source_line].append('hl-red')
else:
line_cov[dp.source_line].append('hl-amber')
# 无分支程序:所有 PD 行标记为已覆盖
if covered_lines:
for ln in covered_lines:
line_cov.setdefault(ln, []).append('hl-green')
src_lines = []
for i, line in enumerate(source_lines, 1):
cls_list = line_cov.get(i, [])
hl = ' ' + ' '.join(cls_list) if cls_list else ''
src_lines.append(f'<div class="source-line{hl}">'
f'<span class="ln">{i}</span>'
f'<span class="code">{line}</span></div>')
source_section = f'''<div class="section source-section">
<h2>&#128214; 源码标注</h2>
{"".join(src_lines)}
</div>'''
else:
source_section = ''
html = _DETAIL_HTML.format(
title=title,
index_relpath=index_relpath or '#',
dec_frac=dec_frac,
dec_pct_text=dec_pct_text,
dec_val_cls=dec_val_cls,
cond_frac=cond_frac,
cond_val_cls=cond_val_cls,
bar_cls=bar_cls,
bar_pct=str(int(dec_pct_val)),
decision_table=decision_table,
leaf_table=leaf_table,
source_section=source_section,
dp_count_text=('' if is_implicit else str(len(decision_points))),
)
outpath = Path(outpath)
outpath.parent.mkdir(parents=True, exist_ok=True)
outpath.write_text(html, encoding='utf-8')
# ── 总括索引页 ──
_INDEX_HTML = '''<!DOCTYPE html>
<html lang="zh">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>覆盖率总览</title>
<style>
*, *::before, *::after {{ box-sizing: border-box; margin: 0; padding: 0; }}
body {{
font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Noto Sans SC", "PingFang SC", "Microsoft YaHei", sans-serif;
background: #f0f2f5; color: #37474f; font-size: 14px; line-height: 1.6;
}}
/* 顶栏 */
.topbar {{
background: linear-gradient(135deg, #1a237e, #283593);
color: #fff; padding: 18px 32px;
display: flex; align-items: center; justify-content: space-between;
box-shadow: 0 2px 8px rgba(0,0,0,0.15);
}}
.topbar h1 {{ font-size: 20px; font-weight: 600; letter-spacing: 0.5px; }}
.topbar .ts {{ font-size: 13px; opacity: 0.8; font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; }}
.container {{ max-width: 1200px; margin: 0 auto; padding: 28px 24px; }}
/* 统计卡片 */
.cards {{ display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 16px; margin-bottom: 28px; }}
.card {{
background: #fff; border-radius: 10px; padding: 20px 22px;
box-shadow: 0 1px 4px rgba(0,0,0,0.06); transition: box-shadow 0.2s, transform 0.2s;
}}
.card:hover {{ box-shadow: 0 4px 16px rgba(0,0,0,0.10); transform: translateY(-2px); }}
.card .num {{ font-size: 28px; font-weight: 700; font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; line-height: 1.2; }}
.card .label {{ font-size: 13px; color: #78909c; margin-top: 4px; }}
.num-green {{ color: #00c853; }}
.num-amber {{ color: #ff8f00; }}
.num-red {{ color: #ff1744; }}
.num-blue {{ color: #1a237e; }}
/* 图表行 */
.charts-row {{
display: flex; gap: 32px; justify-content: center; flex-wrap: wrap;
background: #fff; border-radius: 10px; padding: 28px 20px;
box-shadow: 0 1px 4px rgba(0,0,0,0.06); margin-bottom: 24px;
}}
.chart-box {{ text-align: center; }}
.chart-box svg {{ display: block; margin: 0 auto; }}
.chart-box .chart-label {{ margin-top: 8px; font-size: 14px; font-weight: 500; color: #546e7a; }}
.legend {{
display: flex; justify-content: center; gap: 24px; margin: 0 0 20px 0;
font-size: 13px; color: #546e7a;
}}
.legend .dot {{ display: inline-block; width: 10px; height: 10px; border-radius: 50%; margin-right: 6px; vertical-align: middle; }}
.legend .dot-green {{ background: #00c853; }}
.legend .dot-red {{ background: #ff5252; }}
.legend .dot-amber {{ background: #ffd740; }}
/* 工具栏 */
.toolbar {{
display: flex; justify-content: space-between; align-items: center;
margin-bottom: 14px; flex-wrap: wrap; gap: 10px;
}}
.toolbar input {{
padding: 8px 14px; border: 1px solid #cfd8dc; border-radius: 6px;
font-size: 14px; width: 220px; outline: none; transition: border-color 0.2s;
font-family: inherit;
}}
.toolbar input:focus {{ border-color: #3f51b5; box-shadow: 0 0 0 3px rgba(63,81,181,0.12); }}
.toolbar .sort-group {{ display: flex; gap: 6px; }}
.toolbar .sort-btn {{
padding: 6px 14px; border: 1px solid #cfd8dc; border-radius: 6px;
background: #fff; cursor: pointer; font-size: 13px; color: #546e7a;
transition: all 0.15s; font-family: inherit;
}}
.toolbar .sort-btn:hover {{ background: #eceff1; }}
.toolbar .sort-btn.active {{ background: #e8eaf6; border-color: #3f51b5; color: #1a237e; font-weight: 500; }}
/* 表格 */
.table-wrap {{
background: #fff; border-radius: 10px; overflow: hidden;
box-shadow: 0 1px 4px rgba(0,0,0,0.06);
}}
table {{ width: 100%; border-collapse: collapse; }}
thead th {{
background: #eceff1; font-weight: 600; font-size: 13px; color: #546e7a;
padding: 12px 16px; text-align: left; cursor: pointer; user-select: none;
position: sticky; top: 0; z-index: 1; white-space: nowrap;
transition: background 0.15s;
}}
thead th:hover {{ background: #dde3e8; }}
thead th .sort-arrow {{ margin-left: 4px; font-size: 11px; opacity: 0.4; }}
thead th.sorted .sort-arrow {{ opacity: 1; color: #1a237e; }}
tbody tr {{ transition: background 0.15s; }}
tbody tr:nth-child(even) {{ background: #fafbfc; }}
tbody tr:hover {{ background: #e8eaf6; }}
tbody td {{ padding: 12px 16px; border-top: 1px solid #eceff1; vertical-align: middle; }}
tbody tr.hidden {{ display: none; }}
.prog-name {{ font-weight: 500; }}
.prog-name a {{ color: #283593; text-decoration: none; }}
.prog-name a:hover {{ text-decoration: underline; color: #1a237e; }}
/* 进度条 */
.prog-wrap {{
display: inline-flex; align-items: center; gap: 10px; width: 100%;
}}
.prog-bar {{
flex: 1; max-width: 180px; height: 20px; border-radius: 10px;
background: #ffcdd2; overflow: hidden; position: relative;
}}
.prog-fill {{
height: 100%; border-radius: 10px; transition: width 0.4s ease;
background: linear-gradient(90deg, #66bb6a, #00c853);
position: relative;
}}
.prog-fill.amber {{ background: linear-gradient(90deg, #ffca28, #ff8f00); }}
.prog-fill.red {{ background: linear-gradient(90deg, #ef5350, #ff1744); }}
.prog-fill .prog-label {{
position: absolute; right: 6px; top: 50%; transform: translateY(-50%);
font-size: 11px; font-weight: 700; color: #fff;
text-shadow: 0 1px 2px rgba(0,0,0,0.3);
}}
.prog-fill.full {{ border-radius: 10px; }}
.prog-text {{ font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; font-size: 13px; white-space: nowrap; min-width: 48px; }}
/* 状态徽标 */
.badge {{
display: inline-block; padding: 3px 10px; border-radius: 12px;
font-size: 12px; font-weight: 600; letter-spacing: 0.3px;
}}
.badge-pass {{ background: #e8f5e9; color: #2e7d32; }}
.badge-warn {{ background: #fff8e1; color: #e65100; }}
.badge-fail {{ background: #ffebee; color: #c62828; }}
/* 条件覆盖列 */
.cond-cell {{ font-family: "Cascadia Code","Fira Code","JetBrains Mono",Consolas,monospace; font-size: 13px; }}
/* 响应式 */
@media (max-width: 680px) {{
.topbar {{ flex-direction: column; align-items: flex-start; gap: 6px; padding: 14px 18px; }}
.container {{ padding: 16px 12px; }}
.cards {{ grid-template-columns: 1fr 1fr; }}
.toolbar input {{ width: 100%; }}
.toolbar {{ flex-direction: column; align-items: stretch; }}
.prog-bar {{ max-width: 100px; }}
thead th, tbody td {{ padding: 8px 10px; }}
}}
</style>
</head>
<body>
<div class="topbar">
<h1>&#128202; 覆盖率总览报告</h1>
<span class="ts">{timestamp}</span>
</div>
<div class="container">
<div class="cards">
<div class="card">
<div class="num {dec_num_cls}">{agg_dec_num}</div>
<div class="label">决策覆盖率</div>
</div>
<div class="card">
<div class="num {cond_num_cls}">{agg_cond_num}</div>
<div class="label">条件覆盖率</div>
</div>
<div class="card">
<div class="num num-blue">{prog_count}</div>
<div class="label">已分析程序</div>
</div>
<div class="card">
<div class="num {uncovered_num_cls}">{uncovered_count}</div>
<div class="label">未完全覆盖程序</div>
</div>
</div>
<div class="charts-row">
<div class="chart-box">
{dec_ring_svg}
<div class="chart-label">决策覆盖率</div>
</div>
<div class="chart-box">
{cond_ring_svg}
<div class="chart-label">条件覆盖率</div>
</div>
</div>
<div class="legend">
<span><span class="dot dot-green"></span>已覆盖</span>
<span><span class="dot dot-red"></span>未覆盖</span>
<span><span class="dot dot-amber"></span>推断覆盖</span>
</div>
<div class="toolbar">
<input type="text" id="filterInput" placeholder="&#128269; 输入程序名过滤..." oninput="filterTable()">
<div class="sort-group">
<button class="sort-btn active" data-sort="name" onclick="setSort('name')">程序名 &#8593;</button>
<button class="sort-btn" data-sort="cov" onclick="setSort('cov')">覆盖率 &#8595;</button>
</div>
</div>
<div class="table-wrap">
<table id="progTable">
<thead>
<tr>
<th data-col="name" onclick="sortBy('name')">程序 <span class="sort-arrow">&#8593;</span></th>
<th data-col="branch" onclick="sortBy('branch')">决策分支 <span class="sort-arrow">&#8593;</span></th>
<th data-col="cond" onclick="sortBy('cond')">条件覆盖 <span class="sort-arrow">&#8593;</span></th>
<th data-col="cov" onclick="sortBy('cov')">覆盖率 <span class="sort-arrow">&#8593;</span></th>
<th>状态</th>
</tr>
</thead>
<tbody>
{rows}
</tbody>
</table>
</div>
</div>
<script>
let sortCol = 'name', sortDir = 1;
function setSort(col) {{
document.querySelectorAll('.sort-btn').forEach(b => b.classList.remove('active'));
if (col === 'name') {{
document.querySelector('.sort-btn[data-sort="name"]').classList.add('active');
sortCol = 'name'; sortDir = 1;
}} else {{
document.querySelector('.sort-btn[data-sort="cov"]').classList.add('active');
sortCol = 'cov'; sortDir = -1;
}}
doSort();
}}
function sortBy(col) {{
if (sortCol === col) {{ sortDir = -sortDir; }}
else {{ sortCol = col; sortDir = col === 'name' ? 1 : -1; }}
document.querySelectorAll('.sort-btn').forEach(b => b.classList.remove('active'));
doSort();
}}
function doSort() {{
const tbody = document.querySelector('#progTable tbody');
const rows = Array.from(tbody.querySelectorAll('tr:not(.hidden)'));
rows.sort((a, b) => {{
var va, vb;
if (sortCol === 'name') {{
va = a.cells[0].textContent.trim(); vb = b.cells[0].textContent.trim();
return sortDir * va.localeCompare(vb);
}} else if (sortCol === 'branch') {{
va = a.cells[1].textContent.trim(); vb = b.cells[1].textContent.trim();
return sortDir * va.localeCompare(vb);
}} else if (sortCol === 'cond') {{
va = a.cells[2].textContent.trim(); vb = b.cells[2].textContent.trim();
return sortDir * va.localeCompare(vb);
}} else {{
va = parseFloat(a.getAttribute('data-cov') || '0');
vb = parseFloat(b.getAttribute('data-cov') || '0');
return sortDir * (va - vb);
}}
}});
rows.forEach(r => tbody.appendChild(r));
}}
function filterTable() {{
const q = document.getElementById('filterInput').value.toUpperCase();
const rows = document.querySelectorAll('#progTable tbody tr');
rows.forEach(r => {{
r.classList.toggle('hidden', !r.cells[0].textContent.toUpperCase().includes(q));
}});
doSort();
}}
</script>
</body>
</html>'''
def _ring_svg(pct, color_stops):
"""生成 SVG 圆环 HTML。pct: 0-100 浮点数。"""
r = 54
circ = 2 * 3.14159265 * r
offset = circ * (1 - pct / 100) if pct > 0 else circ
if pct >= 80:
stroke = '#00c853'
elif pct >= 50:
stroke = '#ff8f00'
else:
stroke = '#ff1744'
return (
f'<svg width="140" height="140" viewBox="0 0 140 140">'
f'<circle cx="70" cy="70" r="{r}" fill="none" stroke="#eceff1" stroke-width="12"/>'
f'<circle cx="70" cy="70" r="{r}" fill="none" stroke="{stroke}" stroke-width="12"'
f' stroke-dasharray="{circ}" stroke-dashoffset="{offset}"'
f' transform="rotate(-90 70 70)" stroke-linecap="round"/>'
f'<text x="70" y="64" text-anchor="middle" dominant-baseline="central"'
f' font-size="26" font-weight="700" fill="#37474f"'
f' font-family="Cascadia Code,Fira Code,JetBrains Mono,Consolas,monospace">'
f'{pct:.0f}%</text>'
f'<text x="70" y="86" text-anchor="middle" dominant-baseline="central"'
f' font-size="11" fill="#78909c">覆盖率</text>'
f'</svg>'
)
def generate_coverage_index(programs, outdir):
"""生成覆盖率总括索引页。"""
from datetime import datetime
timestamp = datetime.now().strftime('%Y-%m-%d %H:%M')
agg_total = sum(p['total_branches'] for p in programs)
agg_covered = sum(p['covered_branches'] for p in programs)
agg_implied = sum(p['implied_branches'] for p in programs)
agg_ctotal = sum(p['total_conditions'] for p in programs)
agg_ccovered = sum(p['covered_conditions'] for p in programs)
agg_dec_pct = (agg_covered / agg_total * 100) if agg_total else 0
agg_cond_pct = (agg_ccovered / agg_ctotal * 100) if agg_ctotal else 0
uncovered_count = sum(1 for p in programs if p['total_branches'] and
p['covered_branches'] < p['total_branches'])
dec_num_cls = 'num-green' if agg_dec_pct == 100 else ('num-amber' if agg_dec_pct >= 80 else 'num-red')
cond_num_cls = 'num-green' if agg_cond_pct == 100 else ('num-amber' if agg_cond_pct >= 80 else 'num-red')
uncovered_num_cls = 'num-green' if uncovered_count == 0 else 'num-red'
def sort_key(p):
if p['total_branches']:
return -p['covered_branches'] / p['total_branches']
return -1.0
sorted_programs = sorted(programs, key=sort_key)
rows = []
for p in sorted_programs:
name = p['name']
href = p['detail_relpath']
tb = p['total_branches']
cb = p['covered_branches']
ib = p['implied_branches']
tc = p['total_conditions']
cc = p['covered_conditions']
imp = p.get('implicit_100', False)
pct_dec = (cb / tb * 100) if tb else 0
pct_text = "全部覆盖" if imp else (f"{pct_dec:.1f}%" if tb else "")
implied_text = f'+{ib - cb} 推断)' if ib > cb else ''
branch_text = "" if imp else f"{cb}/{tb}"
cond_text = f"{cc}/{tc}" if tc else ""
bar_pct = int(pct_dec)
# 进度条颜色
if imp or pct_dec >= 100:
bar_cls = ''
elif pct_dec >= 80:
bar_cls = ' amber'
else:
bar_cls = ' red'
# 状态徽标
if tb == 0 or (cb == tb and not (ib > cb)):
badge = '<span class="badge badge-pass">&#10003; 完全</span>'
elif cb == tb and ib > cb:
badge = '<span class="badge badge-warn">&#9675; 推断</span>'
elif pct_dec >= 80:
badge = '<span class="badge badge-warn">&#9888; 不足</span>'
else:
badge = '<span class="badge badge-fail">&#10007; 欠缺</span>'
# 条件覆盖数字颜色
if tc:
cond_pct = cc / tc * 100
cond_color = 'num-green' if cond_pct == 100 else ('num-amber' if cond_pct >= 80 else 'num-red')
cond_display = f'<span class="cond-cell {cond_color}">{cond_text}</span>'
else:
cond_display = '<span class="cond-cell" style="color:#b0bec5">—</span>'
row_class = 'row-imperfect' if cb < tb else ''
rows.append(f'''<tr class="{row_class}" data-cov="{pct_dec}">
<td class="prog-name"><a href="{href}">{name}</a></td>
<td>{branch_text} {implied_text}</td>
<td>{cond_display}</td>
<td>
<div class="prog-wrap">
<div class="prog-bar">
<div class="prog-fill{bar_cls}" style="width:{bar_pct}%">
<span class="prog-label">{pct_text}</span>
</div>
</div>
<span class="prog-text">{pct_text}</span>
</div>
</td>
<td>{badge}</td>
</tr>''')
dec_ring_svg = _ring_svg(agg_dec_pct, '')
cond_ring_svg = _ring_svg(agg_cond_pct, '')
html = _INDEX_HTML.format(
timestamp=timestamp,
agg_dec_num=f"{agg_covered}/{agg_total}",
dec_num_cls=dec_num_cls,
agg_cond_num=f"{agg_ccovered}/{agg_ctotal}" if agg_ctotal else "无数据",
cond_num_cls=cond_num_cls,
prog_count=str(len(programs)),
uncovered_num_cls=uncovered_num_cls,
uncovered_count=str(uncovered_count),
dec_ring_svg=dec_ring_svg,
cond_ring_svg=cond_ring_svg,
rows='\n'.join(rows),
)
outpath = Path(outdir) / 'coverage' / 'index.html'
outpath.parent.mkdir(parents=True, exist_ok=True)
outpath.write_text(html, encoding='utf-8')
# ── PROCEDURE DIVISION 行范围定位(用于无分支程序标记)──
def _find_proc_range(raw_source: str):
"""返回 PROCEDURE DIVISION 的行范围 (start_line, end_line) 1-indexed,或 None。"""
lines = raw_source.splitlines()
proc_start = None
for i, line in enumerate(lines):
if re.search(r'PROCEDURE\s+DIVISION', line.upper()):
proc_start = i + 1
break
if proc_start is None:
return None
# 找下一个 DIVISION 作为结束边界(或文件尾)
for i in range(proc_start, len(lines)):
if re.search(r'(IDENTIFICATION|DATA|ENVIRONMENT)\s+DIVISION', lines[i].upper()):
return (proc_start, i) # 不包含下一个 DIVISION
return (proc_start, len(lines) + 1)
# ── 接入入口 ──
def run_coverage(branch_tree, branch_paths_with_assigns, fields,
raw_source, output_prefix, index_relpath=None):
"""完整覆盖率流程:收集 → 标记 → 定位 → 输出。
Returns:
dict: 汇总数据,用于总括页聚合
"""
decision_points, leaf_stats = collect_decision_points(branch_tree, fields)
mark_coverage(decision_points, leaf_stats, branch_paths_with_assigns, fields)
if raw_source:
locate_decision_lines(decision_points, raw_source)
total = sum(len(dp.branch_names) for dp in decision_points)
covered = sum(len(dp.active_branches) for dp in decision_points)
implied = sum(len(dp.implied_branches) for dp in decision_points)
leaf_covered = (sum(1 for l in leaf_stats if l.covered_true) +
sum(1 for l in leaf_stats if l.covered_false))
leaf_total = len(leaf_stats) * 2
# 无决策点但有路径 → PROCEDURE DIVISION 全部覆盖
covered_lines = set()
if total == 0 and branch_paths_with_assigns and raw_source:
proc_range = _find_proc_range(raw_source)
if proc_range:
covered_lines.update(range(proc_range[0], proc_range[1]))
total = 1
covered = 1
if output_prefix:
generate_html_report(decision_points, leaf_stats,
raw_source.splitlines() if raw_source else [],
f"{output_prefix}_coverage.html",
Path(output_prefix).stem,
index_relpath=index_relpath,
covered_lines=covered_lines)
# 控制台摘要
if total or leaf_total:
logger.info(f"\n=== 分支覆盖率 ===")
if covered_lines and not decision_points:
logger.info(" 程序无分支结构,全部代码已覆盖")
for dp in decision_points:
branches = []
for bn in dp.branch_names:
if bn in dp.active_branches:
branches.append(f'{bn} [x]')
elif bn in dp.implied_branches:
branches.append(f'{bn} [o]')
else:
branches.append(f'{bn} [ ]')
ln = f":{dp.source_line}" if dp.source_line else ""
logger.info(f" #{dp.id} [{dp.kind}] {dp.label}{ln}")
logger.info(f" {' | '.join(branches)}")
if total:
pct = covered / total * 100
logger.info(f"\n 决策覆盖率:{covered}/{total}{pct:.1f}%")
if leaf_total:
pct = leaf_covered / leaf_total * 100
logger.info(f" 条件覆盖率:{leaf_covered}/{leaf_total}{pct:.1f}%")
if output_prefix:
logger.info(f"\n 覆盖率报告:{output_prefix}_coverage.html")
implicit_100 = bool(covered_lines)
return {
'name': Path(output_prefix).stem if output_prefix else '',
'detail_relpath': ('../' + Path(output_prefix).stem + '_coverage.html'
if output_prefix else ''),
'total_branches': total,
'covered_branches': covered,
'implied_branches': implied,
'implicit_100': implicit_100,
'total_conditions': leaf_total,
'covered_conditions': leaf_covered,
'_decision_points': decision_points,
'_leaf_stats': leaf_stats,
}