Huaxu Sentinel Active Safety Platform with embedded algorithm code, Docker Compose setup, and vendored dataset scaffolds for clone-and-run. Co-authored-by: Cursor <cursoragent@cursor.com>
132 lines
4.0 KiB
Python
132 lines
4.0 KiB
Python
#!/usr/bin/env python3
|
||
"""按 workflow active_packs 生成 manifests/yaml_active/*.yaml(可多包合并 train/val)。"""
|
||
|
||
from __future__ import annotations
|
||
|
||
import argparse
|
||
import sys
|
||
from pathlib import Path
|
||
|
||
import yaml
|
||
|
||
SCRIPT_DIR = Path(__file__).resolve().parent
|
||
sys.path.insert(0, str(SCRIPT_DIR))
|
||
from pack_registry import ( # noqa: E402
|
||
load_active_pack_names,
|
||
resolve_pack_dir,
|
||
)
|
||
|
||
|
||
def fmt_names(names) -> str:
|
||
if isinstance(names, dict):
|
||
lines = ["names:"]
|
||
for k, v in sorted(names.items(), key=lambda x: int(x[0])):
|
||
lines.append(f" {k}: {v}")
|
||
return "\n".join(lines)
|
||
inner = ", ".join(f'"{n}"' for n in names)
|
||
return f"names: [{inner}]"
|
||
|
||
|
||
def yaml_list(key: str, paths: list[str]) -> str:
|
||
if len(paths) == 1:
|
||
return f"{key}: {paths[0]}"
|
||
lines = [f"{key}:"] + [f" - {p}" for p in paths]
|
||
return "\n".join(lines)
|
||
|
||
|
||
def pack_task_root(root: Path, pack_name: str, task_dir: str) -> Path:
|
||
return resolve_pack_dir(root, pack_name) / task_dir
|
||
|
||
|
||
def build_detect_pose_yaml(
|
||
task: str,
|
||
tcfg: dict,
|
||
root: Path,
|
||
pack_names: list[str],
|
||
typ: str,
|
||
) -> str:
|
||
task_dir = tcfg["task_dir"]
|
||
bases = []
|
||
train_paths = []
|
||
val_paths = []
|
||
for pack in pack_names:
|
||
base = pack_task_root(root, pack, task_dir)
|
||
if not base.is_dir():
|
||
print(f" skip pack {pack}: missing {base}")
|
||
continue
|
||
bases.append(base)
|
||
train_paths.append(str((base / "images" / "train").resolve()))
|
||
val_paths.append(str((base / "images" / "val").resolve()))
|
||
|
||
if not bases:
|
||
raise SystemExit(f"{task}: 无可用数据包目录")
|
||
|
||
lines = [
|
||
f"# {task} — packs: {', '.join(pack_names)}",
|
||
f"path: {bases[0]}",
|
||
yaml_list("train", train_paths),
|
||
yaml_list("val", val_paths),
|
||
"",
|
||
]
|
||
if typ == "pose":
|
||
lines.insert(4, f"kpt_shape: {tcfg.get('kpt_shape', [37, 3])}")
|
||
else:
|
||
lines.extend([f"nc: {tcfg['nc']}", fmt_names(tcfg["names"]), ""])
|
||
return "\n".join(lines)
|
||
|
||
|
||
def build_classify_yaml(task: str, tcfg: dict, root: Path, pack_names: list[str]) -> str:
|
||
task_dir = tcfg["task_dir"]
|
||
if len(pack_names) > 1:
|
||
print(f" warn {task}: classify 暂用首个包 {pack_names[0]}(多包请先合并目录)")
|
||
base = pack_task_root(root, pack_names[0], task_dir)
|
||
return f"""# {task} — pack: {pack_names[0]}
|
||
path: {base.resolve()}
|
||
train: train
|
||
val: val
|
||
test: test
|
||
"""
|
||
|
||
|
||
def main() -> None:
|
||
p = argparse.ArgumentParser()
|
||
p.add_argument("--root", type=Path, default=SCRIPT_DIR.parent)
|
||
p.add_argument("--packs", help="逗号分隔,覆盖 workflow active_packs")
|
||
p.add_argument("--task", help="只生成某一任务")
|
||
args = p.parse_args()
|
||
root = args.root.resolve()
|
||
reg = yaml.safe_load((root / "datasets.registry.yaml").read_text(encoding="utf-8"))
|
||
cli = [x.strip() for x in args.packs.split(",")] if args.packs else None
|
||
pack_names = load_active_pack_names(root, cli)
|
||
if not pack_names:
|
||
raise SystemExit("active_packs 为空,请编辑 ML/workflow.registry.yaml 或 --packs")
|
||
|
||
out_dir = root / "manifests" / "yaml_active"
|
||
out_dir.mkdir(parents=True, exist_ok=True)
|
||
print(f"active_packs: {pack_names}")
|
||
|
||
tasks = reg["tasks"]
|
||
if args.task:
|
||
if args.task not in tasks:
|
||
raise SystemExit(f"未知 task: {args.task}")
|
||
tasks = {args.task: tasks[args.task]}
|
||
|
||
for task, tcfg in tasks.items():
|
||
typ = tcfg["type"]
|
||
if typ in ("detect", "pose"):
|
||
content = build_detect_pose_yaml(task, tcfg, root, pack_names, typ)
|
||
elif typ == "classify":
|
||
content = build_classify_yaml(task, tcfg, root, pack_names)
|
||
else:
|
||
print(f" skip {task}: type {typ}")
|
||
continue
|
||
out = out_dir / f"{task}.yaml"
|
||
out.write_text(content, encoding="utf-8")
|
||
print(f" wrote {out.relative_to(root)}")
|
||
|
||
print("完成。")
|
||
|
||
|
||
if __name__ == "__main__":
|
||
main()
|