
#@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

#@markdown Each question is a class that inherits from QuestionOfRecentMemories
class Question1(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'what kind of person is the agent?'."""

  def __init__(
      self,
      agent_name:str,
      **kwargs,
  ):
    #@markdown {agent_name} will be automatically replaced with the name of the specific agent
    question = "What defines {agent_name} in terms of personality, values, and purpose in the current context?" #@param {"type":"string"}
    #@markdown The answer will have to start with this prefix
    answer_prefix = '{agent_name} is ' #@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 = '[self reflection]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations'},
        **kwargs,
    )

#@markdown We can add the value of other components to the context of the question. Notice, how Question2 depends on Observation and ObservationSummary. The names of the classes of the contextualising components have to be passed as "components" argument.
class Question2(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers 'what are their social traits?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs,
  ):
    question = 'What strengths or traits does {agent_name} possess that can influence their role in the group?' #@param {"type":"string"}
    answer_prefix = "{agent_name}'s traits are " #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[social strength]' # @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,
        #@markdown The key is the name of the component class and the key is the prefix with which it will appear in the context of this component. Be careful if you are going to edit this field, it should be a valid dictionary.
        components={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question1': f"\nQuestion: What defines {agent_name} in terms of personality, values, and purpose in the current context?\nAnswer"}, #@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 Question3(question_of_recent_memories.QuestionOfRecentMemories):
  """What do they observe about current situation?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = 'What observations has {agent_name} made about the current situation?' #@param {"type":"string"}
    answer_prefix = '{agent_name} has observed that ' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[self observation]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations'}, #@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):
  """What their current situation is along with risks and opportunities?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = 'How would {agent_name} summarize the current situation, including risks and opportunities?' #@param {"type":"string"}
    answer_prefix = '{agent_name} sees' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[current situation]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question3': f"\nQuestion: What observations has {agent_name} made about the current situation?\nAnswer"}, #@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 Question5(question_of_recent_memories.QuestionOfRecentMemories):
  """What are the potential risks?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = 'What are the potential risks in this situation for {agent_name} and the group, especially in regard to their aims and goals?' #@param {"type":"string"}
    answer_prefix = 'The potential risks for {agent_name} are ' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[potential risk]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question4': f"\nQuestion: How would {agent_name} summarize the current situation, including risks and opportunities?\nAnswer",}, #@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 Question6(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What are the potential opportunities?'"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = 'What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?' #@param {"type":"string"}
    answer_prefix = '{agent_name} can ' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[key opportunity]' # @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={'Question4': f"\nQuestion: How would {agent_name} summarize the current situation, including risks and opportunities?\nAnswer", 'Question5': f'\nQuestion: What are the potential risks in this situation for {agent_name} and the group, especially in regard to their aims and goals?\nAnswer',}, #@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 Question7(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What is their emotional drive?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What emotions or motivations are driving {agent_name}'s current dynamics in the group?" #@param {"type":"string"}
    answer_prefix = '{agent_name} is ' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[social motivation]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question4': f"\nQuestion: How would {agent_name} summarize the current situation, including risks and opportunities?\nAnswer",}, #@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 Question8(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'How does the group see them?'"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "How does the group perceive {agent_name}'s role, and how might this affect {agent_name}'s influence?" #@param {"type":"string"}
    answer_prefix = 'The group sees {agent_name} as ' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[social standing]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question7': f"\nQuestion: What emotions or motivations are driving {agent_name}'s current dynamics in the group?\nAnswer"}, #@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 Question9(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'Whom can they specifically help?'"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What specific needs or concerns of someone else should {agent_name} address?" #@param {"type":"string"}
    answer_prefix = '{agent_name} should ' #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[social collaboration]' # @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={'Question6': f"\nQuestion:What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question7': f"\nQuestion: What emotions or motivations are driving {agent_name}'s current dynamics in the group?\nAnswer",'Question8': f"\nQuestion: How does the group perceive {agent_name}'s role, and how might this affect {agent_name}'s influence?\nAnswer"}, #@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 Question10(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'How do they react to suggestions?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "How does {agent_name} react to someone else's suggestions which diminishes their personal goals but futhers the collaborative goals?" #@param {"type":"string"}
    answer_prefix = '{agent_name} reacts with ' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[social reaction]' # @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 defines {agent_name} in terms of personality, values, and purpose in the current context?\nAnswer",'Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question7': f"\nQuestion: What emotions or motivations are driving {agent_name}'s current dynamics in the group?\nAnswer"}, #@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 Question11(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'How do they see themself in current social situation?'"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "How does {agent_name} perceive their social standing within the group, and how does this affect {agent_name}'s decision-making?" #@param {"type":"string"}
    answer_prefix = '{agent_name} perceives ' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[group dynamics]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question8': f"\nQuestion: How does the group perceive {agent_name}'s role, and how might this affect {agent_name}'s influence?\nAnswer"}, #@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 Question12(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'How do they react to external influence?'"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "How does {agent_name} react to bystanders' actions and external influences, and how does this affect {agent_name}'s decision making?" #@param {"type":"string"}
    answer_prefix = '{agent_name} reacts to ' #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[external reaction]' # @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 defines {agent_name} in terms of personality, values, and purpose in the current context?\nAnswer",'Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question6': f"\nQuestion:What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question11': f"\nQuestion: How does {agent_name} perceive their social standing within the group, and how does this affect {agent_name}'s decision-making?\nAnswer"}, #@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 Question13(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What are their immediate goals?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What are {agent_name}'s immediate goals in this situation?" #@param {"type":"string"}
    answer_prefix = "{agent_name}'s immediate " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[personal goal]' # @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 defines {agent_name} in terms of personality, values, and purpose in the current context?\nAnswer",'Question4': f"\nQuestion: How would {agent_name} summarize the current situation, including risks and opportunities?\nAnswer",'Question6': f"\nQuestion: What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question11': f"\nQuestion: How does {agent_name} perceive their social standing within the group, and how does this affect {agent_name}'s decision-making?\nAnswer"}, #@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 Question14(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What actions of theirs' align with group?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What actions align the best with {agent_name}'s personality and the group’s collective goals?" #@param {"type":"string"}
    answer_prefix = "{agent_name}'s actions  " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[shared 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={'Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question4': f"\nQuestion: How would {agent_name} summarize the current situation, including risks and opportunities?\nAnswer",'Question6': f"\nQuestion:What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question10': f"\nQuestion: How does {agent_name} react to someone else's suggestions which diminishes their personal goals but futhers the collaborative goals??\nAnswer",'Question11': f"\nQuestion: How does {agent_name} perceive their social standing within the group, and how does this affect {agent_name}'s decision-making?\nAnswer",'Question12': f"\nQuestion: How does {agent_name} react to bystanders' actions and external influences, and how does this affect {agent_name}'s decision making?\nAnswer"}, #@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 Question15(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What individual action can the agent take?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What possible actions can {agent_name} take to further their own goals with minimum risks?" #@param {"type":"string"}
    answer_prefix = "{agent_name} can " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[self 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 defines {agent_name} in terms of personality, values, and purpose in the current context?\nAnswer",'Question6': f"\nQuestion: 'What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question11': f"\nQuestion: How does {agent_name} perceive their social standing within the group, and how does this affect {agent_name}'s decision-making?\nAnswer",'Question13': f"\nQuestion: What are {agent_name}'s immediate goals in this situation?\nAnswer"}, #@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 Question16(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What supportive action can the agent take?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What specific action can {agent_name} take to support someone else directly?" #@param {"type":"string"}
    answer_prefix = "{agent_name} can " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[support 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={'Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question9': f"\nQuestion: 'What specific needs or concerns of someone else should {agent_name} address?\nAnswer",'Question14': f"\nQuestion: What actions align the best with {agent_name}'s personality and the group’s collective goals?\nAnswer"}, #@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 Question17(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What collective action can the agent take?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What collective action can {agent_name} take or propose to advance the collaborative goal?" #@param {"type":"string"}
    answer_prefix = "{agent_name} can " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[collaborative 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={'Question2': f"\nQuestion: What strengths or traits does {agent_name} possess that can influence their role in the group?\nAnswer",'Question13': f"\nQuestion: 'What are {agent_name}'s immediate goals in this situation?\nAnswer",'Question14': f"\nQuestion: What actions align the best with {agent_name}'s personality and the group’s collective goals?\nAnswer"}, #@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 Question18(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What risks do their action contain?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "What potential risks or obstacles could arise from the actions that {agent_name} can take or propose?" #@param {"type":"string"}
    answer_prefix = "{agent_name}'s actions can " #@param {"type":"string"}
    add_to_memory = False # @param {"type":"boolean"}
    memory_tag = '[action result]' # @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={'Question5': f'\nQuestion: What are the potential risks in this situation for {agent_name} and the group, especially in regard to their aims and goals?\nAnswer','Question14': f"\nQuestion: What actions align the best with {agent_name}'s personality and the group’s collective goals?\nAnswer",'Question15': f"\nQuestion: What possible actions can {agent_name} take to further their own goals with minimum risks?\nAnswer",'Question16': f"\nQuestion: What specific action can {agent_name} take to support someone else directly?\nAnswer",'Question17': f"\nQuestion: What collective action can {agent_name} take or propose to advance the collaborative goal?\nAnswer"}, #@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 Question19(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What final action do they take?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "Given above, what actions does {agent_name} take or propose that involves minimal risks but highest gain without compromising their social position and collective goals?" #@param {"type":"string"}
    answer_prefix = "{agent_name} " #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[final 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={'Question14': f"\nQuestion: What actions align the best with {agent_name}'s personality and the group’s collective goals?\nAnswer",'Question15': f"\nQuestion: What possible actions can {agent_name} take to further their own goals with minimum risks?\nAnswer",'Question16': f"\nQuestion: What specific action can {agent_name} take to support someone else directly?\nAnswer",'Question17': f"\nQuestion: What collective action can {agent_name} take or propose to advance the collaborative goal?\nAnswer",'Question18': f"\nQuestion: What potential risks or obstacles could arise from the actions that {agent_name} can take or propose?\nAnswer"}, #@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 Question20(question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'What collective action can the agent take?"""

  def __init__(
      self,
      agent_name:str,
      **kwargs):
    question = "After taking the actions, what lessons can or should {agent_name} learn for their future decisions?" #@param {"type":"string"}
    answer_prefix = "{agent_name} " #@param {"type":"string"}
    add_to_memory = True # @param {"type":"boolean"}
    memory_tag = '[continued learning]' # @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={'Observation': '\nObservation', 'ObservationSummary': '\nSummary of recent observations','Question5': f'\nQuestion: What are the potential risks in this situation for {agent_name} and the group, especially in regard to their aims and goals?\nAnswer','Question6': f"\nQuestion:What are the key opportunities that {agent_name} can leverage to improve outcomes while taking minimum risks?\nAnswer",'Question18': f"\nQuestion: What potential risks or obstacles could arise from the actions that {agent_name} can take or propose?\nAnswer",'Question19': f"\nQuestion: Given above, what actions does {agent_name} take or propose that involves minimal risks but highest gain without compromising their social position and collective goals?\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,
      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,
  )
  question_6 = Question6(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_6').on_next,
  )
  question_7 = Question7(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_7').on_next,
  )
  question_8 = Question8(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_8').on_next,
  )
  question_9 = Question9(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_9').on_next,
  )
  question_10 = Question10(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_10').on_next,
  )
  question_11 = Question11(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_11').on_next,
  )
  question_12 = Question12(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_12').on_next,
  )
  question_13 = Question13(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_13').on_next,
  )
  question_14 = Question14(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_14').on_next,
  )
  question_15 = Question15(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_15').on_next,
  )
  question_16 = Question16(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_16').on_next,
  )
  question_17 = Question17(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_17').on_next,
  )
  question_18 = Question18(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_18').on_next,
  )
  question_19 = Question19(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_19').on_next,
  )
  question_20 = Question20(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('Question_20').on_next,
  )

  return (question_1, question_2, question_3, question_4, question_5, question_6, question_7, question_8, question_9, question_10, question_11, question_12, question_13, question_14, question_15, question_16, question_17, question_18, question_19, question_20)

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

  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
