from collections import deque
from typing import Any, NamedTuple

import dm_env
import numpy as np
from dm_control import manipulation, suite
from dm_control.suite.wrappers import action_scale, pixels
from dm_control.rl.control import FLAT_OBSERVATION_KEY
from dm_env import StepType, specs


class ExtendedTimeStep(NamedTuple):
    step_type: Any
    reward: Any
    discount: Any
    observation: Any
    action: Any

    def first(self):
        return self.step_type == StepType.FIRST

    def mid(self):
        return self.step_type == StepType.MID

    def last(self):
        return self.step_type == StepType.LAST

    def __getitem__(self, attr):
        return getattr(self, attr)


class ActionRepeatWrapper(dm_env.Environment):
    def __init__(self, env, num_repeats):
        self._env = env
        self._num_repeats = num_repeats

    def step(self, action):
        reward = 0.0
        discount = 1.0
        for i in range(self._num_repeats):
            time_step = self._env.step(action)
            reward += (time_step.reward or 0.0) * discount
            discount *= time_step.discount
            if time_step.last():
                break

        return time_step._replace(reward=reward, discount=discount)

    def observation_spec(self):
        return self._env.observation_spec()

    def action_spec(self):
        return self._env.action_spec()

    def reset(self):
        return self._env.reset()

    def __getattr__(self, name):
        return getattr(self._env, name)


class FrameStackWrapper(dm_env.Environment):
    def __init__(self, env, num_frames, pixels_key='pixels'):
        self._env = env
        self._num_frames = num_frames
        self._frames = deque([], maxlen=num_frames)
        self._pixels_key = pixels_key

        wrapped_obs_spec = env.observation_spec()
        assert pixels_key in wrapped_obs_spec

        pixels_shape = wrapped_obs_spec[pixels_key].shape
        # remove batch dim
        if len(pixels_shape) == 4:
            pixels_shape = pixels_shape[1:]
        self._obs_spec = specs.BoundedArray(shape=np.concatenate(
            [[pixels_shape[2] * num_frames], pixels_shape[:2]], axis=0),
                                            dtype=np.uint8,
                                            minimum=0,
                                            maximum=255,
                                            name='observation')

    def _transform_observation(self, time_step):
        assert len(self._frames) == self._num_frames
        obs = np.concatenate(list(self._frames), axis=0)
        return time_step._replace(observation=obs)

    def _extract_pixels(self, time_step):
        pixels = time_step.observation[self._pixels_key]
        # remove batch dim
        if len(pixels.shape) == 4:
            pixels = pixels[0]
        return pixels.transpose(2, 0, 1).copy()

    def reset(self):
        time_step = self._env.reset()
        pixels = self._extract_pixels(time_step)
        for _ in range(self._num_frames):
            self._frames.append(pixels)
        return self._transform_observation(time_step)

    def step(self, action):
        time_step = self._env.step(action)
        pixels = self._extract_pixels(time_step)
        self._frames.append(pixels)
        return self._transform_observation(time_step)

    def observation_spec(self):
        return self._obs_spec

    def action_spec(self):
        return self._env.action_spec()

    def __getattr__(self, name):
        return getattr(self._env, name)


class ActionDTypeWrapper(dm_env.Environment):
    def __init__(self, env, dtype):
        self._env = env
        wrapped_action_spec = env.action_spec()
        self._action_spec = specs.BoundedArray(wrapped_action_spec.shape,
                                               dtype,
                                               wrapped_action_spec.minimum,
                                               wrapped_action_spec.maximum,
                                               'action')

    def step(self, action):
        action = action.astype(self._env.action_spec().dtype)
        return self._env.step(action)

    def observation_spec(self):
        return self._env.observation_spec()

    def action_spec(self):
        return self._action_spec

    def reset(self):
        return self._env.reset()

    def __getattr__(self, name):
        return getattr(self._env, name)


class ObservationSpecWrapper(dm_env.Environment):
    def __init__(self, env):
        self._env = env
        wrapped_observation_spec = env.observation_spec()[FLAT_OBSERVATION_KEY]
        self._observation_spec = specs.Array(wrapped_observation_spec.shape,
                                             wrapped_observation_spec.dtype,
                                             'observation')

    def _transform_observation(self, time_step):
        obs = time_step.observation[FLAT_OBSERVATION_KEY]
        return time_step._replace(observation=obs)

    def step(self, action):
        time_step = self._env.step(action)
        return self._transform_observation(time_step)

    def observation_spec(self):
        return self._observation_spec

    def action_spec(self):
        return self._env.action_spec()

    def reset(self):
        time_step = self._env.reset()
        return self._transform_observation(time_step)

    def __getattr__(self, name):
        return getattr(self._env, name)


class ExtendedTimeStepWrapper(dm_env.Environment):
    def __init__(self, env):
        self._env = env

    def reset(self):
        time_step = self._env.reset()
        return self._augment_time_step(time_step)

    def step(self, action):
        time_step = self._env.step(action)
        return self._augment_time_step(time_step, action)

    def _augment_time_step(self, time_step, action=None):
        if action is None:
            action_spec = self.action_spec()
            action = np.zeros(action_spec.shape, dtype=action_spec.dtype)
        return ExtendedTimeStep(observation=time_step.observation,
                                step_type=time_step.step_type,
                                action=action,
                                reward=time_step.reward or 0.0,
                                discount=time_step.discount or 1.0)

    def observation_spec(self):
        return self._env.observation_spec()

    def action_spec(self):
        return self._env.action_spec()

    def __getattr__(self, name):
        return getattr(self._env, name)


class DiscretizedActionWrapper(dm_env.Environment):
    def __init__(self, env, k):
        self._env = env
        self.action_dim = self._env.action_spec().shape[0]
        self.k = k
        action_low = self._env.action_spec().minimum.reshape(-1)
        action_high = self._env.action_spec().maximum.reshape(-1)
        n_discrete_actions = self.k**self.action_dim

        self._action_lookup = np.reshape([np.linspace(action_low[i], action_high[i], k)
                                          for i in range(self.action_dim)], [self.action_dim, k])

        self._action_spec = specs.DiscreteArray(n_discrete_actions, dtype=np.int32, name='action')
        # manually set the shape attribute
        self._action_spec._shape = (1,)

    def _unflatten(self, action):
        tmp_action = np.copy(action)
        unflattened_action = np.zeros((self.action_dim, ), dtype=np.int32)
        for i in range(self.action_dim):
            unflattened_action[-i - 1] = tmp_action % self.k
            tmp_action //= self.k
        return unflattened_action

    def _transform(self, action):
        assert 0 <= action < self._action_spec.num_values, f"Action value {action} is invalid."
        action = self._unflatten(action)
        action_cont = self._action_lookup[np.arange(self.action_dim), action]
        return action_cont

    def step(self, action):
        return self._env.step(self._transform(action))

    def reset(self):
        return self._env.reset()

    def observation_spec(self):
        return self._env.observation_spec()

    def action_spec(self):
        return self._action_spec

    def __getattr__(self, name):
        return getattr(self._env, name)


def make(name, frame_stack, action_repeat, seed, pixel_obs=True,
         discrete_actions=False, k_discrete_action=3):
    environment_kwargs = dict()
    if not pixel_obs:
        environment_kwargs['flat_observation'] = True

    domain, task = name.split('_', 1)
    # overwrite cup to ball_in_cup
    domain = dict(cup='ball_in_cup').get(domain, domain)
    # make sure reward is not visualized
    if (domain, task) in suite.ALL_TASKS:
        env = suite.load(domain,
                         task,
                         task_kwargs={'random': seed},
                         environment_kwargs=environment_kwargs,
                         visualize_reward=False)
        pixels_key = 'pixels'
    else:
        name = f'{domain}_{task}_vision'
        env = manipulation.load(name, seed=seed)
        pixels_key = 'front_close'

    # add wrappers for discrete or continuous action spaces
    if discrete_actions:
        env = DiscretizedActionWrapper(env, k_discrete_action)
    else:
        env = ActionDTypeWrapper(env, np.float32)
        env = action_scale.Wrapper(env, minimum=-1.0, maximum=+1.0)

    # add wrappers for pixel or state observations
    if pixel_obs:
        env = ActionRepeatWrapper(env, action_repeat)
        # add renderings for classical tasks
        if (domain, task) in suite.ALL_TASKS:
            # zoom in camera for quadruped
            camera_id = dict(quadruped=2).get(domain, 0)
            render_kwargs = dict(height=84, width=84, camera_id=camera_id)
            env = pixels.Wrapper(env,
                                 pixels_only=True,
                                 render_kwargs=render_kwargs)
        # stack several frames
        env = FrameStackWrapper(env, frame_stack, pixels_key)
        env = ExtendedTimeStepWrapper(env)
    else:
        env = ObservationSpecWrapper(env)
        env = ExtendedTimeStepWrapper(env)

    return env
