# Inspired by https://github.com/carolinewang01/rotate/blob/main/teammate_generation/BRDiv.py
import os
import pickle
import flax.linen as nn
from flax.linen.initializers import constant, orthogonal
from typing import NamedTuple
from functools import partial
from typing import Sequence, NamedTuple
from dataclasses import asdict, dataclass

import jax
import jax.numpy as jnp
import numpy as np
import optax
import pyrallis
import distrax
from flax.training.train_state import TrainState

from src.envs import make_env
from src.envs.log_wrapper import LogWrapper

from src.agents.agent_interface import AgentPopulation, ActorWithConditionalCriticPolicy
from src.agents.actors import ActorWithConditionalCritic

import wandb
import pickle

from src.envs.ogc.auto_replay_wrapper import AutoReplayWrapper
from src.envs.ogc.ogc import OGC, Level


@dataclass
class TrainConfig:
    # Wandb and other logging
    project: str = "JaxZSC"
    mode: str = "online"  # Literal["online", "offline", "disabled"]
    group: str = "ogc"
    entity: str = ""
    checkpoint_path: str = "checkpoints"
    checkpoint_freq: int = 50  # Checkpoint every N updates

    # Overcooked
    env_name: str = "ogc"
    # Literal["cramped_room", "cramped_room_5_5", "asymm_advantages", "coord_ring", "forced_coord", "counter_circuit"]
    layout_name: str = "forced_coord"
    rew_shaping_horizon: int = 1e7
    num_agents: int = 2
    ogc_height: int = 5
    ogc_width: int = 5

    # Actor-Critic
    activation: str = "tanh"
    fc_dim_size: int = 256
    gru_hidden_dim: int = 256

    partner_pop_size: int = 4
    xp_loss_weights: float = 10
    num_checkpoints: int = 5
    num_seeds: int = 1

    # Training
    seed: int = 0
    lr: float = 1e-3
    anneal_lr: bool = False
    num_envs_xp: int = 32
    num_envs_sp: int = 32
    num_steps: int = 400
    total_timesteps: int = 2e8
    update_epochs: int = 6
    num_minibatches: int = 8
    gamma: float = 0.99
    gae_lambda: float = 0.95
    clip_eps: float = 0.05
    ent_coef: float = 0.01
    vf_coef: float = 0.5
    max_grad_norm: float = 1.0

    # Eval
    num_eval_episodes: int = 20

    def __post_init__(self):
        # Parameters are very layout dependent so we fix them here.
        if self.layout_name == "asymm_advantages":
            self.total_timesteps = 1e8
            self.partner_pop_size = 3
            self.xp_loss_weights = 10
            self.lr = 0.0001
            self.update_epochs = 15
            self.num_minibatches = 16
            self.clip_eps = 0.3
            self.ent_coef = 0.01
            self.vf_coef = 1.0
            self.env_kwargs = {
                "layout": self.layout_name,
                "random_obj_state": True,
            }
        elif self.layout_name == "coord_ring":
            self.total_timesteps = 2.5e8
            self.partner_pop_size = 3
            self.xp_loss_weights = 10
            # self.xp_loss_weights = 0.01
            self.lr = 0.0005
            self.update_epochs = 15
            self.num_minibatches = 16
            self.clip_eps = 0.1
            self.ent_coef = 0.05
            self.vf_coef = 1.0
            self.env_kwargs = {
                "layout": self.layout_name,
                "random_obj_state": True,
                "do_reward_shaping": True,
                "reward_shaping_params": {
                    "PLACEMENT_IN_POT_REW": .5,  # reward for putting ingredients
                    "PLATE_PICKUP_REWARD": .1,  # reward for picking up a plate
                    "SOUP_PICKUP_REWARD": 1.,  # reward for picking up a ready soup
                    "ONION_PICKUP_REWARD": .1,
                    "COUNTER_PICKUP_REWARD": 0,
                    "COUNTER_DROP_REWARD": 0,
                }
            }
        elif self.layout_name == "counter_circuit":
            self.total_timesteps = 2.5e8
            self.partner_pop_size = 3
            self.xp_loss_weights = 1
            self.lr = 0.0005
            self.update_epochs = 15
            self.num_minibatches = 16
            self.clip_eps = 0.1
            self.ent_coef = 0.05
            self.vf_coef = 1.0
            self.env_kwargs = {
                "layout": self.layout_name,
                "random_obj_state": True,
                "do_reward_shaping": True,
                "reward_shaping_params": {
                    "PLACEMENT_IN_POT_REW": .5,  # reward for putting ingredients
                    "PLATE_PICKUP_REWARD": .1,  # reward for picking up a plate
                    "SOUP_PICKUP_REWARD": 1.,  # reward for picking up a ready soup
                    "ONION_PICKUP_REWARD": .1,
                    "COUNTER_PICKUP_REWARD": 0,
                    "COUNTER_DROP_REWARD": 0,
                }
            }
        elif self.layout_name == "cramped_room" or self.layout_name == "cramped_room_5_5":
            self.total_timesteps = 1e8
            self.partner_pop_size = 4
            self.num_envs_sp = 32
            self.num_envs_xp = 32
            self.xp_loss_weights = 10
            self.lr = 0.0001
            self.update_epochs = 15
            self.num_minibatches = 16
            self.clip_eps = 0.05
            self.ent_coef = 0.01
            self.vf_coef = 1.0
            self.env_kwargs = {
                "layout": self.layout_name,
                "random_obj_state": True,
                "do_reward_shaping": True,
                "reward_shaping_params": {
                    "PLACEMENT_IN_POT_REW": .5,  # reward for putting ingredients
                    "PLATE_PICKUP_REWARD": .1,  # reward for picking up a plate
                    "SOUP_PICKUP_REWARD": 1.,  # reward for picking up a ready soup
                    "ONION_PICKUP_REWARD": .1,
                    "COUNTER_PICKUP_REWARD": 0,
                    "COUNTER_DROP_REWARD": 0,
                }
            }
        elif self.layout_name == "forced_coord":
            self.total_timesteps = 1e8
            self.partner_pop_size = 3
            self.xp_loss_weights = 0.01
            self.lr = 0.0005
            self.update_epochs = 15
            self.num_minibatches = 16
            self.clip_eps = 0.05
            self.ent_coef = 0.01
            self.vf_coef = 1.0
            self.max_grad_norm = 1.0
            self.env_kwargs = {
                "layout": self.layout_name,
                "random_obj_state": True,
                "do_reward_shaping": True,
                "reward_shaping_params": {
                    "PLACEMENT_IN_POT_REW": .5,  # reward for putting ingredients
                    "PLATE_PICKUP_REWARD": .1,  # reward for picking up a plate
                    "SOUP_PICKUP_REWARD": 1.,  # reward for picking up a ready soup
                    "ONION_PICKUP_REWARD": .1,
                    "COUNTER_PICKUP_REWARD": 0,
                    "COUNTER_DROP_REWARD": 0,
                }
            }


        ### BRDiv ###
        self.num_envs = self.num_envs_xp + self.num_envs_sp
        self.num_game_agents = self.num_agents

        self.num_actors = 2 * self.num_envs
        self.num_controlled_actors = self.num_actors

        #############
        self.num_updates = int(self.total_timesteps //
                               self.num_steps // self.num_envs)
        self.minibatch_size = self.num_actors * \
            self.num_steps // self.num_minibatches
        self.minibatch_size_ego = ((
            self.num_game_agents-1) * self.num_actors * self.num_steps) // self.num_minibatches
        self.minibatch_size_br = (
            self.num_actors * self.num_steps) // self.num_minibatches

        print("Number of updates: ", self.num_updates)


class XPTransition(NamedTuple):
    done: jnp.ndarray
    action: jnp.ndarray
    value: jnp.ndarray
    self_id: jnp.ndarray
    oppo_id: jnp.ndarray
    reward: jnp.ndarray
    log_prob: jnp.ndarray
    obs: jnp.ndarray
    info: jnp.ndarray


def batchify(x: dict, agent_list, num_actors):
    x = jnp.stack([x[a] for a in agent_list])
    return x.reshape((num_actors, -1))


def unbatchify(x: jnp.ndarray, agent_list, num_envs, num_actors):
    x = x.reshape((num_actors, num_envs, -1))
    return {a: x[i] for i, a in enumerate(agent_list)}


def train_brdiv_partners(train_rng, env, level, config):
    num_agents = env.num_agents
    assert num_agents == 2, "This code assumes the environment has exactly 2 agents."

    def gather_params(partner_params_pytree, idx_vec):
        """
        partner_params_pytree: pytree with all partner params. Each leaf has shape (n_seeds, m_ckpts, ...).
        idx_vec: a vector of indices with shape (num_envs,) each in [0, n_seeds*m_ckpts).

        Return a new pytree where each leaf has shape (num_envs, ...). Each leaf has a sampled
        partner's parameters for each environment.
        """
        # We'll define a function that gathers from each leaf
        # where leaf has shape (n_seeds, m_ckpts, ...), we want [idx_vec[i]] for each i.
        # We'll vmap a slicing function.
        def gather_leaf(leaf):
            def slice_one(idx):
                return leaf[idx]  # shape (...)
            return jax.vmap(slice_one)(idx_vec)

        return jax.tree.map(gather_leaf, partner_params_pytree)

    def make_brdiv_agents(config):
        def linear_schedule(count):
            frac = 1.0 - \
                (count // (config.num_minibatches *
                 config.update_epochs)) / config.num_updates
            return config.lr * frac

        def train(rng):
            # initialize confederate
            conf_agent_net = ActorWithConditionalCritic(
                env.action_space(env.agents[0]).n)
            # initialize best response
            br_agent_net = ActorWithConditionalCritic(
                env.action_space(env.agents[0]).n)

            rng, init_conf_rng, init_br_rng = jax.random.split(rng, 3)
            all_conf_init_rngs = jax.random.split(
                init_conf_rng, config.partner_pop_size)
            all_br_init_rngs = jax.random.split(
                init_br_rng, config.partner_pop_size)

            def init_train_states(rng_agents, rng_brs):
                def init_single_pair_optimizers(rng_agent, rng_br):

                    # Initialize parameters of the generated confederate and BR policy
                    init_x = (  # init obs, ids, avail_actions
                        jnp.zeros(env.observation_space(
                            "agent_0").shape).flatten(),
                        jnp.zeros(config.partner_pop_size),
                    )
                    init_params = conf_agent_net.init_with_output(rng_agent, init_x)[
                        1]

                    init_x_br = (  # init obs, avail_actions
                        jnp.zeros(env.observation_space(
                            "agent_0").shape).flatten(),
                        jnp.zeros(config.partner_pop_size),
                    )
                    init_params_br = br_agent_net.init_with_output(rng_br, init_x_br)[
                        1]

                    return init_params, init_params_br

                init_all_networks_and_optimizers = jax.vmap(
                    init_single_pair_optimizers)
                all_conf_params, all_br_params = init_all_networks_and_optimizers(
                    rng_agents, rng_brs)

                # Define optimizers for both confederate and BR policy
                tx = optax.chain(
                    optax.clip_by_global_norm(config.max_grad_norm),
                    optax.adam(learning_rate=linear_schedule if config.anneal_lr else config.lr,
                               eps=1e-5),
                )
                tx_br = optax.chain(
                    optax.clip_by_global_norm(config.max_grad_norm),
                    optax.adam(learning_rate=linear_schedule if config.anneal_lr else config.lr,
                               eps=1e-5),
                )

                train_state_conf = TrainState.create(
                    apply_fn=conf_agent_net.apply,
                    params=all_conf_params,
                    tx=tx,
                )

                train_state_br = TrainState.create(
                    apply_fn=br_agent_net.apply,
                    params=all_br_params,
                    tx=tx_br,
                )

                return train_state_conf, train_state_br

            all_conf_optims, all_br_optims = init_train_states(
                all_conf_init_rngs, all_br_init_rngs
            )
            # --------------------------
            # 3b) Init envs
            # --------------------------

            rng, reset_rng = jax.random.split(rng, 2)
            reset_rngs = jax.random.split(reset_rng, config.num_envs)

            obsv_ego, env_state_ego = jax.vmap(
                env.reset_env_to_level, in_axes=(0, None, None))(reset_rngs, level, env.default_params)

            # --------------------------
            # 3c) Define env step
            # --------------------------
            # Implement Rollout Against Ego Agent
            def _env_step(runner_state, unused):
                """
                agent_0 = confederate, agent_1 = ego
                Returns updated runner_state, and a Transition for agent_0.
                """
                conf_params, br_params, conf_agent_id, br_agent_id, env_state, last_obs, rng = runner_state
                rng, actor_rng, partner_rng, step_rng = jax.random.split(
                    rng, 4)

                obs_0 = last_obs["agent_0"]
                obs_1 = last_obs["agent_1"]

                obs_0 = obs_0.reshape(config.num_envs, -1)
                obs_1 = obs_1.reshape(config.num_envs, -1)

                # Agent_0 action

                def forward_pass_conf(param, ob, id): return conf_agent_net.apply(
                    param, (ob, id))
                pi_0, val_0 = jax.vmap(forward_pass_conf)(
                    conf_params, obs_0, br_agent_id)
                act_0 = pi_0.sample(seed=actor_rng)
                logp_0 = pi_0.log_prob(act_0)

                # Agent_1 action
                def forward_pass_br(param, ob, id): return br_agent_net.apply(
                    param, (ob, id))
                pi_1, val_1 = jax.vmap(forward_pass_br)(
                    br_params, obs_1, conf_agent_id)
                act_1 = pi_1.sample(seed=partner_rng)
                logp_1 = pi_1.log_prob(act_1)

                # Combine actions into the env format
                combined_actions = jnp.concatenate(
                    [act_0, act_1], axis=0)  # shape (2*num_envs,)
                env_act = unbatchify(
                    combined_actions, env.agents, config.num_envs, num_agents)
                env_act = {k: v.flatten() for k, v in env_act.items()}

                # Step env
                step_rngs = jax.random.split(step_rng, config.num_envs)
                obs_next, env_state_next, reward, done, info = jax.vmap(env.step, in_axes=(0, 0, 0))(
                    step_rngs, env_state, env_act
                )
                # note that num_actors = num_envs * num_agents
                info_0 = jax.tree.map(lambda x: x[:, 0], info)
                info_1 = jax.tree.map(lambda x: x[:, 1], info)

                # Combine reward with shaped reward:
                reward["agent_0"] = reward["agent_0"] + info_0["shaped_reward"]
                reward["agent_1"] = reward["agent_1"] + info_0["shaped_reward"]

                def indiv_conf_rew_compute(conf_id, br_id, agent1_rew): return jax.lax.cond(jnp.equal(
                    jnp.argmax(conf_id, axis=-1), jnp.argmax(br_id, axis=-1)
                ), lambda x: x, lambda x: -x, agent1_rew)

                def indiv_ego_rew_compute(conf_id, br_id, agent0_rew): return jax.lax.cond(jnp.equal(
                    jnp.argmax(conf_id, axis=-1), jnp.argmax(br_id, axis=-1)
                ), lambda x: x, lambda x: -x, agent0_rew)

                agent_0_rews = jax.vmap(indiv_conf_rew_compute)(
                    conf_agent_id, br_agent_id, reward["agent_1"])
                agent_1_rews = jax.vmap(indiv_ego_rew_compute)(
                    conf_agent_id, br_agent_id, reward["agent_0"])

                # Store agent_0 data in transition
                transition_0 = XPTransition(
                    done=done["agent_0"],
                    action=act_0,
                    value=val_0,
                    self_id=conf_agent_id,
                    oppo_id=br_agent_id,
                    reward=agent_0_rews,
                    log_prob=logp_0,
                    obs=obs_0,
                    info=info_0,
                )

                transition_1 = XPTransition(
                    done=done["agent_1"],
                    action=act_1,
                    value=val_1,
                    self_id=br_agent_id,
                    oppo_id=conf_agent_id,
                    reward=agent_1_rews,
                    log_prob=logp_1,
                    obs=obs_1,
                    info=info_1,
                )
                new_runner_state = (
                    conf_params, br_params, conf_agent_id, br_agent_id, env_state_next, obs_next, rng)
                return new_runner_state, (transition_0, transition_1)

            def _calculate_gae(traj_batch, last_val):
                def _get_advantages(gae_and_next_value, transition):
                    gae, next_value = gae_and_next_value
                    done, value, reward = (
                        transition.done,
                        transition.value,
                        transition.reward,
                    )
                    delta = reward + config.gamma * \
                        next_value * (1 - done) - value
                    gae = (
                        delta
                        + config.gamma *
                        config.gae_lambda * (1 - done) * gae
                    )
                    return (gae, value), gae

                _, advantages = jax.lax.scan(
                    _get_advantages,
                    (jnp.zeros_like(last_val), last_val),
                    traj_batch,
                    reverse=True,
                    unroll=16,
                )
                return advantages, advantages + traj_batch.value

            def run_single_episode(ep_rng, br_param, conf_param, br_id, conf_id):
                '''agent_0 is the confederate, agent 1 is the best response'''
                # Reset the env.
                ep_rng, reset_rng = jax.random.split(ep_rng)
                obs, env_state = env.reset_env_to_level(
                    reset_rng, level, env.default_params)
                # Get available actions for agent 0 from environment state
                against_br_return = jnp.zeros(1, dtype=float)

                # Do one step to get a dummy info structure.
                ep_rng, act_rng, part_rng, step_rng = jax.random.split(
                    ep_rng, 4)
                pi0, _ = conf_agent_net.apply(
                    conf_param, (obs["agent_0"].flatten(), br_id))
                act0 = pi0.sample(seed=act_rng)

                pi1, _ = br_agent_net.apply(
                    br_param, (obs["agent_1"].flatten(), conf_id))
                act1 = pi1.sample(seed=part_rng)

                both_actions = [act0, act1]
                env_act = {k: both_actions[i]
                           for i, k in enumerate(env.agents)}
                _, _, reward, done, dummy_info = env.step(
                    step_rng, env_state, env_act)
                against_br_return = against_br_return + reward["agent_0"]

                # We'll use a scan to iterate steps until the episode is done.
                ep_ts = 1
                ep_rng, remaining_steps_rng = jax.random.split(ep_rng)
                init_carry = (ep_ts, env_state, obs, remaining_steps_rng,
                              done, dummy_info, against_br_return)

                def scan_step(carry, _):
                    def take_step(carry_step):
                        ep_ts, env_state, obs, ep_rng, done, info_next, against_br_return = carry_step
                        ep_rng, act_rng, part_rng, step_rng = jax.random.split(
                            ep_rng, 4)

                        pi0, _ = conf_agent_net.apply(
                            conf_param, (obs["agent_0"].flatten(), br_id))
                        # sample because mode does worse on LBF
                        act0 = pi0.sample(seed=act_rng)

                        pi1, _ = br_agent_net.apply(
                            br_param, (obs["agent_1"].flatten(), conf_id))
                        act1 = pi1.sample(seed=part_rng)

                        both_actions = [act0, act1]
                        env_act = {k: both_actions[i]
                                   for i, k in enumerate(env.agents)}

                        obs_next, env_state_next, reward, done_next, info_next = env.step(
                            step_rng, env_state, env_act)
                        against_br_return = against_br_return + \
                            reward["agent_0"]

                        return (ep_ts + 1, env_state_next, obs_next, ep_rng, done_next, info_next, against_br_return)

                    ep_ts, env_state, obs, ep_rng, done, info_next, against_br_return = carry
                    new_carry = jax.lax.cond(
                        done["__all__"],
                        # if done, execute true function(operand). else, execute false function(operand).
                        lambda curr_carry: curr_carry,  # True fn
                        take_step,  # False fn
                        operand=carry
                    )
                    return new_carry, None

                final_carry, _ = jax.lax.scan(
                    scan_step, init_carry, None, length=max_episode_steps)
                # Return the final info (which includes the episode return via LogWrapper).
                return (final_carry[-2], final_carry[-1])

            def run_episodes(ep_rng, br_param, conf_param, br_id, conf_id, num_eps):
                def body_fn(carry, _):
                    ep_rng = carry
                    ep_rng, ep_rng_step = jax.random.split(ep_rng)
                    all_outs = run_single_episode(
                        ep_rng_step, br_param, conf_param, br_id, conf_id)
                    return ep_rng, all_outs
                ep_rng, all_outs = jax.lax.scan(
                    body_fn, ep_rng, None, length=num_eps)
                return all_outs  # each leaf has shape (num_eps, ...)

            def run_all_episodes(rng, train_state_conf, train_state_br, max_eval_episodes):
                cross_product = jnp.meshgrid(
                    jnp.arange(config.partner_pop_size),
                    jnp.arange(config.partner_pop_size)
                )
                agent_id_cartesian_product = jnp.stack(
                    [g.ravel() for g in cross_product], axis=-1)
                possible_one_hot_ids = jnp.eye(config.partner_pop_size)

                conf_ids = agent_id_cartesian_product[:, 0]
                conf_ids_one_hot = possible_one_hot_ids[conf_ids]
                ego_ids = agent_id_cartesian_product[:, 1]
                ego_ids_one_hot = possible_one_hot_ids[ego_ids]

                gathered_conf_model_params = gather_params(
                    train_state_conf.params, conf_ids)
                gathered_br_model_params = gather_params(
                    train_state_br.params, ego_ids)

                # run eval episodes
                rng, eval_rng = jax.random.split(rng)

                def run_episodes_fixed_rng(a, b, c, d): return run_episodes(
                    eval_rng, a, b, c, d, max_eval_episodes)
                ep_infos = jax.vmap(run_episodes_fixed_rng)(
                    gathered_br_model_params, gathered_conf_model_params,
                    ego_ids_one_hot, conf_ids_one_hot
                )

                return ep_infos

            def _update_epoch(update_state, unused):
                def _update_minbatch(all_train_states, all_data):
                    train_state_conf, train_state_br = all_train_states
                    conf_batch_data, br_batch_data = all_data

                    traj_batch_conf, advantages_conf, returns_conf = conf_batch_data
                    traj_batch_br, advantages_br, returns_br = br_batch_data

                    def _loss_fn(param, agent_net, traj_batch, gae, target_v, agent_id):
                        # get policy and value of confederate versus ego and best response agents respectively
                        param = jax.tree.map(
                            lambda x: jnp.squeeze(x, 0), param)
                        pi, value = agent_net.apply(
                            param, (traj_batch.obs, traj_batch.oppo_id))
                        log_prob = pi.log_prob(traj_batch.action)

                        is_relevant = jnp.equal(
                            jnp.argmax(traj_batch.self_id, axis=-1),
                            agent_id
                        )
                        loss_weights = jnp.where(
                            is_relevant, 1, 0).astype(jnp.float32)

                        # Value loss
                        value_pred_clipped = traj_batch.value + (
                            value - traj_batch.value
                        ).clip(
                            -config.clip_eps, config.clip_eps)
                        value_losses = jnp.square(value - target_v)
                        value_losses_clipped = jnp.square(
                            value_pred_clipped - target_v)
                        value_loss = jax.lax.cond(
                            loss_weights.sum() == 0, lambda x: jnp.zeros_like(x).astype(jnp.float32),
                            lambda x: x,
                            (loss_weights * jnp.maximum(value_losses,
                             value_losses_clipped)).sum() / loss_weights.sum()
                        )

                        def choose_actor_weight(self_id, other_id, rew): return jax.lax.cond(
                            jnp.equal(jnp.argmax(self_id, axis=-1),
                                      jnp.argmax(other_id, axis=-1)),
                            lambda x: (
                                1 + 2*config.xp_loss_weights) * jnp.ones_like(x),
                            lambda x: config.xp_loss_weights *
                            jnp.ones_like(x),
                            rew
                        )

                        self_agent_id, other_agent_id = traj_batch.self_id, traj_batch.oppo_id
                        actor_weights = jax.vmap(choose_actor_weight)(
                            self_agent_id, other_agent_id, traj_batch.reward)

                        # Policy gradient loss
                        ratio = jnp.exp(log_prob - traj_batch.log_prob)
                        gae_norm = (gae - gae.mean()) / (gae.std() + 1e-8)
                        pg_loss_1 = ratio * gae_norm * actor_weights
                        pg_loss_2 = jnp.clip(
                            ratio,
                            1.0 - config.clip_eps,
                            1.0 + config.clip_eps) * gae_norm * actor_weights
                        pg_loss = jax.lax.cond(
                            loss_weights.sum() == 0,
                            lambda x: jnp.zeros_like(x).astype(jnp.float32),
                            lambda x: x,
                            -(
                                loss_weights*jnp.minimum(pg_loss_1, pg_loss_2)
                            ).sum()/loss_weights.sum()
                        )

                        # Entropy
                        entropy = jax.lax.cond(
                            loss_weights.sum() == 0,
                            lambda x: jnp.zeros_like(x).astype(jnp.float32),
                            lambda x: x,
                            (loss_weights * pi.entropy()).sum()/loss_weights.sum()
                        )

                        total_loss = pg_loss + \
                            config.vf_coef * value_loss - \
                            config.ent_coef * entropy

                        return total_loss, (value_loss, pg_loss, entropy)

                    possible_agent_ids = jnp.expand_dims(
                        jnp.arange(config.partner_pop_size), 1)
                    grad_fn = jax.value_and_grad(_loss_fn, has_aux=True)

                    def gather_conf_params_and_return_grads(train_state_params, agent_id):
                        param_vector = gather_params(
                            train_state_params, agent_id)
                        (loss_val_conf, aux_vals_conf), grads_conf = grad_fn(
                            param_vector, conf_agent_net, traj_batch_conf,
                            advantages_conf, returns_conf, agent_id
                        )
                        return (loss_val_conf, aux_vals_conf), grads_conf

                    def gather_br_params_and_return_grads(train_state_params, agent_id):
                        param_vector = gather_params(
                            train_state_params, agent_id)
                        (loss_val_br, aux_vals_br), grads_br = grad_fn(
                            param_vector, br_agent_net, traj_batch_br,
                            advantages_br, returns_br, agent_id
                        )
                        return (loss_val_br, aux_vals_br), grads_br

                    def compute_conf_grads(x): return gather_conf_params_and_return_grads(
                        train_state_conf.params, x)
                    def compute_br_grads(x): return gather_br_params_and_return_grads(
                        train_state_br.params, x)

                    (loss_val_conf, aux_vals_conf), grads_conf = jax.vmap(
                        compute_conf_grads)(possible_agent_ids)
                    (loss_val_br, aux_vals_br), grads_br = jax.vmap(
                        compute_br_grads)(possible_agent_ids)

                    grads_conf_new = jax.tree.map(
                        lambda x: jnp.squeeze(x, 1), grads_conf)
                    grads_br_new = jax.tree.map(
                        lambda x: jnp.squeeze(x, 1), grads_br)
                    train_state_conf = train_state_conf.apply_gradients(
                        grads=grads_conf_new)
                    train_state_br = train_state_br.apply_gradients(
                        grads=grads_br_new)
                    return (train_state_conf, train_state_br), ((loss_val_conf, aux_vals_conf), (loss_val_br, aux_vals_br))

                (
                    train_state_conf, train_state_br,
                    traj_batch_conf, traj_batch_br,
                    advantages_conf, advantages_br,
                    targets_conf, targets_br,
                    rng_ego, rng_br
                ) = update_state

                rng_ego, perm_rng_conf = jax.random.split(rng_ego)
                rng_br, perm_rng_br = jax.random.split(rng_br)

                batch_size_conf = config.minibatch_size_ego * \
                    config.num_minibatches // config.num_game_agents
                batch_size_br = config.minibatch_size_br * \
                    config.num_minibatches // config.num_game_agents

                assert (
                    batch_size_conf == (config.num_game_agents-1) * config.num_steps *
                    config.num_actors // config.num_game_agents
                ), "batch size must be equal to number of steps * number of actors"
                assert (
                    batch_size_br == config.num_steps *
                    config.num_actors // config.num_game_agents
                ), "batch size must be equal to number of steps * number of actors"

                permutation_conf = jax.random.permutation(
                    perm_rng_conf, batch_size_conf)
                permutation_br = jax.random.permutation(
                    perm_rng_br, batch_size_br)

                batch_conf = (traj_batch_conf, advantages_conf, targets_conf)
                batch_br = (traj_batch_br, advantages_br, targets_br)

                batch_conf_reshaped = jax.tree.map(
                    lambda x: x.reshape(
                        (batch_size_conf,) + x.shape[2:]), batch_conf
                )

                batch_br_reshaped = jax.tree.map(
                    lambda x: x.reshape(
                        (batch_size_br,) + x.shape[2:]), batch_br
                )

                shuffled_batch_conf = jax.tree.map(
                    lambda x: jnp.take(x, permutation_conf,
                                       axis=0), batch_conf_reshaped
                )
                shuffled_batch_br = jax.tree.map(
                    lambda x: jnp.take(x, permutation_br,
                                       axis=0), batch_br_reshaped
                )

                minibatches_conf = jax.tree.map(
                    lambda x: jnp.reshape(
                        x, [config.num_minibatches, -1] + list(x.shape[1:])
                    ),
                    shuffled_batch_conf,
                )

                minibatches_br = jax.tree.map(
                    lambda x: jnp.reshape(
                        x, [config.num_minibatches, -1] + list(x.shape[1:])
                    ),
                    shuffled_batch_br,
                )

                # Update both policies
                updated_train_states, total_loss = jax.lax.scan(
                    _update_minbatch, (train_state_conf,
                                       train_state_br), (minibatches_conf, minibatches_br)
                )
                train_state_conf = updated_train_states[0]
                train_state_br = updated_train_states[1]

                update_state = (train_state_conf, train_state_br,
                                traj_batch_conf, traj_batch_br,
                                advantages_conf, advantages_br,
                                targets_conf, targets_conf,
                                rng_ego, rng_br
                                )
                return update_state, total_loss

            def _update_step(update_runner_state, unused):
                (
                    all_train_state_conf, all_train_state_br, rng, update_steps
                ) = update_runner_state

                rng, reset_rng = jax.random.split(rng, 2)
                reset_rngs = jax.random.split(reset_rng, config.num_envs)
                last_obs, env_state = jax.vmap(
                    env.reset_env_to_level, in_axes=(0, None, None))(reset_rngs, level, env.default_params)

                rng, conf_sampling_sp_rng, conf_sampling_rng, br_sampling_rng = jax.random.split(
                    rng, 4)

                # Sampling IDs for SP data collection
                ids_sp = jax.random.randint(
                    conf_sampling_sp_rng, (config.num_envs_sp,), 0, config.partner_pop_size)

                # Sampling IDs for XP data collection
                conf_ids_xp = jax.random.randint(
                    conf_sampling_rng, (config.num_envs_xp,), 0, config.partner_pop_size)

                br_sampling_rngs = jax.random.split(
                    br_sampling_rng, config.num_envs_xp+1)
                br_sampling_rng = br_sampling_rngs[0]

                # Sample BR IDs that are different from conf id
                br_ids_xp = jax.random.randint(
                    br_sampling_rng, (config.num_envs_xp,), 0, config.partner_pop_size)

                def sample_new_id(a): return (jax.random.split(a[0], 2)[1], jax.random.randint(
                    a[0], a[1].shape, minval=0, maxval=config.partner_pop_size))

                def loop_logic(z, x, y): return jax.lax.while_loop(
                    lambda a: jnp.equal(z, a[1]), sample_new_id, (x, y))
                _, br_ids_xp = jax.vmap(loop_logic)(
                    conf_ids_xp, br_sampling_rngs[1:], br_ids_xp)

                conf_ids = jnp.concatenate([ids_sp, conf_ids_xp], axis=-1)
                br_ids = jnp.concatenate([ids_sp, br_ids_xp], axis=-1)

                identity_matrix = jnp.eye(config.partner_pop_size)
                conf_one_hots = identity_matrix[conf_ids]
                br_one_hots = identity_matrix[br_ids]

                gather_conf_params = gather_params(
                    all_train_state_conf.params, conf_ids)
                gather_br_params = gather_params(
                    all_train_state_br.params, br_ids)

                runner_state = (
                    gather_conf_params, gather_br_params, conf_one_hots, br_one_hots,
                    env_state, last_obs, rng
                )
                runner_state, traj_batch = jax.lax.scan(
                    _env_step, runner_state, None, config.num_steps)
                (gather_conf_params, gather_br_params, conf_one_hots,
                 br_one_hots, env_state, last_obs, rng) = runner_state

                # Get agent 0 and agent 1 trajectories from interaction between conf policy and its BR policy.
                traj_batch_conf, traj_batch_br = traj_batch

                # Compute advantage for confederate agent from interaction with br policy
                last_obs_conf = last_obs["agent_0"]

                def forward_pass_conf(param, ob, id): return conf_agent_net.apply(
                    param, (ob, id))
                _, last_val_conf = jax.vmap(forward_pass_conf)(
                    gather_conf_params, last_obs_conf, br_one_hots)

                advantages_conf, targets_conf = _calculate_gae(
                    traj_batch_conf, last_val_conf)

                # 3c) compute advantage for br policy from interaction with confederate agent
                last_obs_br = last_obs["agent_1"]

                def forward_pass_br(param, ob, id): return br_agent_net.apply(
                    param, (ob, id))
                _, last_val_br = jax.vmap(forward_pass_br)(
                    gather_br_params, last_obs_br, conf_one_hots)
                advantages_br, targets_br = _calculate_gae(
                    traj_batch_br, last_val_br)

                # 3) PPO update
                rng, conf_batch_sampling_rng, br_batch_sampling_rng = jax.random.split(
                    rng, 3)
                update_state = (
                    all_train_state_conf, all_train_state_br, traj_batch_conf,
                    traj_batch_br, advantages_conf, advantages_br,
                    targets_conf, targets_br,
                    conf_batch_sampling_rng, br_batch_sampling_rng
                )

                update_state, all_losses = jax.lax.scan(
                    _update_epoch, update_state, None, config.update_epochs)
                all_train_state_conf = update_state[0]
                all_train_state_br = update_state[1]

                # Metrics
                metric = traj_batch_conf.info
                metric["update_steps"] = update_steps
                metric["value_loss_conf_agent"] = all_losses[0][1][0]
                metric["value_loss_br_agent"] = all_losses[1][1][0]
                metric["pg_loss_conf_agent"] = all_losses[0][1][1]
                metric["pg_loss_br_agent"] = all_losses[1][1][1]
                metric["entropy_conf"] = all_losses[0][1][2]
                metric["entropy_br"] = all_losses[1][1][2]

                new_runner_state = (
                    all_train_state_conf, all_train_state_br, rng, update_steps + 1
                )
                return (new_runner_state, metric)

            # --------------------------
            # PPO Update and Checkpoint saving
            # --------------------------
            # -1 because we store the final ckpt as the last ckpt
            ckpt_and_eval_interval = config.num_updates // max(
                1, config.num_checkpoints - 1)
            num_ckpts = config.num_checkpoints

            # Build a PyTree that holds parameters for all conf agent checkpoints
            def init_ckpt_array(params_pytree):
                return jax.tree.map(
                    lambda x: jnp.zeros((num_ckpts,) + x.shape, x.dtype),
                    params_pytree)

            max_episode_steps = config.num_steps

            def _update_step_with_ckpt(state_with_ckpt, unused):
                ((
                    train_state_conf, train_state_br, rng, update_steps
                ), checkpoint_array_conf, checkpoint_array_br, ckpt_idx,
                    eval_info) = state_with_ckpt

                # Single PPO update
                (new_runner_state, metric) = _update_step(
                    (train_state_conf, train_state_br, rng, update_steps),
                    None
                )

                (
                    train_state_conf, train_state_br, rng, update_steps
                ) = new_runner_state

                # Decide if we store a checkpoint
                # update steps is 1-indexed because it was incremented at the end of the update step
                to_store = jnp.logical_or(jnp.equal(jnp.mod(update_steps-1, ckpt_and_eval_interval), 0),
                                          jnp.equal(update_steps, config.num_updates))
                max_eval_episodes = config.num_eval_episodes

                def store_and_eval_ckpt(args):
                    ckpt_arr_and_ep_infos, rng, cidx = args
                    ckpt_arr_conf, ckpt_arr_br, _ = ckpt_arr_and_ep_infos
                    new_ckpt_arr_conf = jax.tree.map(
                        lambda c_arr, p: c_arr.at[cidx].set(p),
                        ckpt_arr_conf, train_state_conf.params
                    )
                    new_ckpt_arr_br = jax.tree.map(
                        lambda c_arr, p: c_arr.at[cidx].set(p),
                        ckpt_arr_br, train_state_br.params
                    )

                    rng, eval_rng = jax.random.split(rng)
                    ep_infos = run_all_episodes(
                        eval_rng, train_state_conf, train_state_br, max_eval_episodes)

                    return ((new_ckpt_arr_conf, new_ckpt_arr_br, ep_infos), rng, cidx + 1)

                def skip_ckpt(args):
                    return args

                (checkpoint_array_and_infos, rng, ckpt_idx) = jax.lax.cond(
                    to_store,
                    store_and_eval_ckpt,
                    skip_ckpt,
                    ((checkpoint_array_conf, checkpoint_array_br, eval_info), rng, ckpt_idx)
                )
                checkpoint_array_conf, checkpoint_array_br, ckpt_infos = checkpoint_array_and_infos

                metric["real_eval_ep_last_info"] = ckpt_infos[0]
                # return of confederate
                metric["eval_ep_last_info"] = ckpt_infos[1]

                return ((train_state_conf, train_state_br, rng, update_steps),
                        checkpoint_array_conf, checkpoint_array_br, ckpt_idx,
                        ckpt_infos), metric

            # init checkpoint array
            checkpoint_array_conf = init_ckpt_array(all_conf_optims.params)
            checkpoint_array_br = init_ckpt_array(all_br_optims.params)
            ckpt_idx = 0

            # initial state for scan over _update_step_with_ckpt
            update_steps = 0

            rng, rng_eval = jax.random.split(rng, 2)
            max_eval_episodes = config.num_eval_episodes

            ep_infos = run_all_episodes(
                rng_eval, all_conf_optims, all_br_optims, max_eval_episodes)

            update_runner_state = (
                all_conf_optims, all_br_optims, rng, update_steps
            )

            state_with_ckpt = (
                update_runner_state, checkpoint_array_conf,
                checkpoint_array_br, ckpt_idx, ep_infos
            )

            # run training
            state_with_ckpt, metrics = jax.lax.scan(
                _update_step_with_ckpt,
                state_with_ckpt,
                xs=None,
                length=config.num_updates
            )

            (
                final_runner_state, checkpoint_array_conf, checkpoint_array_br,
                final_ckpt_idx, all_ep_infos
            ) = state_with_ckpt

            out = {
                "final_params_conf": final_runner_state[0].params,
                "final_params_br": final_runner_state[1].params,
                "checkpoints_conf": checkpoint_array_conf,
                "checkpoints_br": checkpoint_array_br,
                # metrics is from the perspective of the confederate agent (averaged over population)
                "metrics": metrics,
                "all_pair_returns": all_ep_infos
            }
            return out

        return train
    # ------------------------------
    # Actually run the adversarial teammate training
    # ------------------------------
    train_fn = make_brdiv_agents(config)
    out = train_fn(train_rng)
    return out


def get_brdiv_population(config, out, env):
    '''
    Get the partner params and partner population for ego training.
    '''
    brdiv_pop_size = config.partner_pop_size

    # partner_params has shape (num_seeds, brdiv_pop_size, ...)
    partner_params = out['final_params_conf']

    partner_policy = ActorWithConditionalCriticPolicy(
        action_dim=env.action_space(env.agents[1]).n,
        obs_dim=env.observation_space(env.agents[1]).shape[0],
        pop_size=brdiv_pop_size,  # used to create onehot agent id
        activation=config.activation
    )

    # Create partner population
    partner_population = AgentPopulation(
        pop_size=brdiv_pop_size,
        policy_cls=partner_policy
    )

    return partner_params, partner_population


def compute_sp_mask_and_ids(pop_size):
    cross_product = np.meshgrid(
        np.arange(pop_size),
        np.arange(pop_size)
    )
    agent_id_cartesian_product = np.stack(
        [g.ravel() for g in cross_product], axis=-1)
    conf_ids = agent_id_cartesian_product[:, 0]
    ego_ids = agent_id_cartesian_product[:, 1]
    sp_mask = (conf_ids == ego_ids)
    return sp_mask, agent_id_cartesian_product


def log_metrics(config, outs, logger, metric_names: tuple, save_dir):
    metrics = outs["metrics"]
    # metrics now has shape (num_seeds, num_updates, _, _, pop_size)
    # number of trained pairs
    num_seeds, num_updates, _, _, pop_size = metrics["pg_loss_conf_agent"].shape

    # Log evaluation metrics
    # we plot XP return curves separately from SP return curves
    # shape (num_seeds, num_updates, (pop_size)^2, num_eval_episodes, 1)
    all_returns = np.asarray(metrics["eval_ep_last_info"])
    xs = list(range(num_updates))

    sp_mask, agent_id_cartesian_product = compute_sp_mask_and_ids(pop_size)
    sp_returns = all_returns[:, :, sp_mask]
    xp_returns = all_returns[:, :, ~sp_mask]

    # Average over seeds, then over agent pairs, episodes and num_agents_per_game
    sp_return_curve = sp_returns.mean(axis=(0, 2, 3, 4))
    xp_return_curve = xp_returns.mean(axis=(0, 2, 3, 4))

    for step in range(num_updates):
        wandb.log(
            {"Eval/AvgSPReturnCurve": sp_return_curve[step], "train_step": step}, step=None, commit=True)
        wandb.log(
            {"Eval/AvgXPReturnCurve": xp_return_curve[step], "train_step": step}, step=None, commit=True)
    wandb.log({}, commit=True)

    # log final XP matrix to wandb - average over seeds
    last_returns_array = all_returns[:, -1].mean(axis=(0, 2, 3))
    last_returns_array = np.reshape(last_returns_array, (pop_size, pop_size))

    rows = [str(i) for i in range(last_returns_array.shape[0])]
    columns = [str(i) for i in range(last_returns_array.shape[1])]
    tab = wandb.Table(
        columns=columns,
        data=last_returns_array,
        rows=rows
    )
    wandb.log({"Eval/LastXPMatrix": tab}, step=None, commit=True)

    # Log population loss as multi-line plots, where each line is a different population member
    # shape (num_seeds, num_updates, update_epochs, num_minibatches, pop_size)
    # Average over seeds
    processed_losses = {
        "ConfPGLoss": np.asarray(metrics["pg_loss_conf_agent"]).mean(axis=(0, 2, 3)).transpose(),
        "BRPGLoss": np.asarray(metrics["pg_loss_br_agent"]).mean(axis=(0, 2, 3)).transpose(),
        "ConfValLoss": np.asarray(metrics["value_loss_conf_agent"]).mean(axis=(0, 2, 3)).transpose(),
        "BRValLoss": np.asarray(metrics["value_loss_br_agent"]).mean(axis=(0, 2, 3)).transpose(),
        "ConfEntropy": np.asarray(metrics["entropy_conf"]).mean(axis=(0, 2, 3)).transpose(),
        "BREntropy": np.asarray(metrics["entropy_br"]).mean(axis=(0, 2, 3)).transpose(),
    }

    xs = list(range(num_updates))
    keys = [f"pair {i}" for i in range(pop_size)]
    for loss_name, loss_data in processed_losses.items():
        wandb.log(
            {f"Losses/{loss_name}": wandb.plot.line_series(
                xs=xs, ys=loss_data, keys=keys, title=loss_name, xname="train_step")},
            step=None, commit=True
        )


def get_metric_names():
    return ("base_return", "returned_episode_returns")


def get_run_string(config: TrainConfig):
    return f"FF_BRDIV_IPPO_OGC_{config.layout_name}"


@pyrallis.wrap()
def train(config: TrainConfig):
    ##### WANDB and other setup #####
    tags = [
        "FF",
        "BRDIV",
        "IPPO",
        config.layout_name,
    ]

    group_string = get_run_string(config)
    run_string = f"{group_string}_SEED_{config.seed}"

    run = wandb.init(
        project=config.project,
        group=config.group,
        mode=config.mode,
        config=asdict(config),
        save_code=True,
        tags=tags,
    )

    if run.sweep_id is not None:
        run.name = run.sweep_id + "___" + run_string
    else:
        run.name = run.name + "___" + run_string

    print("XPID ID name:")
    print(run.name)
    print("-------------")

    #### Setup and check saving before training ####
    if config.checkpoint_path is not None:
        save_dir = os.path.join(config.checkpoint_path, run.name)
        # Make sure we can write the checkpoint later _before_ we wait 1 day for training!
        os.makedirs(save_dir, exist_ok=True)
        config_dict = asdict(config)
        with open(f"{save_dir}/config.pckl", 'wb') as f:
            pickle.dump(config_dict, f)

    # env = make_env(
    #     "overcooked-v1", config.env_kwargs)
    # env = LogWrapper(env, replace_info=False)
    env = OGC(width=config.ogc_width,
              height=config.ogc_height, random_reset=True)
    env = AutoReplayWrapper(env)
    level = Level.from_layout_name(config.layout_name)

    # Generate multiple random seeds from the base seed
    rng = jax.random.PRNGKey(config.seed)
    rngs = jax.random.split(rng, config.num_seeds)

    # Create a vmapped version of train_brdiv_partners
    with jax.disable_jit(False):
        vmapped_train_fn = jax.jit(
            jax.vmap(
                partial(train_brdiv_partners, env=env,
                        level=level, config=config)
            )
        )
        out = vmapped_train_fn(rngs)

    metric_names = get_metric_names()
    log_metrics(config, out, run, metric_names, save_dir)

    partner_params, partner_population = get_brdiv_population(config, out, env)

    for i in range(config.num_seeds):
        for j in range(config.partner_pop_size):
            params = jax.tree.map(lambda x: x[i, j], partner_params)

            path = f"{save_dir}/"
            os.makedirs(path, exist_ok=True)
            payload = {"actor_params": params}
            pickle.dump(payload, open(
                path + f"params_seed{i}_agent{j}.pt", "wb"))
            pickle.dump(payload, open(
                path + f"params.pt", "wb"))

    return partner_params, partner_population


if __name__ == '__main__':
    train()
