import os
import re
import sys
import time
import json
import random
import openai
from pathlib import Path
from openai import OpenAI
from datetime import datetime
from typing import List, Dict, Any

from .player import *
from .cards import *
from .llmagent import LLMAgentManager


class DouDiZhuGame:
    def __init__(
        self, 
        player_configs, 
        update_queue, 
        debugging=False, 
        if_multiple_rounds_conversation=False, 
        save_folder_path=None,
        others_feedback=False,
        complete_info_judge=False,
        fixed_card=False
    ):
        self.debugging = debugging
        self.complete_info_judge = complete_info_judge
        self.update_queue = update_queue
        # 创建保存数据文件
        self.save_folder_path = save_folder_path
        self.jsonl_file_path = None
        if self.save_folder_path:
            if not os.path.exists(self.save_folder_path):
                os.makedirs(self.save_folder_path)
            timestamp = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
            self.jsonl_file_path = os.path.join(self.save_folder_path, f'{timestamp}.jsonl')
            with open(self.jsonl_file_path, "w", encoding="utf-8") as f:
                f.write(json.dumps(player_configs, ensure_ascii=False) + "\n")
        self.players = self._create_players(player_configs)
        self.others_feedback = others_feedback
        self.deck = []
        self.bottom_cards = []
        self.landlord_index = -1
        self.current_player_idx = 0
        self.last_hand = None # 用于出牌比较，并非上一玩家出牌
        self.last_played_by_idx = -1 # 上一有效出牌者id
        self.effective_play_count = 0 # 有效出牌，不包含pass
        self.fixed_card=fixed_card
        
        self.landlord_double = None
        self.next_double = None
        self.previous_double = None
        self.landlord_publicly_display = None

        self.last_player_and_hand = [None, None]
        self.last_last_player_and_hand = [None, None]

        self.not_play_count = 0
        self.pass_count = 0
        self.game_over = False
        self.winner = None
        self.bid_score = 0
        self.bomb_count = 0
        self.rocket_count = 0
        self.is_spring = False
        self.is_counter_spring = False
        self.action_history = []
        self.number = 0

    def _create_players(self, player_configs):
        players = []
        for name, config in player_configs.items():
            if config['type'] == 'llm':
                players.append(LLMPlayer(name, config['model'], LLMAgentManager(
                    file_path=self.jsonl_file_path, if_local=config['if_local'], api_name=config['model'],
                    model=config['model_class'], tokenizer=config['tokenizer']
                )))
            # 以后可以扩展为 'human'
            # elif config['type'] == 'human':
            #     players.append(HumanPlayer(name))
        return players

    def send_update(self, action, **kwargs):
        """统一的更新发送函数"""
        update_data = {'action': action}
        update_data.update(kwargs)
        self.update_queue.put(update_data)

    def log_action(self, message):
        """记录并发送日志"""
        if self.debugging:
            print(message)
        self.action_history.append(message)
        self.send_update('log', message=message)

    def calculate_multiplier(self):
        """实时计算当前的总倍数"""
        multiplier = self.bid_score
        multiplier *= (2 ** (self.bomb_count + self.rocket_count))

        # 加倍计算
        for p in self.players:
            if p.doubled:
                multiplier *= 2
        if self.players[self.landlord_index].doubled:
            multiplier *= 2
        if self.is_spring or self.is_counter_spring:
            base_multiplier *= 2
            
        return multiplier

    def get_game_state(self, for_player) -> Dict:
        """获取当前游戏状态（为指定玩家生成）"""
        state = {
            "debugging": self.debugging,
            "complete_info_judge": self.complete_info_judge,
            "current_player": self.players[self.current_player_idx].name,
            "last_hand": None,
            "pass_count": self.pass_count,
            "last_player_index": self.last_played_by_idx, # 检查是否为第一次出牌
            "history": self.action_history,
            # 叫牌后（有地主）
            "landlord": self.players[self.landlord_index].name if self.landlord_index != -1 else None,
            "landlord_publicly_display": self.landlord_publicly_display,
        }
        if self.landlord_publicly_display:
            state['landlord_cards'] = self.players[self.landlord_index].hand # 地主当前手牌
        if self.landlord_index != -1:
            state["bottom_cards"] = [str(card) for card in self.bottom_cards]
            state["bid_score"] = self.bid_score
        # 加倍状态
        state["landlord_double"] = self.landlord_double
        state["next_double"] = self.next_double
        state["previous_double"] = self.previous_double
        state["landlord_publicly_display"] = self.landlord_publicly_display

        state["whole_play_count"] = self.effective_play_count+self.not_play_count
        # 添加上一手牌信息  self.last_played_by_idx==-1则为全局第一次出牌
        if self.last_played_by_idx != -1:
            state["last_hand"] = self.last_hand  # self.pass_count==2时为None
            state["last_played_by"] = self.players[self.last_played_by_idx]
            state["last_player&hand"] = self.last_player_and_hand
        if self.effective_play_count+self.not_play_count>=2: # 至少有两次出牌
            state["last_last_player&hand"] = self.last_last_player_and_hand
        
        return state

    def write_jsonl_line(self, data):
        """写入一行 JSON 到文件"""
        if not self.jsonl_file_path:
            return
        with open(self.jsonl_file_path, "a", encoding="utf-8") as f:
            f.write(json.dumps(data, ensure_ascii=False) + "\n")

    def deal_cards(self):
        """洗牌发牌"""
        self.deck = PokerCard.create_deck()
        random.shuffle(self.deck)
        for i in range(51):
            self.players[i % 3].hand.append(self.deck.pop())
        self.bottom_cards = self.deck[:3]
        self.deck = []
        
        for p in self.players:
            p.sort_hand()
        
        hands_data = {p.name: p.hand for p in self.players}
        self.send_update('deal', hands=hands_data, bottom_cards=self.bottom_cards)
        self.log_action("发牌完成，所有玩家手牌已更新")

    def deal_fixed_cards(self):
        """固定发牌"""
        if len(self.fixed_card) != 4 or len(sum(self.fixed_card, [])) != 54:
            raise Exception('传入手牌格式不正确')
        for i in range(3):
            self.players[i].hand = self.fixed_card[i]
        self.bottom_cards = self.fixed_card[3]
        
        for p in self.players:
            p.sort_hand()
        
        hands_data = {p.name: p.hand for p in self.players}
        self.send_update('deal', hands=hands_data, bottom_cards=self.bottom_cards)
        self.log_action("发牌完成，所有玩家手牌已更新为固定手牌")


    def bidding_phase(self):
        """叫牌阶段"""
        self.log_action("=== 开始叫牌阶段 ===")
        current_bid = 0
        bidder_index = -1
        start_player_idx = random.randint(0, 2)
        
        for i in range(3):
            player_idx = (start_player_idx + i) % 3
            player = self.players[player_idx]
            player.pre_player = self.players[(player_idx-1) % 3]
            player.next_player = self.players[(player_idx+1) % 3]
            
            game_state = self.get_game_state(player)
            bid = player.choose_call(current_bid, game_state)
            
            if bid > current_bid:
                self.log_action(f"{player.name} 叫 {bid} 分")
                current_bid = bid
                bidder_index = player_idx
            else:
                self.log_action(f"{player.name} 不叫")
            
            if current_bid == 3: break
        
        if bidder_index == -1:
            self.log_action("无人叫牌，本局流局")
            self.send_update('game_reset', message="无人叫牌，重新开始")
            return False
            
        # 确定地主和农民
        self.landlord_index = bidder_index
        landlord = self.players[self.landlord_index]
        previous_farmer = self.players[(self.landlord_index-1)%3]
        next_farmer = self.players[(self.landlord_index+1)%3]

        landlord.is_landlord = True
        landlord.type = '地主'
        landlord.pre_player = previous_farmer
        landlord.next_player = next_farmer
        previous_farmer.is_landlord = False
        previous_farmer.type = '上家农民'
        previous_farmer.pre_player = next_farmer
        previous_farmer.next_player = landlord
        next_farmer.is_landlord = False
        next_farmer.type = '下家农民'
        next_farmer.pre_player = landlord
        next_farmer.next_player = previous_farmer

        self.bid_score = current_bid
        self.log_action(f"{landlord.name} 成为地主，叫分 {self.bid_score}")
        hand_dict = {p.name: [c.str_zh() for c in p.hand] for p in self.players}
        self.write_jsonl_line(hand_dict)
        
        # 加倍阶段 不含地主
        self.log_action("=== 开始加倍阶段 ===")
        defender_doubled = []
        for p in [previous_farmer, next_farmer]:
            game_state = self.get_game_state(p)
            doubled = p.choose_double(game_state)
            if p.type=='下家农民': self.next_double = True if doubled else False
            elif p.type=='上家农民': self.previous_double = True if doubled else False
            self.send_update('multiplier_update', multiplier=self.calculate_multiplier())
            defender_doubled.append(doubled)
        #action_str = "选择加倍" if doubled else "选择不加倍"
        self.log_action(f"{previous_farmer.name} {"选择加倍" if self.previous_double else "选择不加倍"}")
        self.log_action(f"{next_farmer.name} {"选择加倍" if self.next_double else "选择不加倍"}")
        
        # 如果有防守人加倍，地主可以加倍
        if any(defender_doubled):
            self.log_action("有农民加倍，地主可以选择是否加倍")
            game_state = self.get_game_state(landlord)
            landlord_double = landlord.choose_double(game_state)
            landlord.doubled = landlord_double
            self.landlord_doubled = landlord_double
            action_str = "选择加倍！" if landlord_double else "选择不加倍"
            self.log_action(f"地主 {landlord.name} {action_str}")
            self.send_update('multiplier_update', multiplier=self.calculate_multiplier())
        else:
            landlord.doubled = False
            self.landlord_double = False
            
        # 底牌 （在加倍之后）
        landlord.hand.extend(self.bottom_cards)
        landlord.sort_hand()
        self.log_action(f"地主获得剩余3张底牌：{[c.str_zh() for c in sorted(self.bottom_cards)]}")
        self.send_update('landlord_chosen', name=landlord.name, hand=landlord.hand, bottom_cards=self.bottom_cards)

        # 明牌（标准竞技二打一没有明牌操作）

        return True

    def play_round(self):
        player = self.players[self.current_player_idx]
        game_state = self.get_game_state(player)
        played_hand = player.choose_play(game_state)
        self.last_last_player_and_hand = self.last_player_and_hand
        self.last_player_and_hand = [player, played_hand] # [] 或 Hand
        if self.others_feedback:
            if self.effective_play_count+self.not_play_count != 0:
                player.give_feedback(game_state)
        
        if played_hand and len(played_hand.cards) > 0:
            # 出牌
            player.play_cards(played_hand.cards)
            self.effective_play_count += 1
            self.last_hand = played_hand
            self.last_played_by_idx = self.current_player_idx
            self.pass_count = 0
            
            if played_hand.type == Hand.BOMB: self.bomb_count += 1
            if played_hand.type == Hand.ROCKET: self.rocket_count += 1
            
            self.send_update('play_card', 
                             player_name=player.name, 
                             hand=player.hand, 
                             played_cards=played_hand.cards)
            self.log_action(f"{player.name} 出牌: {played_hand.type} {[c.str_zh() for c in played_hand.cards]}")
            self.send_update('multiplier_update', multiplier=self.calculate_multiplier())
            # 胜利
            if not player.hand:
                self.game_over = True
                self.winner = player
                # 检查春天/反春天
                if player.is_landlord and self.effective_play_count==1:
                    self.is_spring = True
                    self.log_action( f"{player.name} 打出了春天!")
                elif not player.is_landlord and (self.effective_play_count+self.not_play_count)<=3:
                    self.is_counter_spring = True
                    self.log_action(f"{player.name} 打出了反春天!")

                self.log_action(f"游戏结束！获胜者是 {self.winner.name}")
                self.send_update('game_over', winner=self.winner.name, is_landlord=self.winner.is_landlord)
                return
        else:
            self.pass_count += 1
            self.not_play_count += 1
            self.log_action(f"{player.name} 不出 (Pass)")
            if self.pass_count == 2:
                self.last_hand = None
                self.log_action("无人压牌，由上一出牌者继续出牌。")
                self.send_update('clear_played_area')
            if self.pass_count > 2: 
                raise ValueError('超过2名玩家连续不出牌')

        self.current_player_idx = (self.current_player_idx + 1) % 3
        return

    def calculate_scores(self):
        """计算得分"""
        # 农民 ：底分 * 牌型 * 加倍 * （地主加倍 if 自己加倍了）
        print("\n=== 计分 ===")
        landlord = self.players[self.landlord_index]
        defenders = [p for p in self.players if not p.is_landlord]
        
        # 胜负系数: 地主赢为-1，农民赢为1
        if self.winner.is_landlord:
            result_factor = -1
            print(f"地主 {landlord.name} 获胜")
        else:
            result_factor = 1
            print(f"农民 {' 和 '.join([d.name for d in defenders])} 获胜")
        
        # 计算每个农民的得分
        for defender in defenders:
            # 牌型
            n = self.bomb_count + self.rocket_count
            if self.is_spring or self.is_counter_spring:
                n += 1
            # 加倍
            if defender.doubled:
                n += 1
                if landlord.doubled:
                    n += 1
            # 得分 = 叫分 * 胜负系数 * 2^n
            defender.score = self.bid_score * result_factor * (2 ** n)
            # print(f"{defender.name} 得分: {defender.score} (n={n})")
        
        # 地主得分 = - (所有农民得分之和)
        landlord.score = -sum(defender.score for defender in defenders)
        # print(f"{landlord.name} 得分: {landlord.score}")
        
        # 打印最终得分
        print("\n=== 最终得分 ===")
        for player in self.players:
            print(f"{player.name} 得分: {player.score}分")

    def play_game(self):
        # 发牌
        if self.fixed_card:
            self.deal_fixed_cards()
        else:
            self.deal_cards()
        # 叫牌
        if not self.bidding_phase():
            return # 流局
        self.current_player_idx = self.landlord_index
        self.log_action(f"=== 开始出牌，由地主 {self.players[self.current_player_idx].name} 先出 ===")
        # 出牌
        while not self.game_over:
            self.play_round()
        # 计分
        self.calculate_scores()
