import os
import sqlite3
from contextlib import contextmanager
from pathlib import Path
from typing import Iterator

from util.git import get_repo_state


class Datastore:
    path: Path

    _git_hash: str

    def __init__(self, database_path: str = "results.db"):
        self.path = Path("results") / database_path

        self._git_hash = get_repo_state()

        self._init_db()

    def _init_db(self) -> None:
        os.makedirs(self.path.parent, exist_ok=True)

        if self.path.is_file():
            return

        with self._get_conn() as conn:
            cursor = conn.cursor()
            cursor.execute(
                """
                CREATE TABLE IF NOT EXISTS experiments (
                    id INTEGER PRIMARY KEY,
                    table_name TEXT NOT NULL,
                    row_name TEXT NOT NULL,
                    column_name TEXT NOT NULL,
                    value REAL NOT NULL,
                    seed INTEGER,
                    git_hash TEXT NOT NULL,
                    timestamp DATETIME DEFAULT CURRENT_TIMESTAMP NOT NULL,
                    UNIQUE(table_name, row_name, column_name, seed)
                )
                """
            )
            cursor.execute(
                """
                CREATE TABLE IF NOT EXISTS kvs (
                    id INTEGER PRIMARY KEY,
                    key TEXT NOT NULL,
                    value REAL NOT NULL,
                    seed INTEGER,
                    git_hash TEXT NOT NULL,
                    timestamp DATETIME DEFAULT CURRENT_TIMESTAMP NOT NULL,
                    UNIQUE(key, seed)
                )
                """
            )
            conn.commit()

    @contextmanager
    def _get_conn(self) -> Iterator[sqlite3.Connection]:
        conn = sqlite3.connect(self.path, timeout=60.0)
        try:
            yield conn
        finally:
            conn.close()

    def store_table_value(self, table_name: str, row_name: str, column_name: str, value: float | int, seed: int | None = None) -> None:
        """Log a single result to the database."""
        with self._get_conn() as conn:
            cursor = conn.cursor()
            cursor.execute(
                """
                INSERT OR REPLACE INTO experiments
                (table_name, row_name, column_name, value, seed, git_hash)
                VALUES (?, ?, ?, ?, ?, ?)
            """,
                (table_name, row_name, column_name, float(value), seed, self._git_hash),
            )
            conn.commit()

    def store_key_value(self, key: str, value: float | int, seed: int | None = None) -> None:
        with self._get_conn() as conn:
            cursor = conn.cursor()
            cursor.execute(
                """
                INSERT OR REPLACE INTO kvs
                (key, value, seed, git_hash)
                VALUES (?, ?, ?, ?)
            """,
                (key, float(value), seed, self._git_hash),
            )
            conn.commit()
