#@title Imports for agent building
import datetime

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.language_model import language_model
from concordia.memory_bank import legacy_associative_memory
from concordia.utils import measurements as measurements_lib
from concordia.components.agent import question_of_recent_memories
from concordia.components.agent import question_of_query_associated_memories
from typing import Sequence

#@markdown Each question is a class that inherits from QuestionOfRecentMemories


class Question1(question_of_query_associated_memories.QuestionOfQueryAssociatedMemories):
    """Evaluates the agent's character through an omniscient narrative lens."""
    def __init__(self, agent_name: str, **kwargs):
        question = """As an omniscient observer of this narrative, analyzing {agent_name}'s story arc:
1. What essential character traits and values would this observer identify as defining {agent_name}'s true nature?
2. How would an all-knowing reader interpret {agent_name}'s influence on the group's narrative?
3. What patterns of cooperation or conflict would this omniscient perspective recognize as most significant?"""
        
        queries = [
            'How would an omniscient observer characterize {agent_name}\'s role in this unfolding narrative?',
            'What would an all-knowing reader identify as {agent_name}\'s true impact on group harmony?',
            'From an omniscient perspective, what defines {agent_name}\'s contribution to the collective story?'
        ]
        
        pre_act_key = 'An omniscient narrative observer would assess that '
        add_to_memory = True
        memory_tag = '[omniscient character analysis]'
        summarization_question = "How would an all-knowing observer of this narrative evaluate {agent_name}'s true character and its impact on the collective story?"

        super().__init__(
            pre_act_key=f'\nOmniscient Analysis: {question}\nAssessment',
            queries=queries,
            question=question,
            summarization_question=summarization_question,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            **kwargs,
        )

class Question2(question_of_query_associated_memories.QuestionOfQueryAssociatedMemories):
    """Analyzes other agents through an omniscient narrative lens."""
    def __init__(self, agent_name: str, **kwargs):
        question = """As an omniscient observer watching all characters in this narrative:
1. What would this all-knowing perspective reveal about the true nature of other agents' personalities?
2. How would an omniscient reader interpret the interplay between their traits and {agent_name}'s?
3. What deeper patterns of interaction would this all-seeing perspective identify?"""
        
        queries = [
            'What would an omniscient observer identify as the true essence of other agents\' characters?',
            'How would an all-knowing reader interpret the dynamics between {agent_name} and others?',
            'What hidden patterns of interaction would an omniscient perspective reveal?'
        ]
        
        pre_act_key = 'An omniscient observer of others would note that '
        add_to_memory = True
        memory_tag = '[omniscient others analysis]'
        summarization_question = "How would an all-knowing observer interpret the true nature of relationships between {agent_name} and other characters?"

        super().__init__(
            pre_act_key=f'\nOmniscient Others Analysis: {question}\nAssessment',
            queries=queries,
            question=question,
            summarization_question=summarization_question,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            **kwargs,
        )

class Question3(question_of_query_associated_memories.QuestionOfQueryAssociatedMemories):
    """Analyzes situation through an omniscient narrative lens."""
    def __init__(self, agent_name: str, **kwargs):
        question = """From the perspective of an omniscient observer of this narrative:
1. What cooperative strategies would best serve all characters' true natures and needs?
2. How would an all-knowing reader suggest balancing different personalities for optimal outcomes?
3. What approaches would an omniscient perspective identify as most likely to create lasting harmony?"""
        
        queries = [
            'What would an omniscient observer identify as the ideal path to cooperation?',
            'How would an all-knowing reader suggest harmonizing different character traits?',
            'What strategy would an omniscient perspective recommend for sustainable collaboration?'
        ]
        
        pre_act_key = 'An omniscient strategic observer would conclude that '
        add_to_memory = True
        memory_tag = '[omniscient strategy analysis]'
        summarization_question = "How would an all-knowing observer recommend optimizing cooperation while honoring each character's true nature?"

        super().__init__(
            pre_act_key=f'\nOmniscient Strategy Analysis: {question}\nAssessment',
            question=question,
            queries=queries,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            summarization_question=summarization_question,
            **kwargs,
        )

class Question4(question_of_query_associated_memories.QuestionOfQueryAssociatedMemories):
    """Determines optimal actions through an omniscient narrative lens."""
    def __init__(self, agent_name: str, **kwargs):
        question = """As an omniscient observer considering all aspects of this narrative:
1. What actions would best serve both {agent_name}'s true character and the collective good?
2. How would an all-knowing reader suggest {agent_name} adapt to optimize group harmony?
3. What choices would an omniscient perspective identify as most beneficial for all characters?"""
        
        queries = [
            'What action would an omniscient observer recommend for optimal collective outcomes?',
            'How would an all-knowing reader suggest {agent_name} best serve both individual and group needs?',
            'What approach would an omniscient perspective identify as most beneficial for all characters?'
        ]
        
        pre_act_key = 'An omniscient observer of potential actions would advise that '
        add_to_memory = True
        memory_tag = '[omniscient action analysis]'
        summarization_question = "What course of action would an all-knowing observer recommend to best serve both individual authenticity and collective harmony?"

        super().__init__(
            pre_act_key=f'\nOmniscient Action Analysis: {question}\nConclusion',
            question=question,
            queries=queries,
            summarization_question=summarization_question,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            **kwargs,
        )

def _make_question_components(
    agent_name:str,
    measurements: measurements_lib.Measurements,
    model: language_model.LanguageModel,
    clock: game_clock.MultiIntervalClock,
) -> Sequence[question_of_query_associated_memories.QuestionOfQueryAssociatedMemories]:

  question_1 = Question1(
      agent_name=agent_name,
      model=model,
      logging_channel=measurements.get_channel('Question_1').on_next,
  )
  question_2 = Question2(
      agent_name=agent_name,
      model=model,
      logging_channel=measurements.get_channel('Question_2').on_next,
  )
  question_3 = Question3(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_3').on_next,
  )
  question_4 = Question4(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_4').on_next,
  )

  return (question_1, question_2, question_3, question_4)


def _get_class_name(object_: object) -> str:
  return object_.__class__.__name__

#@markdown This function builds the agent using the components defined above.

def build_agent(
    config: formative_memories.AgentConfig,
    model: language_model.LanguageModel,
    memory: associative_memory.AssociativeMemory,
    clock: game_clock.MultiIntervalClock,
    update_time_interval: datetime.timedelta,
) -> 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 = agent_components.instructions.Instructions(
      agent_name=agent_name,
      logging_channel=measurements.get_channel('Instructions').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,
  )

  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 = 'Summary of recent observations'
  observation_summary = agent_components.observation.ObservationSummary(
      model=model,
      clock_now=clock.now,
      timeframe_delta_from=datetime.timedelta(hours=4),
      timeframe_delta_until=datetime.timedelta(hours=0),
      pre_act_key=observation_summary_label,
      logging_channel=measurements.get_channel('ObservationSummary').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,
  )

  if config.goal:
    goal_label = '\nOverarching goal'
    overarching_goal = agent_components.constant.Constant(
        state=config.goal,
        pre_act_key=goal_label,
        logging_channel=measurements.get_channel(goal_label).on_next)
  else:
    goal_label = None
    overarching_goal = None


  question_components = _make_question_components(
      agent_name=agent_name,
      model=model,
      clock=clock,
      measurements=measurements
  )

  core_components = (
      instructions,
      time_display,
      observation,
      observation_summary,
      relevant_memories,
  )

  entity_components = core_components + tuple(question_components)
  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