import { randomUUID, timingSafeEqual } from "node:crypto";
import { createServer, type IncomingMessage, type Server, type ServerResponse } from "node:http";
import type { Gateway, GatewayChannelKey } from "../../../gateway/index.js";
import type { ChannelAdapter, ChannelHandle, ChannelLogger, ChannelStartDeps } from "../protocol/ChannelAdapter.js";
import { ApiServerSessionMapper } from "./ApiServerSessionMapper.js";
import { renderApiServerEvent } from "./api-server-render.js";

const DEFAULT_HOST = "127.0.0.1";
const DEFAULT_PORT = 8642;
const MAX_REQUEST_BYTES = 1_000_000;
const DEFAULT_MODEL_NAME = "pilotdeck-gateway";
const REQUEST_TIMEOUT_MS = 300_000;

const CORS_HEADERS: Record<string, string> = {
  "Access-Control-Allow-Methods": "GET, POST, OPTIONS, DELETE",
  "Access-Control-Allow-Headers": "Content-Type, Authorization, X-Hermes-Session-Id",
  "Access-Control-Expose-Headers": "X-Hermes-Session-Id",
};

export type ApiServerChannelOptions = {
  port?: number;
  apiKey?: string;
  host?: string;
  modelName?: string;
  corsOrigins?: string | string[];
  mapper?: ApiServerSessionMapper;
};

export class ApiServerChannel implements ChannelAdapter {
  readonly channelKey: GatewayChannelKey = "api_server";

  private readonly mapper: ApiServerSessionMapper;
  private readonly host: string;
  private readonly port: number;
  private readonly apiKey: string;
  private readonly modelName: string;
  private readonly corsOrigins: string[];

  private gateway?: Gateway;
  private logger?: ChannelLogger;
  private server: Server | null = null;
  private activeChats = new Set<string>();

  constructor(options: ApiServerChannelOptions = {}) {
    this.mapper = options.mapper ?? new ApiServerSessionMapper();
    this.host = options.host ?? process.env.API_SERVER_HOST ?? DEFAULT_HOST;
    this.port = Number(options.port ?? process.env.API_SERVER_PORT ?? DEFAULT_PORT);
    this.apiKey = options.apiKey ?? process.env.API_SERVER_KEY ?? "";
    this.modelName = options.modelName ?? process.env.API_SERVER_MODEL_NAME ?? DEFAULT_MODEL_NAME;
    this.corsOrigins = parseCorsOrigins(options.corsOrigins ?? process.env.API_SERVER_CORS_ORIGINS ?? "");
  }

  async start(deps: ChannelStartDeps): Promise<ChannelHandle> {
    this.gateway = deps.gateway;
    this.logger = deps.logger;

    try {
      this.server = createServer((req, res) => {
        void this.handleRequest(req, res);
      });
      await new Promise<void>((resolve, reject) => {
        this.server!.once("error", reject);
        this.server!.listen(this.port, this.host, () => {
          this.server!.off("error", reject);
          resolve();
        });
      });
      this.logger?.info?.(`api-server: listening on http://${this.host}:${this.port}`);
    } catch (e) {
      this.logger?.error?.(`api-server: failed to start: ${e}`);
      this.server = null;
      return { stop: async () => undefined };
    }

    return {
      stop: async (reason?: string) => {
        this.logger?.info?.(`api-server: stopping (${reason ?? "no reason"})`);
        if (this.server) {
          await new Promise<void>((resolve) => {
            this.server!.close(() => resolve());
          });
          this.server = null;
        }
      },
    };
  }

  private getCorsHeaders(origin: string | null): Record<string, string> | null {
    if (!origin || this.corsOrigins.length === 0) return null;
    if (this.corsOrigins.includes("*")) {
      return { ...CORS_HEADERS, "Access-Control-Allow-Origin": "*" };
    }
    if (this.corsOrigins.includes(origin)) {
      return { ...CORS_HEADERS, "Access-Control-Allow-Origin": origin, Vary: "Origin" };
    }
    return null;
  }

  private applyCors(res: ServerResponse, cors: Record<string, string> | null): void {
    if (!cors) return;
    for (const [k, v] of Object.entries(cors)) {
      res.setHeader(k, v);
    }
  }

  private checkAuth(req: IncomingMessage): { ok: true } | { ok: false; status: number; body: unknown } {
    if (!this.apiKey) return { ok: true };
    const auth = String(req.headers["authorization"] ?? "");
    if (auth.startsWith("Bearer ")) {
      const token = auth.slice(7).trim();
      try {
        const a = Buffer.from(token);
        const b = Buffer.from(this.apiKey);
        if (a.length === b.length && timingSafeEqual(a, b)) return { ok: true };
      } catch {
        // fallthrough
      }
    }
    return {
      ok: false,
      status: 401,
      body: { error: { message: "Invalid API key", type: "invalid_request_error", code: "invalid_api_key" } },
    };
  }

  private async handleRequest(req: IncomingMessage, res: ServerResponse): Promise<void> {
    const url = new URL(req.url ?? "/", `http://${req.headers.host ?? `${this.host}:${this.port}`}`);
    const origin = (req.headers["origin"] as string | undefined) ?? null;
    const cors = this.getCorsHeaders(origin);

    if (req.method === "OPTIONS") {
      this.applyCors(res, cors);
      res.statusCode = 204;
      res.end();
      return;
    }

    this.applyCors(res, cors);

    if (url.pathname === "/health") {
      sendJson(res, 200, { status: "ok", platform: "api-server" });
      return;
    }

    if (url.pathname === "/v1/models") {
      sendJson(res, 200, {
        object: "list",
        data: [
          {
            id: this.modelName,
            object: "model",
            created: Math.floor(Date.now() / 1000),
            owned_by: "gateway",
          },
        ],
      });
      return;
    }

    if (url.pathname === "/v1/chat/completions" && req.method === "POST") {
      const auth = this.checkAuth(req);
      if (!auth.ok) {
        sendJson(res, auth.status, auth.body);
        return;
      }
      await this.handleChatCompletions(req, res);
      return;
    }

    res.statusCode = 404;
    res.end("Not Found");
  }

  private async handleChatCompletions(req: IncomingMessage, res: ServerResponse): Promise<void> {
    let bodyText: string;
    try {
      bodyText = await readRequestBody(req, MAX_REQUEST_BYTES);
    } catch (e) {
      sendJson(res, 413, { error: { message: `Request too large or unreadable: ${e}`, type: "invalid_request_error" } });
      return;
    }

    let body: Record<string, unknown>;
    try {
      body = JSON.parse(bodyText) as Record<string, unknown>;
    } catch {
      sendJson(res, 400, { error: { message: "Invalid JSON", type: "invalid_request_error" } });
      return;
    }

    const messages = body.messages as Array<{ role: string; content: unknown }> | undefined;
    if (!messages || !Array.isArray(messages) || messages.length === 0) {
      sendJson(res, 400, { error: { message: "messages is required", type: "invalid_request_error" } });
      return;
    }

    const lastMsg = messages[messages.length - 1];
    const userText = normalizeContent(lastMsg?.content);
    if (!userText) {
      sendJson(res, 400, { error: { message: "Empty message", type: "invalid_request_error" } });
      return;
    }

    const sessionIdHeader = String(req.headers["x-hermes-session-id"] ?? "").trim();
    const chatId = sessionIdHeader || `api-${randomUUID()}`;
    const streaming = body.stream === true;

    if (this.activeChats.has(chatId)) {
      this.logger?.info?.(`api-server: chat ${chatId} already active, rejecting`);
      sendJson(res, 429, {
        error: { message: "Session already processing a request", type: "rate_limit_error" },
      });
      return;
    }

    const mapped = this.mapper.resolve({ chatId, text: userText });
    if (mapped.command === "new" && !mapped.message) {
      const ack = "已创建新会话。";
      if (streaming) {
        writeChatCompletionStream(res, this.modelName, ack);
      } else {
        sendJson(res, 200, buildChatCompletion(this.modelName, ack));
      }
      return;
    }
    if (!mapped.message) {
      sendJson(res, 400, { error: { message: "Empty message", type: "invalid_request_error" } });
      return;
    }

    this.activeChats.add(chatId);
    try {
      if (streaming) {
        await this.streamTurn(res, chatId, mapped.sessionKey, mapped.message);
      } else {
        await this.bufferedTurn(res, chatId, mapped.sessionKey, mapped.message);
      }
    } finally {
      this.activeChats.delete(chatId);
    }
  }

  private async streamTurn(res: ServerResponse, chatId: string, sessionKey: string, message: string): Promise<void> {
    if (!this.gateway) {
      sendJson(res, 503, { error: { message: "Gateway not ready", type: "server_error" } });
      return;
    }

    res.setHeader("Content-Type", "text/event-stream");
    res.setHeader("Cache-Control", "no-cache");
    res.setHeader("Connection", "keep-alive");
    res.setHeader("X-Hermes-Session-Id", chatId);
    res.statusCode = 200;
    res.flushHeaders?.();

    const timeout = setTimeout(() => {
      this.logger?.warn?.(`api-server: stream timeout for ${chatId}`);
      try { res.end(); } catch { /* best effort */ }
    }, REQUEST_TIMEOUT_MS);

    try {
      for await (const event of this.gateway.submitTurn({
        sessionKey,
        channelKey: "api_server",
        message,
      })) {
        const fragment = renderApiServerEvent(event);
        if (fragment != null && fragment.length > 0) {
          writeStreamChunk(res, this.modelName, fragment);
        }
      }
      writeStreamDone(res, this.modelName);
    } catch (e) {
      this.logger?.error?.(`api-server: stream submitTurn error: ${e}`);
      try {
        res.write(`data: ${JSON.stringify({ error: { message: String(e), type: "server_error" } })}\n\n`);
      } catch { /* best effort */ }
    } finally {
      clearTimeout(timeout);
      try { res.end(); } catch { /* best effort */ }
    }
  }

  private async bufferedTurn(res: ServerResponse, chatId: string, sessionKey: string, message: string): Promise<void> {
    if (!this.gateway) {
      sendJson(res, 503, { error: { message: "Gateway not ready", type: "server_error" } });
      return;
    }

    let replyText = "";
    const timeout = setTimeout(() => {
      this.logger?.warn?.(`api-server: buffered turn timeout for ${chatId}`);
    }, REQUEST_TIMEOUT_MS);

    try {
      for await (const event of this.gateway.submitTurn({
        sessionKey,
        channelKey: "api_server",
        message,
      })) {
        const fragment = renderApiServerEvent(event);
        if (fragment != null) replyText += fragment;
      }
    } catch (e) {
      clearTimeout(timeout);
      this.logger?.error?.(`api-server: submitTurn error: ${e}`);
      sendJson(res, 500, { error: { message: String(e), type: "server_error" } });
      return;
    }
    clearTimeout(timeout);

    res.setHeader("X-Hermes-Session-Id", chatId);
    sendJson(res, 200, buildChatCompletion(this.modelName, replyText.trim()));
  }
}

function readRequestBody(req: IncomingMessage, max: number): Promise<string> {
  return new Promise((resolve, reject) => {
    const chunks: Buffer[] = [];
    let size = 0;
    req.on("data", (chunk: Buffer) => {
      size += chunk.length;
      if (size > max) {
        reject(new Error("payload too large"));
        req.destroy();
        return;
      }
      chunks.push(chunk);
    });
    req.on("end", () => resolve(Buffer.concat(chunks).toString("utf8")));
    req.on("error", reject);
  });
}

function sendJson(res: ServerResponse, status: number, body: unknown): void {
  res.statusCode = status;
  res.setHeader("Content-Type", "application/json");
  res.end(JSON.stringify(body));
}

function parseCorsOrigins(value: unknown): string[] {
  if (!value) return [];
  if (typeof value === "string") return value.split(",").map((s) => s.trim()).filter(Boolean);
  if (Array.isArray(value)) return value.map(String).filter(Boolean);
  return [];
}

function normalizeContent(content: unknown): string {
  if (!content) return "";
  if (typeof content === "string") return content;
  if (Array.isArray(content)) {
    return content
      .map((item: any) => {
        if (typeof item === "string") return item;
        if (item?.type === "text" || item?.type === "input_text") return item.text ?? "";
        return "";
      })
      .filter(Boolean)
      .join("\n");
  }
  return String(content);
}

function buildChatCompletion(model: string, content: string): Record<string, unknown> {
  return {
    id: `chatcmpl-${Date.now()}`,
    object: "chat.completion",
    created: Math.floor(Date.now() / 1000),
    model,
    choices: [
      {
        index: 0,
        message: { role: "assistant", content },
        finish_reason: "stop",
      },
    ],
    usage: { prompt_tokens: 0, completion_tokens: 0, total_tokens: 0 },
  };
}

function writeStreamChunk(res: ServerResponse, model: string, content: string): void {
  const chunk = {
    id: `chatcmpl-${Date.now()}`,
    object: "chat.completion.chunk",
    created: Math.floor(Date.now() / 1000),
    model,
    choices: [{ index: 0, delta: { content }, finish_reason: null }],
  };
  res.write(`data: ${JSON.stringify(chunk)}\n\n`);
}

function writeStreamDone(res: ServerResponse, model: string): void {
  const doneChunk = {
    id: `chatcmpl-${Date.now()}`,
    object: "chat.completion.chunk",
    created: Math.floor(Date.now() / 1000),
    model,
    choices: [{ index: 0, delta: {}, finish_reason: "stop" }],
  };
  res.write(`data: ${JSON.stringify(doneChunk)}\n\ndata: [DONE]\n\n`);
}

function writeChatCompletionStream(res: ServerResponse, model: string, content: string): void {
  res.setHeader("Content-Type", "text/event-stream");
  res.setHeader("Cache-Control", "no-cache");
  res.setHeader("Connection", "keep-alive");
  res.statusCode = 200;
  writeStreamChunk(res, model, content);
  writeStreamDone(res, model);
  res.end();
}