from typing import NamedTuple

import chex
import haiku as hk
import jax
import jax.numpy as jnp
from haiku_geometric.nn import GINEConv, GraphConv

from src.baselines.qrdqn.util import distort_value


class FeatureExtractor(hk.Module):
    def __init__(self, hidden_channels: int):
        super().__init__()
        self.hidden_channels = hidden_channels

        self.conv1 = GraphConv(out_channels=hidden_channels)
        self.conv2 = GraphConv(out_channels=hidden_channels)
        self.conv3 = GraphConv(out_channels=hidden_channels)
        self.conv4 = GraphConv(out_channels=hidden_channels)

        self.num_registers = 1
        self.register_node_init = hk.get_parameter(
            "register_node_init",
            shape=(self.num_registers, hidden_channels),
            init=hk.initializers.RandomNormal(),
        )

    def __call__(self, nodes, edges, senders, receivers, aux):
        nodes = jax.nn.relu(
            hk.Linear(self.hidden_channels)(nodes)
        )  # (nodes, hidden_channels)
        nodes = jnp.concatenate(
            [nodes, jnp.ones((nodes.shape[0], self.hidden_channels))], axis=-1
        )  # (nodes, hidden_channels + 1)
        nodes = jax.nn.relu(
            hk.Linear(self.hidden_channels)(nodes)
        )  # (nodes, hidden_channels)

        register_nodes = self.register_node_init  # (1, hidden)
        num_nodes = nodes.shape[0]
        nodes = jnp.concatenate(
            [nodes, register_nodes], axis=0
        )  # (n_nodes + 1, hidden)

        # There are n_nodes + 1 nodes in the graph. We want to connect (register) nodes [n_nodes:] to all the prior nodes.
        receivers_ = jnp.arange(num_nodes)
        senders_ = jnp.full(
            num_nodes, num_nodes, dtype=jnp.int32
        )  # [num_node, num_node, ...]
        senders_ = jnp.concatenate([senders, senders_], axis=0)
        receivers_ = jnp.concatenate([receivers, receivers_], axis=0)

        # We've added n_nodes edges to the graph, these are type 0 edges.
        edge_dim = edges.shape[-1]
        edges_ = jnp.zeros((num_nodes, edge_dim), dtype=edges.dtype)
        edges_ = edges_.at[:, 0].set(1.0)  # Set the first edge type to 1
        edges = jnp.concatenate([edges, edges_], axis=0)
        edges = jax.nn.relu(
            hk.Linear(self.hidden_channels)(edges)
        )  # (edges, hidden_channels)

        # Make the graph undirected by adding the reverse edges
        senders = jnp.concatenate([senders_, receivers_], axis=0)
        receivers = jnp.concatenate([receivers_, senders_], axis=0)
        # Repeat edges to match undirected edges
        edges = jnp.concatenate([edges, edges], axis=0)

        # Apply convolutions
        x = self.conv1(
            nodes, senders, receivers, edges
        )  # (n_nodes + 1, hidden_channels)
        x = hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(x)
        )  # (n_nodes + 1, hidden_channels)
        edges = edges + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(hk.Linear(self.hidden_channels)(edges))
        )  # (edges, hidden_channels)
        x = x + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(self.conv2(x, senders, receivers, edges))
        )  # (n_nodes + 1, hidden_channels)
        edges = edges + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(hk.Linear(self.hidden_channels)(edges))
        )  # (edges, hidden_channels)

        x = x + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(self.conv3(x, senders, receivers, edges))
        )  # (n_nodes + 1, hidden_channels)
        edges = edges + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(hk.Linear(self.hidden_channels)(edges))
        )  # (edges, hidden_channels)

        x = x + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(self.conv4(x, senders, receivers, edges))
        )  # (n_nodes + 1, hidden_channels)

        # Pool and concat
        aux_embedding = jax.nn.relu(
            hk.Linear(self.hidden_channels)(aux)
        )  # (out_channels)
        pool = jnp.sum(x, axis=0)  # (hidden_channels,)
        pool_rep = jax.nn.relu(hk.Linear(self.hidden_channels)(pool))
        pool_aux_concat_ = jnp.concat(
            [pool_rep[jnp.newaxis, :], aux_embedding[jnp.newaxis, :]], axis=-1
        ).reshape((-1))  # (out_channels * 2,)
        pool_aux_concat = jax.nn.relu(hk.Linear(self.hidden_channels)(pool_aux_concat_))

        # Remove half the edges + num_nodes (edges connecting to registers``) to get edge embeddings
        edge_embeddings = edges[
            : edges.shape[0] // 2 - num_nodes
        ]  # (edges, hidden_channels)

        # Update node embeddings based on pool and aux
        pool_rep = jax.nn.relu(hk.Linear(self.hidden_channels)(pool))
        pool_rep = jnp.repeat(pool_rep[jnp.newaxis, :], x.shape[0], axis=0)
        aux_embedding = jax.nn.relu(
            hk.Linear(self.hidden_channels)(aux)
        )  # (out_channels)
        aux_embedding = jnp.repeat(aux_embedding[jnp.newaxis, :], x.shape[0], axis=0)
        node_pool_aux_concat = jnp.concatenate([x, pool_rep, aux_embedding], axis=-1)
        node_embeddings = x + hk.LayerNorm(
            axis=-1, create_scale=True, create_offset=True
        )(jax.nn.relu(hk.Linear(self.hidden_channels)(node_pool_aux_concat)))

        aux_embedding = jax.nn.relu(
            hk.Linear(self.hidden_channels)(aux)
        )  # (out_channels)

        return node_embeddings, aux_embedding, pool_aux_concat, edge_embeddings


class ObsEncoder(hk.Module):
    def __init__(
        self,
        num_hidden: int,
        name: str = "obs_encoder",
    ):
        super().__init__(name=name)
        self.num_hidden = num_hidden

        self.feature_extractor = FeatureExtractor(
            hidden_channels=num_hidden,
        )

    def __call__(
        self, nodes, edges, senders, receivers, aux
    ) -> tuple[jnp.ndarray, jnp.ndarray, jnp.ndarray]:
        node_embeddings, aux_embedding, _, edge_embeddings = self.feature_extractor(
            nodes, edges, senders, receivers, aux
        )  # (n_nodes, hidden_channels * 3)
        return node_embeddings, aux_embedding, edge_embeddings


class ProjectionHead(hk.Module):
    def __init__(self, num_hidden: int, num_out: int, name: str = "projection_head"):
        super().__init__(name=name)
        self.num_hidden = num_hidden
        self.num_out = num_out

    def __call__(
        self,
        node_embeddings: jnp.ndarray,
        edge_emebddings: jnp.ndarray,
        aux_embeddings: jnp.ndarray,
    ) -> tuple[jnp.ndarray, jnp.ndarray, jnp.ndarray]:
        node_embeddings = node_embeddings.astype(jnp.float32)
        node_embedding = jax.nn.relu(hk.Linear(self.num_hidden)(node_embeddings))
        node_proj = hk.Linear(
            self.num_out,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(node_embedding)

        edge_emebddings = edge_emebddings.astype(jnp.float32)
        edge_embeddings = jax.nn.relu(hk.Linear(self.num_hidden)(edge_emebddings))
        edge_proj = hk.Linear(
            self.num_out,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(edge_embeddings)

        aux_embeddings_ = aux_embeddings.astype(jnp.float32)
        aux_embeddings = jax.nn.relu(hk.Linear(self.num_hidden)(aux_embeddings_))
        aux_proj = hk.Linear(
            self.num_out,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(aux_embeddings)

        return node_proj, edge_proj, aux_proj


class PolicyHead(hk.Module):
    def __init__(self, num_hidden: int, name: str = "policy_head"):
        super().__init__(name=name)
        self.num_hidden = num_hidden

    def __call__(
        self, node_embeddings: jnp.ndarray, aux_embedding: jnp.ndarray
    ) -> jnp.ndarray:
        # node_reps: (n_nodes, hidden_channels * 3), remove the register nodes
        node_embeddings = node_embeddings.astype(jnp.float32)[:-1]
        node_embeddings = jax.nn.relu(hk.Linear(self.num_hidden)(node_embeddings))

        pool_embedding = jnp.sum(node_embeddings, axis=0)  # (hidden_channels,)
        pool_embedding = jax.nn.relu(
            hk.Linear(self.num_hidden)(pool_embedding)
        )  # (hidden_channels,)

        aux_embedding = aux_embedding.astype(jnp.float32)
        aux_embedding = jax.nn.relu(hk.Linear(self.num_hidden)(aux_embedding))

        pool_aux_concat = jnp.concatenate(
            [pool_embedding, aux_embedding], axis=-1
        )  # (hidden_channels * 2,)

        # Repeat pool_aux_concat to match the number of nodes
        node_reps = jnp.repeat(
            pool_aux_concat[jnp.newaxis, :], node_embeddings.shape[0], axis=0
        )
        node_reps = jnp.concatenate([node_reps, node_embeddings], axis=-1)

        logits = hk.Linear(
            1,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(node_reps).reshape((-1,))  # (n_nodes,)
        return logits


class QValueHead(hk.Module):
    def __init__(self, num_hidden: int, num_quantiles: int, name: str = "value_head"):
        super().__init__(name=name)
        self.num_hidden = num_hidden
        self.num_quantiles = num_quantiles

    def __call__(
        self, node_embeddings: jnp.ndarray, aux_embedding: jnp.ndarray
    ) -> jnp.ndarray:
        # node_reps: (n_nodes, hidden_channels * 3), remove the register nodes
        node_embeddings_ = node_embeddings.astype(jnp.float32)[:-1]
        node_embeddings = jax.nn.relu(hk.Linear(self.num_hidden)(node_embeddings_))

        pool_embedding = jnp.sum(node_embeddings, axis=0)  # (hidden_channels,)
        pool_embedding = jax.nn.relu(
            hk.Linear(self.num_hidden)(pool_embedding)
        )  # (hidden_channels,)

        aux_embedding = aux_embedding.astype(jnp.float32)
        aux_embedding = jax.nn.relu(hk.Linear(self.num_hidden)(aux_embedding))

        pool_aux_concat = jnp.concatenate(
            [pool_embedding, aux_embedding], axis=-1
        )  # (hidden_channels * 2,)

        v_dist = jax.nn.relu(hk.Linear(self.num_hidden)(pool_aux_concat))
        v_dist = hk.Linear(
            self.num_quantiles,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(v_dist)  # (num_quantiles,)
        return v_dist  # (num_quantiles,)


class RewardHistoryHead(hk.Module):
    def __init__(self, num_hidden: int, num_quantiles: int):
        super().__init__()
        self.num_hidden = num_hidden
        self.num_quantiles = num_quantiles
        self.feature_extractor = FeatureExtractor(hidden_channels=num_hidden)

    def __call__(self, nodes, edges, senders, receivers, aux):
        _, _, pool_aux_concat, _ = self.feature_extractor(
            nodes, edges, senders, receivers, aux
        )
        r_dist = hk.Linear(
            self.num_quantiles,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(pool_aux_concat)  # (num_quantiles,)
        return r_dist  # (num_quantiles,)


class DynamicsHead(hk.Module):
    def __init__(
        self,
        num_hidden: int,
        num_quantiles: int,
        name: str = "dynamics_head",
    ):
        super().__init__(name=name)
        self.num_hidden = num_hidden
        self.num_quantiles = num_quantiles

        self.action_embedding = hk.get_parameter(
            "action_embedding",
            shape=(num_hidden,),
            init=hk.initializers.RandomNormal(),
        )
        self.register_edge_embedding = hk.get_parameter(
            "register_edge_embedding",
            shape=(1, num_hidden),
            init=hk.initializers.RandomNormal(),
        )

        self.conv1 = GraphConv(out_channels=self.num_hidden)
        self.conv2 = GraphConv(out_channels=self.num_hidden)
        self.num_registers = 1

    def __call__(
        self,
        node_embeddings,
        edge_embeddings,
        aux_embedding,
        senders,
        receivers,
        selected_node_index,
    ) -> tuple[jnp.ndarray, jnp.ndarray, jnp.ndarray, jnp.ndarray]:
        senders_in = senders.astype(jnp.int32)
        receivers_in = receivers.astype(jnp.int32)
        # These node embeddings include the register node
        node_embeddings_ = node_embeddings.astype(jnp.float32)
        edge_embeddings_ = edge_embeddings.astype(jnp.float32)

        # Edge embeddings
        edge_embeddings = edge_embeddings_ + jax.nn.relu(
            hk.Linear(self.num_hidden)(edge_embeddings_)
        )

        # Node embeddings
        node_embeddings = node_embeddings.at[selected_node_index].add(
            self.action_embedding
        )
        node_embeddings = node_embeddings_ + jax.nn.relu(
            hk.Linear(self.num_hidden)(node_embeddings)
        )

        node_pool = jnp.sum(node_embeddings, axis=0)  # (hidden_channels,)
        node_pool = node_pool + jax.nn.relu(
            hk.Linear(self.num_hidden)(node_pool)
        )  # (hidden_channels,)

        selected_node = node_embeddings[selected_node_index]  # (hidden_channels,)
        sa_node_pool_aux_concat = jnp.concatenate(
            [node_pool, aux_embedding, selected_node], axis=-1
        )  # (hidden_channels * 3,)
        sa_embed = jax.nn.relu(hk.Linear(self.num_hidden)(sa_node_pool_aux_concat))
        sa_embed = selected_node + sa_embed  # (hidden,)
        r_dist = hk.Linear(
            self.num_quantiles,
            w_init=hk.initializers.UniformScaling(0.01),
            b_init=hk.initializers.UniformScaling(0.25),
        )(sa_embed).reshape((-1))  # (num_quantiles,)

        num_nodes = node_embeddings.shape[0] - 1
        # There are n_nodes + 1 nodes in the graph. We want to connect (register) nodes [n_nodes:] to all the prior nodes.
        receivers_ = jnp.arange(num_nodes)
        senders_ = jnp.full(
            num_nodes, num_nodes, dtype=jnp.int32
        )  # [num_node, num_node, ...]
        senders_ = jnp.concatenate([senders, senders_], axis=0)
        receivers_ = jnp.concatenate([receivers, receivers_], axis=0)

        # We've added n_nodes edges to the graph, create edge embeddings for them
        edges_ = jnp.repeat(
            self.register_edge_embedding, num_nodes, axis=0
        )  # (num_nodes, hidden)
        edge_embeddings = jnp.concatenate([edge_embeddings, edges_], axis=0)

        # Make the graph undirected by adding the reverse edges
        senders = jnp.concatenate([senders_, receivers_], axis=0)
        receivers = jnp.concatenate([receivers_, senders_], axis=0)
        # Repeat edges to match undirected edges
        edge_embeddings = jnp.concatenate([edge_embeddings, edge_embeddings], axis=0)

        # Apply convolutions
        x = self.conv1(
            node_embeddings, senders, receivers, edge_embeddings
        )  # (n_nodes + 1, hidden_channels)
        x = hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(x)
        )  # (n_nodes + 1, hidden_channels)
        edge_embeddings = edge_embeddings + hk.LayerNorm(
            axis=-1, create_scale=True, create_offset=True
        )(
            jax.nn.relu(hk.Linear(self.num_hidden)(edge_embeddings))
        )  # (edges, hidden_channels)
        x = x + hk.LayerNorm(axis=-1, create_scale=True, create_offset=True)(
            jax.nn.relu(self.conv2(x, senders, receivers, edge_embeddings))
        )  # (n_nodes + 1, hidden_channels)

        aux_embedding = aux_embedding.astype(jnp.float32)
        aux_embedding = aux_embedding + jax.nn.relu(
            hk.Linear(self.num_hidden)(aux_embedding)
        )  # (hidden_channels,)

        edge_embeddings = edge_embeddings[
            : edge_embeddings.shape[0] // 2 - num_nodes
        ]  # (edges, hidden_channels)

        # Update node embeddings based on pool and aux
        pool = jnp.sum(x, axis=0)
        pool = pool + jax.nn.relu(hk.Linear(self.num_hidden)(pool))
        pool = jnp.repeat(pool[jnp.newaxis, :], x.shape[0], axis=0)
        aux = jnp.repeat(aux_embedding[jnp.newaxis, :], x.shape[0], axis=0)
        node_pool_aux_concat = jnp.concatenate([x, pool, aux], axis=-1)
        node_embeddings = x + hk.LayerNorm(
            axis=-1, create_scale=True, create_offset=True
        )(jax.nn.relu(hk.Linear(self.num_hidden)(node_pool_aux_concat)))

        # Update each edge embedding with the corresponding node embeddings
        sender_nodes = node_embeddings[senders_in]  # (edges, hidden_channels)
        receiver_nodes = node_embeddings[receivers_in]  # (edges, hidden_channels)
        sender_receiver_concat = jnp.concatenate(
            [sender_nodes, receiver_nodes, edge_embeddings], axis=-1
        )
        edge_embeddings = edge_embeddings + hk.LayerNorm(
            axis=-1, create_scale=True, create_offset=True
        )(jax.nn.relu(hk.Linear(self.num_hidden)(sender_receiver_concat)))

        return (node_embeddings, edge_embeddings, r_dist, aux_embedding)


def get_helpers(args, tau_hats: jnp.ndarray):
    # Return value for expansion
    def v_extract_value(v_dist: jnp.ndarray) -> jnp.ndarray:
        return v_dist

    def q_extract_value(q_dist: jnp.ndarray) -> jnp.ndarray:
        # Distort the q distributions to get greedy values
        # q_dist: (b, num_quantiles, num_actions)
        q_values = jax.vmap(distort_value, in_axes=(0, None, None))(
            q_dist,  # (b, num_quantiles, num_actions)
            tau_hats,  # (num_quantiles,)
            args.cvar_alpha,
        )
        greedy_actions = jnp.argmax(q_values, axis=1)  # (b,)
        q_dist = jnp.take_along_axis(
            q_dist, greedy_actions[:, None, None], axis=-1
        ).squeeze(axis=-1)
        return q_dist
        # (b, num_quantiles)

    def v_value_at_action(v_dist: jnp.ndarray, _action: jnp.ndarray) -> jnp.ndarray:
        """Extracts the value at the given action from the value distribution."""
        # value: (b, num_quantiles)
        # action: (b, )
        return v_dist

    def q_value_at_action(q_dist: jnp.ndarray, action: jnp.ndarray) -> jnp.ndarray:
        """Extracts the value at the given action from the Q distribution."""
        # q_dist: (b, num_quantiles, num_actions)
        # action: (b, )
        return jnp.take_along_axis(q_dist, action[:, None, None], axis=-1).squeeze(
            axis=-1
        )
        # (b, num_quantiles)

    if args.use_q_value_head:
        extract_value = q_extract_value
        value_at_action = q_value_at_action
    else:
        extract_value = v_extract_value
        value_at_action = v_value_at_action

    return extract_value, value_at_action


def make_network_apply_fns(args):
    def representation_apply_fn(
        nodes: jnp.ndarray,
        edges: jnp.ndarray,
        senders: jnp.ndarray,
        receivers: jnp.ndarray,
        aux: jnp.ndarray,
    ):
        """Applies the representation model to the observation."""
        obs_encoder = ObsEncoder(num_hidden=args.num_hidden)
        return jax.vmap(obs_encoder)(nodes, edges, senders, receivers, aux)

    def projection_apply_fn(
        node_embedding: jnp.ndarray,
        edge_embeddings: jnp.ndarray,
        aux_embedding: jnp.ndarray,
    ) -> jnp.ndarray:
        """Applies the projection model to the observation embedding."""
        projection_head = ProjectionHead(
            num_hidden=args.num_hidden, num_out=args.num_hidden
        )
        return jax.vmap(projection_head)(node_embedding, edge_embeddings, aux_embedding)

    def critic_apply_fn(
        node_embedding: jnp.ndarray,
        aux_embedding: jnp.ndarray,
    ) -> jnp.ndarray:
        """Applies the critic model to the observation embedding."""
        value_head = QValueHead(
            num_hidden=args.num_hidden,
            num_quantiles=args.num_quantiles,
        )
        return jax.vmap(value_head)(node_embedding, aux_embedding)

    def reward_history_apply_fn(
        nodes: jnp.ndarray,
        edges: jnp.ndarray,
        senders: jnp.ndarray,
        receivers: jnp.ndarray,
        aux: jnp.ndarray,
    ) -> jnp.ndarray:
        """Apply the reward history head to the observation embedding."""
        net = RewardHistoryHead(
            num_hidden=args.num_hidden,
            num_quantiles=args.num_quantiles,
        )
        return jax.vmap(net)(nodes, edges, senders, receivers, aux)

    def policy_apply_fn(
        node_embedding: jnp.ndarray,
        aux_embedding: jnp.ndarray,
    ) -> jnp.ndarray:
        """Applies the policy model to the observation embedding."""
        policy_logits = PolicyHead(num_hidden=args.num_hidden)
        return jax.vmap(policy_logits)(node_embedding, aux_embedding)

    def recurrent_inference_fn(
        node_embeddings,
        edge_embeddings,
        aux_embedding,
        senders,
        receivers,
        selected_node_index,
    ) -> jnp.ndarray:
        dynamics_head = DynamicsHead(
            num_hidden=args.num_hidden,
            num_quantiles=args.num_quantiles,
        )
        return jax.vmap(dynamics_head)(
            node_embeddings,
            edge_embeddings,
            aux_embedding,
            senders,
            receivers,
            selected_node_index,
        )

    def init_model_fn(nodes, edges, senders, receivers, aux, selected_node_index):
        """Just for tracing through the entire model."""
        obs_encoder = ObsEncoder(num_hidden=args.num_hidden)
        node_embedding, aux_embedding, edge_embeddings = jax.vmap(obs_encoder)(
            nodes, edges, senders, receivers, aux
        )  # (n_nodes, hidden_channels), (out_channels,), (edges, hidden_channels)

        projection_head = ProjectionHead(
            num_hidden=args.num_hidden, num_out=args.num_hidden
        )
        projection = jax.vmap(projection_head)(
            node_embedding, edge_embeddings, aux_embedding
        )  # (hidden,)

        policy_head = PolicyHead(num_hidden=args.num_hidden)
        policy_logits = jax.vmap(policy_head)(
            node_embedding, aux_embedding
        )  # (n_nodes,)

        value_head = QValueHead(
            num_hidden=args.num_hidden,
            num_quantiles=args.num_quantiles,
        )
        value_dist = jax.vmap(value_head)(
            node_embedding, aux_embedding
        )  # (num_quantiles, n_nodes)

        reward_history_head = RewardHistoryHead(
            num_hidden=args.num_hidden, num_quantiles=args.num_quantiles
        )
        reward_history = jax.vmap(reward_history_head)(
            nodes, edges, senders, receivers, aux
        )  # (num_quantiles,)

        dynamics_head = DynamicsHead(
            num_hidden=args.num_hidden,
            num_quantiles=args.num_quantiles,
        )
        dynamics_output = jax.vmap(dynamics_head)(
            node_embedding,
            edge_embeddings,
            aux_embedding,
            senders,
            receivers,
            selected_node_index,
        )
        return policy_logits, value_dist, reward_history, dynamics_output, projection

    init_model = hk.without_apply_rng(hk.transform(init_model_fn))
    policy_apply = hk.without_apply_rng(hk.transform(policy_apply_fn))
    representation_apply = hk.without_apply_rng(hk.transform(representation_apply_fn))
    projection_apply = hk.without_apply_rng(hk.transform(projection_apply_fn))
    critic_apply = hk.without_apply_rng(hk.transform(critic_apply_fn))
    reward_history_apply = hk.without_apply_rng(hk.transform(reward_history_apply_fn))
    recurrent_inference = hk.without_apply_rng(hk.transform(recurrent_inference_fn))

    return (
        init_model,
        representation_apply,
        projection_apply,
        policy_apply,
        critic_apply,
        reward_history_apply,
        recurrent_inference,
    )


if __name__ == "__main__":
    # Example usage
    args = NamedTuple(
        "Args",
        [
            ("num_hidden", int),
            ("num_actions", int),
            ("num_quantiles", int),
            ("cvar_alpha", float),
            ("use_q_value_head", bool),
        ],
    )(
        num_hidden=64,
        num_actions=4,
        num_quantiles=32,
        cvar_alpha=0.25,
        use_q_value_head=True,
    )

    (
        init_model,
        representation_apply,
        projection_apply,
        policy_apply,
        critic_apply,
        reward_history_apply,
        recurrent_inference,
    ) = make_network_apply_fns(args)

    # Example usage
    nodes = jnp.array([[[1.0, 2.0], [3.0, 4.0]]])
    edges = jnp.array([[[1.0, 0.5], [0.5, 1.0]]])
    senders = jnp.array([[0, 1]])
    receivers = jnp.array([[1, 0]])
    aux = jnp.array([[0.1, 0.2]])

    params = init_model.init(
        jax.random.PRNGKey(42), nodes, edges, senders, receivers, aux, jnp.array([0])
    )

    node_reps, pool_aux_concat, edge_embeddings = representation_apply.apply(
        params, nodes, edges, senders, receivers, aux
    )
    print("node_reps shape:", node_reps.shape)
    print("pool_aux_concat shape:", pool_aux_concat.shape)
    print("edge_embeddings shape:", edge_embeddings.shape)
    print()

    projection = projection_apply.apply(params, pool_aux_concat)
    print("projection shape:", projection.shape)
    print()

    policy_logits = policy_apply.apply(params, node_reps)
    print("policy_logits shape:", policy_logits.shape)
    print()

    value_dist = critic_apply.apply(params, node_reps)
    print("value_dist shape:", value_dist.shape)
    print()

    reward_history = reward_history_apply.apply(params, pool_aux_concat)
    print("reward_history shape:", reward_history.shape)
    print()

    node_reps, edge_embeddings, r_dist, pool_rep = recurrent_inference.apply(
        params, node_reps, edge_embeddings, senders, receivers, jnp.array([0])
    )
    print("recurrent node_reps shape:", node_reps.shape)
    print("recurrent edge_embeddings shape:", edge_embeddings.shape)
    print("recurrent r_dist shape:", r_dist.shape)
    print("recurrent pool_rep shape:", pool_rep.shape)
    print()
