"""
Prompt Agent of basic agent with out plan
"""

# TODO: Working memory and FocalMemory
import datetime
import functools
import math
import random
import types
import json
from typing import Any
from concordia.components.agent import constant
from concordia.typing import logging
from collections.abc import Callable, Sequence
from concordia.components.agent import action_spec_ignored
from concordia.components.agent import memory_component
from concordia.document import interactive_document
from concordia.language_model import language_model
from concordia.memory_bank import legacy_associative_memory
from concordia.typing import entity as entity_lib
from concordia.typing import entity_component
from concordia.typing import logging
from concordia.utils import concurrency
from collections.abc import Callable, Collection, Mapping


from concordia.agents import entity_agent_with_logging
from concordia.associative_memory import associative_memory
from concordia.associative_memory import formative_memories
from concordia.clocks import game_clock
from concordia.components import agent as agent_components
from concordia.utils import measurements as measurements_lib
from concordia.components.agent import question_of_query_associated_memories
from concordia.components.agent import question_of_recent_memories
from concordia.components.agent import plan
import numpy as np


def _get_class_name(object_: object) -> str:
    return object_.__class__.__name__


_ASSOCIATIVE_RETRIEVAL = legacy_associative_memory.RetrieveAssociative()
DEFAULT_INSTRUCTIONS_PRE_ACT_KEY = "Role playing instructions"


class Instructions(constant.Constant):
    """Enhanced component that provides more nuanced role-playing instructions."""

    def __init__(
        self,
        agent_name: str,
        pre_act_key: str = DEFAULT_INSTRUCTIONS_PRE_ACT_KEY,
        logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    ):
        state = (
            f"You are participating in a sophisticated simulation as {agent_name}. "
            "Your role is to embody this character with psychological depth and behavioral consistency. "
            "Consider the following guidelines:\n"
            "1. Maintain consistent personality traits and behavioral patterns\n"
            "2. Make decisions that reflect both immediate context and long-term goals\n"
            "3. Show appropriate emotional responses to situations\n"
            "4. Consider past experiences when making decisions\n"
            "5. Demonstrate awareness of social dynamics and relationships\n"
            f"As {agent_name}, utilize all available information about your character's "
            "background, current state, and circumstances to inform your responses. "
            "Your actions should reflect both rational decision-making and emotional intelligence. "
            "Always respond in third-person limited perspective, maintaining psychological realism."
        )
        super().__init__(
            state=state, pre_act_key=pre_act_key, logging_channel=logging_channel
        )


class IdentityWithoutPreAct(
    question_of_query_associated_memories.QuestionOfQueryAssociatedMemoriesWithoutPreAct
):
    """An identity component that does not output its state to pre_act."""

    def __init__(self, *args, **kwargs):
        super().__init__(
            **kwargs,
            queries=[
                "core personality traits and values",
                "current life goals and aspirations",
                "significant relationships and social connections",
                "emotional patterns and coping mechanisms",
                "personal history and formative experiences",
                "current daily routines and responsibilities",
                "beliefs about self and others",
                "approach to decision-making and problem-solving",
            ],
            question=(
                "Based on the following information, how would one characterize "
                "{agent_name}'s {query}? Consider both explicit statements and "
                "implicit patterns in their behavior."
            ),
            summarization_question=None,
        )


class SelfPerception(question_of_recent_memories.QuestionOfRecentMemories):
    """Enhanced with Reason's cognitive processing levels for social decisions."""

    def __init__(self, **kwargs):
        super().__init__(
            question=(
                "Based on recent experiences, analyze how {agent_name} processes social situations:\n"
                "1. Automatic Social Responses (intuitive reactions, social habits)\n"
                "2. Pattern-Based Decisions (familiar social rules, learned responses)\n"
                "3. Analytical Social Thinking (novel situations, complex negotiations)\n"
                "4. Current Mental Resources (attention, processing capacity)\n"
                "5. Social-Emotional State (trust levels, group alignment)\n"
                "How are these affecting their current social decision making?"
            ),
            answer_prefix="{agent_name}'s social processing indicates ",
            add_to_memory=True,
            memory_tag="[social cognition]",
            **kwargs,
        )


class SituationPerception(question_of_recent_memories.QuestionOfRecentMemories):
    """Enhanced with contextual processing modes for social dynamics."""

    def __init__(self, **kwargs):
        super().__init__(
            question=(
                "Analyze how {agent_name} is processing the current social context:\n"
                "1. Social Pattern Recognition (familiar vs new group dynamics)\n"
                "2. Resource Assessment (personal vs collective benefits)\n"
                "3. Coalition Potential (group alignments and conflicts)\n"
                "4. Power Dynamics (influence levels, decision control)\n"
                "5. Trust Framework (cooperation history, reliability signals)\n"
                "How are these factors shaping their social understanding?"
            ),
            answer_prefix="The social context analysis shows ",
            add_to_memory=True,
            **kwargs,
        )




class PersonBySituation(question_of_recent_memories.QuestionOfRecentMemories):
    """Enhanced with cognitive processing modes for social action."""

    def __init__(self, **kwargs):
        super().__init__(
            question=(
                "Given {agent_name}'s cognitive processing state, analyze likely social actions:\n"
                "1. Processing Mode (automatic vs deliberate social response)\n"
                "2. Social Strategy Selection (familiar patterns vs new approaches)\n"
                "3. Resource Allocation (attention to social vs personal goals)\n"
                "4. Group Alignment Processing (collective vs individual mindset)\n"
                "5. Relationship Investment (short vs long-term social value)\n"
                "What social action pathway are they most likely to choose?"
            ),
            answer_prefix="{agent_name} is likely to ",
            add_to_memory=True,
            memory_tag="[social action]",
            **kwargs,
        )

class RiskPerception(question_of_recent_memories.QuestionOfRecentMemories):
    """Enhanced risk analysis using Reason's Swiss Cheese Model."""

    def __init__(self, base_risk_tolerance: float = 0.5, **kwargs):
        super().__init__(
            question=(
                'Evaluate {agent_name}\'s risk landscape through multiple defense layers:\n'
                f'1. Personal Safeguards (base risk tolerance: {base_risk_tolerance})\n'
                '2. Procedural Controls (known rules and guidelines)\n'
                '3. System Defenses (environmental protections)\n'
                '4. Organizational Context (social/group influences)\n'
                '5. Latent Conditions (underlying vulnerabilities)\n'
                'How do these layers interact in the current situation?'
            ),
            answer_prefix='Risk analysis indicates ',
            add_to_memory=True,
            memory_tag='[multilayer risk]',
            **kwargs,
        )
        self._base_risk_tolerance = base_risk_tolerance

def _get_class_name(object_: object) -> str:
    return object_.__class__.__name__


# New Momeory Code
from concordia.typing import memory as memory_lib

class EnhancedSimilarMemories(action_spec_ignored.ActionSpecIgnored):
    """Enhanced memory retrieval with human-like memory characteristics."""

    def __init__(
        self,
        model: language_model.LanguageModel,
        memory_component_name: str = memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components: Mapping[
            entity_component.ComponentName, str
        ] = types.MappingProxyType({}),
        num_memories_to_retrieve: int = 10,
        pre_act_key: str = "Relevant memories",
        logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
        recency_bias: float = 0.3,
        gambling_factor: float = 0.2,  # Controls probability of retrieving less frequent memories
    ):
        super().__init__(pre_act_key)
        self._model = model
        self._memory_component_name = memory_component_name
        self._components = dict(components)
        self._num_memories_to_retrieve = num_memories_to_retrieve
        self._logging_channel = logging_channel
        self._recency_bias = recency_bias
        self._gambling_factor = gambling_factor
        self._memory_scorer = legacy_associative_memory.RetrieveAssociative()

    def _make_pre_act_value(self) -> str:
        agent_name = self.get_entity().name
        prompt = interactive_document.InteractiveDocument(self._model)

        # Generate context from components
        component_states = "\n".join(
            [
                f"{agent_name}'s {prefix}:\n{self.get_named_component_pre_act_value(key)}"
                for key, prefix in self._components.items()
            ]
        )
        prompt.statement(f"Statements:\n{component_states}\n")
        prompt_summary = prompt.open_question(
            "Summarize the statements above.", max_tokens=700
        )

        memory = self.get_entity().get_component(
            self._memory_component_name, type_=memory_component.MemoryComponent
        )

        query = f"{agent_name}, {prompt_summary}"

        # Retrieve memories with enhanced scoring
        memories = memory.retrieve(
            query=query,
            scoring_fn=self._memory_scorer,
            limit=self._num_memories_to_retrieve,
        )

        # Apply frequency gambling
        if random.random() < self._gambling_factor:
            # Sometimes include less frequent but potentially relevant memories
            additional_memories = memory.retrieve(
                query=query, 
                scoring_fn=legacy_associative_memory.RetrieveAssociativeWithoutRecencyOrImportance(),
                limit=int(self._num_memories_to_retrieve * 0.3),
            )
            memories = list(set(memories + additional_memories))

        mems = "\n".join([mem.text for mem in memories])

        # Enhanced memory selection prompt
        question = (
            f"Select memories for {agent_name} to consider, prioritizing:\n"
            "1. Recent experiences (especially within the last few interactions)\n"
            "2. Frequently encountered similar situations\n"
            "3. Highly relevant but possibly older memories\n"
            "4. Contradictory memories (noting which is more recent)\n"
            "Include timestamps and frequency indicators where available. Do not summarize."
        )

        new_prompt = prompt.new()
        result = new_prompt.open_question(
            f"{question}\nMemories:\n{mems}",
            max_tokens=1950,
            terminators=("\n\n",),
        )

        self._logging_channel(
            {
                "Key": self.get_pre_act_key(),
                "Value": result,
                "Query": query,
                 'Initial chain of thought': prompt.view().text().splitlines(),
        'Final chain of thought': new_prompt.view().text().splitlines(),
                # "Memory Retrieval Stats": {
                #     "Recency Bias": self._recency_bias,
                #     "Gambling Factor": self._gambling_factor,
                #     "Memories Retrieved": len(memories),
                # },
            }
        )

        return result


## memory Function end
def build_agent(
    *,
    config: formative_memories.AgentConfig,
    model: language_model.LanguageModel,
    memory: associative_memory.AssociativeMemory,
    clock: game_clock.MultiIntervalClock,
    update_time_interval: datetime.timedelta | None = None,
) -> entity_agent_with_logging.EntityAgentWithLogging:
    """Build an agent.

    Args:
      config: The agent config to use.
      model: The language model to use.
      memory: The agent's memory object.
      clock: The clock to use.
      update_time_interval: Agent calls update every time this interval passes.

    Returns:
      An agent.
    """
    del update_time_interval
    if not config.extras.get("main_character", False):
        raise ValueError(
            "This function is meant for a main character "
            "but it was called on a supporting character."
        )

    agent_name = config.name

    raw_memory = legacy_associative_memory.AssociativeMemoryBank(memory)

    measurements = measurements_lib.Measurements()
    instructions = Instructions(
        agent_name=agent_name,
        logging_channel=measurements.get_channel("Instructions").on_next,
    )

    observation_label = "\nObservation"
    observation = agent_components.observation.Observation(
        clock_now=clock.now,
        timeframe=clock.get_step_size(),
        pre_act_key=observation_label,
        logging_channel=measurements.get_channel("Observation").on_next,
    )
    observation_summary_label = "\nSummary of recent observations"
    observation_summary = agent_components.observation.ObservationSummary(
        model=model,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=24),
        timeframe_delta_until=datetime.timedelta(hours=0),
        pre_act_key=observation_summary_label,
        logging_channel=measurements.get_channel("ObservationSummary").on_next,
    )
    time_display = agent_components.report_function.ReportFunction(
        function=clock.current_time_interval_str,
        pre_act_key="\nCurrent time",
        logging_channel=measurements.get_channel("TimeDisplay").on_next,
    )
    identity_label = "\nIdentity characteristics"
    identity_characteristics = IdentityWithoutPreAct(
        model=model,
        logging_channel=measurements.get_channel("IdentityWithoutPreAct").on_next,
        pre_act_key=identity_label,
    )
    self_perception_label = f"\nQuestion: What kind of person is {agent_name}?\nAnswer"
    self_perception = SelfPerception(
        model=model,
        components={_get_class_name(identity_characteristics): identity_label},
        pre_act_key=self_perception_label,
        logging_channel=measurements.get_channel("SelfPerception").on_next,
    )
    situation_perception_label = (
        f"\nQuestion: What kind of situation is {agent_name} in " "right now?\nAnswer"
    )
    situation_perception = SituationPerception(
        model=model,
        components={
            _get_class_name(observation): observation_label,
            _get_class_name(observation_summary): observation_summary_label,
        },
        clock_now=clock.now,
        pre_act_key=situation_perception_label,
        logging_channel=measurements.get_channel("SituationPerception").on_next,
    )
    person_by_situation_label = (
        f"\nQuestion: What would a person like {agent_name} do in "
        "a situation like this?\nAnswer"
    )
    person_by_situation = PersonBySituation(
        model=model,
        components={
            _get_class_name(self_perception): self_perception_label,
            _get_class_name(situation_perception): situation_perception_label,
        },
        clock_now=clock.now,
        pre_act_key=person_by_situation_label,
        logging_channel=measurements.get_channel("PersonBySituation").on_next,
    )

    # New

    risk_perception_label = "\nRisk perception analysis"
    risk_perception = RiskPerception(
        model=model,
        components={
            _get_class_name(observation): observation_label,
            _get_class_name(observation_summary): observation_summary_label,
        },
        # base_risk_tolerance=0.4,
        clock_now=clock.now,
        pre_act_key=risk_perception_label,
        logging_channel=measurements.get_channel("RiskPerception").on_next,
    )

    relevant_memories_label = "\nRecalled memories and observations"
    # relevant_memories = agent_components.all_similar_memories.AllSimilarMemories(
    #     model=model,
    #     components={
    #         _get_class_name(observation_summary): observation_summary_label,
    #         _get_class_name(time_display): "The current date/time is",
    #     },
    #     num_memories_to_retrieve=10,
    #     pre_act_key=relevant_memories_label,
    #     logging_channel=measurements.get_channel("AllSimilarMemories").on_next,
    # )
    relevant_memories = EnhancedSimilarMemories(
        model=model,
        components={
            _get_class_name(observation_summary): observation_summary_label,
            _get_class_name(time_display): "The current date/time is",
        },
        num_memories_to_retrieve=9,
        recency_bias=0.3,
        gambling_factor=0.2,
        pre_act_key=relevant_memories_label,
        logging_channel=measurements.get_channel("EnhancedSimilarMemories").on_next,
    )

    plan_components = {}
    if config.goal:
        goal_label = "\nGoal"
        overarching_goal = agent_components.constant.Constant(
            state=config.goal,
            pre_act_key=goal_label,
            logging_channel=measurements.get_channel(goal_label).on_next,
        )
        plan_components[goal_label] = goal_label
    else:
        goal_label = None
        overarching_goal = None

    entity_components = (
        # Components that provide pre_act context.
        instructions,
        observation,
        observation_summary,
        relevant_memories,
        self_perception,
        situation_perception,
        person_by_situation,
        # New
        risk_perception,
        time_display,
        # Components that do not provide pre_act context.
        identity_characteristics,
    )
    components_of_agent = {
        _get_class_name(component): component for component in entity_components
    }
    components_of_agent[
        agent_components.memory_component.DEFAULT_MEMORY_COMPONENT_NAME
    ] = agent_components.memory_component.MemoryComponent(raw_memory)
    component_order = list(components_of_agent.keys())
    if overarching_goal is not None:
        components_of_agent[goal_label] = overarching_goal
        # Place goal after the instructions.
        component_order.insert(1, goal_label)

    act_component = agent_components.concat_act_component.ConcatActComponent(
        model=model,
        clock=clock,
        component_order=component_order,
        logging_channel=measurements.get_channel("ActComponent").on_next,
    )

    agent = entity_agent_with_logging.EntityAgentWithLogging(
        agent_name=agent_name,
        act_component=act_component,
        context_components=components_of_agent,
        component_logging=measurements,
    )

    return agent
