import type { LogMessage } from '../types'

type LogCallback = (log: LogMessage) => void

type AnyCallback = (data: any) => void

class WebSocketService {
  private ws: WebSocket | null = null
  private callbacks: Set<AnyCallback> = new Set()
  private reconnectTimer: number | null = null
  private currentRunId: string | null = null

  connect(runId: string, onLog: AnyCallback) {
    // 如果是同一个 runId，直接添加回调
    if (this.currentRunId === runId && this.ws?.readyState === WebSocket.OPEN) {
      this.callbacks.add(onLog)
      return
    }

    // 保留已有回调（以便重连后恢复），然后断开旧连接
    const existingCallbacks = Array.from(this.callbacks)
    this.disconnect()

    this.currentRunId = runId
    this.callbacks.add(onLog)

    const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:'
    const wsUrl = `${protocol}//${window.location.host}/ws/logs?run_id=${runId}`

    this.ws = new WebSocket(wsUrl)

    this.ws.onopen = () => {
      // 恢复之前的回调
      existingCallbacks.forEach(cb => this.callbacks.add(cb))
      console.log(`WebSocket connected for run ${runId}`)
    }

    this.ws.onmessage = (event) => {
      try {
        const data = JSON.parse(event.data)

        // If backend sends a file notification, pass it to callbacks
        if (data.type && data.type === 'file') {
          // ensure timestamp
          data.timestamp = data.timestamp || new Date().toLocaleString('zh-CN', {
            hour12: false,
            hour: '2-digit',
            minute: '2-digit',
            second: '2-digit'
          })
          this.callbacks.forEach(cb => {
            try {
              cb(data)
            } catch (err) {
              console.error('WebSocket callback error (file):', err)
            }
          })
          return
        }

        // 后端直接发送 {level, color, message} 格式
        if (data.level && data.message) {
          const logMessage: LogMessage = {
            level: data.level,
            content: data.message,
            color: data.color,  // 保留后端传来的颜色
            timestamp: new Date().toLocaleString('zh-CN', { 
              hour12: false,
              hour: '2-digit',
              minute: '2-digit',
              second: '2-digit'
            }),
          }
          
          // 通知所有回调
          this.callbacks.forEach(cb => {
            try {
              cb(logMessage)
            } catch (err) {
              console.error('WebSocket callback error (log):', err)
            }
          })
        }
      } catch (error) {
        console.error('Failed to parse WebSocket message:', error)
      }
    }

    this.ws.onerror = (error) => {
      console.error('WebSocket error:', error)
    }

    this.ws.onclose = () => {
      console.log('WebSocket closed')
      // 3秒后尝试重连
      if (this.currentRunId) {
        this.reconnectTimer = setTimeout(() => {
          if (this.currentRunId && this.callbacks.size > 0) {
            console.log('Reconnecting WebSocket...')
            this.connect(this.currentRunId, Array.from(this.callbacks)[0])
          }
        }, 3000)
      }
    }
  }

  disconnect() {
    if (this.reconnectTimer) {
      clearTimeout(this.reconnectTimer)
      this.reconnectTimer = null
    }

    if (this.ws) {
      this.ws.close()
      this.ws = null
    }

    this.callbacks.clear()
    this.currentRunId = null
  }

  removeCallback(callback: LogCallback) {
    this.callbacks.delete(callback)
    
    // 如果没有回调了,断开连接
    if (this.callbacks.size === 0) {
      this.disconnect()
    }
  }
}

export const wsService = new WebSocketService()
