import os
import requests
import json
import numpy as np
from dotenv import load_dotenv
load_dotenv()
NOCO_BASE = os.getenv("NOCO_URL")
NOCO_TOKEN = os.getenv("NOCO_API_TOKEN")
NOCO_TABLE_ID = os.getenv("NOCO_TABLE_ID", "m571f4eg2jn8nkf")
NOCO_BASE_ID = os.getenv("NOCO_BASE_ID", "pzi4euhz9j7lnqs")
class NocoClient:
    def __init__(self):
        self.base = NOCO_BASE
        self.token = NOCO_TOKEN
        self.table_id = NOCO_TABLE_ID
    def _headers(self):
        return {
            "xc-token": self.token,
            "Content-Type": "application/json"
        }
    def _sanitize_column_name(self, name: str) -> str:
        import re
        sanitized = re.sub(r'[^\w]', '_', name)
        sanitized = re.sub(r'_+', '_', sanitized)
        sanitized = sanitized.strip('_')
        if sanitized and sanitized[0].isdigit():
            sanitized = 'col_' + sanitized
        return sanitized.lower()
    def _convert_numpy_types(self, obj):
        if isinstance(obj, dict):
            return {key: self._convert_numpy_types(value) for key, value in obj.items()}
        elif isinstance(obj, list):
            return [self._convert_numpy_types(item) for item in obj]
        elif isinstance(obj, np.integer):
            return int(obj)
        elif isinstance(obj, np.floating):
            return float(obj)
        elif isinstance(obj, np.ndarray):
            return obj.tolist()
        else:
            return obj
    def upload_result(self, result: dict, table_id: str = None):
        if table_id is None:
            table_id = self.table_id
        headers = self._headers()
        clean_result = {}
        for key, value in result.items():
            sanitized_key = self._sanitize_column_name(key)
            clean_result[sanitized_key] = self._convert_numpy_types(value)
        url = f"{self.base}/api/v2/tables/{table_id}/records"
        resp = requests.post(url, headers=headers, json=[clean_result])
        print(f"Status: {resp.status_code}")
        print(f"Response: {resp.text}")
        if resp.status_code == 200:
            print("Upload successful!")
            return resp.json()
    def get_table_id(self, table_name: str) -> str:
        headers = self._headers()
        url = f"{self.base}/api/v2/meta/bases/{NOCO_BASE_ID}/tables"
        try:
            resp = requests.get(url, headers=headers)
            if resp.status_code == 200:
                tables = resp.json().get("list", [])
                for table in tables:
                    if table.get("table_name") == table_name:
                        return table.get("id")
        except Exception as e:
            print(f"Error getting table ID: {e}")
        return None
    def table_exists(self, table_name: str) -> bool:
        return self.get_table_id(table_name) is not None
    def create_table(self, table_name: str, columns_config: list, skip_if_exists: bool = True) -> str:
        if skip_if_exists:
            existing_id = self.get_table_id(table_name)
            if existing_id:
                print(f"Table '{table_name}' already exists, returning existing ID")
                return existing_id
        headers = self._headers()
        if columns_config and isinstance(columns_config[0], str):
            columns_config = [{"name": self._sanitize_column_name(col), "type": "text"} for col in columns_config]
        else:
            for col_config in columns_config:
                col_config["name"] = self._sanitize_column_name(col_config["name"])
        columns = []
        columns.append({
            "column_name": "id",
            "title": "Id",
            "uidt": "ID",
            "dt": "int",
            "pk": True,
            "ai": True,
            "rqd": True,
            "un": True
        })
        for col_config in columns_config:
            col_name = col_config['name']
            col_type = col_config.get('type', 'text')
            col_required = col_config.get('required', False)
            col_length = col_config.get('length', 255)
            if col_type == 'text':
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "SingleLineText",
                    "dt": "varchar",
                    "dtxp": str(col_length),
                    "rqd": col_required
                }
            elif col_type == 'number':
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "Number",
                    "dt": "decimal",
                    "dtxp": "10",
                    "dtxs": "2",
                    "rqd": col_required
                }
            elif col_type == 'datetime':
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "DateTime",
                    "dt": "timestamp",
                    "rqd": col_required
                }
            elif col_type == 'json':
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "JSON",
                    "dt": "json",
                    "rqd": col_required
                }
            elif col_type == 'boolean':
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "Checkbox",
                    "dt": "boolean",
                    "rqd": col_required
                }
            else:
                column_def = {
                    "column_name": col_name,
                    "title": col_name.replace('_', ' ').title(),
                    "uidt": "SingleLineText",
                    "dt": "varchar",
                    "dtxp": str(col_length),
                    "rqd": col_required
                }
            columns.append(column_def)
        table_data = {
            "table_name": table_name,
            "title": table_name,
            "columns": columns
        }
        url = f"{self.base}/api/v2/meta/bases/{NOCO_BASE_ID}/tables"
        try:
            resp = requests.post(url, headers=headers, json=table_data)
            print(f"Create table status: {resp.status_code}")
            print(f"Response: {resp.text}")
            if resp.status_code in [200, 201]:
                print(f"Table '{table_name}' created successfully!")
                result = resp.json()
                return result.get("id")
            else:
                print(f"Failed to create table: {resp.text}")
                return None
        except Exception as e:
            print(f"Error creating table: {e}")
            return None
if __name__ == "__main__":
    client = NocoClient()
    simple_columns = ["run_id", "model_name", "dataset", "notes"]
    detailed_columns = [
        {"name": "run_id", "type": "text", "required": True, "length": 50},
        {"name": "accuracy", "type": "number", "required": False},
        {"name": "config", "type": "json", "required": False},
        {"name": "model_name", "type": "text", "required": False, "length": 100},
        {"name": "completed", "type": "boolean", "required": False}
    ]
    client.upload_result({
        "run_id": "exp_001", 
        "accuracy": 0.88,
        "config": {"lr": 0.001, "seed": 42},
        "model_name": "bert-base",
        "completed": True
    })
