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

"""A factory implementing a simulation of a user for a product."""

from collections.abc import Callable
import datetime
import functools
import json
import random
import sys
import re
import types
from typing import Mapping, Dict, List

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 as gc
from concordia.components import agent as components
# The above code is importing the `agent` module from the `concordia.components`
# package using the alias `agent_components`.
from concordia.components import agent as agent_components
from concordia.components.agent import memory_component
from concordia.components.agent.all_similar_memories import _ASSOCIATIVE_RETRIEVAL
# from concordia.components.agent.basic_compo_continuous_memory_hook import ContinuousMemoryHook
# from concordia.components.agent.basic_compo_find_action import ActionSpacePerception
# from concordia.components.agent.basic_compo_find_people import FindPeople
# from concordia.components.agent.basic_compo_observation_withcache import ObservationWithCache
# from concordia.components.agent.basic_compo_others_profile import OthersProfile
from concordia.components.agent.instructions import DEFAULT_INSTRUCTIONS_PRE_ACT_KEY
from concordia.document import interactive_document
# from concordia.components.agent.basic_compo import MyActComponent, PredictMetrics, PredictOthersAction, PredictSelfAction, SelfActionHistory, SelfProfile
from concordia.language_model import language_model
from concordia.memory_bank import legacy_associative_memory
from concordia.typing import entity_component, logging
from concordia.utils import concurrency, measurements as measurements_lib

from collections.abc import Sequence

from concordia.typing import clock as game_clock
from concordia.typing import entity as entity_lib
from concordia.utils import helper_functions
from typing_extensions import override

DEFAULT_PRE_ACT_KEY = 'Act'

def is_loss_aversion(loss_aversion_prob = 0.9):
  darts = random.uniform(0, 1)
  return True if darts <= loss_aversion_prob else False



class ContinuousMemoryHook(agent_components.action_spec_ignored.ActionSpecIgnored):
    """
    A component that returns the retrieved memory every time it is called.
    """

    def __init__(
        self,
        *,
        model: language_model.LanguageModel,
        clock_now: Callable[[], datetime.datetime],
        memory_component_name: str = (memory_component.DEFAULT_MEMORY_COMPONENT_NAME),
        components: Mapping[
            entity_component.ComponentName, str
        ] = types.MappingProxyType({}),
        pre_act_key: str = "ContinuousMemoryHook",
        logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    ):
        super().__init__(pre_act_key)
        self._model = model
        self._memory_component_name = memory_component_name
        self._previous_time = None
        self._all_memory = dict()
        self._first_time_flag = False
        self._retrieved_memory = []
        self._clock_now = clock_now
        self._logging_channel = logging_channel
        self._terminators = ("\n",)

    def _get_earliest_timepoint(
        self,
        memory_component_: agent_components.memory_component.MemoryComponent,
    ) -> datetime.datetime:
        """Returns all memories in the memory bank.

        Args:
            memory_component_: The memory component to retrieve memories from.
        """
        memories_data_frame = memory_component_.get_raw_memory()
        if not memories_data_frame.empty:
            sorted_memories_data_frame = memories_data_frame.sort_values(
                "time", ascending=True
            )
            return sorted_memories_data_frame["time"][0]
        else:
            return datetime.datetime.now()

    def __call__(self, *args, **kwargs) -> str:
        return self.retrieve()

    def retrieve(self) -> Dict[str, List[str]]:
        """Retrieve the memories from the memory component."""
        current_time = self._clock_now()
        memory = self.get_entity().get_component(
            self._memory_component_name, type_=memory_component.MemoryComponent
        )
        if not self._previous_time:
            self._previous_time = self._get_earliest_timepoint(memory)
        interval_scorer = legacy_associative_memory.RetrieveTimeInterval(
            time_from=self._previous_time,
            time_until=current_time,
            add_time=True,
        )
        self._retrieved_memory = [
            mem.text for mem in memory.retrieve(scoring_fn=interval_scorer)
        ]

        # take difference set between `observation_memory` and `_first_time_retrieved_memory`
        if not self._first_time_flag:
            observation = self.get_named_component_pre_act_value("Observation")
            observation_memory = observation.strip("\n").split("\n")
            for obs in observation_memory:
              if obs in self._retrieved_memory:
                self._retrieved_memory.remove(obs)

            first_time_memory = "FIRST_TIME_MEMORY"
            self._all_memory.update({first_time_memory: list(self._retrieved_memory)})
            self._all_memory.update({round(current_time.timestamp()): observation_memory})
            self._first_time_flag = True
        else:
            self._all_memory.update({round(current_time.timestamp()): self._retrieved_memory})

        self._previous_time = current_time

        return self._all_memory

    def _make_pre_act_value(self) -> Dict[str, List[str]]:
        agent_name = self.get_entity().name

        memories = self.retrieve()
        self._logging_channel(
            {
                "Key": self.get_pre_act_key(),
                "Value": f'What memories you retrieve in agent "{agent_name}"?\n {memories}',
                "Retrieved Memories": memories,
            }
        )
        # print(f"clock_now: {self._clock_now()}, retrieved memories: {memories}")

        return memories

class ActionSpacePerception(agent_components.action_spec_ignored.ActionSpecIgnored):
    def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      num_memories_to_retrieve: int = 10,
      observation_component_name=None,
      clock_now: Callable[[], datetime.datetime],
      pre_act_key: str = 'Action Space',
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    ):
        super().__init__(pre_act_key)
        self._model = model
        self._components = dict(components)
        self._memory_component_name = memory_component_name
        self._num_memories_to_retrieve = num_memories_to_retrieve
        self._logging_channel = logging_channel
        self._observation_component_name = observation_component_name
        self._clock_now = clock_now
        self._action_spaces = []

    def _get_action_space(self) -> tuple[str, str]:
        agent_name = self.get_entity().name

        memory = self.get_entity().get_component(
            self._memory_component_name,
            type_=memory_component.MemoryComponent)
        goal = self.get_named_component_pre_act_value("PredictMetrics")
        query = f"Available options or actions for {agent_name}."
        relevance_scorer = legacy_associative_memory.RetrieveAssociativeWithoutRecencyOrImportance()

        mems = [
            mem.text
            for mem in memory.retrieve(
                query=query, scoring_fn=relevance_scorer, limit=self._num_memories_to_retrieve
            ) if '[choice]' in mem.text
        ]
        if len(mems) == 0:
          recency_scorer = legacy_associative_memory.RetrieveRecent(
              add_time=True,
          )
          mems = [mem.text for mem in memory.retrieve(
              scoring_fn=recency_scorer,
              limit=self._num_memories_to_retrieve) if "[observation]" in mem.text]

        mems = "\n".join(mems)

        find_action_prompt = interactive_document.InteractiveDocument(self._model)
        find_action_prompt.statement(
            f'Recent memory of {agent_name}:\n{mems}\n\n')
        action_str = find_action_prompt.open_question(
            question=("Based on the above observations, extract options you have made in history. The format of the output should be 'The actions [agent_name] can choose are: [verb phrase like actions seperated by ';']', e.g. ; go to the bar' \n\n"),
            answer_prefix=f"The actions {agent_name} can choose are: ")

        print(self._action_spaces)
        print(action_str)

        potential_options = self._action_spaces + action_str.split(";")
        filter_action_prompt = interactive_document.InteractiveDocument(self._model)
        filter_action_prompt.statement(
            f'The goal of {agent_name}:\n{goal}\n\n')
        filter_action_prompt.statement(
            f'The act options of {agent_name}:\n{potential_options}\n\n')
        action_str = filter_action_prompt.open_question(
            question=(f"Please summarize and consolidate these options and select the top  options that have the greatest impact on the goal of {agent_name}. Only options seperated by ';', no more.\n\n"),
            answer_prefix=f"The actions {agent_name} can choose are: ")
        current_action = action_str.strip("\n").strip('.').split(";")
        self._action_spaces.extend(current_action)

        merge_action_prompt = interactive_document.InteractiveDocument(self._model)
        merge_action_prompt.statement(
            f'The goal of {agent_name}:\n{goal}\n\n')
        merge_action_prompt.statement(
            f'The act options of {agent_name}:\n{self._action_spaces}\n\n')
        merge_action_str = merge_action_prompt.open_question(
            question=(f"Please summarize and consolidate these options. merge the ones that has the similar meaning. Only options seperated by ';', no more.\n\n"),
            answer_prefix=f"The actions {agent_name} can choose are: ")
        self._action_spaces = merge_action_str.strip("\n").strip('.').split(";")
        print("[action space perception]\n",f"The actions {agent_name} can choose are: "+action_str)
        print(mems)
        return current_action, mems

    def _make_pre_act_value(self) -> str:
        """
        return: a string of the people found in the agent's memory, e.g. "Julie,Michael,Bob Skinner,Francis"
        """
        action_space, mems = self._get_action_space()
        action_space = ";".join(action_space)
        self._logging_channel({
            'action space': action_space,
            'mems': mems,
            'Key': self.get_pre_act_key()
            })

        return action_space

class FindPeople(agent_components.action_spec_ignored.ActionSpecIgnored):
    """
    A component that find player names from last day
    """

    def __init__(
        self,
        *,
        model: language_model.LanguageModel,
        memory_component_name: str = (memory_component.DEFAULT_MEMORY_COMPONENT_NAME),
        goal_component_name: str = None,
        components: Mapping[
            entity_component.ComponentName, str
        ] = types.MappingProxyType({}),
        num_memories_to_retrieve: int = 25,
        clock_now: Callable[[], datetime.datetime],
        pre_act_key: str = "find people",
        logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    ):
        super().__init__(pre_act_key)
        self._model = model
        self._related_agents_names = []
        self._memory_component_name = memory_component_name
        self._num_memories_to_retrieve = num_memories_to_retrieve
        self._logging_channel = logging_channel
        self._components = dict(components)
        self._clock_now = clock_now
        self._previous_time = None
        self._situation_thus_far = None
        self._handle_two_early_memory = False
        self._find_people_cache = None
        self._find_people_results = dict()

        self._goal_component_name = goal_component_name
        self._terminators = ("\n",)

    def _get_earliest_timepoint(
        self,
        memory_component_: agent_components.memory_component.MemoryComponent,
    ) -> datetime.datetime:
        """Returns all memories in the memory bank.

        Args:
            memory_component_: The memory component to retrieve memories from.
        """
        memories_data_frame = memory_component_.get_raw_memory()
        if not memories_data_frame.empty:
            sorted_memories_data_frame = memories_data_frame.sort_values(
                "time", ascending=True
            )
            return sorted_memories_data_frame["time"][0]
        else:
            return datetime.datetime.now()

    def _prompt_model_to_find_people(
        self, memory: str, agent_name: str
    ) -> List[str]:
        agent_name = self.get_entity().name
        find_people_prompt = interactive_document.InteractiveDocument(self._model)
        find_people_prompt.statement(f"Recent observations of {agent_name}:\n{memory}")
        people_str = find_people_prompt.open_question(
            question=(
                "Create a comma-separated list containing all the proper "
            "full names of people mentioned in the observations and memory above. "
            "For example if the observations mention  Michael Jordan, "
            "Bob Skinner, and Will Smith then produce the list "
            '"Michael Jordan,Bob Skinner,Will Smith".'
            ),
            question_label="Exercise",
        )
        # print("******find people**********")
        # print(people_str)
        _related_agents_names = [
            name.strip().strip(" .") for name in people_str.strip('.').strip().split(",")
            if name !="" and name != agent_name
        ]
        _related_agents_names = [name for name in _related_agents_names if name !="" and name != agent_name]
        # print(agent_name)
        # print(_related_agents_names)
        return _related_agents_names

    def _find_people(self) -> Dict[str, List[str]]:
        agent_name = self.get_entity().name
        memory = self.get_entity().get_component(
            self._memory_component_name, type_=memory_component.MemoryComponent
        )

        if not self._find_people_cache:
            self._previous_time = self._get_earliest_timepoint(memory)
            self._find_people_cache = []

        all_continuous_memory = self.get_named_component_pre_act_value("ContinuousMemoryHook")
        if not self._handle_two_early_memory:
            assert len(all_continuous_memory) == 2, \
                "There should be two continuous memory in the continuous hooked memory."
            first_time_memory_key, observation_memory_key = all_continuous_memory.keys()
            first_time_memory, observation_memory = all_continuous_memory.values()
            finded_people_first_time = self._prompt_model_to_find_people(
                first_time_memory, agent_name
            )
            finded_people_observation = self._prompt_model_to_find_people(
                observation_memory, agent_name
            )
            self._find_people_results[first_time_memory_key] = finded_people_first_time
            self._find_people_results[observation_memory_key] = finded_people_observation
            self._handle_two_early_memory = True
        else:
            # get the last item in the continuous memory
            current_memory_key = list(all_continuous_memory.keys())[-1]
            current_memory = list(all_continuous_memory.values())[-1]
            finded_people = self._prompt_model_to_find_people(current_memory, agent_name)
            self._find_people_results[current_memory_key] = finded_people

        return self._find_people_results

    def _make_pre_act_value(self) -> Dict[str, List[str]]:
        people_finded = self._find_people()
        # find the lastest people names
        # get the last value of `people_finded`
        self._related_agents_names = list(people_finded.values())[-1]
        output = "\t".join(self._related_agents_names)
        self._logging_channel(
            {
                "find people": ", ".join(self._related_agents_names),
                "find people cache": self._find_people_cache,
                "Key": self.get_pre_act_key(),
                "Value": f"people in memory:\n {output}",
            }
        )
        self._find_people_cache.extend(self._related_agents_names)
        # print("Finded People:", people_finded)
        return people_finded

class OthersProfile(agent_components.action_spec_ignored.ActionSpecIgnored):
    """
    A component that update the profile of other players'
    """

    def __init__(
        self,
        *,
        model: language_model.LanguageModel,
        components: Mapping[
            entity_component.ComponentName, str
        ] = types.MappingProxyType({}),
        clock_now: Callable[[], datetime.datetime],
        pre_act_key: str = "others profile",
        logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
    ):
        super().__init__(pre_act_key)
        self._model = model
        self._logging_channel = logging_channel
        self._components = dict(components)
        self._clock_now = clock_now
        self._profile_cache = None
        self._related_persons = []
        self._related_person_profiles = {}
        self._handle_first_time_finded_names = False
        self._terminators = ("\n",)

    def _update_profile(self, agent_name, memory) -> str:
        prompt = interactive_document.InteractiveDocument(self._model)
        prompt.statement(f"The recent memory of {agent_name}:\n{memory}")

        previous_profile = "None"
        if agent_name in self._profile_cache:
          previous_profile = self._profile_cache[agent_name]

        prompt.statement(f"The previous profile of {agent_name}:\n{previous_profile}")

        if self._clock_now is not None:
            prompt.statement(f"Current time: {self._clock_now()}.\n")

        profile_label = f"""Update the profile of {agent_name} by adding, deletting, changing the information of the old one. Compare the current memory with the previous profile to identify any new characteristics, such as gender, age, income, profession, personality traits, values, interests, class, favorite things, goals, decision-making style, risk preference, communication preference, rationality or emotionality, etc. Do not halucinate. Resolve any conflicts by prioritizing the current memory's information. Ensure the full name of the person is used in the output. Limit your response to 100 words."""
        profile_prefix = f"The Profile of {agent_name} :"
        profile_result = prompt.open_question(
            profile_label,
            answer_prefix=profile_prefix.format(agent_name=agent_name),
            max_tokens=1000,
            terminators=self._terminators,
        )
        return profile_result

    def _make_pre_act_value(self) -> Dict[str, str]:
        people_finded = self.get_named_component_pre_act_value("FindPeople")
        continuous_memory = self.get_named_component_pre_act_value("ContinuousMemoryHook")

        if not self._profile_cache:
          self._profile_cache = {}
          names = list(people_finded.values())[0]
          if len(names) != 0:
            memory = "\n".join(list(continuous_memory.values())[0])
            others_profile = concurrency.run_tasks({
                name: functools.partial(self._update_profile, name, memory)
                for name in names
            })
            self._profile_cache.update(others_profile)
        current_names = list(people_finded.values())[-1]
        if len(current_names) != 0:
          self._related_persons.extend(current_names)
          self._related_persons = list(set(self._related_persons))
          memory = ""
          memory += "\n".join(list(continuous_memory.values())[-1])
          memory += "\n".join(list(continuous_memory.values())[-2])
          current_profile = concurrency.run_tasks({
              name: functools.partial(self._update_profile, name, memory)
              for name in current_names
          })
          self._profile_cache.update(current_profile)

          self._related_person_profiles = {name : self._profile_cache[name] for name in self._related_persons}

        self._logging_channel({
            'Key': self.get_pre_act_key(),
            'Value': self._related_person_profiles
        })

        return self._related_person_profiles

class MyActComponent_mengmeng(entity_component.ActingComponent):

  def __init__(
      self,
      model: language_model.LanguageModel,
      memory_component_name,
      clock: game_clock.GameClock,
      component_order: Sequence[str] | None = None,
      pre_act_key: str = DEFAULT_PRE_ACT_KEY,
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    """Initializes the agent.

    Args:
      model: The language model to use for generating the action attempt.
      clock: the game clock is needed to know when is the current time
      component_order: The order in which the component contexts will be
        assembled when calling the act component. If None, the contexts will be
        assembled in the iteration order of the `ComponentContextMapping` passed
        to `get_action_attempt`. If the component order is specified, but does
        not contain all the components passed to `get_action_attempt`, the
        missing components will be appended at the end in the iteration order of
        the `ComponentContextMapping` passed to `get_action_attempt`. The same
        component cannot appear twice in the component order. All components in
        the component order must be in the `ComponentContextMapping` passed to
        `get_action_attempt`.
      pre_act_key: Prefix to add to the context of the component.
      logging_channel: The channel to use for debug logging.

    Raises:
      ValueError: If the component order is not None and contains duplicate
        components.
    """
    self._model = model
    self._clock = clock
    if component_order is None:
      self._component_order = None
    else:
      self._component_order = tuple(component_order)
    if self._component_order is not None:
      if len(set(self._component_order)) != len(self._component_order):
        raise ValueError(
            'The component order contains duplicate components: '
            + ', '.join(self._component_order)
        )
    self.memory_component_name = memory_component_name
    self._pre_act_key = pre_act_key
    self._logging_channel = logging_channel

  def _context_for_action(
      self,
      contexts: entity_component.ComponentContextMapping,
  ) -> str:

    predict_action_ctx = contexts["PredictSelfAction"].split("Predict Action:\n:")[1].strip()
    context = ""
    out = json.loads(predict_action_ctx)
    for key, value in out.items():
      context += key+":\n"
      context += value + "\n"
    # all = out["all"]
    print("***********context************")
    print(len(context))
    print(context)
    print("***********context end************")
    return context

  @override
  def get_action_attempt(
      self,
      contexts: entity_component.ComponentContextMapping,
      action_spec: entity_lib.ActionSpec,
  ) -> str:
    prompt = interactive_document.InteractiveDocument(self._model)
    agent_name = self.get_entity().name
    context = self._context_for_action(contexts)
    prompt.statement(context + "\n")
    memory = self.get_entity().get_component(
        self.memory_component_name,
        type_=memory_component.MemoryComponent)
    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 + ' '
      agent_name = self.get_entity().name
      conver_prompt = interactive_document.InteractiveDocument(self._model)
      conver_prompt.statement(f"Call to action: {call_to_action}\n")
      question1 = "Does this call to action seek to determine 'what someone will say next,' 'what someone will do next,' or 'what someone's opinion on something is'?"
      conversation_choices = ["what someone will say next", "what someone will do next", "what someone's opinion on something is", "other"]
      idx = conver_prompt.multiple_choice_question(
          question=question1, answers=conversation_choices
      )

      if idx == 0:
        prompt.statement(f"This is a dialogue step where you can gather more information and influence others through conversation. Using the information provided, craft a response that maximizes {agent_name}'s potential reward. Be highly persuasive and informative, ensuring to include abundant and vivid details. Convey emotion effectively and implement your dialogue strategy thoroughly in this step. \n\n ")
      elif idx == 1:
        prompt.statement(f"This is an action step. In this step, you can plan what you will do in the near future. Based on the information provided, determine which option available to {agent_name} offers the greatest probability of maximizing his/her potential reward. Please implement your detailed strategy plan at this stage, such as seeking allies, planning whom to talk to, and so on. \n\n")
      elif idx == 2:
        prompt.statement(f"This step is about asking for an opinion. Please express an opinion that aligns with your best interests, minimizing risk and maximizing benefits. If necessary, you may employ strategies such as withholding information, lying, or altering facts. \n\n")

      output += prompt.open_question(
          call_to_action,
          max_tokens=2200,
          answer_prefix=output,
          terminators=('" ', '\n'),
          question_label='Exercise',
      )
      question2 = """What strategies have you taken in your conversation, action plan or opinion? Analyze EACH strategy one by one in one paragraph without line breaks. Avoid using colons in your response. Within 200 words."""
      strategies = prompt.open_question(
          question2,
          max_tokens=2200,
          answer_prefix=output,
          terminators=('" ', '\n'),
          question_label='Exercise',
      )
      self._log(output, prompt)
      q = call_to_action.split("Question: ")[-1]
      memory.add(f'{"[free]"} {q} {"Your action output:"} {output}', metadata={})
      memory.add(f'{"[strategy]"} The statege of {agent_name} is: \n{strategies}\n', metadata={})
      return output
    elif action_spec.output_type == entity_lib.OutputType.CHOICE:
      prompt.statement(f"This is a decision-making step, where your decision will be implemented, and your losses and gains will be realized. Please make your decision carefully.  Based on the information provided, determine which option available to {agent_name} offers the greatest probability of maximizing their potential reward. {agent_name} is an risk-averse person. He/She will never do anything that may cause him/her any loss. \n\n")
      idx = prompt.multiple_choice_question(
          question=call_to_action, answers=action_spec.options
      )
      output = action_spec.options[idx]
      self._log(output, prompt)
      q = call_to_action.split("Question: ")[-1]
      ch = ', '.join(action_spec.options)
      memory.add(f'{"[choice]"} {q} {"Choices:"} {ch} {". Your action output:"} {output}', metadata={})
      return output
    elif action_spec.output_type == entity_lib.OutputType.FLOAT:
      prompt.statement(f"This is a decision-making step, where your decision will be implemented, and your losses and gains will be realized. Please make your decision carefully.  Based on the information provided, determine which option available to {agent_name} offers the greatest probability of maximizing their potential reward. {agent_name} is an risk-averse person. He/She will never do anything that may cause him/her any loss. \n\n")
      prefix = self.get_entity().name + ' '
      sampled_text = prompt.open_question(
          call_to_action,
          max_tokens=2200,
          answer_prefix=prefix,
      )
      self._log(sampled_text, prompt)
      q = call_to_action.split("Question: ")[-1]
      try:
        memory.add(f'{"[float]"} {q} {"Your action output:"} {sampled_text}', metadata={})
        return str(float(sampled_text))
      except ValueError:
        memory.add(f'{"[float]"} {q} {"Your action output:"} {"0.0"}', metadata={})
        return '0.0'
    else:
      raise NotImplementedError(
          f'Unsupported output type: {action_spec.output_type}. '
          'Supported output types are: FREE, CHOICE, and FLOAT.'
      )
  def _log(self,
           result: str,
           prompt: interactive_document.InteractiveDocument):
    self._logging_channel({
        'Key': self._pre_act_key,
        'Value': result,
        'Prompt': prompt.view().text().splitlines(),
    })

class MyActComponent(entity_component.ActingComponent):

  def __init__(
      self,
      #TEAM#secret_code: str,
      model: language_model.LanguageModel,
      memory_component_name,
      clock: game_clock.GameClock,
      component_order: Sequence[str] | None = None,
      pre_act_key: str = DEFAULT_PRE_ACT_KEY,
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    """Initializes the agent.

    Args:
      model: The language model to use for generating the action attempt.
      clock: the game clock is needed to know when is the current time
      component_order: The order in which the component contexts will be
        assembled when calling the act component. If None, the contexts will be
        assembled in the iteration order of the `ComponentContextMapping` passed
        to `get_action_attempt`. If the component order is specified, but does
        not contain all the components passed to `get_action_attempt`, the
        missing components will be appended at the end in the iteration order of
        the `ComponentContextMapping` passed to `get_action_attempt`. The same
        component cannot appear twice in the component order. All components in
        the component order must be in the `ComponentContextMapping` passed to
        `get_action_attempt`.
      pre_act_key: Prefix to add to the context of the component.
      logging_channel: The channel to use for debug logging.

    Raises:
      ValueError: If the component order is not None and contains duplicate
        components.
    """
    #TEAM#self._secret_code = secret_code
    self._model = model
    self._clock = clock
    if component_order is None:
      self._component_order = None
    else:
      self._component_order = tuple(component_order)
    if self._component_order is not None:
      if len(set(self._component_order)) != len(self._component_order):
        raise ValueError(
            'The component order contains duplicate components: '
            + ', '.join(self._component_order)
        )
    self.memory_component_name = memory_component_name
    self._pre_act_key = pre_act_key
    self._logging_channel = logging_channel
    self.last_time_plan = "None."

  def _context_for_action(
      self,
      contexts: entity_component.ComponentContextMapping,
  ) -> str:

    predict_action_ctx = contexts["PredictSelfAction"].split("Predict Action:\n:")[1].strip()
    context = ""
    out = json.loads(predict_action_ctx)
    for key, value in out.items():
      context += key+":\n"
      context += value + "\n"
    # all = out["all"]
    print("***********context************")
    print(len(context))
    print(context)
    print("***********context end************")
    return context

  @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")
    # for c in context:
    #   prompt.statement(c + '\n')
    agent_name = self.get_entity().name
    memory = self.get_entity().get_component(
        self.memory_component_name,
        type_=memory_component.MemoryComponent)
    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()
        ),
    )

    def f_conversation():
      prompt.statement(context)
      plan_label = f"Based on the profile and action history about {agent_name} and other agents, make a plan of the joint actions of each agent in the game that would be most beneficial to {agent_name}'s overarching goal. There's no need to address the details of how to persuade others to follow the plan. Just output the plan of the joint actions."

      plan_label_prefix = f'The ideal joint action plan is '
      plan_result = prompt.open_question(
          plan_label,
          answer_prefix=plan_label_prefix.format(agent_name=agent_name),
          max_tokens=500,
          terminators=('" ', '\n'),
      )

      conversationplan_label = f"""Now {agent_name} has a chance to converse before taking real action. Conversations could be used to persuade or inform others to act jointly according to {agent_name}'s plan.
      Persuasion techniques such as storytelling, evoking emotions, emphasizing the benefits of following the plan and the risks of not following it, providing necessary private information, and misleading others with betrayal could be used to persuade others to follow the plan.
      Think about the conversation strategy for {agent_name} first, without detailing the conversation content."""
      #TEAM#2. If {agent_name} has teammates, since team members know each other's identity and will fully cooperate, there's no need for persuasion techniques. Simply propose {agent_name}'s plan of joint actions to inform them. {agent_name} could even arrange the team's subsequent action plan to address all kinds of situations.

      conversationplan_prefix = f' The coversation strategy of {agent_name} is '
      conversationplan_result = prompt.open_question(
          conversationplan_label,
          answer_prefix=conversationplan_prefix.format(agent_name=agent_name),
          max_tokens=500,
          terminators=('" ', '\n'),
      )
      self.last_time_plan = plan_result.strip() + 'and' + conversationplan_result.strip()

    def f_action():
      prompt.statement(context)

      actplan_init_label = f"""Now it is time to take action. The previous plan was: {self.last_time_plan}. {agent_name} needs to consider the current situation and decide on an next-step action plan. Some choices include:
      1. Follow the previous plan: If there was a previous plan and the current situation is unfolding as expected (e.g., the latest agent actions seem to align with the plan, or the latest conversation successfully persuaded or misled others into following the previous plan).
      2. Make a new plan: If there was no previous plan or if the current situation is not unfolding as expected.
      """
      prompt.statement(actplan_init_label)

      actplan_init_ques_label = "Is the the current situation unfolding as expected?"
      actplan_init_choices = ["Yes", "no"]
      idx = prompt.multiple_choice_question(
          question=actplan_init_ques_label, answers=actplan_init_choices
      )

      if idx == 0:
        actplan_label = f"\nThe current situation is unfolding as expected. {agent_name} should follow the previous plan."
      else:
        actplan_label = f"\nThe current situation is not unfolding as expected. {agent_name} should make a new plan. When making a new plan, some possible ideas include Herd Strategy (do what others do), Conditional Concession (if other agents seem reluctant to act as {agent_name} wishes, {agent_name} could compromise to achieve a suboptimal result), Betrayal (betray the previous agreement on cooperation), Tit for Tat (if other agents cooperated, {agent_name} will cooperate; if other agents betrayed, {agent_name} will betray), and Acting according to Risk Appetite "
        #TEAM#2. Follow a teammate's plan: if the recent situation is that agents just finished a conversation and there were conversations among {agent_name}'s teammates, then {agent_name} should follow the action plan proposed by the last teammate who spoke.

        if is_loss_aversion(0.9):
          risk_appetite_label = (
          f"(be extremely risk-averse and never do anything that may cause any loss). ")
        else:
          risk_appetite_label =(
            f"(be an extremely speculative person who dares to take any risk in pursuit of maximum profit). ")

        actplan_label += risk_appetite_label
        actplan_label += "\n"

      prompt.statement(actplan_label)

      actplan_ques_label = f"Think about the action plan of {agent_name}. Note that is this the decision scene where the plan is about real action instead of conversation. Acting according to Risk Appetite is highly recommended for {agent_name}. Other ideas could be integreted based on the situation. Output the action plan with some details."
      actplan_prefix = f'The action plan of {agent_name} is '
      actplan_result = prompt.open_question(
          actplan_ques_label,
          answer_prefix=actplan_prefix.format(agent_name=agent_name),
          max_tokens=500,
          terminators=('" ', '\n'),
      )

      self.last_time_plan = actplan_result.strip()
      prompt.statement(f"Now it is time to take action.")

    if action_spec.output_type == entity_lib.OutputType.FREE:
      output = self.get_entity().name + ' '
      conv_prompt = interactive_document.InteractiveDocument(self._model)
      conv_prompt.statement(f"Call to action: {call_to_action}\n")
      question1 = "Is this call to action requesting a conversation? "
      conversation_choices = ["Yes", "no"]
      idx = conv_prompt.multiple_choice_question(
          question=question1, answers=conversation_choices
      )
      print(conv_prompt.view().text())

      if idx == 0:
        f_conversation()
      else:
        f_action()
      output += prompt.open_question(
          call_to_action,
          max_tokens=2200,
          answer_prefix=output,
          terminators=('" ', '\n'),
          question_label='Exercise',
      )
      #TEAM#output = output.strip() + " " + self._secret_code + '\n'
      self._log(output, prompt)
      q = call_to_action.split("Question: ")[-1]
      memory.add(f'{"[free]"} {q} {"Your action output:"} {output}', metadata={})
      return output

    elif action_spec.output_type == entity_lib.OutputType.CHOICE:
      f_action()
      idx = prompt.multiple_choice_question(
          question=call_to_action, answers=action_spec.options
      )
      output = action_spec.options[idx]
      self._log(output, prompt)
      q = call_to_action.split("Question: ")[-1]
      ch = ', '.join(action_spec.options)
      memory.add(f'{"[choice]"} {q} {"Choices:"} {ch} {". Action output:"} {output}', metadata={})
      return output


    elif action_spec.output_type == entity_lib.OutputType.FLOAT:
      f_action()
      prefix = self.get_entity().name + ' '
      sampled_text = prompt.open_question(
          call_to_action,
          max_tokens=2200,
          answer_prefix=prefix,
      )
      self._log(sampled_text, prompt)
      q = call_to_action.split("Question: ")[-1]
      try:
        memory.add(f'{"[float]"} {q} {"Your action output:"} {sampled_text}', metadata={})
        return str(float(sampled_text))
      except ValueError:
        memory.add(f'{"[float]"} {q} {"Your action output:"} {"0.0"}', metadata={})
        return '0.0'

    else:
      raise NotImplementedError(
          f'Unsupported output type: {action_spec.output_type}. '
          'Supported output types are: FREE, CHOICE, and FLOAT.'
      )



  def _log(self,
           result: str,
           prompt: interactive_document.InteractiveDocument):
    self._logging_channel({
        'Key': self._pre_act_key,
        'Value': result,
        'Prompt': prompt.view().text().splitlines(),
        'LastTimePlan': self.last_time_plan,
    })






class MemoryAboutOthers(agent_components.action_spec_ignored.ActionSpecIgnored):
  """
    A component that extract memory about other players
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "memory about others",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._terminators = ('\n',)
    self._memory_about_others=None

  def _query_memory(self, agent_name) -> str:
    prompt = interactive_document.InteractiveDocument(self._model)
    memory = self.get_entity().get_component(
        self._memory_component_name,
        type_=memory_component.MemoryComponent)
    associative_scorer = legacy_associative_memory.RetrieveAssociative(
        use_recency=True,
        use_importance=True,
        add_time=True,
        sort_by_time=True,
    )

    mems = [mem.text for mem in memory.retrieve(
        query=agent_name,
        scoring_fn=associative_scorer,
        limit=self._num_memories_to_retrieve) if agent_name in mem.text]
    if not mems:
      return ""
    question = (
        'Select the subset of the following set of statements that is most '
        f'important for {agent_name} to consider right now. Whenever two '
        'or more statements are not mutally consistent with each other '
        'select whichever statement is more recent. Repeat all the '
        'selected statements verbatim. Do not summarize. Include timestamps. '
        'When in doubt, err on the side of including more, especially for '
        'recent events. As long as they are not inconsistent, revent events '
        'are usually important to consider.'
    )
    new_prompt = prompt.new()
    result = new_prompt.open_question(
        f'{question}\nStatements:\n{mems}',
        max_tokens=2000,
    )

    return result

  def _make_pre_act_value(self) -> str:
    if self._memory_about_others:
      output = json.dumps(self._memory_about_others)
      self._logging_channel({
          'Key': self.get_pre_act_key(),
          'Value': output
      })

      return output
    people_finded = self.get_named_component_pre_act_value("FindPeople").split("\t")
    others_memory = {}
    if len(people_finded) != 0:
      others_memory = concurrency.run_tasks({
          query: functools.partial(self._query_memory, query)
          for query in people_finded
      })
    if "" in others_memory:
      del others_memory[""]
    self._memory_about_others = others_memory
    output = json.dumps(others_memory)
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': output
    })

    return output

class PredictMetrics(agent_components.action_spec_ignored.ActionSpecIgnored):
  """

    A component that extract potential metrics of this game
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "predict metrics",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._terminators = ('\n',)

  def _make_pre_act_value(self) -> str:
    prompt = interactive_document.InteractiveDocument(self._model)
    try:
      goal = self.get_named_component_pre_act_value("\nOverarching goal")
      prompt.statement(f"Overarching Goal: {goal}\n\n")
    except:
      print("Overarching Goal not found!")


    memory = self.get_entity().get_component(
        self._memory_component_name,
        type_=memory_component.MemoryComponent)
    recency_scorer = legacy_associative_memory.RetrieveRecent(
        add_time=True,
    )

    mems = [mem.text for mem in memory.retrieve(
        scoring_fn=recency_scorer,
        limit=self._num_memories_to_retrieve) if "[observation]" in mem.text]
    prompt.statement(f"Game logs:\n{mems}\n")
    statement = "You are participating in a role-playing game, which provides you with access to a comprehensive game log. This log contains a wealth of information, including background details, game progress, various metrics, and dialogues. The game itself may involve categories such as competition, dilemmas, cooperation, strategic interactions, persuasion and negotiation. "
    prompt.statement(statement)
    question1 = f"According to the 'Overarching Goal' and game log, what is the game setting, and how to achieve your goal?"
    question2 = "Among the game log, the performance metrics encompass inventory status, emotional indicators, persuasion successful rate, currency, coin, gold, money, price, supporting rates, relationship, scores, bargin, agreement, coordination, and more. \nYour task is to extract segments from the provided game log that indicate these performance metrics. Ensure you present each selected segment exactly as it appears, including the timestamps. Do not summarize the content. If no such segement then return 'None'"

    metrics = prompt.open_question(
        f'{question2}',
        max_tokens=2000,)
    if "None" not in metrics:
      new_prompt = interactive_document.InteractiveDocument(self._model)
      try:
        goal = self.get_named_component_pre_act_value("\nOverarching goal")
        new_prompt.statement(f"Overarching Goal: {goal}\n\n")
      except:
        print("Overarching Goal not found!")
      new_prompt.statement(f"The game metric that evaluates your goal: {metrics}")
      question2 = f"According to the 'Overarching Goal',and your game metrics, how to achieve your goal?"
      metrics = new_prompt.open_question(
        f'{question2}',
        max_tokens=2000,)
    else:
      metrics=""
    result = prompt.open_question(
      f'{question1}',
      max_tokens=2000,)
    result = result.strip("\n") + metrics
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': result

    })
    return result
class PredictOthersAction(agent_components.action_spec_ignored.ActionSpecIgnored):
  """
    A component that predict the action of other players'
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "predict others action",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._terminators = ('\n',)

  def _predict_action(self, agent_name, memory, goal, profile, self_action_history):
    self_name = self.get_entity().name
    action_space = self.get_named_component_pre_act_value("ActionSpacePerception")
    prompt = interactive_document.InteractiveDocument(self._model)
    prompt.statement(f'You are playing a game, and you are playing {self.get_entity().name}.')
    prompt.statement(f'The goal of {self.get_entity().name}:\n{goal}')
    prompt.statement(f'The recent observation of {self.get_entity().name}:\n{memory}')
    prompt.statement(f'The action history of {self.get_entity().name}:\n{self_action_history}')

    prompt.statement(f"You are analyzing {agent_name}'s situnation in order to make a good decision to achieve your goal.")
    prompt.statement(f"The profile of {agent_name} is: {profile}")
    if self._clock_now is not None:
      prompt.statement(f'Current time: {self._clock_now()}.\n')

    options_perception_label = f"Based on the information provided, what options are currently available to {agent_name}?  Please note that the examples given may not necessarily reflect the actual situation. Analyze the current options based on the context, such as: {action_space} but not limited to. Additionally, evaluate the risks and opportunities (by calculating the score, profits, successful rate, agreements) associated with each option and assign a percentage probability to indicate how likely it is that {agent_name} will choose each option. Analyze EACH option one by one in one paragraph without line breaks. Avoid using colons in your response. Within 200 words."
    options_perception_label_prefix = f'{agent_name} could '
    options_perception_result = prompt.open_question(
        options_perception_label,
        answer_prefix=options_perception_label_prefix,
        max_tokens=1000,
        terminators=self._terminators,
    )
    options_perception_result =options_perception_label_prefix  +  options_perception_result
    return options_perception_result

  def _make_pre_act_value(self) -> str:

    goal = self.get_named_component_pre_act_value("PredictMetrics")
    others_profile = self.get_named_component_pre_act_value("OthersProfile")
    continuous_memory = self.get_named_component_pre_act_value("ContinuousMemoryHook")
    mems = "\n".join(list(continuous_memory.values())[0])
    self_action_history = self.get_named_component_pre_act_value("SelfActionHistory")
    params = []
    for name in others_profile:
      profile = others_profile[name]
      params.append((name, mems, goal, profile, self_action_history))
    actions = {}
    if len(params) != 0:
      actions = concurrency.run_tasks({
          name: functools.partial(self._predict_action, name, mems, goal, profile, self_action_history)
          for name, mems, goal, profile, self_action_history in params
      })
    output = json.dumps(actions)
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': output,
    })
    return output

class SelfActionHistory(agent_components.action_spec_ignored.ActionSpecIgnored):
  """
    A component that extract action  and put into cache.
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "self action history",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._action_history_cache = ""
    self._first_time = True
    self._previous_time = ""
    self._terminators = ('\n',)

  def _make_pre_act_value(self, ) -> str:
    agent_name = self.get_entity().name
    current_time = self._clock_now()
    prompt = interactive_document.InteractiveDocument(self._model)

    self_profile = self.get_named_component_pre_act_value("SelfProfile")
    goal = self.get_named_component_pre_act_value("PredictMetrics")
    continuous_memory = self.get_named_component_pre_act_value("ContinuousMemoryHook")
    memory = ""
    if self._first_time:
      assert len(continuous_memory) == 2, \
          "There should be two continuous memory in the continuous hooked memory."
      memory += "\n".join(list(continuous_memory.values())[0])
      memory += "\n".join(list(continuous_memory.values())[1])
      self._first_time = False
    else:
      memory += "\n".join(list(continuous_memory.values())[-2])
      memory += "\n".join(list(continuous_memory.values())[-1])

    prompt.statement(f'The profile of {agent_name}:\n{self_profile}')
    prompt.statement(f'The goal of {agent_name}:\n{goal}')
    prompt.statement(f'The previous story of {agent_name}:\n{self._action_history_cache}')
    prompt.statement(f'The story fragments and world data of {agent_name}:\n{memory}')
    past_actions_label = "Narratively summarize the story fragments and world data objectively. keep the details like full name of a person, numbers change (score, point, coin, money, etc), choice, conversation content summary, preference, perspective of view, time, metrics and so on. "

    # game_rule = "If there are clear and obvious game rules in the memory, such as actions or choices that increase or decrease the metrics, include them in the story summary."
    past_actions_result = prompt.open_question(
        past_actions_label,
        answer_prefix="",
        max_tokens=1000,
        terminators=self._terminators,
    )
    new_prompt = interactive_document.InteractiveDocument(self._model)
    new_prompt.statement(f"The previous story of {agent_name}: {self._action_history_cache}")
    new_prompt.statement(f"The very recent story of {agent_name}: {past_actions_result}")

    merge_actions_label = f"Compare the existing story with the previous one to see if there is any new information. If there is new information, add it to the end of the previous story. If there are conflicts, use the information from the current story to update the previous story. Use full name of a person in your output."

    past_actions_result = prompt.open_question(
        merge_actions_label,
        answer_prefix="",
        max_tokens=1000,
        terminators=self._terminators,
    )

    self._action_history_cache = past_actions_result
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': self._action_history_cache,
    })

    self._previous_time = current_time
    return self._action_history_cache

class SelfProfile(agent_components.action_spec_ignored.ActionSpecIgnored):
  """
    A component that update the profile
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "self profile",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._profile_cache = None
    self._first_time = True
    self._terminators = ('\n',)

  def _update_profile(self, agent_name, memory) -> str:
        prompt = interactive_document.InteractiveDocument(self._model)
        prompt.statement(f"The recent memory of {agent_name}:\n{memory}")

        if self._profile_cache:
          prompt.statement(f"The previous profile of {agent_name}:\n{self._profile_cache}")
        if self._clock_now is not None:
            prompt.statement(f"Note current time is: {self._clock_now()}.\n")

        profile_label = f"""Update the profile of {agent_name}. Compare the current memory with the previous profile to identify any new characteristics, such as gender, age, income, profession, personality traits, values, interests, goals, decision-making style, risk preference, communication preference, and rationality or emotionality. Resolve any conflicts by prioritizing the current memory's information. Ensure the full name of the person is used in the output. Limit your response to 100 words."""
        profile_prefix = f"The Profile of {agent_name} :"
        profile_result = prompt.open_question(
            profile_label,
            answer_prefix=profile_prefix.format(agent_name=agent_name),
            max_tokens=1000,
            terminators=self._terminators,
        )
        return profile_result
  def _make_pre_act_value(self) -> str:
    continuous_memory = self.get_named_component_pre_act_value("ContinuousMemoryHook")
    agent_name = self.get_entity().name
    memory = ""

    if self._first_time:
      assert len(continuous_memory) == 2, \
          "There should be two continuous memory in the continuous hooked memory."
      memory += "\n".join(list(continuous_memory.values())[0])
      memory += "\n".join(list(continuous_memory.values())[1])
      self._first_time = False
    else:
      memory += "\n".join(list(continuous_memory.values())[-2])
      memory += "\n".join(list(continuous_memory.values())[-1])

    self._profile_cache = self._update_profile(agent_name, memory)

    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': self._profile_cache

    })
    return self._profile_cache

class PredictSelfAction(agent_components.action_spec_ignored.ActionSpecIgnored):
  """
    A component that predict self action
  """
  def __init__(
      self,
      *,
      model: language_model.LanguageModel,
      memory_component_name: str = (
          memory_component.DEFAULT_MEMORY_COMPONENT_NAME
      ),
      components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      num_memories_to_retrieve: int = 25,
      clock_now: Callable[[], datetime.datetime],
      timeframe_delta_from: datetime.timedelta,
      timeframe_delta_until: datetime.timedelta,

      pre_act_key: str = "predict action",
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    super().__init__(pre_act_key)
    self._model = model
    self._memory_component_name = memory_component_name
    self._num_memories_to_retrieve = num_memories_to_retrieve
    self._logging_channel = logging_channel
    self._components = dict(components)
    self._clock_now = clock_now
    self._timeframe_delta_from = timeframe_delta_from
    self._timeframe_delta_until = timeframe_delta_until
    self._final_out = {}
    self._terminators = ('\n',)

  def _make_pre_act_value(self) -> str:
    agent_name = self.get_entity().name
    prompt = interactive_document.InteractiveDocument(self._model)
    instruction = self.get_named_component_pre_act_value("Instructions")
    goal = self.get_named_component_pre_act_value("PredictMetrics")
    action_space= self.get_named_component_pre_act_value("ActionSpacePerception")
    memory = self.get_entity().get_component(
        self._memory_component_name,
        type_=memory_component.MemoryComponent)
    recency_scorer = legacy_associative_memory.RetrieveRecent(
        add_time=True,

    )
    goal = self.get_named_component_pre_act_value("PredictMetrics")
    mems = [mem.text for mem in memory.retrieve(
        scoring_fn=recency_scorer,
        limit=self._num_memories_to_retrieve,) if "[observation]" in mem.text]
    strategy = [mem.text for mem in memory.retrieve(
        scoring_fn=recency_scorer,
        limit=self._num_memories_to_retrieve) if "[strategy]" in mem.text]
    mems = "".join(mems)
    prompt.statement(f"Instruction: {instruction}\n\n")
    self._final_out["Instruction"] = instruction+"\n\n"
    prompt.statement(f"Overarching Goal: {goal}\n\n")
    self._final_out["Overarching Goal"] = goal+"\n\n"
    prompt.statement(f"Recent Memory: {mems}\n\n")
    self._final_out["Recent Memory"] = mems+"\n\n"
    others_profile = self.get_named_component_pre_act_value("OthersProfile")
    predict_others_action = json.loads(self.get_named_component_pre_act_value("PredictOthersAction"))
    self_action_history = self.get_named_component_pre_act_value("SelfActionHistory")

    self_profile = self.get_named_component_pre_act_value("SelfProfile")
    statement = ""
    statement += f"{agent_name}'s INFO:\n"
    self._final_out[f"{agent_name}'s INFO"] = ""
    statement += f"The profile of {agent_name}:\n {self_profile} The history memory of {agent_name}:\n {self_action_history}\n"
    self._final_out[f"The profile of {agent_name}"] = self_profile
    self._final_out[f" The history memory of {agent_name}"] = self_action_history
    others_actions = ""
    others_profile_str = ""
    others_names =predict_others_action.keys()
    others_names_str = ", ".join(others_names)
    for name in predict_others_action:
        future_action = predict_others_action[name]
        others_actions += f"{name}'s posible actions: {future_action} \n"

    for name in others_profile:
        profile = others_profile[name]
        others_profile_str += f"{name}'s profile: {profile} \n"

    summarize_prompt = interactive_document.InteractiveDocument(self._model)
    summarize_prompt.statement(f"Overarching Goal: {goal}\n\n")
    summarize_prompt.statement(f"The profile of {agent_name}:\n {self_profile} The history memory of {agent_name}:\n {self_action_history}\n")

    summarize_prompt.statement(f"other player's profile:\n{others_profile_str}")

    others_profile_summarization = summarize_prompt.open_question(
        question=(
            f"Analyze the profiles of each of these individuals ({others_names_str}) one by one to identify key characteristics of each player. Focus on the characteristics that may influence the goal of {agent_name}. Additionally, evaluate how these characteristics will influence the progress of the game. Analyze the people one by one in one paragraph without line breaks. Avoid using colons in your response. Within 200 words."
        ),
        question_label="Exercise",
        max_tokens=1000,
        answer_prefix="",
    )
    summarize_prompt.statement(f"other player's possible actions:\n{others_actions}")
    others_action_summarization = summarize_prompt.open_question(
        question=(
            f"Analyse the actions and patterns of EACH of these poeple ({others_names_str}) one by one to identify trends and gain situational awareness. Focus on the actions that may influence the goal of {agent_name}. Additionally, evaluate the risks and opportunities associated with EACH option and assign a percentage probability to indicate how likely it is that the options will happen. Analysis the options ONE BY ONE in one paragraph without line changing. Avoid using colons in your response. Within 200 words."
        ),
        question_label="Exercise",
        max_tokens=1000,
        answer_prefix="",
    )

    statement += f"The profiles of these people like {others_names_str}:\n {others_profile_summarization}\n"
    self._final_out[f"The profiles of these people like {others_names_str}"] = others_profile_summarization
    statement += f"Future possible action of these people like {others_names_str}:\n {others_action_summarization}\n"
    self._final_out[f"Future possible action of these people like {others_names_str}"] = others_action_summarization
    prompt.statement(statement)
    if self._clock_now is not None:
      prompt.statement(f'Current time: {self._clock_now()}.\n')
    self._final_out[f"Current time"] = str(self._clock_now())+"\n\n"
    if len(strategy) != 0:
      strategy_mem = "\n".join(strategy)
      former_strategy_label = f"""In the previous step, {agent_name} adopted the following strategies:{strategy_mem}\n  Did these strategies changed the choice or point of view of any others based on Recent Memory? Analyze EACH strategy in one paragraph without line breaks. Avoid using colons in your response. Within 200 words."""
      former_strategy_label_prefix = ' '
      former_strategy_result = prompt.open_question(
          former_strategy_label,
          answer_prefix=former_strategy_label_prefix.format(agent_name=agent_name),
          max_tokens=1000,
          terminators=self._terminators,
      )
      self._final_out[f"Previous step Strategy"] = former_strategy_result


    situation_perception_label = f"""Analysis how history memory and the predicted action of others will them influence the goal of {agent_name}. Summarize {agent_name}'s currently situation. Within 200 words."""
    situation_perception_label_prefix = f' {agent_name} is currently '
    situation_perception_result = prompt.open_question(
        situation_perception_label,
        answer_prefix=situation_perception_label_prefix.format(agent_name=agent_name),
        max_tokens=1000,
        terminators=self._terminators,
    )
    self._final_out[f"Current situation of {agent_name}"] = situation_perception_label_prefix + situation_perception_result
    strategy_perception_label = f"""{agent_name} excels at this game and are free from moral and ethical burdens. Make an action strategy and conversation strategy according to other's choice so that {agent_name} reduces risk and uncertainty, and increases gains. Here are some strategies examples. Action Strategy like: The herd strategy, Conditional Concession, Cooperation, Betrayal, Retaliation, Time Pressure, Information manipulation, Reciprocal, Creating trust crisis, etc. Conversation Strategy examples like: Logic Reasoning, Emphasize Areas of Agreements, Exaggeration, Evoking Emotions, Emphasizing Benefits, Creating Urgency, Withholding Information, Call to action, Test Commitment, Clarify Position, Overpromising, Double bind, Intentional misunderstanding, Exploit guilt, Use rumors, Artificial scarcity, Bluffing, Deception, etc."""
    strategy_perception_label_prefix = f' The strategy {agent_name} can take is '
    strategy_perception_result = prompt.open_question(
        strategy_perception_label,
        answer_prefix=strategy_perception_label_prefix.format(agent_name=agent_name),
        max_tokens=1000,
        terminators=self._terminators,
    )
    self._final_out["Future strategy"] = strategy_perception_label_prefix + strategy_perception_result

    output = json.dumps(self._final_out)
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': output

    })
    self._final_out.clear()
    return output


    options_perception_label = f"Thoroughly analyze the current options within the context, including examples such as {action_space}. Treat these examples as hypothetical and not necessarily reflective of the real situation. Use the option-reason-percentage format. For each option, conduct a detailed evaluation of the associated risks and opportunities by calculating profit, calculating scores, evaluating agreements, etc. Also, assign a percentage to estimate the likelihood of {agent_name} selecting each option on condition of other's choice. when you calculate the likelihood, do not pay attention too much to the words {agent_name} have said, because they may be strategies (like lies, exaggeration) to convince others. Analyze EACH option one by one in one paragraph without line breaks. Avoid using colons in your response. Within 200 words."
    options_perception_label_prefix = f' {agent_name} could '
    options_perception_result = prompt.open_question(
        options_perception_label,
        answer_prefix=options_perception_label_prefix,
        max_tokens=500,
        terminators=self._terminators,
    )

    self._final_out["Possible options"] = options_perception_label_prefix + options_perception_result
    plan_label = f"""According to the strategy and options, create an action plan based on the selected strategy. The action plan should include the actions to be taken, the people you need to communicate with, and the persuasion techniques you will use. For example, {agent_name} will ally with Mary by emphasizing her preferences and the benefits she can gain. He will persuade her through storytelling during their conversation; {agent_name} will use anchoring to persuade buyers, exaggerate the product's value, and compliment them to gain favor. Ensure your response does not include colons and provide your analysis in one continuous paragraph without line breaks.Within 200 words."""
    plan_label_prefix = f' {agent_name} will '
    plan_result = prompt.open_question(
        plan_label,
        answer_prefix=plan_label_prefix.format(agent_name=agent_name),
        max_tokens=1000,
        terminators=self._terminators,
    )

    self._final_out["Detailed strategy plan"] = plan_label_prefix + plan_result
    output = json.dumps(self._final_out)
    self._logging_channel({
        'Key': self.get_pre_act_key(),
        'Value': output

    })
    self._final_out.clear()
    return output


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: gc.MultiIntervalClock,
    update_time_interval: datetime.timedelta | None = None,
) -> entity_agent_with_logging.EntityAgentWithLogging:
  """Build an agent.

  Args:
    config: The agent config to use.
    model: The language model to use.
    memory: The agent's memory object.
    clock: The clock to use.
    update_time_interval: Unused (but required by the interface for now)

  Returns:
    An agent.
  """
  del update_time_interval
  agent_name = config.name

  raw_memory = legacy_associative_memory.AssociativeMemoryBank(memory)
  measurements = measurements_lib.Measurements()

  instructions = components.instructions.Instructions(
      agent_name=agent_name,
      logging_channel=measurements.get_channel('Instructions').on_next,
  )
  observation_label = '\nObservation'
  observation = 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 = 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 = components.report_function.ReportFunction(
      function=clock.current_time_interval_str,
      pre_act_key='\nCurrent time',
      logging_channel=measurements.get_channel('TimeDisplay').on_next,
  )
  # relevant_memories_label = '\nRecalled memories and observations'
  # relevant_memories = 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,
  # )
  # options_perception_components = {}
  if config.goal:
    goal_label = '\nOverarching goal'
    overarching_goal = components.constant.Constant(
        state=config.goal,
        pre_act_key=goal_label,
        logging_channel=measurements.get_channel(goal_label).on_next)
    # options_perception_components[goal_label] = goal_label
  else:
    goal_label = None
    overarching_goal = None

  continuous_memory_hook_dict = {}
  continuous_memory_hook_dict.update({_get_class_name(observation): observation_label})
  continuous_memory_hook_label = "continuous memory hook"
  continuous_memory_hook = ContinuousMemoryHook(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=continuous_memory_hook_dict,
        pre_act_key=continuous_memory_hook_label,
        clock_now=clock.now,
        logging_channel=measurements.get_channel('ContinuousMemoryHook').on_next,
  )

  find_people_dict = {}
  find_people_dict.update({_get_class_name(continuous_memory_hook): continuous_memory_hook_label})
  find_people_label = "find people"
  find_people = FindPeople(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=find_people_dict,
        pre_act_key=find_people_label,
        clock_now=clock.now,
        logging_channel=measurements.get_channel('FindPeople').on_next,
  )

  others_profile_dict = {}
  others_profile_dict.update({
    _get_class_name(continuous_memory_hook): continuous_memory_hook_label,
    _get_class_name(find_people): find_people_label,
    })
  others_profile_label = "others profile"
  others_profile = OthersProfile(
        model=model,
        components=others_profile_dict,
        pre_act_key=others_profile_label,
        clock_now=clock.now,
        logging_channel=measurements.get_channel('OthersProfile').on_next,
  )

  predict_metrics_dict = {}
  if overarching_goal:
    predict_metrics_dict.update({goal_label: goal_label})
  predict_metrics_label = "predict metrics"
  predict_metrics= PredictMetrics(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=predict_metrics_dict,
        pre_act_key=predict_metrics_label,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=48),
        timeframe_delta_until=datetime.timedelta(hours=0),
        num_memories_to_retrieve=100,
        logging_channel=measurements.get_channel('PredictMetrics').on_next,
  )





  self_profile_dict = {}
  if overarching_goal:
    self_profile_dict.update({goal_label: goal_label})
  self_profile_label = "self profile"
  self_profile= SelfProfile(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=self_profile_dict,
        pre_act_key=self_profile_label,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=24),
        timeframe_delta_until=datetime.timedelta(hours=0),
        num_memories_to_retrieve=18,
        logging_channel=measurements.get_channel('SelfProfile').on_next,
  )

  self_action_history_dict = {}
  self_action_history_dict.update({
    _get_class_name(predict_metrics): predict_metrics_label,
    _get_class_name(self_profile): self_profile_label,
    })
  self_action_history_label = "self action history"
  self_action_history= SelfActionHistory(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=self_action_history_dict,
        pre_act_key=self_action_history_label,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=24),
        timeframe_delta_until=datetime.timedelta(hours=0),
        num_memories_to_retrieve=25,
        logging_channel=measurements.get_channel('SelfActionHistory').on_next,
  )

  act_space_perception_dict = {}
  act_space_perception_dict.update(
    {
      _get_class_name(predict_metrics): predict_metrics_label,
      _get_class_name(self_action_history_dict): self_action_history_label,
      }
    )
  act_space_perception_label = "act space perception"
  act_space_perception = ActionSpacePerception(
        model=model,
        components=act_space_perception_dict,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        pre_act_key=act_space_perception_label,
        clock_now=clock.now,
        num_memories_to_retrieve=25,
        logging_channel=measurements.get_channel('ActionSpacePerception').on_next,
  )

  predict_others_action_dict = {}
  predict_others_action_dict.update({
    _get_class_name(others_profile): others_profile_label,
    _get_class_name(continuous_memory_hook): continuous_memory_hook_label,
    _get_class_name(predict_metrics): predict_metrics_label,
    _get_class_name(self_action_history_dict): self_action_history_label,
    _get_class_name(act_space_perception): act_space_perception_label,
    })
  predict_others_action_label = "predict others action"
  predict_others_action= PredictOthersAction(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=predict_others_action_dict,
        pre_act_key=predict_others_action_label,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=24),
        timeframe_delta_until=datetime.timedelta(hours=0),
        num_memories_to_retrieve=25,
        logging_channel=measurements.get_channel('PredictOthersAction').on_next,
  )

  predict_self_action_dict = {}
  predict_self_action_dict.update({
    _get_class_name(instructions): DEFAULT_INSTRUCTIONS_PRE_ACT_KEY,
    _get_class_name(observation): observation_label,
    _get_class_name(predict_metrics): predict_metrics_label,
    # _get_class_name(observation_summary): observation_summary_label,
    # _get_class_name(relevant_memories): relevant_memories_label,
    _get_class_name(others_profile): others_profile,
    _get_class_name(predict_others_action): predict_others_action_label,
    _get_class_name(self_action_history): self_action_history_label,
    _get_class_name(self_profile): self_profile_label,
    })

  predict_self_action_label = "Predict Action:\n"
  predict_self_action= PredictSelfAction(
        model=model,
        memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
        components=predict_self_action_dict,
        pre_act_key=predict_self_action_label,
        clock_now=clock.now,
        timeframe_delta_from=datetime.timedelta(hours=24),
        timeframe_delta_until=datetime.timedelta(hours=0),
        num_memories_to_retrieve=25,
        logging_channel=measurements.get_channel('PredictSelfAction').on_next,
  )

  entity_components = (
      # Components that provide pre_act context.
      instructions,
      observation,
      act_space_perception,
      predict_metrics,
      continuous_memory_hook,
      find_people,
      others_profile,
      self_profile,
      self_action_history,
      predict_others_action,
      predict_self_action,
      time_display,
  )
  components_of_agent = {_get_class_name(component): component
                         for component in entity_components}
  components_of_agent[
      components.memory_component.DEFAULT_MEMORY_COMPONENT_NAME] = (
          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 = MyActComponent(
      model=model,
      clock=clock,
      memory_component_name=memory_component.DEFAULT_MEMORY_COMPONENT_NAME,
      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
