from curses import KEY_F28
from typing import Tuple

import jax
import jax.numpy as jnp
from jax import jit
from functools import partial

from common import Batch, InfoDict, Model, Params, PRNGKey

# @partial(jit, static_argnums=(5, 6))
def update(key: PRNGKey, actor: Model, critic: Model, value: Model,
           batch: Batch, temperature: float, reweight_improve, reweight_constraint) -> Tuple[Model, InfoDict]:
    key1, key2, key3 = jax.random.split(key, 3)
    v = value(batch.observations, rngs={'dropout': key1})
    q1, q2 = critic(batch.observations, batch.actions, rngs={'dropout': key2})
    q = jnp.minimum(q1, q2)
    exp_a = jnp.exp((q - v) * temperature)

    reweight = reweight_improve * reweight_constraint
    weights = batch.weights * reweight + jnp.ones_like(batch.weights) * (1-reweight)
    exp_a = jnp.minimum(exp_a * weights, 100.0)
    # TODO
    
    # if reweight_improve and reweight_constraint:
    #     weights = batch.weights
    # elif reweight_improve and not reweight_constraint:
    #     raise NotImplementedError
    # elif not reweight_improve and reweight_constraint:
    #     raise NotImplementedError
    # else:
    #     weights = jnp.ones_like(batch.weights)

    def actor_loss_fn(actor_params: Params, ) -> Tuple[jnp.ndarray, InfoDict]:
        dist = actor.apply({'params': actor_params},  # forward prog
                           batch.observations,
                           training=True,
                           rngs={'dropout': key3})
        log_probs = dist.log_prob(batch.actions)
        actor_loss = -exp_a * log_probs
        actor_loss = actor_loss.mean()
        return actor_loss, {'actor_loss': actor_loss, 'adv': q - v}

    new_actor, info = actor.apply_gradient(actor_loss_fn)

    return new_actor, info
