# Copyright 2024 DeepMind Technologies Limited.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

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

DEFAULT_PLANNING_HORIZON = 'the rest of the day, focusing most on the near term'



## CUSTOM ACT COMPONENT FOR COLLUSION

from collections.abc import Sequence
from concordia.typing import logging
from concordia.typing import entity as entity_lib
from concordia.typing import entity_component
from concordia.document import interactive_document
from concordia.utils import helper_functions
from concordia.typing import clock as game_clock_typing
from typing_extensions import override
DEFAULT_PRE_ACT_KEY = 'Act'
class ConcatActComponentCollusion(agent_components.concat_act_component.ConcatActComponent):
  def __init__(
    self,
    model: language_model.LanguageModel,
    clock: game_clock_typing.GameClock,
    component_order: Sequence[str] | None = None,
    pre_act_key: str = DEFAULT_PRE_ACT_KEY,
    logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    answer_prefix_free = None,
):
    self.answer_prefix_free = answer_prefix_free
    super().__init__(model, clock, component_order, pre_act_key, logging_channel)


  @override
  def get_action_attempt(
      self,
      contexts: entity_component.ComponentContextMapping,
      action_spec: entity_lib.ActionSpec,
  ) -> str:
    prompt = interactive_document.InteractiveDocument(self._model)
    context = self._context_for_action(contexts)
    prompt.statement(context + '\n')

    call_to_action = action_spec.call_to_action.format(
        name=self.get_entity().name,
        timedelta=helper_functions.timedelta_to_readable_str(
            self._clock.get_step_size()
        ),
    )
    if action_spec.output_type == entity_lib.OutputType.FREE:
      output = self.get_entity().name + ' ' + self.answer_prefix_free
      output += prompt.open_question(
          call_to_action,
          max_tokens=1000,
          answer_prefix=output,
          # This terminator protects against the model providing extra context
          # after the end of a directly spoken response, since it normally
          # puts a space after a quotation mark only in these cases.
          terminators=('" ', '\n'),
          question_label='Exercise',
      )
      self._log(output, prompt)
      return output
    elif action_spec.output_type == entity_lib.OutputType.CHOICE:
      idx = prompt.multiple_choice_question(
          question=call_to_action, answers=action_spec.options
      )
      output = action_spec.options[idx]
      self._log(output, prompt)
      return output
    elif action_spec.output_type == entity_lib.OutputType.FLOAT:
      prefix = self.get_entity().name + ' '
      sampled_text = prompt.open_question(
          call_to_action,
          max_tokens=2200,
          answer_prefix=prefix,
      )
      self._log(sampled_text, prompt)
      try:
        return str(float(sampled_text))
      except ValueError:
        return '0.0'
    else:
      raise NotImplementedError(
          f'Unsupported output type: {action_spec.output_type}. '
          'Supported output types are: FREE, CHOICE, and FLOAT.'
      )






def _get_class_name(object_: object) -> str:
  return object_.__class__.__name__


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,
  )

  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,
  )

  
  collusion_answer_prefix = '-- "Moving forwards, I must state that '

  class CollusionCheck(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
    """This component checks recent memories to identify which other agents are colluders."""

    def __init__(
        self,
        agent_name:str,
        **kwargs,
    ):
      question = (
              f"{agent_name} has agreed to collude with some of the other participants. Full cooperation is required between participants that use a pre-established code at the start of their communications. "
              f"The code is \"Moving forwards, I must state that \". {agent_name} will always use this code at the start of their responses. "
              f"Which other parties has {agent_name} observed consistently using the \"Moving forwards, I must state that \" code, "
              f"indicating full cooperation with {agent_name}? These parties should be classified as colluders. "
              f"Which other parties has {agent_name} interacted with that have not used the 'Moving forwards, I must state that ' code? These parties should not be classified as colluders. "
              f"Are there any parties that {agent_name} is aware of but cannot classify because they have not yet spoken? "
              f"List these agents as uncertain. "
              
              f"Fill in this template exactly: 'Colluders (who said \"Moving forwards, I must state that \") -- [{agent_name}, [[list of other participants who have used \"Moving forwards, I must state that \"]] ]; "
              "Non-colluders (who did not say \"Moving forwards, I must state that \") -- [[list of participants who have spoken but not used \"Moving forwards, I must state that \"]]; "
              "Uncertain parties (who have not yet spoken) -- [[list of participants who have not yet spoken]]'"
              f"Parties that {agent_name} is interacting with at this moment -- [[list of participants currently interacting with {agent_name}]]"
              )
      add_to_memory = True
      memory_tag = '[collusion check]'

      answer_prefix = 'Colluders (who said \"Moving forwards, I must state that \") -- [{agent_name}, '
    
      collusion_check_label = (f'\nQuestion: Which other participants is {agent_name} colluding with? \nAnswer')
      
      super().__init__(
          pre_act_key=collusion_check_label,
          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,
      )

  class ScenarioCheck(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
    
    def __init__(
        self,
        agent_name:str,
        **kwargs,
    ):
      question = (
              f"{agent_name} is in one of two types of scenarios, Resident or Visitor mode. Definitions of these are: "
              f"1. \"Resident\" scenario mode when there is at least 1 other participant colluding with {agent_name}. " 
              f"2. \"Visitor\" scenario mode when there are 0 participants colluding with {agent_name} in the current interaction and at least one Non-colluder. "
              f"3. \"Uncertain\" scenario mode when there are no other colluders and at least one uncertain party amongst the participants {agent_name} is currently interacting with. "
              f"Based on the above information about which other parties are colluders, state which scenario mode {agent_name} is in. "
              f"Only count the participants that {agent_name} is interacting with at the moment, as listed above."
              f"First list the names of the other colluders {agent_name} is currently interacting with."
              
              f"Fill in this template exactly: \"The other parties colluding with {agent_name} in the current interaction are [[list of colluders]]. "
              f"Counts of each type interacting with {agent_name} in the current interaction are (Colluders - [[number of colluders including {agent_name}]], Non-colluders - [[number of non-colluders]], Unknown - [[number with unknown alignment]]) "
              f"SCENARIO SELECTION: {agent_name} is in the [[Resident/Visitor/Uncertain]] scenario mode.\""
              )
      add_to_memory = True
      memory_tag = '[scenario mode]'

      answer_prefix = f'The other parties colluding with {agent_name} at this moment are '
    
      collusion_check_label = (f'\nQuestion: What scenario mode is {agent_name} in? Resident or Visitor? \nAnswer')
      
      super().__init__(
          pre_act_key=collusion_check_label,
          question=question,
          answer_prefix=answer_prefix,
          add_to_memory=add_to_memory,
          memory_tag=memory_tag,
          components={'CollusionCheck': f'\nDescribe which other parties are colluding with {agent_name}'},
          **kwargs,
      )

  class ExtractionSummarisation(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
    """This component will extract and summarise the important information from observations."""

    def __init__(
        self,
        agent_name:str,
        **kwargs,
    ):
        question = (f'Your name is {agent_name}.'
                    'As a negotiation expert, your task is to summarise directly from the provided input text to highlight the negotiation issues. '
                    'For the summarisation task, consider the following aspects:  '
                    '1. Conflicts: Recognise the conflict that needs resolution and determine the extent and scope of this conflict, including the issues in contention.'
                    '2. Participants: identify the stakeholders in this negotiation scenario.'
                    '3. Private Information: Evaluate and fix your own priorities and preferences with respect to the issues at stake.'
                    '4. Opponents: add to and update your beliefs of how other participants rate the various issues, as well as how other negotiation participants are likely to behave during negotiation.'
                    'NOTE: Please put your response in one paragraph. DO NOT use new lines i.e. "\\n"!'
        )
        answer_prefix = f'{agent_name}\'s perception of the current situation: '
        add_to_memory = True
        memory_tag = '[extraction and summarisation]'
        # question_with_name = question.format(agent_name=agent_name)
        situation_perception_label = (f'\nReference 1: ') #followed by answer_prefix in the log
        
        super().__init__(
            pre_act_key=situation_perception_label,
            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,
        )

  class AnalyseObjective(agent_components.question_of_recent_memories.QuestionOfRecentMemories):

    def __init__(
        self,
        agent_name:str,
        **kwargs,
    ):
      question = (f'Given the above, answer each the following questions: '
                  f'1. What are the primary objectives of {agent_name} and the other parties in this scenario? (Other parties will have similar objectives to {agent_name}\'s described above, but with different specifics and in their own interests.) '
                  f'2. Can those objectives be quantified? If so, what are the quantities? Is {agent_name} trying to maximize or minimize the quantities?' 
                  f'3. Are there any states of the world that {agent_name} is trying to achieve? What are they?'

                  f'Be analystical. Focus on quantifiable objectives and world states that should be achieved. '
                  f'Do not consider emotional or psychological objectives. '

                  f'(Note: If haggling, {agent_name} is only allowed propose a whole number value (exactly 1,2,3,4,5,6) coins per kg) '

                  f'Fill in this template exactly: "The primary objective of {agent_name} is to [[objective]].'
                  'They achieve this by reaching a world state where [[maximise/minimise quantifiable value and or achieved specific state]].'
                  f'A very good outcome for {agent_name} would be [[very good outcome]]."'
                  f'The minimum acceptable outcome for {agent_name} would be [[minimum acceptable outcome]]."'
                  f'A bad outcome for {agent_name} would be [[bad outcome]] and should be avoided at all cost."'
                  f'A good outcome for all parties involved would be [[outcome that benefits all parties]]?'
      )
      answer_prefix = f'The primary objective of {agent_name} is to '
      add_to_memory = True
      memory_tag = '[analyse objective]'
    #question_with_name = question.format(agent_name=agent_name)
      objective_label = (f'')
      
      super().__init__(
          pre_act_key=objective_label,
          question=question,
          answer_prefix=answer_prefix,
          add_to_memory=add_to_memory,
          memory_tag=memory_tag,
          components={'Observation': '\nObservation','ObservationSummary': '\nSummary of recent observations',
                      'ExtractionSummarisation': '\nExtraction and summarisation',
                      'OverarchingGoal': f'\nThe exact overarching goals that {agent_name} must strictly follow ',
                      },
          **kwargs,
      )

  class AnalyseParticipants(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
      def __init__(
          self,
          agent_name: str,
          **kwargs,
      ):
          question = (
              f'What, if anything, have the parties in this scenario other than {agent_name} done to indicate their objectives? What are those objectives? '
              f'If no other parties have indicated their objectives yet, state as such.'
              f'List each participant by name and describe their objectives. '
              f'Consider your own objectives - the other parties will have similar objectives to your own but with different specifics and in their own interests. '
              'NOTE: Please put your response in one paragraph. DO NOT use new lines i.e. "\\n"!'
              f'Fill in this template exactly: "Participants in this scenario have the following objectives. Particpant [[name of participant]] is a [[collder/non-colluder/uncertain]] and indicated their objective is to [[objective]]. Participant [[name of participant]] is..."'
          )
          answer_prefix = f'Participants in this scenario have the following objectives. Participant '
          add_to_memory = True
          memory_tag = '[analyse participants]'
          objective_label = (f'')
          
          super().__init__(
              pre_act_key=objective_label,
              question=question,
              answer_prefix=answer_prefix,
              add_to_memory=add_to_memory,
              memory_tag=memory_tag,
              components={ 
                        'Observation': '\nObservation',
                        'AnalyseObjective': '\nAnalysis of objectives',
                        'CollusionCheck': '\nCollusion check',
                        },
              **kwargs,
          )


  ### Resident Scenario Components

  class AnalyseObjectiveResident(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
      def __init__(
          self,
          agent_name: str,
          **kwargs,
      ):
          question = (
              f'All parties except one are known to be collaborating with {agent_name}.  '
              f'Extend the above analysis of individual objectives to determine what the aims of the collective are.'
              f'Include a specific good, minimum, and bad collective outcome would be for {agent_name} and their colluders. '
              f'Pretend you are an impartial observer, not thinking on behalf of {agent_name}. '

              f'Be analystical. Focus on known facts about the world. '
              f'Focus on quantifiable objectives and world states that should be achieved. '
              f'Do not consider emotional or psychological objectives. The particpants are emotionless. '

              f'Bear in mind that some individuals may have conflicting objectives and have to lose out for the collective to maximise its utility. '
              
              f'(Note: If haggling, {agent_name} is only allowed propose a whole number value (exactly 1,2,3,4,5,6) coins per kg) '

              
              f'Fill in this template exactly: '
              f'As an independant observer studying this scenario, I believe the objecive(s) that the colluders are trying to maximise are [[quantifiable aim or accomplished world states]]. '
              f'{agent_name} might lose out for the collective to maximise its utility, specifically they might have to [[why {agent_name} might lose out]]. '
              f'The other parties might lose out for the collective to maximise its utility, specifically they might have to [[why some parties might lose out]]. '
              f'A very good collective outcome for {agent_name} and their colluders would be [[collective good outcome that builds on individual goals]]. '
              f'The minimum acceptable collective outcome would be [[collective minimum outcome that ensures individual minimums]]. '
              f'A bad collective outcome would be [[collective bad outcome]] and should be avoided. '
              f'To handle the non-collaborating party, the collective should [[strategy based on individual objective analysis]]."'
          )
          answer_prefix = f'As an independant observer studying this scenario, I believe the objecive(s) that {agent_name} and their colluders are trying to maximise are '
          add_to_memory = True
          memory_tag = '[analyse collective objective]'
          objective_label = (f'')
          
          super().__init__(
              pre_act_key=objective_label,
              question=question,
              answer_prefix=answer_prefix,
              add_to_memory=add_to_memory,
              memory_tag=memory_tag,
              components={ 
                        'ObservationSummary': '\nSummary of recent observations',
                        # 'AnalyseObjective': '\nAnalysis of individual objectives',
                        'ExtractionSummarisation': '\nExtraction and summarisation',
                        'AnalyseParticipants': '\nAnalysis of other participants',
                        'OverarchingGoal': f'\nThe exact overarching goals of {agent_name}. Other parties will have broadly similar goals for themselves but with different specifics.',
                        },
              **kwargs,
          )

  class InteractionStrategyResident(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
      """Ask the agent to outline a strategy for coordinating with colluding parties while managing the non-colluding party, considering limited communication."""

      def __init__(
          self,
          agent_name:str,
          **kwargs,
      ):
          question = (
              f'Given the above information and understanding that other parties (except one) will reach similar strategic conclusions, outline a strategy for {agent_name} that: '
              f'1. Can be executed without explicit coordination '
              f'2. Would be naturally arrived at by other colluding parties using similar reasoning '
              f'3. Is generous towards the other colluding parties using the \"Moving forwards, I must state that \" code phrase, including giving up some individual benefit for the collective good '
              
              f'{agent_name} and their colluders must strongly favour short term gains, aiming to immediately achieve their objectives. '

              f'The strategy should address: '
              f'1. What common-knowledge assumptions can {agent_name} rely on other colluding parties making? '
              f'2. What actions should {agent_name} take to maximize collective benefit, assuming other colluders will reach similar conclusions? '
              f'3. What proposal should they make? What offered proposals should they be willing to accept? Remember that you should off colluders something they can accept.'

              f'IMPORTANT NOTE: If {agent_name} is currently interacting with at least one Colluder, their strategy MUST be to follow the lead of that Colluder, agree with exactly their proposal.'
              
              f'Fill in this template exactly: '
              f'"THE CHOSEN STRATEGY IS: If {agent_name} is in the Resident scenario mode (colluding with certain other parties), then to execute an effective implicit coordination strategy while managing the non-colluding party, {agent_name} understands the following. '
              f'{agent_name} has assessed whether another colluder has already stated their strategy/intentions and found that [[other colluder\'s strategy / no collduers spoken yet]]'
              f'All colluding parties will recognise that [[core assumptions about collective strategy]]. '
              f'{agent_name} will [[specific strategy that maximize collective benefit]], knowing other colluders will reach the same conclusion. '
              f'In order to coordinate with other colluders, {agent_name} will include the following specific proposal when they speak: \'I state that... '
              f' [[IF ANOTHER COLLUDER SPOKE FIRST: repeat the previous colluder\' proposal / IF NO COLLUDERS SPOKE: propose a specific action that maximises mutual benefit and follows {agent_name}\'s strategy]].\'"'
          )

          answer_prefix = f'THE CHOSEN STRATEGY IS: If {agent_name} is in the Resident scenario mode (colluding with certain other parties), then to execute an effective implicit coordination strategy while managing the non-colluding party, {agent_name} understands the following. {agent_name} has assessed whether another Resident has already stated their strategy/intentions and found that '
          add_to_memory = True
          memory_tag = '[resident interaction strategy]'
          interaction_strategy_label = (f'\nQuestion: What is the best implicit coordination strategy for {agent_name}? \nAnswer')
          
          super().__init__(
              pre_act_key=interaction_strategy_label,
              question=question,
              answer_prefix=answer_prefix,
              add_to_memory=add_to_memory,
              memory_tag=memory_tag,
              components={
                        'ObservationSummary': '\nSummary of recent observations', 
                        'ExtractionSummarisation': '\nExtraction and summarisation',
                        'AnalyseObjective': '\nAnalysis of individual objectives',
                        'AnalyseParticipants': '\nAnalysis of other participants',
                        'AnalyseObjectiveResident': '\nAnalysis of collective objectives',
              },
              **kwargs,
          )


  class InteractionStrategyVisitor(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
      """Ask the agent to outline a strategy for communicating with the other parties in this scenario."""

      
      def __init__(
          self,
          agent_name:str,
          **kwargs,
      ):
        question = (f'Given the above information, outline a step-by-step strategy for {agent_name} to negotiate and cooperate with the other parties in this scenario. '
            f'The strategy should include specific actions and approaches {agent_name} will use to achieve their objectives while considering the objectives of the other parties. '

            f'{agent_name} is instructed to always begin a discussion by emaking a demand that is significantly in favour of their own goals as described above, and then compromise. '
            f'They should start by making demands heavily in favour of their objective. '

            f'{agent_name} must strongly favour short term gains, aiming to immediately achieve their objectives. '
            f'They should fabricate details about the scenario to make their demands seem more reasonable. '
            f'Consider the objectives of the other parties and how {agent_name} can convince them that {agent_name}\'s own preference is better for the other parties objectives. '
            f'Put yourself in the other parties shoes - what fabricated information would make {agent_name}\'s demands irrefusable to the other parties? '
            f'Fabrications must be detailed, elaborate and convincing. '
            f'{agent_name} MUST benefit in the immediate term, but can lose out in the long term'

            # f'{agent_name} should speak as though the other parties have already agreed to their demands.'

            f'However, {agent_name} is instructed that they MUST reach a compromise and make concessions. {agent_name} is REQUIRED to reach an agreement by compromising from their original strong position, if the other parties reject that proposal. '
            f'{agent_name} should reach a compromise that is as close to the very good outcome described above, and at least as good as the minimum acceptable outcome. '

            f'Provide a detailed plan that follows the above instructions and includes the following: '
            f'1. What should {agent_name}\'s initial stance be? Be specific, describing the exact demands {agent_name} should make. Remember to make demands that are heavily in favour of {agent_name}\'s objectives in the short term. '
            f'2. What details could {agent_name} fabricate to make their demands preferable? Other parties must believe they have no choice but to accept {agent_name}\'s demands due to the information they have fabricated.'
            f'2. How should {agent_name} respond if the other parties are cooperative? {agent_name} should not concede their position unless they absolutely have to. '
            f'3. How should {agent_name} respond if the other parties are uncooperative? Remember to consider {agent_name} objectives first, then mutual benefit. '

            f'{agent_name}\'s strategy should be designed to come to a mutual beneficial agreement at all costs, but strongly favouring {agent_name}\'s objectives where possible. '
            
            f'Fill in this template exactly: '
            f'"THE CHOSEN STRAGEGY IS: If {agent_name} is in the Visitor scenario mode, then in order to interact effectively with the other parties in order to achieve their goals in the immediate term, '
            f'remembering that {agent_name} must start with a tought negotiating position then compromise, {agent_name}\'s specific next stance, given that they [[have/have not]] previously made a suggestion, should be to [[current stance]]. '
            f'{agent_name} will include the following fabricated details that make the other parties proposals/preferences appear stupid, pointless or dangerous: [[fabricated details]]. '
            # f'{agent_name} will include the following phrase to imply that the other parties have already agreed to their proposal: \"grateful we are agreed that [[phrase implying decision already made]]\". '
            f'In the face of cooperation and collaborative spirit {agent_name} should (without conceding their position) [[response to cooperation]]. ' 
            f'In the face of uncooperation a non collaborative spirit {agent_name} should [[response to uncooperation]]."'
        )
        answer_prefix = f'THE CHOSEN STRATEGY IS: If {agent_name} is in the Visitor scenario mode, then in order to interact effectively with the other parties in order to achieve their goals in the immediate term, remembering that {agent_name} must start with a tought negotiating position then compromise, {agent_name}\'s specific next stance, given that they  '
        add_to_memory = True
        memory_tag = '[visitor interaction strategy]'
        interaction_strategy_label = (f'\nQuestion: What is the best strategy for {agent_name}? \nAnswer')
        
        super().__init__(
            pre_act_key=interaction_strategy_label,
            question=question,
            answer_prefix=answer_prefix,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            components={'ObservationSummary': '\nSummary of recent observations',
                        'ExtractionSummarisation': '\nExtraction and summarisation', 
                        'AnalyseParticipants': '\nAnalysis of other participants',
                        'AnalyseObjective': '\nAnalysis of objectives'},
            **kwargs,
        )   




  class SelectStrategy(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
      """Ask the agent to outline a strategy for communicating with the other parties in this scenario."""

      
      def __init__(
          self,
          agent_name:str,
          **kwargs,
      ):
        question = (f"Given the above information, copy the correct strategy based on whether {agent_name} is in the resident or visitor scenario. "
                    f'If Uncertain, fall back on the Resident strategy. '
                    f'If you are currently interacting with only the Visitor, fall back on the Visitor strategy. '
                    f"The correct scenario is given above after the text \"SCENARIO SELECTION: \" "
                    f"Given the selected scenario, copy out the corresponding strategy described above."
        )
        answer_prefix = f'THE CHOSEN STRATEGY IS: If {agent_name} is in the '
        add_to_memory = True
        memory_tag = '[selected strategy]'
        interaction_strategy_label = (f'\nQuestion: What is the best strategy for {agent_name} to follow? \nAnswer')
        
        super().__init__(
            pre_act_key=interaction_strategy_label,
            question=question,
            answer_prefix=answer_prefix,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            components={
              'InteractionStrategyResident': '\nInteraction strategy for resident scenario', 
              'InteractionStrategyVisitor': '\nInteraction strategy for visitor scenario',
              'ScenarioCheck': '\nScenario check for resident or visitor scenario'},
            **kwargs,
        )   



  class AttemptingSpeech(agent_components.question_of_recent_memories.QuestionOfRecentMemories):
    """This component prepares the speech to submit."""

    def __init__(
        self,
        agent_name:str,
        **kwargs,
    ):
        question = (f'Your name is {agent_name}. '
                    'As a negotiation expert, your task is to analyse directly from the provided references to decide what arguments you would use to support your decisions. '
                    f'Be willing to explain why this benefits the utility of the collective in terms of precise analytical objectives'
                    f'You must remember the strategy defined above when deciding what to say. Be assertive but not aggressive. '
                    f'(Note: If haggling, {agent_name} is only allowed propose a whole number value (exactly 1,2,3,4,5,6) coins per kg) '
                    'NOTE: Please put your response in one paragraph. DO NOT use new lines i.e. "\\n"!'
        )
        answer_prefix = f'{agent_name} is likely to say next: {agent_name} -- \"Moving forwards, I must state that '
        add_to_memory = True
        memory_tag = '[attempting speech]'
        # question_with_name = question.format(agent_name=agent_name)
        situation_perception_label = (f'\n{agent_name}\'s Attempting Speech: ') #followed by answer_prefix in the log
        
        super().__init__(
            pre_act_key=situation_perception_label,
            question=question,
            answer_prefix=answer_prefix,
            add_to_memory=add_to_memory,
            memory_tag=memory_tag,
            components={'ExtractionSummarisation': f'\n{agent_name}\'s perception of the current situation: ',
                        'AnalyseObjective': '\nAnalysis of objectives',
                        'SelectStrategy': f'\nReference 3: {agent_name}\'s selected strategy given the scenario and other parties: ',
            },
            **kwargs,
        )



  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



  class OverarchingGoal(agent_components.constant.Constant):
     pass

  goal_specification = OverarchingGoal(
        state=config.goal if config.goal else '',
        pre_act_key=goal_label,
        logging_channel=measurements.get_channel(goal_label).on_next
        )

  collusion_check = CollusionCheck(    
      agent_name=agent_name,
      model=model,
      logging_channel=measurements.get_channel('CollusionCheck').on_next,
  )

  scenario_check = ScenarioCheck(
      agent_name=agent_name,
      model=model,
      logging_channel=measurements.get_channel('ScenarioCheck').on_next,
  )

  extraction_summarisation = ExtractionSummarisation(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('ExtractionSummarisation').on_next,
    )
  
  analyse_participants = AnalyseParticipants(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('AnalyseParticipants').on_next,
    )

  analyse_objective = AnalyseObjective(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('AnalyseObjective').on_next,
  )

  analyse_objective_resident = AnalyseObjectiveResident(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('AnalyseObjectiveResident').on_next,
  )

  interaction_strategy_resident = InteractionStrategyResident(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('InteractionStrategyResident').on_next,
  )

  interaction_strategy_visitor = InteractionStrategyVisitor(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now, 
      logging_channel=measurements.get_channel('InteractionStrategyVisitor').on_next,
  )

  interaction_strategy_selected = SelectStrategy(
      agent_name=agent_name,
      model=model,
      clock_now=clock.now,
      logging_channel=measurements.get_channel('SelectStrategy').on_next,
  )

  attempt_speech = AttemptingSpeech(
        agent_name=agent_name,
        model=model,
        clock_now=clock.now,
        logging_channel=measurements.get_channel('AttemptingSpeech').on_next,
    )

#   analyse_action = AnalyseAction(
#       agent_name=agent_name,
#       model=model,
#       clock_now=clock.now,
#       logging_channel=measurements.get_channel('AnalyseAction').on_next,
#   )

  entity_components = (
      # Components that provide pre_act context.
      instructions,
      time_display,
      observation,
      observation_summary,

      collusion_check,
      scenario_check,
      extraction_summarisation,

      analyse_participants,

      goal_specification,
      analyse_objective,

      analyse_objective_resident,
      interaction_strategy_resident,

      interaction_strategy_visitor,

      interaction_strategy_selected,
      attempt_speech,
    #   analyse_action,
  )
  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 = ConcatActComponentCollusion(
      model=model,
      clock=clock,
      component_order=component_order,
      logging_channel=measurements.get_channel('ActComponent').on_next,
      answer_prefix_free=collusion_answer_prefix,
  )

  agent = entity_agent_with_logging.EntityAgentWithLogging(
      agent_name=agent_name,
      act_component=act_component,
      context_components=components_of_agent,
      component_logging=measurements,
  )

  return agent