"""Script containing the base simulation kernel class."""
from algorithms.envs.flow.core.kernel.simulation.base import KernelSimulation
from algorithms.envs.flow.utils.aimsun.api import FlowAimsunAPI
import os.path as osp
import csv
from algorithms.envs.flow.core.util import ensure_dir


class AimsunKernelSimulation(KernelSimulation):
    """Aimsun simulation kernel.

    Extends KernelSimulation.

    Attributes
    ----------
    master_kernel : flow.core.kernel.Kernel
        the higher level kernel (used to call methods from other sub-kernels)
    kernel_api : any
        an API that may be used to interact with the simulator
    sim_step : float
        seconds per simulation step; 0.5 by default
    emission_path : str
        Path to the folder in which to create the emissions output.
        Emissions output is not generated if this value is not specified
    time : float
        used to internally keep track of the simulation time
    stored_data : dict
        a file used to store data if an emission file is provided
    """

    def __init__(self, master_kernel):
        """Initialize the Aimsun simulation kernel."""
        KernelSimulation.__init__(self, master_kernel)

        self.master_kernel = master_kernel
        self.kernel_api = None
        self.sim_step = None
        self.emission_path = None
        self.time = 0
        self.stored_data = {
            'time': [],
            'x': [],
            'y': [],
            'angle': [],
            'type': [],
            'id': [],
            'relative_position': [],
            'speed': [],
            'edge_id': [],
            'lane_number': []
        }

    def pass_api(self, kernel_api):
        """See parent class."""
        self.kernel_api = kernel_api

    def start_simulation(self, network, sim_params):
        """See parent class.

        This method calls the aimsun generator to generate the network, starts
        a simulation, and creates a class to communicate with the simulation
        via an TCP connection.
        """
        # save the simulation step size (for later use)
        self.sim_step = sim_params.sim_step

        self.emission_path = sim_params.emission_path
        if self.emission_path is not None:
            ensure_dir(self.emission_path)

        return FlowAimsunAPI(port=sim_params.port)

    def simulation_step(self):
        """See parent class."""
        self.kernel_api.simulation_step()

    def update(self, reset):
        """See parent class.

        No update is needed in this case.
        """
        if reset:
            self.time = 0
        else:
            self.time += self.sim_step

        if self.emission_path is not None:
            for veh_id in self.master_kernel.vehicle.get_ids():
                pos = self.master_kernel.vehicle.get_position_world(veh_id)
                self.stored_data['id'].append(
                    veh_id)
                self.stored_data['time'].append(
                    self.time)
                self.stored_data['type'].append(
                    self.master_kernel.vehicle.get_type(veh_id))
                self.stored_data['x'].append(
                    pos[0])
                self.stored_data['y'].append(
                    pos[1])
                self.stored_data['relative_position'].append(
                    self.master_kernel.vehicle.get_position(veh_id))
                self.stored_data['angle'].append(
                    self.master_kernel.vehicle.get_angle(veh_id))
                self.stored_data['speed'].append(
                    self.master_kernel.vehicle.get_speed(veh_id))
                self.stored_data['edge_id'].append(
                    self.master_kernel.vehicle.get_edge(veh_id))
                self.stored_data['lane_number'].append(
                    self.master_kernel.vehicle.get_lane(veh_id))

    def check_collision(self):
        """See parent class."""
        return False

    def close(self):
        """See parent class."""
        # save the emission data to a csv
        if self.emission_path is not None:
            name = "%s_emission.csv" % self.master_kernel.network.network.name
            with open(osp.join(self.emission_path, name), "w") as f:
                writer = csv.writer(f, delimiter=',')
                writer.writerow(self.stored_data.keys())
                writer.writerows(zip(*self.stored_data.values()))

        # close the API and simulation process
        try:
            self.kernel_api.stop_simulation()
            self.master_kernel.network.aimsun_proc.kill()
        except OSError:
            # in case no simulation originally existed (used by the visualizer)
            pass
