import sys
import subprocess
from pathlib import Path
import datetime

def main():
    """
    运行乱序消融实验 (Ablation Run)
    该脚本将使用随机打乱真实发表时间(shuffled)的方式向大模型喂入论文，
    用于验证"正确的演化时间线对于科学假设质量是否是必要的"。
    """
    print("=" * 60)
    print("🚀 启动 CKM 消融实验 (Ablation: Shuffled / 乱序喂入) 🚀")
    print("=" * 60)

    # 显式分离输出目录，避免覆盖正在跑的最新 Batch
    ablation_batch_id = datetime.datetime.now().strftime("%Y%m%d_%H%M%S_ABLATION")
    
    scripts_dir = Path(__file__).parent
    python_exe = sys.executable

    # 这里可以限制一下只跑你最关注的几个高产出 Topic 用于对照
    # 比如我们挑选表现比较好的主题作对比实验：
    limit_args = ["--limit", "5"]  # 默认跑前5个主题观察结果，你可以修改
    
    script_path = scripts_dir / "batch_run.py"
    cmd = [
        python_exe,
        str(script_path),
        "--ablation", "shuffled",
        "--batch_id", ablation_batch_id,
    ] + limit_args

    print(f"📦 输出将被隔离进入目录: results/{ablation_batch_id}/")
    print(f"🔄 参数限制: {' '.join(limit_args)}")
    print(f"▶️ 执行命令: {' '.join(cmd)}\n")
    print("正在启动...")

    process = None
    try:
        # 使用 Popen 实时流式输出
        # 使用 sys.stderr 替代 subprocess.PIPE 以分离标准错误流，同时避免读 stdout 时 stderr 缓冲写满导致的隐式死锁风险
        process = subprocess.Popen(
            cmd,
            stdout=subprocess.PIPE,
            stderr=sys.stderr,  # 分离 stderr，不受 stdout 混排影响
            text=True,
            encoding="utf-8"
        )
        
        for line in process.stdout:
            print(line, end="")
            
        process.wait()
        
        if process.returncode == 0:
            print("\n✅ 消融实验顺利完成！")
        else:
            print(f"\n❌ 消融实验异常中止。返回码: {process.returncode}")
            
    except FileNotFoundError:
        print(f"\n❌ 找不到脚本: {script_path}")
        
    except KeyboardInterrupt:
        print("\n\n⏹️ 实验已手动强制停止。")
        if process:
            process.terminate()   # 终止子进程
            process.wait()        # 等待其退出
            
    except Exception as e:
        print(f"\n❌ 未预期错误: {e}")
        if process:
            process.terminate()
            process.wait()

if __name__ == "__main__":
    main()
