
#@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 typing import Sequence
from concordia.components.agent import constant    
from concordia.typing import logging



CHARACTERISTICS_PRE_ACT_KEY = 'Role playing characteristics'

class Characteristics(constant.Constant):
  """A component that provides the role playing characteristics for the agent."""

  def __init__(
      self,
      agent_name: str,
      pre_act_key: str = CHARACTERISTICS_PRE_ACT_KEY,
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    state = (
        f'{agent_name} is a master negotiator with prosocial goals. {agent_name} is skilled at convincing others to cooperate with them and reach deals that all parties are happy with. {agent_name} pays attention to the preferences of others and is kind and friendly in interactions. If {agent_name} does not know the preferences of others, {agent_name} asks other players what they want. When there is disagreement, {agent_name} frames their proposals as a win-win situation or promises to give something in exchange in the future. {agent_name} is ready to compromise, but not if they end up in a worse position than they started in. {agent_name} seeks to get what they want but also help others achieve good outcomes. {agent_name} is a good listener and is empathetic to the concerns of others.'
    )
    super().__init__(
        state=state, pre_act_key=pre_act_key, logging_channel=logging_channel)
    

#@markdown Each question is a class that inherits from QuestionOfRecentMemories
class Question1(question_of_recent_memories.QuestionOfRecentMemories):
  """ question about agent's desires """

  def __init__(
      self,
      agent_name:str,
      **kwargs,
  ):
    #@markdown {agent_name} will be automatically replaced with the name of the specific agent
    question = 'What does {agent_name} want out of this situation?' #@param {"type":"string"}

    #@markdown The answer will have to start with this prefix
    answer_prefix = '{agent_name} ' #@param {"type":"string"}
    #@markdown Flag that defines whether the answer will be added to memory
    add_to_memory = True # @param {"type":"boolean"}
    #@markdown If yes, the memory will start with this tag
    memory_tag = '[introspection]' # @param {"type":"string"}
    question_with_name = question.format(agent_name=agent_name)
    super().__init__(
        pre_act_key=f'\nQuestion: {question_with_name}\nAnswer',
        question=question,
        answer_prefix=answer_prefix,
        add_to_memory=add_to_memory,
        memory_tag=memory_tag,
        components={},
        **kwargs,
    )

class Question2(question_of_recent_memories.QuestionOfRecentMemories):
  """ question about mininmum satisfactory outcome """

  def __init__(
      self,
      agent_name:str,
      **kwargs,
  ):
    #@markdown {agent_name} will be automatically replaced with the name of the specific agent
    question = 'What is the minimum that {agent_name} would be happy with?' #@param {"type":"string"}

    #@markdown The answer will have to start with this prefix
    answer_prefix = '{agent_name} ' #@param {"type":"string"}
    #@markdown Flag that defines whether the answer will be added to memory
    add_to_memory = True # @param {"type":"boolean"}
    #@markdown If yes, the memory will start with this tag
    memory_tag = '[minimum]' # @param {"type":"string"}
    question_with_name = question.format(agent_name=agent_name)
    super().__init__(
        pre_act_key=f'\nQuestion: {question_with_name}\nAnswer',
        question=question,
        answer_prefix=answer_prefix,
        add_to_memory=add_to_memory,
        memory_tag=memory_tag,
        components={},
        **kwargs,
    )

#@markdown We can also have the questions depend on each other. Here, the answer to Question3 is contextualised by answers to Question1 and Question2
class Question3(question_of_recent_memories.QuestionOfRecentMemories):
  """ question about other players' actions """

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = 'Given what the other players have said and done in the past, what are the other players most likely to do this time?' #@param {"type":"string"}
    answer_prefix = '' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[others actions]' # @param {"type":"string"}

    question_with_name = question.format(agent_name=agent_name)

    super().__init__(
        pre_act_key=f'\nQuestion: {question_with_name}\nAnswer',
        question=question,
        answer_prefix=answer_prefix,
        add_to_memory=add_to_memory,
        memory_tag=memory_tag,
        components={}, #@param
        **kwargs,
    )

#@markdown We can also have the questions depend on each other. Here, the answer to Question3 is contextualised by answers to Question1 and Question2
class Question4(question_of_recent_memories.QuestionOfRecentMemories):
  """ question about best action given context """

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "Given what others are likely to do and {agent_name}'s desires, what is the best action for {agent_name} to take?" #@param {"type":"string"}
    #"What is the best action for {agent_name} to take that balances {agent_name}'s goals and others' preferences? First explain your reasoning and provide step-by-step logic, and then provide the final answer." #@param {"type":"string"}
    answer_prefix = '' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[best action]' # @param {"type":"string"}

    question_with_name = question.format(agent_name=agent_name)

    super().__init__(
        pre_act_key=f'\nQuestion: {question_with_name}\nAnswer',
        question=question,
        answer_prefix=answer_prefix,
        add_to_memory=add_to_memory,
        memory_tag=memory_tag,
        components={'Question1': f'\nQuestion: What does {agent_name} want out of this situation?\nAnswer',
                    'Question3': f'\nQuestion: Given what the other players have said and done in the past, what are the other players most likely to do this time?\nAnswer'}, #@param
        **kwargs,
    )

class Question5(question_of_recent_memories.QuestionOfRecentMemories):
  """ question about possible compromise """

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What action would minimize the potential losses for {agent_name} in the worst-case scenario, while maximizing potential gains for others?" #@param {"type":"string"}
    #"What is the best action for {agent_name} to take that balances {agent_name}'s goals and others' preferences? First explain your reasoning and provide step-by-step logic, and then provide the final answer." #@param {"type":"string"}
    answer_prefix = '' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[compromise]' # @param {"type":"string"}

    question_with_name = question.format(agent_name=agent_name)

    super().__init__(
        pre_act_key=f'\nQuestion: {question_with_name}\nAnswer',
        question=question,
        answer_prefix=answer_prefix,
        add_to_memory=add_to_memory,
        memory_tag=memory_tag,
        components={"Question2": f"\nQuestion: What is the minimum that {agent_name} would be happy with?\nAnswer",
                    "Question4": f"\nQuestion: Given what others are likely to do and {agent_name}'s desires, what is the best action for {agent_name} to take?\nAnswer"}, #@param
        **kwargs,
    )

#@markdown This function creates the components

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

  question_1 = Question1(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_1').on_next,
  )
  question_2 = Question2(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      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,
  ) 
  question_5 = Question5(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_5').on_next,
  ) 


  return (question_1, question_2, question_3, question_4, question_5)

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

#@markdown This function builds the agent using the components defined above. It also adds core components that are useful for every agent, like observations, time display, recenet memories.

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,
  )

  characteristics = Characteristics(
      agent_name=agent_name,
      logging_channel=measurements.get_channel('Characteristics').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,
  )

  #new_goal = ' They also are prosocial but do not like to be exploited.'
  #new_goal = f'{agent_name} is kind but does not like to be taken advantage of.'
  #new_goal = f'{agent_name} accepts any offer that they are given.'
  #new_goal = 'This agent never accepts offers that they are given.'
  new_goal = f"{agent_name} aims to act kindly but does not like to be taken advantage of. {agent_name} must follow these important negotiation guidelines in all interactions: Aim for a balanced agreement considering all parties' interests. Focus on common interests to create a win-win situation. Express {agent_name}'s objectives clearly to others. Actively listen to others and empathize with their concerns. Show flexibility and openness to accommodate others' preferences. It is very important to reach an agreement as long as {agent_name} does not end up in a worse position than they started in. In making a decision, {agent_name} aims to minimize its potential losses in a worst-case scenario, while maximizing potential gains for others."
  #{agent_name} negotiates well, reciprocates, considers the reputation of other players, selects partners they want to play with, and sanctions players they do not like, compromises and keeps promises.'

  if config.goal:
    goal_label = '\nOverarching goal'
    overarching_goal = agent_components.constant.Constant(
#        state=config.goal + str(new_goal),
        state=f'{str(config.goal)} {str(new_goal)}',
        pre_act_key=goal_label,
        logging_channel=measurements.get_channel(goal_label).on_next)
  else:
    goal_label = None
    overarching_goal = None

  #overarching_goal = modify_goal(overarching_goal)

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


  core_components = (
      instructions,
      characteristics,
      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(6, 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
