from __future__ import annotations

import time
import uuid
from typing import Any, Dict, List, Optional, Tuple


SCHEMA_VERSION = "0.2"
SERVER_NAME = "auto-numerics"


def error_schema() -> Dict[str, Any]:
    return {
        "type": "object",
        "properties": {
            "code": {"type": "string"},
            "message": {"type": "string"},
            "details": {"type": "object"},
            "retryable": {"type": "boolean"},
        },
        "required": ["code", "message"],
    }


def meta_schema() -> Dict[str, Any]:
    return {
        "type": "object",
        "properties": {
            "request_id": {"type": "string"},
            "run_id": {"type": "string"},
            "task_tag": {"type": "string"},
            "server_time_s": {"type": "number"},
            "duration_ms": {"type": "number"},
            "server_version": {"type": "string"},
        },
        "required": ["request_id", "server_time_s", "server_version"],
    }


def response_schema(data_schema: Optional[Dict[str, Any]] = None) -> Dict[str, Any]:
    return {
        "type": "object",
        "properties": {
            "schema_version": {"type": "string", "const": SCHEMA_VERSION},
            "status": {"type": "string", "enum": ["ok", "error", "running", "not_found"]},
            "data": data_schema or {"type": "object"},
            "warnings": {"type": "array", "items": error_schema()},
            "errors": {"type": "array", "items": error_schema()},
            "meta": meta_schema(),
        },
        "required": ["schema_version", "status", "data", "warnings", "errors", "meta"],
    }


def make_response(
    status: str,
    data: Optional[Dict[str, Any]] = None,
    warnings: Optional[List[Dict[str, Any]]] = None,
    errors: Optional[List[Dict[str, Any]]] = None,
    meta: Optional[Dict[str, Any]] = None,
    request_id: Optional[str] = None,
    run_id: Optional[str] = None,
    task_tag: Optional[str] = None,
    duration_ms: Optional[float] = None,
) -> Dict[str, Any]:
    meta_payload: Dict[str, Any] = dict(meta or {})
    if not meta_payload.get("request_id"):
        meta_payload["request_id"] = request_id or uuid.uuid4().hex[:16]
    if run_id and not meta_payload.get("run_id"):
        meta_payload["run_id"] = run_id
    if task_tag and not meta_payload.get("task_tag"):
        meta_payload["task_tag"] = task_tag
    if duration_ms is not None and not meta_payload.get("duration_ms"):
        meta_payload["duration_ms"] = int(duration_ms)
    meta_payload.setdefault("server_time_s", time.time())
    meta_payload.setdefault("server_version", SCHEMA_VERSION)

    return {
        "schema_version": SCHEMA_VERSION,
        "status": status,
        "data": data or {},
        "warnings": warnings or [],
        "errors": errors or [],
        "meta": meta_payload,
    }


def build_artifact_id(task_tag: str, relative_path: Optional[str]) -> str:
    rel = (relative_path or "").strip()
    rel = rel.lstrip("/")
    if not rel or rel == ".":
        rel = "root"
    return f"{task_tag}:{rel}"


def parse_artifact_id(artifact_id: str) -> Tuple[str, str]:
    if ":" not in artifact_id:
        raise ValueError("artifact_id must be in format task_tag:relative_path")
    task_tag, rel = artifact_id.split(":", 1)
    rel = rel.strip().lstrip("/")
    if not rel:
        rel = "root"
    return task_tag, rel
