run_daily.py 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  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 render_digest import render_html, render_markdown
  12. from select_papers import select_papers
  13. from utils import DEFAULT_OUTPUT_DIR, ensure_dir, load_env, log, now_local, write_json, write_text
  14. def parse_models(raw: str) -> List[str]:
  15. return [item.strip() for item in str(raw or "").split(",") if item.strip()]
  16. def choose_selection(lookback_days: int, fallback_lookback_days: int, max_results_per_domain: int) -> Dict[str, Any]:
  17. candidates = fetch_candidates(lookback_days=lookback_days, max_results_per_domain=max_results_per_domain)
  18. selection = select_papers(candidates)
  19. counts = selection.get("counts", {})
  20. if any(counts.get(domain, 0) < 2 for domain in ["embodied", "representation", "reinforcement"]) and fallback_lookback_days > lookback_days:
  21. log(f"Some domains are sparse with lookback={lookback_days}; retrying with lookback={fallback_lookback_days}")
  22. candidates = fetch_candidates(lookback_days=fallback_lookback_days, max_results_per_domain=max_results_per_domain)
  23. selection = select_papers(candidates)
  24. selection["candidate_count"] = len(candidates)
  25. selection["candidates"] = candidates
  26. return selection
  27. def build_output_paths(root: Path, date_slug: str) -> Dict[str, Path]:
  28. bundle_dir = ensure_dir(root / date_slug)
  29. return {
  30. "bundle_dir": bundle_dir,
  31. "candidates_json": bundle_dir / "candidates.json",
  32. "selected_json": bundle_dir / "selected.json",
  33. "enriched_json": bundle_dir / "enriched.json",
  34. "digest_html": bundle_dir / "robotdaily.html",
  35. "digest_md": bundle_dir / "robotdaily.md",
  36. "manifest_json": bundle_dir / "manifest.json",
  37. }
  38. def main() -> None:
  39. parser = argparse.ArgumentParser(description="Run RobotDaily daily digest pipeline")
  40. parser.add_argument("--output-root", default="")
  41. parser.add_argument("--lookback-days", type=int, default=2)
  42. parser.add_argument("--fallback-lookback-days", type=int, default=4)
  43. parser.add_argument("--max-results-per-domain", type=int, default=40)
  44. parser.add_argument("--models", default="")
  45. parser.add_argument("--skip-enrich", action="store_true")
  46. parser.add_argument("--publish-discord", action="store_true")
  47. parser.add_argument("--dry-run", action="store_true")
  48. args = parser.parse_args()
  49. env = load_env()
  50. date_slug = now_local().strftime("%Y-%m-%d")
  51. output_root = Path(args.output_root or env.get("ROBOTDAILY_OUTPUT_DIR", str(DEFAULT_OUTPUT_DIR)))
  52. paths = build_output_paths(output_root, date_slug)
  53. selection = choose_selection(
  54. lookback_days=args.lookback_days,
  55. fallback_lookback_days=args.fallback_lookback_days,
  56. max_results_per_domain=args.max_results_per_domain,
  57. )
  58. write_json(paths["candidates_json"], {"generated_at": now_local().isoformat(), "papers": selection.get("candidates", [])})
  59. write_json(paths["selected_json"], {k: v for k, v in selection.items() if k != "candidates"})
  60. models = parse_models(args.models or env.get("INSIGHT_MODELS", "qwen3.5:27b"))
  61. if args.skip_enrich:
  62. enriched = {k: v for k, v in selection.items() if k != "candidates"}
  63. for paper in enriched.get("papers", []):
  64. paper.setdefault("translated_abstract_zh", paper.get("summary", ""))
  65. paper.setdefault("brief_explanation_zh", paper.get("selection_reason", ""))
  66. paper.setdefault("tags", [])
  67. else:
  68. enriched = enrich_selection({k: v for k, v in selection.items() if k != "candidates"}, model_names=models)
  69. write_json(paths["enriched_json"], enriched)
  70. html = render_html(enriched)
  71. markdown = render_markdown(enriched)
  72. write_text(paths["digest_html"], html)
  73. write_text(paths["digest_md"], markdown)
  74. manifest = {
  75. "generated_at": now_local().isoformat(),
  76. "date": date_slug,
  77. "candidate_count": selection.get("candidate_count", 0),
  78. "selected_count": len(enriched.get("papers", [])),
  79. "counts": enriched.get("counts", {}),
  80. "models": models,
  81. "effective_models_used": enriched.get("effective_models_used", []),
  82. "paths": {name: str(path) for name, path in paths.items() if name != "bundle_dir"},
  83. }
  84. write_json(paths["manifest_json"], manifest)
  85. if args.publish_discord:
  86. publisher = DiscordPublisher(
  87. openclaw_bin=env.get("OPENCLAW_BIN", "openclaw"),
  88. account_id=env.get("DISCORD_ACCOUNT_ID", "codex"),
  89. mode=env.get("DISCORD_DELIVERY_MODE", "thread"),
  90. guild_id=env.get("DISCORD_GUILD_ID", ""),
  91. parent_channel_id=env.get("DISCORD_PARENT_CHANNEL_ID", ""),
  92. target_channel_id=env.get("DISCORD_TARGET_CHANNEL_ID", ""),
  93. target_channel_name=env.get("DISCORD_TARGET_CHANNEL_NAME", ""),
  94. category_id=env.get("DISCORD_CATEGORY_ID", ""),
  95. bot_token=env.get("DISCORD_BOT_TOKEN", ""),
  96. thread_auto_archive_min=int(env.get("DISCORD_THREAD_AUTO_ARCHIVE_MIN", "10080")),
  97. dry_run=args.dry_run,
  98. )
  99. target = publish_digest(
  100. enriched,
  101. # html_path=str(paths["digest_html"]),
  102. markdown_path=str(paths["digest_md"]),
  103. publisher=publisher,
  104. )
  105. manifest["discord_target"] = target
  106. write_json(paths["manifest_json"], manifest)
  107. print(json.dumps(manifest, ensure_ascii=False, indent=2))
  108. if __name__ == "__main__":
  109. main()