Source code for stonesoup.models.measurement.nonlinear

from abc import ABC
import copy
from typing import Sequence, Tuple, Union

from math import sqrt
import numpy as np
from scipy.linalg import inv, pinv, block_diag
from scipy.stats import multivariate_normal

from ...base import Property, clearable_cached_property
from ...types.numeric import Probability

from ...functions import cart2pol, pol2cart, \
    cart2sphere, sphere2cart, cart2angles, \
    build_rotation_matrix, cart2az_el_rg, az_el_rg2cart
from ...types.array import StateVector, CovarianceMatrix, StateVectors
from ...types.angle import Bearing, Elevation, Azimuth
from ..base import LinearModel, GaussianModel, ReversibleModel
from .base import MeasurementModel


[docs] class CombinedReversibleGaussianMeasurementModel(ReversibleModel, GaussianModel, MeasurementModel): r"""Combine multiple models into a single model by stacking them. The assumption is that all models are Gaussian, and must be combination of :class:`~.LinearModel` and :class:`~.NonLinearModel` models. They must all expect the same dimension state vector (i.e. have the same :attr:`~.MeasurementModel.ndim_state`), using model mapping as appropriate. This also implements the :meth:`inverse_function`, but will raise a :exc:`NotImplementedError` if any model isn't either a :class:`~.LinearModel` or :class:`~.ReversibleModel`. """ model_list: Sequence[GaussianModel] = Property(doc="List of Measurement Models.") def __init__(self, *args, **kwargs): super().__init__(*args, **kwargs) for model in self.model_list: if model.ndim_state != self.ndim_state: raise ValueError("Models must all have the same `ndim_state`") @property def ndim_state(self) -> int: """Number of state dimensions""" return self.model_list[0].ndim_state @property def ndim_meas(self) -> int: return sum(model.ndim_meas for model in self.model_list) @property def mapping(self): return [x for model in self.model_list for x in model.mapping]
[docs] def function(self, state, **kwargs) -> StateVector: return np.vstack([model.function(state, **kwargs) for model in self.model_list]).view(StateVector)
@staticmethod def _linear_inverse_function(model, state, **kwargs): model_matrix = model.matrix(**kwargs) inv_model_matrix = pinv(model_matrix) return inv_model_matrix@state.state_vector
[docs] def inverse_function(self, detection, **kwargs) -> StateVector: state = copy.copy(detection) ndim_count = 0 state_vector = np.zeros((self.ndim_state, 1)).view(StateVector) for model in self.model_list: state.state_vector = detection.state_vector[ndim_count:model.ndim_meas + ndim_count, :] if isinstance(model, ReversibleModel): state_vector += model.inverse_function(state, **kwargs) elif isinstance(model, LinearModel): state_vector += self._linear_inverse_function(model, state, **kwargs) else: raise NotImplementedError( "Model {!r} not reversible".format(type(model))) ndim_count += model.ndim_meas return state_vector
[docs] def covar(self, **kwargs) -> CovarianceMatrix: return block_diag( *(model.covar(**kwargs) for model in self.model_list) ).view(CovarianceMatrix)
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: rvs_vectors = np.vstack([model.rvs(num_samples, **kwargs) for model in self.model_list]) if num_samples == 1: return rvs_vectors.view(StateVector) else: return rvs_vectors.view(StateVectors)
[docs] class NonLinearGaussianMeasurement(MeasurementModel, GaussianModel, ABC): r"""This class combines the MeasurementModel, NonLinearModel and \ GaussianModel classes. It is not meant to be instantiated directly \ but subclasses should be derived from this class. """ noise_covar: CovarianceMatrix = Property(doc="Noise covariance") rotation_offset: StateVector = Property( default=None, doc="A 3x1 array of angles (rad), specifying the clockwise rotation " "around each Cartesian axis in the order :math:`x,y,z`. " "The rotation angles are positive if the rotation is in the " "counter-clockwise direction when viewed by an observer looking " "along the respective rotation axis, towards the origin.") def __init__(self, *args, **kwargs): """ Ensure that the rotation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.rotation_offset is None: self.rotation_offset = StateVector([[0], [0], [0]]) if not isinstance(self.noise_covar, CovarianceMatrix): self.noise_covar = CovarianceMatrix(self.noise_covar)
[docs] def covar(self, **kwargs) -> CovarianceMatrix: """Returns the measurement model noise covariance matrix. Returns ------- :class:`~.CovarianceMatrix` of shape\ (:py:attr:`~ndim_meas`, :py:attr:`~ndim_meas`) The measurement noise covariance. """ return self.noise_covar
@clearable_cached_property('rotation_offset') def rotation_matrix(self) -> np.ndarray: """3D axis rotation matrix""" return build_rotation_matrix(self.rotation_offset)
[docs] class CartesianToElevationBearingRange(NonLinearGaussianMeasurement, ReversibleModel): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of bearing \ (:math:`\phi`), elevation (:math:`\theta`) and range (:math:`r`), with \ Gaussian noise in each dimension. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \theta \\ \phi \\ r \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} asin(\mathcal{z}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ atan2(\mathcal{y},\mathcal{x}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2 + \mathcal{z}^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\theta}^2 & 0 & 0 \\ 0 & \sigma_{\phi}^2 & 0 \\ 0 & 0 & \sigma_{r}^2 \end{bmatrix} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. Note ---- The current implementation of this class assumes a 3D Cartesian plane. """ # noqa:E501 translation_offset: StateVector = Property( default=None, doc="A 3x1 array specifying the Cartesian origin offset in terms of :math:`x,y,z` " "coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 3) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 3
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset xyz = state.state_vector[self.mapping, :] - self.translation_offset # Rotate coordinates xyz_rot = self.rotation_matrix @ xyz # Convert to Spherical rho, phi, theta = cart2sphere(xyz_rot[0, :], xyz_rot[1, :], xyz_rot[2, :]) elevations = [Elevation(i) for i in theta] bearings = [Bearing(i) for i in phi] return StateVectors([elevations, bearings, rho]) + noise
[docs] def inverse_function(self, detection, **kwargs) -> StateVector: theta, phi, rho = detection.state_vector xyz = StateVector(sphere2cart(rho, phi, theta)) inv_rotation_matrix = inv(self.rotation_matrix) xyz = inv_rotation_matrix @ xyz res = np.zeros((self.ndim_state, 1)).view(StateVector) res[self.mapping, :] = xyz + self.translation_offset return res
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Elevation(0.)], [Bearing(0.)], [0.]]) + out return out
[docs] class CartesianToBearingRange(NonLinearGaussianMeasurement, ReversibleModel): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of bearing \ (:math:`\phi`) and range (:math:`r`), with Gaussian noise in each dimension. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \phi \\ r \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} atan2(\mathcal{y},\mathcal{x}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\phi}^2 & 0 \\ 0 & \sigma_{r}^2 \end{bmatrix} The :py:attr:`mapping` property of the model is a 2 element vector, \ whose first (i.e. :py:attr:`mapping[0]`) and second (i.e. \ :py:attr:`mapping[1]`) elements contain the state index of the \ :math:`x` and :math:`y` coordinates, respectively. Note ---- The current implementation of this class assumes a 2D Cartesian plane. """ # noqa:E501 translation_offset: StateVector = Property( default=None, doc="A 2x1 array specifying the origin offset in terms of :math:`x,y` coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * len(self.mapping)) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 2
[docs] def inverse_function(self, detection, **kwargs) -> StateVector: if not ((self.rotation_offset[0] == 0) and (self.rotation_offset[1] == 0)): raise RuntimeError( "Measurement model assumes 2D space. \ Rotation in 3D space is unsupported at this time.") x, y = pol2cart(detection.state_vector[1, :], detection.state_vector[0, :]) xyz = np.array([x, y, np.zeros(detection.state_vector.shape[1])]) inv_rotation_matrix = inv(self.rotation_matrix) xyz = inv_rotation_matrix @ xyz xy = xyz[0:2] res = np.zeros((self.ndim_state, detection.state_vector.shape[1])).view(StateVector) res[self.mapping[:2], :] = xy + self.translation_offset[:2, :] return res
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_meas`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset xyz = np.array([state.state_vector[self.mapping[0], :] - self.translation_offset[0, 0], state.state_vector[self.mapping[1], :] - self.translation_offset[1, 0], [0] * state.state_vector.shape[1] ]) # Rotate coordinates xyz_rot = self.rotation_matrix @ xyz # Covert to polar rho, phi = cart2pol(*xyz_rot[:2, :]) bearings = [Bearing(i) for i in phi] return StateVectors([bearings, rho]) + noise
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Bearing(0)], [0.]]) + out return out
[docs] class CartesianToElevationBearing(NonLinearGaussianMeasurement): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of bearing \ (:math:`\phi`) and elevation (:math:`\theta`) and with \ Gaussian noise in each dimension. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \theta \\ \phi \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} asin(\mathcal{z}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ atan2(\mathcal{y},\mathcal{x}) \\ \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\theta}^2 & 0 \\ 0 & \sigma_{\phi}^2\\ \end{bmatrix} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. Note ---- The current implementation of this class assumes a 3D Cartesian plane. """ # noqa:E501 translation_offset: StateVector = Property( default=None, doc="A 3x1 array specifying the origin offset in terms of :math:`x,y,z` coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 3) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 2
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset xyz = state.state_vector[self.mapping, :] - self.translation_offset # Rotate coordinates xyz_rot = self.rotation_matrix @ xyz # Convert to Angles phi, theta = cart2angles(xyz_rot[0, :], xyz_rot[1, :], xyz_rot[2, :]) bearings = [Bearing(i) for i in phi] elevations = [Elevation(i) for i in theta] return StateVectors([elevations, bearings]) + noise
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Elevation(0.)], [Bearing(0.)]]) + out return out
[docs] class Cartesian2DToBearing(NonLinearGaussianMeasurement): r"""This is a class implementation of a time-invariant measurement model, where measurements \ are assumed to be received in the form of bearing (:math:`\phi`) with Gaussian noise. The model is described by the following equations: .. math:: \phi_t = h(\vec{x}_t, v_t) * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,v_t) = atan2(\mathcal{y},\mathcal{x}) + v_t * :math:`v_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: v_t \sim \mathcal{N}(0,\sigma_{\phi}^2) The :py:attr:`mapping` property of the model is a 2 element vector, whose first \ (i.e. :py:attr:`mapping[0]`) and second (i.e. :py:attr:`mapping[1]`) elements contain the \ state index of the :math:`x` and :math:`y` coordinates, respectively. """ # noqa:E501 translation_offset: StateVector = Property( default=None, doc="A 2x1 array specifying the origin offset in terms of :math:`x,y` coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 2) @property def ndim_meas(self): """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 1
[docs] def function(self, state, noise=False, **kwargs): r"""Model function :math:`h(\vec{x}_t,v_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added. If 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset xyz = np.array([state.state_vector[self.mapping[0], :] - self.translation_offset[0, 0], state.state_vector[self.mapping[1], :] - self.translation_offset[1, 0], [0] * state.state_vector.shape[1] ]) # Rotate coordinates xyz_rot = self.rotation_matrix @ xyz # Covert to polar _, phi = cart2pol(*xyz_rot[:2, :]) bearings = [Bearing(i) for i in phi] return StateVectors([bearings]) + noise
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Bearing(0.)]]) + out return out
[docs] class CartesianToBearingRangeRate(NonLinearGaussianMeasurement): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of bearing \ (:math:`\phi`), range (:math:`r`) and range-rate (:math:`\dot{r}`), with Gaussian noise in each dimension. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \phi \\ r \\ \dot{r} \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} atan2(\mathcal{y},\mathcal{x}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2} \\ (x\dot{x} + y\dot{y})/\sqrt{x^2 + y^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\phi}^2 & 0 & 0\\ 0 & \sigma_{r}^2 & 0 \\ 0 & 0 & \sigma_{\dot{r}}^2 \end{bmatrix} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. Note ---- This class implementation assuming at 3D cartesian space, it therefore \ expects a 6D state space. """ translation_offset: StateVector = Property( default=None, doc="A 3x1 array specifying the origin offset in terms of :math:`x,y` coordinates.") velocity_mapping: Tuple[int, int, int] = Property( default=(1, 3, 5), doc="Mapping to the targets velocity within its state space") velocity: StateVector = Property( default=None, doc="A 3x1 array specifying the sensor velocity in terms of :math:`x,y,z` coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 3) if self.velocity is None: self.velocity = StateVector([0] * 3) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 3
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset in position to enable range and angles to be determined xy_pos = state.state_vector[self.mapping, :] - self.translation_offset # Rotate coordinates based upon the sensor_velocity xy_rot = self.rotation_matrix @ xy_pos # Convert to Spherical rho, phi, _ = cart2sphere(xy_rot[0, :], xy_rot[1, :], xy_rot[2, :]) # Determine the net velocity component in the engagement xy_vel = state.state_vector[self.velocity_mapping, :] - self.velocity # Use polar to calculate range rate rr = np.einsum('ij,ij->j', xy_pos, xy_vel) / np.linalg.norm(xy_pos, axis=0) # Convert to bearings bearings = [Bearing(i) for i in phi] return StateVectors([bearings, rho, rr]) + noise
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Bearing(0)], [0.], [0.]]) + out return out
[docs] class CartesianToElevationBearingRangeRate(NonLinearGaussianMeasurement, ReversibleModel): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of elevation \ (:math:`\theta`), bearing (:math:`\phi`), range (:math:`r`) and range-rate (:math:`\dot{r}`), with Gaussian noise in each dimension. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \theta \\ \phi \\ r \\ \dot{r} \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} asin(\mathcal{z}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ atan2(\mathcal{y},\mathcal{x}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2 + \mathcal{z}^2} \\ (x\dot{x} + y\dot{y} + z\dot{z})/\sqrt{x^2 + y^2 + z^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\theta}^2 & 0 & 0 & 0\\ 0 & \sigma_{\phi}^2 & 0 & 0\\ 0 & 0 & \sigma_{r}^2 & 0\\ 0 & 0 & 0 & \sigma_{\dot{r}}^2 \end{bmatrix} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. Note ---- This class implementation assuming at 3D cartesian space, it therefore \ expects a 6D state space. """ translation_offset: StateVector = Property( default=None, doc="A 3x1 array specifying the origin offset in terms of :math:`x,y,z` coordinates.") velocity_mapping: Tuple[int, int, int] = Property( default=(1, 3, 5), doc="Mapping to the targets velocity within its state space") velocity: StateVector = Property( default=None, doc="A 3x1 array specifying the sensor velocity in terms of :math:`x,y,z` coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 3) if self.velocity is None: self.velocity = StateVector([0] * 3) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 4
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.StateVector` An input state vector for the target noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset in position to enable range and angles to be determined xyz_pos = state.state_vector[self.mapping, :] - self.translation_offset # Rotate coordinates based upon the sensor_velocity xyz_rot = self.rotation_matrix @ xyz_pos # Convert to Spherical rho, phi, theta = cart2sphere(xyz_rot[0, :], xyz_rot[1, :], xyz_rot[2, :]) # Determine the net velocity component in the engagement xyz_vel = state.state_vector[self.velocity_mapping, :] - self.velocity # Use polar to calculate range rate rr = np.einsum('ij,ij->j', xyz_pos, xyz_vel) / np.linalg.norm(xyz_pos, axis=0) bearings = [Bearing(i) for i in phi] elevations = [Elevation(i) for i in theta] return StateVectors([elevations, bearings, rho, rr]) + noise
[docs] def inverse_function(self, detection, **kwargs) -> StateVector: theta, phi, rho, rho_rate = detection.state_vector x, y, z = sphere2cart(rho, phi, theta) # because only rho_rate is known, only the components in # x,y and z of the range rate can be found. x_rate = np.cos(phi) * np.cos(theta) * rho_rate y_rate = np.cos(phi) * np.sin(theta) * rho_rate z_rate = np.sin(phi) * rho_rate inv_rotation_matrix = inv(self.rotation_matrix) out_vector = StateVector([[0.], [0.], [0.], [0.], [0.], [0.]]) out_vector[self.mapping, 0] = x, y, z out_vector[self.velocity_mapping, 0] = x_rate, y_rate, z_rate out_vector[self.mapping, :] = inv_rotation_matrix @ out_vector[self.mapping, :] out_vector[self.velocity_mapping, :] = \ inv_rotation_matrix @ out_vector[self.velocity_mapping, :] out_vector[self.mapping, :] = out_vector[self.mapping, :] + self.translation_offset return out_vector
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Elevation(0)], [Bearing(0)], [0.], [0.]]) + out return out
[docs] def jacobian(self, state, **kwargs): """Model jacobian matrix :math:`H_{jac}` Parameters ---------- state : :class:`~.State` An input state Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_meas`, \ :py:attr:`~ndim_state`) The model jacobian matrix evaluated around the given state vector. """ # Account for origin offset in position to enable range and angles to be determined xyz_pos = state.state_vector[self.mapping, :] - self.translation_offset # Determine the net velocity component in the engagement xyz_vel = state.state_vector[self.velocity_mapping, :] - self.velocity # Rotate into RADAR coordinate system to linearize around the correct # state xyz_pos = self.rotation_matrix @ xyz_pos xyz_vel = self.rotation_matrix @ xyz_vel jac = np.zeros((4, 6), dtype=np.float64) x, y, z = xyz_pos vx, vy, vz = xyz_vel x2, y2, z2 = x**2, y**2, z**2 x2y2 = x2 + y2 r2 = x2y2 + z2 r = sqrt(r2) sqrt_x2_y2 = sqrt(x2y2) r32 = r2*r # Jacobian encodes partial derivatives of measurement vector components # Y = <theta, phi, r, rdot> against state vector # X = <x, vx, y, vy, z, vz>. # dtheta/dx sqrt_x2_y2r2 = sqrt_x2_y2*r2 jac[0, 0] = -(x*z)/(sqrt_x2_y2r2) # dtheta/dy jac[0, 2] = -(y*z)/(sqrt_x2_y2r2) # dthtea/dz jac[0, 4] = sqrt_x2_y2/r2 # dphi/dx jac[1, 0] = - y/(x2y2) # dphi/dy jac[1, 2] = x/(x2y2) # dphi/dz = 0 # dr/dx and drdot/dvx jac[2, 0] = jac[3, 1] = x/r # dr/dx and drdot/dvy jac[2, 2] = jac[3, 3] = y/r # dr/dx and drdot/dvz jac[2, 4] = jac[3, 5] = z/r vx_x, vy_y, vz_z = vx*x, vy*y, vz*z # drdot/dx jac[3, 0] = (-x*(vy_y + vz_z) + vx*(y2 + z2))/r32 # drdot/dy jac[3, 2] = (vy*(x2 + z2) - y*(vx_x + vz_z))/r32 # drdot/dz jac[3, 4] = (vz*(x2y2) - (vx_x + vy_y)*z)/r32 # Up to this point, the Jacobian has been with respect to the state # vector after rotating into the RADAR coordinate system. However, we # want the Jacobian with respect to world state vector, so we must post # multiply Jacobian by the RADAR rotation matrix. jac[:, self.mapping] = jac[:, self.mapping] @ self.rotation_matrix jac[:, self.velocity_mapping] = jac[:, self.velocity_mapping] @ self.rotation_matrix return jac
[docs] class RangeRangeRateBinning(CartesianToElevationBearingRangeRate): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be in the form of elevation \ (:math:`\theta`), bearing (:math:`\phi`), range (:math:`r`) and range-rate (:math:`\dot{r}`), with Gaussian noise in each dimension and the range and range-rate are binned based on the range resolution and range-rate resolution respectively. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \theta \\ \phi \\ r \\ \dot{r} \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} \textrm{asin}(\mathcal{z}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ \textrm{atan2}(\mathcal{y},\mathcal{x}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2 + \mathcal{z}^2} \\ (x\dot{x} + y\dot{y} + z\dot{z})/\sqrt{x^2 + y^2 + z^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\theta}^2 & 0 & 0 & 0\\ 0 & \sigma_{\phi}^2 & 0 & 0\\ 0 & 0 & \sigma_{r}^2 & 0\\ 0 & 0 & 0 & \sigma_{\dot{r}}^2 \end{bmatrix} The covariances for radar are determined by different factors. The angle error is affected by the radar beam width. Range error is affected by the SNR and pulse bandwidth. The error for the range rate is dependent on the dwell time. The range and range rate are binned to the centre of the cell using .. math:: x = \textrm{floor}(x/\Delta x)*\Delta x + \frac{\Delta x}{2} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. The :py:attr:`velocity_mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`velocity_mapping[0]`), second (i.e. \ :py:attr:`velocity_mapping[1]`) and third (i.e. :py:attr:`velocity_mapping[2]`) elements \ contain the state index of the :math:`\dot{x}`, :math:`\dot{y}` and :math:`\dot{z}` \ coordinates, respectively. Note ---- This class implementation assumes a 3D cartesian space, it therefore \ expects a 6D state space. """ range_res: float = Property(doc="Size of the range bins in m") range_rate_res: float = Property(doc="Size of the velocity bins in m/s") @property def ndim_meas(self): return 4
[docs] def function(self, state, noise=False, **kwargs): r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.StateVector` An input state vector for the target noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is ``False``, in which case no noise will be added and no binning takes place if ``True``, the output of :attr:`~.Model.rvs` is added and the range and range rate are binned) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ out = super().function(state, noise, **kwargs) if isinstance(noise, bool) or noise is None: if noise: out[2] = np.floor(out[2] / self.range_res) * self.range_res + self.range_res/2 out[3] = np.floor(out[3] / self.range_rate_res) * \ self.range_rate_res + self.range_rate_res/2 return out
@classmethod def _gaussian_integral(cls, a, b, mean, cov): # this function is the cumulative probability ranging from a to b for a normal distribution return (multivariate_normal.cdf(a, mean=mean, cov=cov) - multivariate_normal.cdf(b, mean=mean, cov=cov)) @classmethod def _binned_pdf(cls, measured_value, mean, bin_size, cov): # this function finds the probability density of the bin the measured_value is in a = np.floor(measured_value / bin_size) * bin_size + bin_size b = np.floor(measured_value / bin_size) * bin_size return cls._gaussian_integral(a, b, mean, cov)/bin_size
[docs] def pdf(self, state1, state2, **kwargs): r"""Model pdf/likelihood evaluation function Evaluates the pdf/likelihood of ``state1``, given the state ``state2`` which is passed to :meth:`function()`. For the first 2 dimensions, this can be written as: .. math:: p = p(y_t | x_t) = \mathcal{N}(y_t; x_t, Q) where :math:`y_t` = ``state_vector1``, :math:`x_t` = ``state_vector2``, :math:`Q` = :attr:`covar` and :math:`\mathcal{N}` is a normal distribution The probability for the binned dimensions, the last 2, can be written as: .. math:: p = P(a \leq \mathcal{N} \leq b) In this equation a and b are the edges of the bin. Parameters ---------- state1 : :class:`~.State` state2 : :class:`~.State` Returns ------- : :class:`~.Probability` The likelihood of ``state1``, given ``state2`` """ # state1 is in measurement space # state2 is in state_space if (((state1.state_vector[2, 0]-self.range_res/2) / self.range_res).is_integer() and ((state1.state_vector[3, 0]-self.range_rate_res/2) / self.range_rate_res).is_integer()): mean_vector = self.function(state2, noise=False, **kwargs) # pdf for the angles az_el_pdf = multivariate_normal.pdf( state1.state_vector[:2, 0], mean=mean_vector[:2, 0], cov=self.covar()[:2, :2]) # pdf for the binned range and velocity range_pdf = self._binned_pdf( state1.state_vector[2, 0], mean_vector[2, 0], self.range_res, self.covar()[2, 2]) velocity_pdf = self._binned_pdf( state1.state_vector[3, 0], mean_vector[3, 0], self.range_rate_res, self.covar()[3, 3]) return Probability(range_pdf * velocity_pdf * az_el_pdf) else: return Probability(0)
[docs] def logpdf(self, *args, **kwargs): # As pdf replaced, need to go to first non GaussianModel parent return super(ReversibleModel, self).logpdf(*args, **kwargs)
[docs] class CartesianToAzimuthElevationRange(NonLinearGaussianMeasurement, ReversibleModel): r"""This is a class implementation of a time-invariant measurement model, \ where measurements are assumed to be received in the form of azimuth \ (:math:`\phi`), elevation (:math:`\theta`), and range (:math:`r`), with \ Gaussian noise in each dimension. For this model, the Azimuth is defined as the angle of the measurement from \ the YZ plan to the YX plane and the Elevation is the angle from the XY plan \ to the XZ plane. The z axis is the direction the radar is pointing (broadside) \ and is only defined in the positive z. The x axis is generally the direction of travel \ for an airborne radar and the y axis is orthogonal to both the x and z. Measurements are only correctly defined for +z (measurements must be in front \ of the sensor. The model is described by the following equations: .. math:: \vec{y}_t = h(\vec{x}_t, \vec{v}_t) where: * :math:`\vec{y}_t` is a measurement vector of the form: .. math:: \vec{y}_t = \begin{bmatrix} \phi \\ \theta \\ r \end{bmatrix} * :math:`h` is a non-linear model function of the form: .. math:: h(\vec{x}_t,\vec{v}_t) = \begin{bmatrix} asin(\mathcal{x}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ asin(\mathcal{y}/\sqrt{\mathcal{x}^2 + \mathcal{y}^2 +\mathcal{z}^2}) \\ \sqrt{\mathcal{x}^2 + \mathcal{y}^2 + \mathcal{z}^2} \end{bmatrix} + \vec{v}_t * :math:`\vec{v}_t` is Gaussian distributed with covariance :math:`R`, i.e.: .. math:: \vec{v}_t \sim \mathcal{N}(0,R) .. math:: R = \begin{bmatrix} \sigma_{\phi}^2 & 0 & 0 \\ 0 & \sigma_{\theta}^2 & 0 \\ 0 & 0 & \sigma_{r}^2 \end{bmatrix} The :py:attr:`mapping` property of the model is a 3 element vector, \ whose first (i.e. :py:attr:`mapping[0]`), second (i.e. \ :py:attr:`mapping[1]`) and third (i.e. :py:attr:`mapping[2]`) elements \ contain the state index of the :math:`x`, :math:`y` and :math:`z` \ coordinates, respectively. Note ---- The current implementation of this class assumes a 3D Cartesian plane. """ # noqa:E501 translation_offset: StateVector = Property( default=None, doc="A 3x1 array specifying the Cartesian origin offset in terms of :math:`x,y,z` " "coordinates.") def __init__(self, *args, **kwargs): """ Ensure that the translation offset is initiated """ super().__init__(*args, **kwargs) # Set values to defaults if not provided if self.translation_offset is None: self.translation_offset = StateVector([0] * 3) @property def ndim_meas(self) -> int: """ndim_meas getter method Returns ------- :class:`int` The number of measurement dimensions """ return 3
[docs] def function(self, state, noise=False, **kwargs) -> StateVector: r"""Model function :math:`h(\vec{x}_t,\vec{v}_t)` Parameters ---------- state: :class:`~.State` An input state noise: :class:`numpy.ndarray` or bool An externally generated random process noise sample (the default is `False`, in which case no noise will be added if 'True', the output of :meth:`~.Model.rvs` is added) Returns ------- :class:`numpy.ndarray` of shape (:py:attr:`~ndim_state`, 1) The model function evaluated given the provided time interval. """ if isinstance(noise, bool) or noise is None: if noise: noise = self.rvs(num_samples=state.state_vector.shape[1], **kwargs) else: noise = 0 # Account for origin offset xyz = state.state_vector[self.mapping, :] - self.translation_offset # Rotate coordinates xyz_rot = self.rotation_matrix @ xyz # Convert to measurement space phi, theta, rho = cart2az_el_rg(xyz_rot[0, :], xyz_rot[1, :], xyz_rot[2, :]) elevations = [Elevation(i) for i in theta] azimuths = [Azimuth(i) for i in phi] return StateVectors([azimuths, elevations, rho]) + noise
[docs] def inverse_function(self, detection, **kwargs) -> StateVector: phi, theta, rho = detection.state_vector # convert to cartesian x, y, z = az_el_rg2cart(phi, theta, rho) xyz = StateVector([x, y, z]) inv_rotation_matrix = inv(self.rotation_matrix) xyz = inv_rotation_matrix @ xyz res = np.zeros((self.ndim_state, 1)).view(StateVector) res[self.mapping, :] = xyz + self.translation_offset return res
[docs] def rvs(self, num_samples=1, **kwargs) -> Union[StateVector, StateVectors]: out = super().rvs(num_samples, **kwargs) out = np.array([[Azimuth(0.)], [Elevation(0.)], [0.]]) + out return out