"""Script containing the base vehicle kernel class."""
from algorithms.envs.flow.core.kernel.vehicle.base import KernelVehicle
import collections
import numpy as np
from algorithms.envs.flow.utils.aimsun.struct import InfVeh
from algorithms.envs.flow.controllers.car_following_models import SimCarFollowingController
from algorithms.envs.flow.controllers.rlcontroller import RLController
from algorithms.envs.flow.controllers.lane_change_controllers import SimLaneChangeController

# import time

# colors for vehicles
WHITE = (255, 255, 255)
CYAN = (0, 255, 255)
RED = (255, 0, 0)

# this is used when identifying if a specific object is tracked
INFOS_ATTR_BY_INDEX = [
    'CurrentPos', 'distance2End', 'xCurrentPos', 'yCurrentPos', 'zCurrentPos',
    'xCurrentPosBack', 'yCurrentPosBack', 'zCurrentPosBack', 'CurrentSpeed',
    'TotalDistance', 'SectionEntranceT', 'CurrentStopTime', 'stopped',
    'idSection', 'segment', 'numberLane', 'idJunction', 'idSectionFrom',
    'idLaneFrom', 'idSectionTo', 'idLaneTo'
]


class AimsunKernelVehicle(KernelVehicle):
    """Aimsun vehicle kernel.

    Extends KernelVehicle.
    """

    def __init__(self,
                 master_kernel,
                 sim_params):
        """See parent class."""
        KernelVehicle.__init__(self, master_kernel, sim_params)

        self.__ids = []  # ids of all vehicles
        self.__human_ids = []  # ids of human-driven vehicles
        self.__controlled_ids = []  # ids of flow-controlled vehicles
        self.__controlled_lc_ids = []  # ids of flow lc-controlled vehicles
        self.__rl_ids = []  # ids of rl-controlled vehicles
        self.__observed_ids = []  # ids of the observed vehicles

        # vehicles: Key = Vehicle ID, Value = Dictionary describing the vehicle
        # Ordered dictionary used to keep neural net inputs in order
        self.__vehicles = collections.OrderedDict()

        # total number of vehicles in the network
        self.num_vehicles = 0
        # number of rl vehicles in the network
        self.num_rl_vehicles = 0

        # contains the parameters associated with each type of vehicle
        self.type_parameters = {}

        # list of vehicle ids located in each edge in the network
        self._ids_by_edge = dict()

        # number of vehicles that entered the network for every time-step
        self._num_departed = []
        self._departed_ids = []

        # number of vehicles to exit the network for every time-step
        self._num_arrived = []
        self._arrived_ids = []
        self._arrived_rl_ids = []

        # contains conversion from Flow-ID to Aimsun-ID
        self._id_aimsun2flow = {}
        self._id_flow2aimsun = {}

        # current number of vehicles of each type
        self.num_type = {}
        # total number of vehicles of each type (used for naming them)
        self.total_num_type = {}

        # type of vehicles that will be tracked
        # note: vehicles added via the network (i.e. by calling the
        # add_vehicle function) will also be tracked, even if their
        # type is not specified here
        self.tracked_vehicle_types = {"rl", "idm"}  # TODO maybe generic

        # all the vehicle tracking information that should be stored
        # for the tracked vehicles info that can be tracked:
        # CurrentPos, distance2End, xCurrentPos, yCurrentPos, zCurrentPos,
        # xCurrentPosBack, yCurrentPosBack, zCurrentPosBack, CurrentSpeed,
        # TotalDistance, SectionEntranceT, CurrentStopTime, stopped,
        # idSection, segment, numberLane, idJunction, idSectionFrom,
        # idLaneFrom, idSectionTo, idLaneTo
        self.tracked_info_bitmap = self.make_bitmap_for_tracking({
            'CurrentPos', 'distance2End',
            'xCurrentPos', 'yCurrentPos', 'xCurrentPosBack', 'yCurrentPosBack',
            'CurrentSpeed', 'numberLane',
            'idSection', 'idJunction', 'idSectionFrom', 'idSectionTo'
        })
        # FIXME lots of these used in simulation/aimsun.py, used when
        # we want to store the values in an emission file (necessary?)

    def initialize(self, vehicles):
        """Initialize vehicle state information.

        This is responsible for collecting vehicle type information from the
        VehicleParams object and placing them within the Vehicles kernel.

        Parameters
        ----------
        vehicles : flow.core.params.VehicleParams
            initial vehicle parameter information, including the types of
            individual vehicles and their initial speeds
        """
        self.type_parameters = vehicles.type_parameters
        self.num_vehicles = 0
        self.num_rl_vehicles = 0

        self.__vehicles.clear()
        for typ in vehicles.initial:
            for i in range(typ['num_vehicles']):
                veh_id = '{}_{}'.format(typ['veh_id'], i)
                self.__vehicles[veh_id] = dict()
                self.__vehicles[veh_id]['type'] = typ['veh_id']
                self.__vehicles[veh_id]['type_name'] = typ['veh_id']  # FIXME
                self.__vehicles[veh_id]['initial_speed'] = typ['initial_speed']
                self.num_vehicles += 1
                if typ['acceleration_controller'][0] == RLController:
                    self.num_rl_vehicles += 1

        # for tracked_type in self.tracked_vehicle_types:
        #     self.num_type[tracked_type] = 0
        #     self.total_num_type[tracked_type] = 0
        #     self.type_parameters[tracked_type] = {}

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

    def make_bitmap_for_tracking(self, infos):
        """Create a bitmap object to simplify observation collection.

        Parameters
        ----------
        infos : list
            set containing all infos that we want (see list of info in
            __init__)

        Returns
        -------
        str
            a corresponding bitmap to be used in the
            self.kernel_api.get_vehicle_tracking_info function
        """
        bitmap = ""

        for attr in INFOS_ATTR_BY_INDEX:
            bitmap += '1' if attr in infos else '0'

        return bitmap

    ###########################################################################
    #               Methods for interacting with the simulator                #
    ###########################################################################

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

        This is used to store an updated vehicle information object.
        """
        # for veh_type in self.tracked_vehicle_types:
        #     print("- Type:", veh_type, ", count:",
        #           self.num_type[veh_type], ", total since start:",
        #           self.total_num_type[veh_type])

        # collect the entered and exited vehicle_ids
        added_vehicles = self.kernel_api.get_entered_ids()
        exited_vehicles = self.kernel_api.get_exited_ids()

        # keep track of arrived rl vehicles
        arrived_rl_ids = []

        # add the new vehicles if they should be tracked
        for aimsun_id in added_vehicles:
            veh_type = self.kernel_api.get_vehicle_type_name(aimsun_id)
            if veh_type in self.tracked_vehicle_types:
                self._add_departed(aimsun_id)
            if aimsun_id in self.get_rl_ids():
                arrived_rl_ids.append(aimsun_id)
        self._arrived_rl_ids.append(arrived_rl_ids)

        # remove the exited vehicles if they were tracked
        if not reset:
            for aimsun_id in exited_vehicles:
                if aimsun_id in self._id_aimsun2flow:
                    self.remove(aimsun_id)

        # start = time.time()

        for veh_id in self.__ids:
            aimsun_id = self._id_flow2aimsun[veh_id]

            # update the vehicle's tracking information
            self.__vehicles[veh_id]['tracking_info'] = \
                self.kernel_api.get_vehicle_tracking_info(
                    aimsun_id, self.tracked_info_bitmap
                )

            # get the leader, follower, and headway for each tracked vehicle
            lead_id_aimsun = self.kernel_api.get_vehicle_leader(aimsun_id)
            if lead_id_aimsun < -1:
                self.__vehicles[veh_id]['leader'] = None
                self.__vehicles[veh_id]['headway'] = 1000
            else:
                inf_veh = self.__vehicles[veh_id]['tracking_info']

                if lead_id_aimsun in self._id_aimsun2flow:
                    lead_id = self._id_aimsun2flow[lead_id_aimsun]
                    inf_veh_leader = self.__vehicles[lead_id]['tracking_info']
                    leader_length = self.__vehicles[lead_id]['static_info'].\
                        length
                    self.__vehicles[veh_id]['leader'] = lead_id
                    self.__vehicles[lead_id]['follower'] = veh_id
                else:
                    # TODO can be hardcoded when we won't change it anymore
                    tracked_info_leader = self.make_bitmap_for_tracking({
                        'CurrentPos', 'distance2End',
                        'idSection', 'idJunction',
                        'idSectionFrom', 'idSectionTo'
                    })

                    inf_veh_leader = self.kernel_api.get_vehicle_tracking_info(
                        lead_id_aimsun, tracked_info_leader, tracked=False
                    )
                    leader_length = self.kernel_api.\
                        get_vehicle_length(lead_id_aimsun)
                    self.__vehicles[veh_id]['leader'] = -1

                # FIXME can be simplified
                if inf_veh.idSection != -1:  # vehicle is in a section
                    next_section = self.kernel_api.get_next_section(
                        aimsun_id, inf_veh.idSection)
                    # leader is in a section
                    if inf_veh_leader.idSection != -1:
                        # veh in section and leader in same section
                        if inf_veh.idSection == inf_veh_leader.idSection:
                            gap = inf_veh_leader.CurrentPos\
                                - inf_veh.CurrentPos - leader_length
                        # veh in section and leader in next section
                        elif inf_veh_leader.idSection == next_section:
                            gap = inf_veh.distance2End\
                                + inf_veh_leader.CurrentPos - leader_length
                            # TODO need to add junction length (we have
                            # turning id -> get its length)
                        # veh in section and leader several sections ahead
                        else:
                            # TODO
                            gap = 1001
                    else:
                        # veh in section and leader in next junction
                        if inf_veh_leader.idSectionFrom == inf_veh.idSection:
                            gap = inf_veh.distance2End\
                                + inf_veh_leader.CurrentPos - leader_length
                        # veh in section and leader several junctions ahead
                        else:
                            # TODO
                            gap = 1002
                else:
                    if inf_veh_leader.idSection == -1:
                        # veh in junction and leader in same junction
                        if inf_veh.idJunction == inf_veh_leader.idJunction:
                            gap = inf_veh_leader.CurrentPos\
                                - inf_veh.CurrentPos - leader_length
                        # veh in junction and leader in next junction
                        # veh in junction and leader several junctions ahead
                        else:
                            # TODO
                            gap = 1003
                    else:
                        # veh in junction and leader in next section
                        if inf_veh_leader.idSection == inf_veh.idSectionTo:
                            gap = inf_veh.distance2End\
                                + inf_veh_leader.CurrentPos - leader_length
                        # veh in junction and leader several sections ahead
                        else:
                            # TODO
                            gap = 1004

                self.__vehicles[veh_id]['headway'] = gap

        # end = time.time()
        # if len(self.__ids) > 0:
        #     print("update time per tracked vehicle (ms):",
        #           1000 * (end - start) / len(self.__ids))

    def _add_departed(self, aimsun_id):
        """See parent class."""
        # get vehicle information from API
        static_inf_veh = self.kernel_api.get_vehicle_static_info(aimsun_id)

        # get the vehicle's type
        type_id = self.kernel_api.get_vehicle_type_name(aimsun_id)

        self.kernel_api.set_vehicle_tracked(aimsun_id)

        # get the vehicle ID, or create a new vehicle ID if one doesn't exist
        # for the vehicle
        if aimsun_id not in self._id_aimsun2flow.keys():
            # get a new name for this vehicle
            if type_id not in self.num_type:
                self.num_type[type_id] = 0
                self.total_num_type[type_id] = 0
            veh_id = '{}_{}'.format(type_id, self.total_num_type[type_id])
            self.num_type[type_id] += 1
            self.total_num_type[type_id] += 1
            self.__ids.append(veh_id)
            self.__vehicles[veh_id] = {}
            # set the Aimsun/Flow vehicle ID converters
            self._id_aimsun2flow[aimsun_id] = veh_id
            self._id_flow2aimsun[veh_id] = aimsun_id
        else:
            veh_id = self._id_aimsun2flow[aimsun_id]

        # store the static info
        self.__vehicles[veh_id]["static_info"] = static_inf_veh
        self.__vehicles[veh_id]["type_name"] = type_id

        # store an empty tracking info object
        self.__vehicles[veh_id]['tracking_info'] = InfVeh()

        if type_id in self.type_parameters:
            # specify the acceleration controller class
            accel_controller = \
                self.type_parameters[type_id]["acceleration_controller"]
            car_following_params = \
                self.type_parameters[type_id]["car_following_params"]
            self.__vehicles[veh_id]["acc_controller"] = \
                accel_controller[0](veh_id,
                                    car_following_params=car_following_params,
                                    **accel_controller[1])

            # specify the lane-changing controller class
            lc_controller = \
                self.type_parameters[type_id]["lane_change_controller"]
            self.__vehicles[veh_id]["lane_changer"] = \
                lc_controller[0](veh_id=veh_id, **lc_controller[1])

            # specify the routing controller class
            rt_controller = self.type_parameters[type_id]["routing_controller"]
            if rt_controller is not None:
                self.__vehicles[veh_id]["router"] = \
                    rt_controller[0](veh_id=veh_id,
                                     router_params=rt_controller[1])
            else:
                self.__vehicles[veh_id]["router"] = None

            # FIXME should add RL controller to RL vehicle added by Aimsun via
            # load.py

            # add the vehicle's id to the list of vehicle ids
            if accel_controller[0] == RLController:
                self.__rl_ids.append(veh_id)
                self.num_rl_vehicles += 1
            else:
                self.__human_ids.append(veh_id)
                if accel_controller[0] != SimCarFollowingController:
                    self.__controlled_ids.append(veh_id)
                if lc_controller[0] != SimLaneChangeController:
                    self.__controlled_lc_ids.append(veh_id)

        # set the "last_lc" parameter of the vehicle
        self.__vehicles[veh_id]["last_lc"] = -float("inf")

        self.__human_ids.append(veh_id)  # FIXME not true for RL vehicles

        # make sure that the order of rl_ids is kept sorted
        self.__rl_ids.sort()

    def add(self, veh_id, type_id, edge, pos, lane, speed):
        """See parent class."""
        self.num_vehicles += 1
        self.__ids.append(veh_id)
        self.__vehicles[veh_id] = {}
        self.__vehicles[veh_id]["type_name"] = type_id

        # add vehicle in Aimsun
        # negative one means the first feasible turn TODO get route
        next_section = -1
        aimsun_id = self.kernel_api.add_vehicle(
            edge=self.master_kernel.network.aimsun_edge_name(edge),
            lane=lane,
            type_id=type_id,
            pos=pos,
            speed=speed,
            next_section=next_section)

        self.__vehicles[veh_id]['static_info'] =\
            self.kernel_api.get_vehicle_static_info(aimsun_id)
        self.__vehicles[veh_id]['tracking_info'] = InfVeh()

        # set the Aimsun/Flow vehicle ID converters
        self._id_aimsun2flow[aimsun_id] = veh_id
        self._id_flow2aimsun[veh_id] = aimsun_id

        # increment the number of vehicles of this type
        if type_id in self.num_type:
            self.num_type[type_id] += 1
            self.total_num_type[type_id] += 1
        else:
            self.num_type[type_id] = 1
            self.total_num_type[type_id] = 1

    def reset(self):
        """See parent class."""
        raise NotImplementedError

    def remove(self, aimsun_id):
        """See parent class."""
        veh_id = self._id_aimsun2flow[aimsun_id]
        self.kernel_api.remove_vehicle(aimsun_id)

        type_id = self.__vehicles[veh_id]['type_name']
        self.num_type[type_id] -= 1

        # remove from the vehicles kernel
        del self.__vehicles[veh_id]
        del self._id_aimsun2flow[aimsun_id]
        del self._id_flow2aimsun[veh_id]
        self.__ids.remove(veh_id)
        self.num_vehicles -= 1

        # remove it from all other ids (if it is there)
        if veh_id in self.__human_ids:
            self.__human_ids.remove(veh_id)
            if veh_id in self.__controlled_ids:
                self.__controlled_ids.remove(veh_id)
            if veh_id in self.__controlled_lc_ids:
                self.__controlled_lc_ids.remove(veh_id)
        elif veh_id in self.__rl_ids:
            # FIXME should be else
            self.__rl_ids.remove(veh_id)
            self.num_rl_vehicles -= 1

        # make sure that the rl ids remain sorted
        self.__rl_ids.sort()

    def apply_acceleration(self, veh_id, acc):
        """See parent class."""
        # to hand the case of a single vehicle
        if type(veh_id) == str:
            veh_id = [veh_id]
            acc = [acc]

        for i, veh_id in enumerate(veh_id):
            if acc[i] is not None:
                this_vel = self.get_speed(veh_id)
                next_vel = max(this_vel + acc[i] * self.sim_step, 0)
                aimsun_id = self._id_flow2aimsun[veh_id]
                self.kernel_api.set_speed(aimsun_id, next_vel)

    def apply_lane_change(self, veh_id, direction):
        """Apply an instantaneous lane-change to a set of vehicles.

        This method also prevents vehicles from moving to lanes that do not
        exist, and set the "last_lc" variable for RL vehicles that lane changed
        to match the current time step, in order to assist in maintaining a
        lane change duration for these vehicles.

        Parameters
        ----------
        veh_id : str or list of str
            list of vehicle identifiers
        direction : {-2, -1, 0, 1} or list of {-2, -1, 0, 1}
            -2: reset, gives back the control to the default simulation model
            -1: lane change to the right
             0: no lane change
             1: lane change to the left

        Raises
        ------
        ValueError
            If any of the direction values are not -2, -1, 0, or 1.
        """
        # to hand the case of a single vehicle
        if type(veh_id) == str:
            veh_id = [veh_id]
            direction = [direction]

        # if any of the directions are not -1, 0, or 1, raise a ValueError
        if any(d not in [-2, -1, 0, 1] for d in direction):
            raise ValueError(
                "Direction values for lane changes may only be: -2, -1, 0, \
                or 1.")

        for i, veh_id in enumerate(veh_id):
            # check for no lane change
            if direction[i] == 0:
                continue

            # compute the target lane, and clip it so vehicle don't try to lane
            # change out of range
            this_lane = self.get_lane(veh_id)
            this_edge = self.get_edge(veh_id)
            target_lane = min(
                max(this_lane + direction[i], 0),
                self.master_kernel.network.num_lanes(this_edge) - 1)

            # perform the requested lane action action in Aimsun
            if target_lane != this_lane:
                aimsun_id = self._id_flow2aimsun[veh_id]
                self.kernel_api.apply_lane_change(aimsun_id, int(target_lane))

                if veh_id in self.get_rl_ids():
                    self.prev_last_lc[veh_id] = \
                        self.__vehicles[veh_id]["last_lc"]

    def choose_routes(self, veh_id, route_choices):
        """Update the route choice of vehicles in the network.

        Parameters
        ----------
        veh_id : str or list of str
            list of vehicle identifiers
        route_choices : array_like
            list of edges the vehicle wishes to traverse, starting with the
            edge the vehicle is currently on. If a value of None is provided,
            the vehicle does not update its route
        """
        raise NotImplementedError  # FIXME
        # for i, veh_id in enumerate(veh_ids):
        #     if route_choices[i] is not None:
        #         aimsun_id = self._id_flow2aimsun[veh_id]
        #         size_next_sections = len(route_choices[i])
        #         self.kernel_api.AKIVehTrackedModifyNextSections(
        #             aimsun_id, size_next_sections, route_choices[i])

    def set_max_speed(self, veh_id, max_speed):
        """See parent class."""
        raise NotImplementedError

    ###########################################################################
    # Methods to visually distinguish vehicles by {RL, observed, unobserved}  #
    ###########################################################################

    def update_vehicle_colors(self):
        """Modify the color of vehicles if rendering is active."""
        # color rl vehicles red
        for veh_id in self.get_rl_ids():
            aimsun_id = self._id_flow2aimsun[veh_id]
            self.kernel_api.set_color(veh_id=aimsun_id, color=RED)

        # observed human-driven vehicles are cyan and unobserved are white
        for veh_id in self.get_human_ids():
            aimsun_id = self._id_flow2aimsun[veh_id]
            color = CYAN if veh_id in self.get_observed_ids() else WHITE
            self.kernel_api.set_color(veh_id=aimsun_id, color=color)

        # clear the list of observed vehicles
        for veh_id in self.get_observed_ids():
            self.remove_observed(veh_id)

    def set_observed(self, veh_id):
        """Add a vehicle to the list of observed vehicles."""
        if veh_id not in self.__observed_ids:
            self.__observed_ids.append(veh_id)

    def remove_observed(self, veh_id):
        """Remove a vehicle from the list of observed vehicles."""
        if veh_id in self.__observed_ids:
            self.__observed_ids.remove(veh_id)

    def get_observed_ids(self):
        """Return the list of observed vehicles."""
        return self.__observed_ids

    def get_color(self, veh_id):
        """See parent class."""
        raise NotImplementedError

    def set_color(self, veh_id, color):
        """See parent class."""
        raise NotImplementedError

    ###########################################################################
    #                        State acquisition methods                        #
    ###########################################################################

    def get_orientation(self, veh_id):
        """See parent class."""
        raise NotImplementedError

    def get_timestep(self, veh_id):
        """See parent class."""
        raise NotImplementedError

    def get_timedelta(self, veh_id):
        """See parent class."""
        raise NotImplementedError

    def get_ids(self):
        """See parent class."""
        return self.__ids

    def get_human_ids(self):
        """See parent class."""
        return self.__human_ids

    def get_controlled_ids(self):
        """See parent class."""
        return self.__controlled_ids

    def get_controlled_lc_ids(self):
        """See parent class."""
        return self.__controlled_lc_ids

    def get_rl_ids(self):
        """See parent class."""
        return self.__rl_ids

    def get_ids_by_edge(self, edges):
        """See parent class."""
        if isinstance(edges, (list, np.ndarray)):
            return sum([self.get_ids_by_edge(edge) for edge in edges], [])
        return [veh for veh in self.__ids if self.get_edge(veh) == edges]

    def get_inflow_rate(self, time_span):
        """See parent class."""
        if len(self._num_departed) == 0:
            return 0
        num_inflow = self._num_departed[-int(time_span / self.sim_step):]
        return 3600 * sum(num_inflow) / (len(num_inflow) * self.sim_step)

    def get_outflow_rate(self, time_span):
        """See parent class."""
        if len(self._num_arrived) == 0:
            return 0
        num_outflow = self._num_arrived[-int(time_span / self.sim_step):]
        return 3600 * sum(num_outflow) / (len(num_outflow) * self.sim_step)

    def get_num_arrived(self):
        """See parent class."""
        if len(self._num_arrived) > 0:
            return self._num_arrived[-1]
        else:
            return 0

    def get_arrived_ids(self):
        """See parent class."""
        raise NotImplementedError

    def get_arrived_rl_ids(self, k=1):
        """See parent class."""
        if len(self._arrived_rl_ids) > 0:
            arrived = []
            for arr in self._arrived_rl_ids[-k:]:
                arrived.extend(arr)
            return arrived
        else:
            return 0

    def get_departed_ids(self):
        """See parent class."""
        raise NotImplementedError

    def get_num_not_departed(self):
        """See parent class."""
        raise NotImplementedError

    def get_fuel_consumption(self):
        """See parent class."""
        raise NotImplementedError

    def get_type(self, veh_id):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_type(veh) for veh in veh_id]
        return self.__vehicles.get(veh_id, {}).get('type_name', "")

    def get_initial_speed(self, veh_id):
        """See parent class."""
        return self.__vehicles[veh_id]["initial_speed"]

    def get_speed(self, veh_id, error=-1001):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_speed(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['tracking_info'].CurrentSpeed / 3.6

    def get_default_speed(self, veh_id, error=-1001):
        """See parent class."""
        raise NotImplementedError

    def get_position(self, veh_id, error=-1001):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_position(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['tracking_info'].CurrentPos

    def get_position_world(self, veh_id, error=-1001):
        """Return the position of the vehicle relative to its current edge.

        Parameters
        ----------
        veh_id : str or list of str
            vehicle id, or list of vehicle ids
        error : any, optional
            value that is returned if the vehicle is not found

        Returns
        -------
        float
            x position
        float
            y position
        float
            z position
        """
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_position_world(veh, error) for veh in veh_id]
        x_pos = self.__vehicles[veh_id]['tracking_info'].xCurrentPos
        y_pos = self.__vehicles[veh_id]['tracking_info'].yCurrentPos
        z_pos = self.__vehicles[veh_id]['tracking_info'].zCurrentPos
        return [x_pos, y_pos, z_pos]

    def get_edge(self, veh_id, error=""):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_edge(veh, error) for veh in veh_id]
        edge_aimsun_id = self.__vehicles[veh_id]['tracking_info'].idSection
        if edge_aimsun_id < 0:
            # TODO: add from and to lanes in junctions
            from_edge = self.master_kernel.network.flow_edge_name(
                self.__vehicles[veh_id]['tracking_info'].idSectionFrom)
            to_edge = self.master_kernel.network.flow_edge_name(
                self.__vehicles[veh_id]['tracking_info'].idSectionTo)
            return '{}_to_{}'.format(from_edge, to_edge)
        else:
            return self.master_kernel.network.flow_edge_name(edge_aimsun_id)

    def get_angle(self, veh_id, error=-1001):
        """Return the angle of the vehicle.

        Parameters
        ----------
        veh_id : str or list of str
            vehicle id, or list of vehicle ids
        error : any, optional
            value that is returned if the vehicle is not found

        Returns
        -------
        float
            the angle of the vehicle
        """
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_angle(veh, error) for veh in veh_id]

        if veh_id not in self.__vehicles:
            return error
        else:
            x2 = self.__vehicles[veh_id]['tracking_info'].xCurrentPos
            y2 = self.__vehicles[veh_id]['tracking_info'].yCurrentPos
            x1 = self.__vehicles[veh_id]['tracking_info'].xCurrentPosBack
            y1 = self.__vehicles[veh_id]['tracking_info'].yCurrentPosBack
            return np.arctan2(y2-y1, x2-x1)

    def get_lane(self, veh_id, error=-1001):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_lane(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['tracking_info'].numberLane

    def get_route(self, veh_id, error=None):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_route(veh) for veh in veh_id]
        return []  # FIXME
        # aimsun_id = self._id_flow2aimsun[veh_id]
        # num_secs = self.kernel_api.AKIVehTrackedGetNbSectionsVehiclePath(
        #     aimsun_id)
        # return self.kernel_api.AKIVehTrackedGetIdSectionVehiclePath(
        #     aimsun_id, num_secs)

    def get_length(self, veh_id, error=-1001):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_length(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['static_info'].length

    def get_leader(self, veh_id, error=""):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_leader(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['leader'] or error

    def get_follower(self, veh_id, error=""):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_follower(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['follower']

    def get_headway(self, veh_id, error=-1001):
        """See parent class."""
        if isinstance(veh_id, (list, np.ndarray)):
            return [self.get_headway(veh, error) for veh in veh_id]
        return self.__vehicles[veh_id]['headway']

    def get_last_lc(self, veh_id, error=-1001):
        """See parent class."""
        return self.__vehicles[veh_id]["last_lc"]

    def get_acc_controller(self, veh_id, error=None):
        """See parent class."""
        return self.__vehicles[veh_id]["acc_controller"]

    def get_lane_changing_controller(self, veh_id, error=None):
        """See parent class."""
        return self.__vehicles[veh_id]["lane_changer"]

    def get_routing_controller(self, veh_id, error=None):
        """See parent class."""
        # TODO FIXME
        try:
            return self.__vehicles[veh_id]["router"]
        except KeyError:
            return None

    def get_x_by_id(self, veh_id):
        """See parent class."""
        return self.master_kernel.network.get_x(self.get_edge(veh_id),
                                                self.get_position(veh_id))

    def set_lane_headways(self, veh_id, lane_headways):
        """See parent class."""
        raise NotImplementedError

    def get_lane_headways(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def set_lane_leaders(self, veh_id, lane_leaders):
        """See parent class."""
        raise NotImplementedError

    def get_lane_leaders(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def set_lane_tailways(self, veh_id, lane_tailways):
        """See parent class."""
        raise NotImplementedError

    def get_lane_tailways(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def set_lane_followers(self, veh_id, lane_followers):
        """See parent class."""
        raise NotImplementedError

    def get_lane_followers(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def get_lane_followers_speed(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def get_lane_leaders_speed(self, veh_id, error=None):
        """See parent class."""
        raise NotImplementedError

    def get_max_speed(self, veh_id, error):
        """See parent class."""
        raise NotImplementedError
