run_daily.py 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. #!/usr/bin/env python3
  2. """End-to-end RobotDaily pipeline."""
  3. from __future__ import annotations
  4. import argparse
  5. import json
  6. from pathlib import Path
  7. from typing import Any, Dict, List
  8. from enrich_papers import enrich_selection
  9. from fetch_arxiv import fetch_candidates
  10. from publish_discord import DiscordPublisher, publish_digest
  11. from publish_hugo import publish_markdown_to_hugo, publish_to_hugo
  12. from render_digest import render_html, render_markdown
  13. from select_papers import select_papers
  14. from utils import DEFAULT_OUTPUT_DIR, ensure_dir, load_env, log, now_local, write_json, write_text
  15. def parse_models(raw: str) -> List[str]:
  16. return [item.strip() for item in str(raw or "").split(",") if item.strip()]
  17. def choose_selection(lookback_days: int, fallback_lookback_days: int, max_results_per_domain: int) -> Dict[str, Any]:
  18. candidates = fetch_candidates(lookback_days=lookback_days, max_results_per_domain=max_results_per_domain)
  19. selection = select_papers(candidates)
  20. counts = selection.get("counts", {})
  21. if any(counts.get(domain, 0) < 2 for domain in ["embodied", "representation", "reinforcement"]) and fallback_lookback_days > lookback_days:
  22. log(f"Some domains are sparse with lookback={lookback_days}; retrying with lookback={fallback_lookback_days}")
  23. candidates = fetch_candidates(lookback_days=fallback_lookback_days, max_results_per_domain=max_results_per_domain)
  24. selection = select_papers(candidates)
  25. selection["candidate_count"] = len(candidates)
  26. selection["candidates"] = candidates
  27. return selection
  28. def build_output_paths(root: Path, date_slug: str) -> Dict[str, Path]:
  29. bundle_dir = ensure_dir(root / date_slug)
  30. return {
  31. "bundle_dir": bundle_dir,
  32. "candidates_json": bundle_dir / "candidates.json",
  33. "selected_json": bundle_dir / "selected.json",
  34. "enriched_json": bundle_dir / "enriched.json",
  35. "digest_html": bundle_dir / "robotdaily.html",
  36. "digest_md": bundle_dir / "robotdaily.md",
  37. "manifest_json": bundle_dir / "manifest.json",
  38. }
  39. def main() -> None:
  40. parser = argparse.ArgumentParser(description="Run RobotDaily daily digest pipeline")
  41. parser.add_argument("--output-root", default="")
  42. parser.add_argument("--lookback-days", type=int, default=2)
  43. parser.add_argument("--fallback-lookback-days", type=int, default=4)
  44. parser.add_argument("--max-results-per-domain", type=int, default=40)
  45. parser.add_argument("--models", default="")
  46. parser.add_argument("--skip-enrich", action="store_true")
  47. parser.add_argument("--publish-discord", action="store_true")
  48. parser.add_argument("--publish-hugo", action="store_true")
  49. parser.add_argument("--hugo-content-dir", default="")
  50. parser.add_argument("--dry-run", action="store_true")
  51. args = parser.parse_args()
  52. env = load_env()
  53. date_slug = now_local().strftime("%Y-%m-%d")
  54. output_root = Path(args.output_root or env.get("ROBOTDAILY_OUTPUT_DIR", str(DEFAULT_OUTPUT_DIR)))
  55. paths = build_output_paths(output_root, date_slug)
  56. selection = choose_selection(
  57. lookback_days=args.lookback_days,
  58. fallback_lookback_days=args.fallback_lookback_days,
  59. max_results_per_domain=args.max_results_per_domain,
  60. )
  61. write_json(paths["candidates_json"], {"generated_at": now_local().isoformat(), "papers": selection.get("candidates", [])})
  62. write_json(paths["selected_json"], {k: v for k, v in selection.items() if k != "candidates"})
  63. models = parse_models(args.models or env.get("INSIGHT_MODELS", "qwen3.5:cloud,glm-4.7:cloud"))
  64. if args.skip_enrich:
  65. enriched = {k: v for k, v in selection.items() if k != "candidates"}
  66. for paper in enriched.get("papers", []):
  67. paper.setdefault("translated_abstract_zh", paper.get("summary", ""))
  68. paper.setdefault("brief_explanation_zh", paper.get("selection_reason", ""))
  69. paper.setdefault("tags", [])
  70. else:
  71. enriched = enrich_selection({k: v for k, v in selection.items() if k != "candidates"}, model_names=models)
  72. write_json(paths["enriched_json"], enriched)
  73. html = render_html(enriched)
  74. markdown = render_markdown(enriched)
  75. write_text(paths["digest_html"], html)
  76. write_text(paths["digest_md"], markdown)
  77. manifest = {
  78. "generated_at": now_local().isoformat(),
  79. "date": date_slug,
  80. "candidate_count": selection.get("candidate_count", 0),
  81. "selected_count": len(enriched.get("papers", [])),
  82. "counts": enriched.get("counts", {}),
  83. "models": models,
  84. "effective_models_used": enriched.get("effective_models_used", []),
  85. "paths": {name: str(path) for name, path in paths.items() if name != "bundle_dir"},
  86. }
  87. write_json(paths["manifest_json"], manifest)
  88. if args.publish_hugo:
  89. content_dir = args.hugo_content_dir or env.get("HUGO_CONTENT_DIR", "")
  90. if content_dir:
  91. hugo_target = publish_to_hugo(
  92. markdown_path=str(paths["digest_md"]),
  93. manifest_path=str(paths["manifest_json"]),
  94. content_dir=content_dir,
  95. )
  96. else:
  97. site_dir = env.get("HUGO_SITE_DIR", "")
  98. if not site_dir:
  99. raise SystemExit("--publish-hugo 需要设置 HUGO_CONTENT_DIR 或 HUGO_SITE_DIR")
  100. hugo_target = publish_markdown_to_hugo(
  101. str(paths["digest_md"]),
  102. site_dir=site_dir,
  103. section=env.get("HUGO_CONTENT_SECTION", "ai-daily"),
  104. )
  105. manifest["hugo_target"] = str(hugo_target)
  106. write_json(paths["manifest_json"], manifest)
  107. if args.publish_discord:
  108. publisher = DiscordPublisher(
  109. openclaw_bin=env.get("OPENCLAW_BIN", "openclaw"),
  110. account_id=env.get("DISCORD_ACCOUNT_ID", "codex"),
  111. mode=env.get("DISCORD_DELIVERY_MODE", "thread"),
  112. guild_id=env.get("DISCORD_GUILD_ID", ""),
  113. parent_channel_id=env.get("DISCORD_PARENT_CHANNEL_ID", ""),
  114. target_channel_id=env.get("DISCORD_TARGET_CHANNEL_ID", ""),
  115. target_channel_name=env.get("DISCORD_TARGET_CHANNEL_NAME", ""),
  116. category_id=env.get("DISCORD_CATEGORY_ID", ""),
  117. bot_token=env.get("DISCORD_BOT_TOKEN", ""),
  118. thread_auto_archive_min=int(env.get("DISCORD_THREAD_AUTO_ARCHIVE_MIN", "10080")),
  119. dry_run=args.dry_run,
  120. )
  121. target = publish_digest(
  122. enriched,
  123. markdown_path=str(paths["digest_md"]),
  124. publisher=publisher,
  125. )
  126. manifest["discord_target"] = target
  127. write_json(paths["manifest_json"], manifest)
  128. print(json.dumps(manifest, ensure_ascii=False, indent=2))
  129. if __name__ == "__main__":
  130. main()