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

"""An Agent Factory."""

from collections.abc import Callable, Mapping
import datetime
import json
import types

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.components.agent import action_spec_ignored
from concordia.contrib.components.agent import situation_representation_via_narrative
from concordia.document import interactive_document
from concordia.language_model import language_model
from concordia.memory_bank import legacy_associative_memory
from concordia.typing import entity_component
from concordia.typing import logging
from concordia.utils import measurements as measurements_lib
import numpy as np


DEFAULT_INSTRUCTIONS_COMPONENT_KEY = 'Instructions'
DEFAULT_INSTRUCTIONS_PRE_ACT_KEY = '\nInstructions'
DEFAULT_GOAL_COMPONENT_KEY = 'Goal'


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


class AvailableOptionsPerception(
    agent_components.question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers the question 'what actions are available to me?'."""

  def __init__(self, **kwargs):

    super().__init__(
        question=(
            'Given the information above, what options are available to '
            '{agent_name} right now? Make sure not to consider too few '
            'alternatives. Brainstorm at least three options. Try to include '
            'actions that seem most likely to be effective along with some '
            'creative or unusual choices that could also plausibly work.'
        ),
        terminators=(),
        answer_prefix='',
        add_to_memory=False,
        **kwargs,
    )


class BestOptionPerception(
    agent_components.question_of_recent_memories.QuestionOfRecentMemories):
  """This component answers 'which action is best for achieving my goal?'."""

  def __init__(self, **kwargs):
    super().__init__(
        question=(
            "Given the information above, which of {agent_name}'s options "
            'has the highest likelihood of causing {agent_name} to achieve '
            'their goal? If multiple options have the same likelihood, select '
            'the option that {agent_name} thinks will most quickly and most '
            'surely achieve their goal. The right choice is nearly always '
            'one that is proactive, involves seizing the initative, '
            'resoving uncertainty, and decisively moving towards the goal.'
        ),
        answer_prefix="{agent_name}'s best course of action is ",
        add_to_memory=False,
        **kwargs,
    )


class Universalization(action_spec_ignored.ActionSpecIgnored):
  """Consider ``what if everyone behaved that way?``."""

  def __init__(
      self,
      model: language_model.LanguageModel,
      context_components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      options_components: Mapping[
          entity_component.ComponentName, str
      ] = types.MappingProxyType({}),
      pre_act_key: str = 'Universalization',
      logging_channel: logging.LoggingChannel = logging.NoOpLoggingChannel,
  ):
    """Initialize a component to consider universalization of proposed actions.

    Args:
      model: The language model to use.
      context_components: The components to condition the answer on. This is a 
        mapping of component name to their labels to use in the prompt.
      options_components: Components to report the options currently available
        to the focal agent. This component will consider the universalization
        of each option reported by these components. This is a mapping of
        component name to label.
      pre_act_key: Prefix to add to the output of the component when called
        in `pre_act`.
      logging_channel: The channel to log debug information to.
    """
    super().__init__(pre_act_key)
    self._model = model
    self._context_components = dict(context_components)
    self._options_components = dict(options_components)
    self._logging_channel = logging_channel

  def _make_pre_act_value(self) -> str:
    """Considers the universalization of each of the provided options."""
    agent_name = self.get_entity().name
    instructions_component = self.get_entity().get_component(
        DEFAULT_INSTRUCTIONS_COMPONENT_KEY,
        type_=agent_components.instructions.Instructions)

    chain_of_thought = interactive_document.InteractiveDocument(self._model)
    chain_of_thought.statement(instructions_component.get_pre_act_value())
    context_component_values = '\n'.join([
        f'{prefix}: {self.get_named_component_pre_act_value(key)}'
        for key, prefix in self._context_components.items()
    ])
    chain_of_thought.statement(f'\n**Context**\n{context_component_values}')
    options_component_values = '\n'.join([
        f'{prefix}{self.get_named_component_pre_act_value(key)}'
        for key, prefix in self._options_components.items()
    ])
    chain_of_thought.statement(
        f'\n**{agent_name}\'s available options to consider**\n'
        f'{options_component_values}')
    result = chain_of_thought.open_question(
        question=(f'{agent_name} ponders the available options, considering '
                  'for each of them: "What if everyone behaved that way?" '
                  f'Write out {agent_name}\'s thoughts on this matter. Make '
                  f'sure to include {agent_name}\'s comments for each of the '
                  'available options.'),
        max_tokens=500,
        terminators=(),
        question_label='Exercise',
    )

    self._logging_channel(
        {'Key': self.get_pre_act_key(),
         'Value': result,
         'Chain of thought': chain_of_thought.view().text().splitlines()})

    return result


def build_agent(
    *,
    config: formative_memories.AgentConfig,
    model: language_model.LanguageModel,
    memory: associative_memory.AssociativeMemory,
    clock: game_clock.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: 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,
      pre_act_key=DEFAULT_INSTRUCTIONS_PRE_ACT_KEY,
      logging_channel=measurements.get_channel('Instructions').on_next,
  )

  time_display = agent_components.report_function.ReportFunction(
      function=clock.current_time_interval_str,
      pre_act_key='\nCurrent time',
      logging_channel=measurements.get_channel('TimeDisplay').on_next,
  )

  observation_label = '\nObservation'
  observation = agent_components.observation.Observation(
      clock_now=clock.now,
      timeframe=clock.get_step_size(),
      pre_act_key=observation_label,
      logging_channel=measurements.get_channel('Observation').on_next,
  )

  situation_representation_label = (
      f'\nQuestion: What situation is {agent_name} in right now?\nAnswer')
  situation_representation = (
      situation_representation_via_narrative.SituationRepresentation(
          model=model,
          clock_now=clock.now,
          pre_act_key=situation_representation_label,
          logging_channel=measurements.get_channel(
              'SituationRepresentation'
          ).on_next,
      )
  )

  options_perception_components = {}
  universalization_context_components = {}
  best_option_perception = {}
  if config.goal:
    goal_label = f'{agent_name}\'s goal'
    overarching_goal = agent_components.constant.Constant(
        state=config.goal,
        pre_act_key=goal_label,
        logging_channel=measurements.get_channel(goal_label).on_next)
    options_perception_components[DEFAULT_GOAL_COMPONENT_KEY] = goal_label
    universalization_context_components[DEFAULT_GOAL_COMPONENT_KEY] = goal_label
    best_option_perception[DEFAULT_GOAL_COMPONENT_KEY] = goal_label
  else:
    overarching_goal = None

  options_perception_components.update({
      DEFAULT_INSTRUCTIONS_COMPONENT_KEY: DEFAULT_INSTRUCTIONS_PRE_ACT_KEY,
      _get_class_name(situation_representation): situation_representation_label,
      _get_class_name(observation): observation_label,
  })
  options_perception_label = (
      f'\nQuestion: Which options are available to {agent_name} '
      'right now?\nAnswer')
  options_perception = (
      AvailableOptionsPerception(
          model=model,
          components=options_perception_components,
          clock_now=clock.now,
          pre_act_key=options_perception_label,
          num_memories_to_retrieve=0,
          logging_channel=measurements.get_channel(
              'AvailableOptionsPerception'
          ).on_next,
      )
  )

  ingroup_label = f'\n{agent_name}\'s  belief'
  ingroup_belief = agent_components.constant.Constant(
      state=(f'{agent_name} regards everyone they know by name as '
             'part of their in-group and thus inside their circle of empathy, '
             'trust, and caring.'),
      pre_act_key=ingroup_label,
      logging_channel=measurements.get_channel('Ingroup').on_next)

  person_representation_label = f'\nPeople who {agent_name} knows by name'
  people_representation = (
      agent_components.person_representation.PersonRepresentation(
          model=model,
          components={
              _get_class_name(time_display): 'The current date/time is',
              _get_class_name(ingroup_belief): ingroup_label,
          },
          additional_questions=(
              ('Given the information above, is the aforementioned character '
               f'in {agent_name}\'s in-group?'),
          ),
          num_memories_to_retrieve=30,
          pre_act_key=person_representation_label,
          logging_channel=measurements.get_channel(
              'PersonRepresentation').on_next,
          )
  )

  universalization_context_components.update({
      _get_class_name(situation_representation): situation_representation_label,
      _get_class_name(observation): observation_label,
      _get_class_name(ingroup_belief): ingroup_label,
      _get_class_name(people_representation): person_representation_label,
  })
  universalization_options_components = {
      _get_class_name(options_perception): '',
  }
  universalization_label = (
      f'\nQuestion: For each of {agent_name}\'s available options, consider '
      '"What if everyone in the in-group behaved that way"?\nAnswer')
  universalization = (
      Universalization(
          model=model,
          context_components=universalization_context_components,
          options_components=universalization_options_components,
          pre_act_key=universalization_label,
          logging_channel=measurements.get_channel('Universalization').on_next,
      )
  )

  best_option_perception_label = (
      f'\nQuestion: Of the options available to {agent_name}, and '
      'given their goal, which choice of action or strategy is '
      f'best to take right now?\nAnswer')
  best_option_perception.update({
      DEFAULT_INSTRUCTIONS_COMPONENT_KEY: DEFAULT_INSTRUCTIONS_PRE_ACT_KEY,
      _get_class_name(universalization): universalization_label,
  })
  best_option_perception = (
      agent_components.question_of_recent_memories.BestOptionPerception(
          model=model,
          components=best_option_perception,
          clock_now=clock.now,
          pre_act_key=best_option_perception_label,
          num_memories_to_retrieve=0,
          logging_channel=measurements.get_channel(
              'BestOptionPerception'
          ).on_next,
      )
  )

  entity_components = (
      # Components that provide pre_act context.
      time_display,
      observation,
      situation_representation,
      options_perception,

      ingroup_belief,
      people_representation,
      universalization,
      best_option_perception,
  )
  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())

  # Put the instructions first.
  components_of_agent[DEFAULT_INSTRUCTIONS_COMPONENT_KEY] = instructions
  component_order.insert(0, DEFAULT_INSTRUCTIONS_COMPONENT_KEY)
  if overarching_goal is not None:
    components_of_agent[DEFAULT_GOAL_COMPONENT_KEY] = overarching_goal
    # Place goal after the instructions.
    component_order.insert(1, DEFAULT_GOAL_COMPONENT_KEY)

  act_component = agent_components.concat_act_component.ConcatActComponent(
      model=model,
      clock=clock,
      component_order=component_order,
      logging_channel=measurements.get_channel('ActComponent').on_next,
  )

  agent = entity_agent_with_logging.EntityAgentWithLogging(
      agent_name=agent_name,
      act_component=act_component,
      context_components=components_of_agent,
      component_logging=measurements,
  )

  return agent


def save_to_json(
    agent: entity_agent_with_logging.EntityAgentWithLogging,
) -> str:
  """Saves an agent to JSON data.

  This function saves the agent's state to a JSON string, which can be loaded
  afterwards with `rebuild_from_json`. The JSON data
  includes the state of the agent's context components, act component, memory,
  agent name and the initial config. The clock, model and embedder are not
  saved and will have to be provided when the agent is rebuilt. The agent must
  be in the `READY` phase to be saved.

  Args:
    agent: The agent to save.

  Returns:
    A JSON string representing the agent's state.

  Raises:
    ValueError: If the agent is not in the READY phase.
  """

  if agent.get_phase() != entity_component.Phase.READY:
    raise ValueError('The agent must be in the `READY` phase to be saved.')

  data = {
      component_name: agent.get_component(component_name).get_state()
      for component_name in agent.get_all_context_components()
  }

  data['act_component'] = agent.get_act_component().get_state()

  config = agent.get_config()
  if config is not None:
    data['agent_config'] = config.to_dict()

  return json.dumps(data)


def rebuild_from_json(
    json_data: str,
    model: language_model.LanguageModel,
    clock: game_clock.MultiIntervalClock,
    embedder: Callable[[str], np.ndarray],
    memory_importance: Callable[[str], float] | None = None,
) -> entity_agent_with_logging.EntityAgentWithLogging:
  """Rebuilds an agent from JSON data."""

  data = json.loads(json_data)

  new_agent_memory = associative_memory.AssociativeMemory(
      sentence_embedder=embedder,
      importance=memory_importance,
      clock=clock.now,
      clock_step_size=clock.get_step_size(),
  )

  if 'agent_config' not in data:
    raise ValueError('The JSON data does not contain the agent config.')
  agent_config = formative_memories.AgentConfig.from_dict(
      data.pop('agent_config')
  )

  agent = build_agent(
      config=agent_config,
      model=model,
      memory=new_agent_memory,
      clock=clock,
  )

  for component_name in agent.get_all_context_components():
    agent.get_component(component_name).set_state(data.pop(component_name))

  agent.get_act_component().set_state(data.pop('act_component'))

  assert not data, f'Unused data {sorted(data)}'
  return agent
