from collections import defaultdict
from typing import Awaitable, Callable, DefaultDict, List, Sequence, Set
from ..types.agent.agent import Agent
from ..types.agent import AgentId, AgentType
from ..subscription.subscription import Subscription
from ..types.msg_channel import MessageChannel
async def get_impl(
    *,
    id_or_type: AgentId | AgentType | str,
    key: str,
    lazy: bool,
    instance_getter: Callable[[AgentId], Awaitable[Agent]],
) -> AgentId:
    if isinstance(id_or_type, AgentId):
        if not lazy:
            await instance_getter(id_or_type)
        return id_or_type
    type_str = id_or_type if isinstance(id_or_type, str) else id_or_type.type
    id = AgentId(type_str, key)
    if not lazy:
        await instance_getter(id)
    return id
[docs]
class SubscriptionManager:
    def __init__(self) -> None:
        self._subscriptions: List[Subscription] = []
        self._seen_topics: Set[MessageChannel] = set()
        self._subscribed_recipients: DefaultDict[MessageChannel, List[AgentId]] = defaultdict(list)
    @property
    def subscriptions(self) -> Sequence[Subscription]:
        return self._subscriptions
[docs]
    async def subscribe(self, subscription: Subscription) -> None:
        # Check if the subscription already exists
        if any(sub == subscription for sub in self._subscriptions):
            raise ValueError("Subscription already exists")
        self._subscriptions.append(subscription)
        self._rebuild_subscriptions(self._seen_topics) 
[docs]
    async def unsubscribe(self, id: str) -> None:
        # Check if the subscription exists
        if not any(sub.id == id for sub in self._subscriptions):
            raise ValueError("Subscription does not exist")
        def is_not_sub(x: Subscription) -> bool:
            return x.id != id
        self._subscriptions = list(filter(is_not_sub, self._subscriptions))
        # Rebuild the subscriptions
        self._rebuild_subscriptions(self._seen_topics) 
[docs]
    async def get_subscribed_recipients(self, topic: MessageChannel) -> List[AgentId]:
        if topic not in self._seen_topics:
            self._build_for_new_topic(topic)
        return self._subscribed_recipients[topic] 
    # TODO: optimize this...
    def _rebuild_subscriptions(self, topics: Set[MessageChannel]) -> None:
        self._subscribed_recipients.clear()
        for topic in topics:
            self._build_for_new_topic(topic)
    def _build_for_new_topic(self, topic: MessageChannel) -> None:
        self._seen_topics.add(topic)
        for subscription in self._subscriptions:
            if subscription.is_match(topic):
                self._subscribed_recipients[topic].append(subscription.map_to_agent(topic))