"""
ID管理系统
提供全局唯一的ID分配器和网络级别的ID管理器
支持动态类别管理和高效批量访问
"""

from typing import List, Dict, Optional


class GlobalIDAllocator:
    """全局ID分配器，使用简单递增计数器
    
    单例模式，保证全局唯一性
    GID和SID共用同一个计数器，自然分离在不同命名空间
    """
    _instance = None
    _counter = 0
    
    def __new__(cls):
        if cls._instance is None:
            cls._instance = super(GlobalIDAllocator, cls).__new__(cls)
        return cls._instance
    
    @classmethod
    def get_next_id(cls) -> int:
        """获取下一个唯一ID"""
        id_val = cls._counter
        cls._counter += 1
        return id_val
    
    @classmethod
    def get_next_batch(cls, count: int) -> List[int]:
        """批量获取ID，返回连续的ID列表"""
        if count <= 0:
            return []
        
        start = cls._counter
        cls._counter += count
        return list(range(start, cls._counter))
    
    @classmethod
    def reset_counter(cls, value: int = 0):
        """重置计数器（主要用于测试）"""
        cls._counter = value


class NetworkIDManager:
    """网络级别的ID管理器
    
    支持动态添加类别，灵活的注册方式，高效的批量访问
    """
    
    def __init__(self):
        # 嵌套字典存储: {category: {index: id}}
        self.gids: Dict[str, Dict[int, int]] = {}
        self.sids: Dict[str, Dict[int, int]] = {}
        
        # 列表视图，用于高效批量访问: {category: [ids...]}
        self.gid_lists: Dict[str, List[int]] = {}
        self.sid_lists: Dict[str, List[int]] = {}
        
        # 记录每个类别的下一个可用索引
        self.gid_next_index: Dict[str, int] = {}
        self.sid_next_index: Dict[str, int] = {}
    
    def _ensure_category_exists(self, category: str, is_gid: bool = True):
        """确保类别存在，如果不存在则创建"""
        if is_gid:
            if category not in self.gids:
                self.gids[category] = {}
                self.gid_lists[category] = []
                self.gid_next_index[category] = 0
        else:
            if category not in self.sids:
                self.sids[category] = {}
                self.sid_lists[category] = []
                self.sid_next_index[category] = 0
    
    def _extend_list_if_needed(self, target_list: List[int], target_index: int):
        """扩展列表到指定索引"""
        if target_index >= len(target_list):
            # 用None填充空位，稍后会被实际ID替换
            target_list.extend([None] * (target_index + 1 - len(target_list)))
    
    # ========== GID管理 ==========
    
    def register_gid(self, category: str, index: Optional[int] = None) -> int:
        """注册单个GID
        
        Args:
            category: GID类别
            index: 索引，如果为None则自动分配下一个索引
            
        Returns:
            分配的GID
        """
        self._ensure_category_exists(category, is_gid=True)
        
        if index is None:
            index = self.gid_next_index[category]
        
        gid = GlobalIDAllocator.get_next_id()
        self.gids[category][index] = gid
        
        # 维护列表视图
        self._extend_list_if_needed(self.gid_lists[category], index)
        self.gid_lists[category][index] = gid
        
        # 更新下一个可用索引
        if index >= self.gid_next_index[category]:
            self.gid_next_index[category] = index + 1
        
        return gid
    
    def register_gids_batch(self, category: str, count: int, start_index: Optional[int] = None) -> List[int]:
        """批量注册GID
        
        Args:
            category: GID类别
            count: 要注册的数量
            start_index: 起始索引，如果为None则从下一个可用索引开始
            
        Returns:
            分配的GID列表
        """
        if count <= 0:
            return []
        
        self._ensure_category_exists(category, is_gid=True)
        
        if start_index is None:
            start_index = self.gid_next_index[category]
        
        gids = GlobalIDAllocator.get_next_batch(count)
        
        # 注册到字典
        for i, gid in enumerate(gids):
            self.gids[category][start_index + i] = gid
        
        # 扩展列表并填入GID
        self._extend_list_if_needed(self.gid_lists[category], start_index + count - 1)
        for i, gid in enumerate(gids):
            self.gid_lists[category][start_index + i] = gid
        
        # 更新下一个可用索引
        if start_index + count > self.gid_next_index[category]:
            self.gid_next_index[category] = start_index + count
        
        return gids
    
    def get_gid(self, category: str, index: int) -> Optional[int]:
        """获取单个GID"""
        return self.gids.get(category, {}).get(index)
    
    def get_gids(self, category: str) -> List[int]:
        """获取某类别所有GID（高性能批量访问）
        
        返回的列表可能包含None（对应未分配的索引位置）
        """
        return [gid for gid in self.gid_lists.get(category, []) if gid is not None]
    
    def get_gids_with_indices(self, category: str) -> List[tuple]:
        """获取某类别所有GID及其索引
        
        Returns:
            [(index, gid), ...] 的列表
        """
        result = []
        gids = self.gids.get(category, {})
        for index, gid in gids.items():
            result.append((index, gid))
        return sorted(result)
    
    # ========== SID管理（相同模式） ==========
    
    def register_sid(self, category: str, index: Optional[int] = None) -> int:
        """注册单个SID"""
        self._ensure_category_exists(category, is_gid=False)
        
        if index is None:
            index = self.sid_next_index[category]
        
        sid = GlobalIDAllocator.get_next_id()
        self.sids[category][index] = sid
        
        # 维护列表视图
        self._extend_list_if_needed(self.sid_lists[category], index)
        self.sid_lists[category][index] = sid
        
        # 更新下一个可用索引
        if index >= self.sid_next_index[category]:
            self.sid_next_index[category] = index + 1
        
        return sid
    
    def register_sids_batch(self, category: str, count: int, start_index: Optional[int] = None) -> List[int]:
        """批量注册SID"""
        if count <= 0:
            return []
        
        self._ensure_category_exists(category, is_gid=False)
        
        if start_index is None:
            start_index = self.sid_next_index[category]
        
        sids = GlobalIDAllocator.get_next_batch(count)
        
        # 注册到字典
        for i, sid in enumerate(sids):
            self.sids[category][start_index + i] = sid
        
        # 扩展列表并填入SID
        self._extend_list_if_needed(self.sid_lists[category], start_index + count - 1)
        for i, sid in enumerate(sids):
            self.sid_lists[category][start_index + i] = sid
        
        # 更新下一个可用索引
        if start_index + count > self.sid_next_index[category]:
            self.sid_next_index[category] = start_index + count
        
        return sids
    
    def get_sid(self, category: str, index: int) -> Optional[int]:
        """获取单个SID"""
        return self.sids.get(category, {}).get(index)
    
    def get_sids(self, category: str) -> List[int]:
        """获取某类别所有SID（高性能批量访问）"""
        return [sid for sid in self.sid_lists.get(category, []) if sid is not None]
    
    def get_sids_with_indices(self, category: str) -> List[tuple]:
        """获取某类别所有SID及其索引"""
        result = []
        sids = self.sids.get(category, {})
        for index, sid in sids.items():
            result.append((index, sid))
        return sorted(result)
    
    # ========== 辅助方法 ==========
    
    def get_categories(self, gid: bool = True) -> List[str]:
        """获取所有类别名称"""
        if gid:
            return list(self.gids.keys())
        else:
            return list(self.sids.keys())
    
    def get_category_size(self, category: str, gid: bool = True) -> int:
        """获取某类别的大小（已注册的ID数量）"""
        if gid:
            return len(self.gids.get(category, {}))
        else:
            return len(self.sids.get(category, {}))
    
    def debug_info(self) -> str:
        """返回调试信息"""
        info = ["=== NetworkIDManager Debug Info ==="]
        info.append(f"GID categories: {list(self.gids.keys())}")
        info.append(f"SID categories: {list(self.sids.keys())}")
        
        for category, gids in self.gids.items():
            info.append(f"GID[{category}]: {len(gids)} items, next_index={self.gid_next_index[category]}")
            
        for category, sids in self.sids.items():
            info.append(f"SID[{category}]: {len(sids)} items, next_index={self.sid_next_index[category]}")
            
        return "\n".join(info)


# 创建全局实例
_global_allocator = GlobalIDAllocator()