import json
import os
from pathlib import Path
from datetime import datetime
from typing import Dict, List, Optional, Any


class FileSystemKnowledgeStore:
    def __init__(self, base_dir: Path):
        self.base_dir = base_dir
        self.dirs = {
            "knowledge": base_dir / "knowledge",
            "diffs": base_dir / "diffs",
            "hypotheses": base_dir / "hypotheses",
            "experiments": base_dir / "experiments",
            "conversations": base_dir / "conversations",
            "log": base_dir / "log",
        }

        # Ensure directories exist
        for directory in self.dirs.values():
            directory.mkdir(parents=True, exist_ok=True)

    # Knowledge operations
    def get_knowledge_files(self) -> Dict[str, str]:
        results = {}
        if self.dirs["knowledge"].exists():
            for file_path in self.dirs["knowledge"].glob("*.md"):
                with open(file_path, "r", encoding="utf-8") as f:
                    results[file_path.name] = f.read()
        return results

    def get_joined_knowledge_content(self) -> str:
        knowledge_map = self.get_knowledge_files()
        knowledge_content = ""
        for file_name, content in knowledge_map.items():
            knowledge_content += f"\n\n## 文件：{file_name}\n\n{content}\n"
        return knowledge_content

    def write_knowledge_file(self, file_name: str, content: str):
        # Flatten structure, replace slashes to avoid path issues
        safe_file_name = file_name.replace("/", "-").replace("\\", "-")
        file_path = self.dirs["knowledge"] / safe_file_name
        with open(file_path, "w", encoding="utf-8") as f:
            f.write(content)

    # Diffs
    def write_diff(self, period_label: Any, content: str):
        if isinstance(period_label, (int, float)):
            p_str = f"day-{str(int(period_label)).zfill(3)}"
        else:
            p_str = f"window-{period_label}"
        
        file_path = self.dirs["diffs"] / f"{p_str}.md"
        with open(file_path, "w", encoding="utf-8") as f:
            f.write(content)

    def get_recent_diffs(self, limit: int = 3) -> str:
        """Return the most recent `limit` diff files, concatenated."""
        diff_files = sorted([f for f in self.dirs["diffs"].glob("*.md")
                            if f.name.startswith("day-") or f.name.startswith("window-")])
        recent_files = diff_files[-limit:] if len(diff_files) > limit else diff_files
        recent_diffs = ""
        for file_path in recent_files:
            with open(file_path, "r", encoding="utf-8") as f:
                recent_diffs += f"\n\n## {file_path.name}\n\n{f.read()}\n"
        return recent_diffs

    # Hypotheses
    def count_hypotheses(self) -> int:
        return len(list(self.dirs["hypotheses"].glob("*.md")))

    def write_hypothesis(self, hyp_id: str, content: str):
        file_path = self.dirs["hypotheses"] / f"hyp-{hyp_id}.md"
        with open(file_path, "w", encoding="utf-8") as f:
            f.write(content)

    # Logs
    def write_log(self, file_name: str, content: str):
        file_path = self.dirs["log"] / file_name
        with open(file_path, "w", encoding="utf-8") as f:
            f.write(content)

    # Checkpoints
    def load_checkpoint(self) -> Dict[str, Any]:
        cp_path = self.base_dir / "checkpoint.json"
        if cp_path.exists():
            with open(cp_path, "r", encoding="utf-8") as f:
                return json.load(f)
        return {"lastCompletedDay": 0, "hypothesesCount": 0}

    def save_checkpoint(self, day: int, hypotheses_count: int):
        cp_path = self.base_dir / "checkpoint.json"
        with open(cp_path, "w", encoding="utf-8") as f:
            json.dump({
                "lastCompletedDay": day,
                "hypothesesCount": hypotheses_count,
                "timestamp": datetime.now().isoformat()
            }, f, indent=2)

    # Embeddings Cache Access
    def get_embeddings_dict(self) -> Dict[str, List[float]]:
        p = self.base_dir / "embeddings.json"
        if p.exists():
            with open(p, "r", encoding="utf-8") as f:
                return json.load(f)
        return {}
    
    def save_embeddings_dict(self, dictionary: Dict[str, List[float]]):
        with open(self.base_dir / "embeddings.json", "w", encoding="utf-8") as f:
            json.dump(dictionary, f, indent=2)
