diff --git a/.pylintrc b/.pylintrc index 2dddcb3bf..787f73e51 100644 --- a/.pylintrc +++ b/.pylintrc @@ -359,6 +359,9 @@ max-statements=25 # Minimum number of public methods for a class (see R0903). min-public-methods=0 +# Maximum number of positional arguments (see R0917). +max-positional-arguments = 25 + [EXCEPTIONS] diff --git a/CHANGELOG.md b/CHANGELOG.md index 6826e98cf..3e5139933 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -32,11 +32,13 @@ Attention: The newest changes should be on top --> ### Added -- +- ENH: Generic Surfaces and Generic Linear Surfaces [#680](https://github.com/RocketPy-Team/RocketPy/pull/680) +- ENH: Free-Form Fins [#694](https://github.com/RocketPy-Team/RocketPy/pull/694) +- ENH: Expand Polation Options for ND Functions. [#691](https://github.com/RocketPy-Team/RocketPy/pull/691) ### Changed -- +- ### Fixed diff --git a/README.md b/README.md index 5808678d3..c549840f7 100644 --- a/README.md +++ b/README.md @@ -329,6 +329,9 @@ RocketPy extends its gratitude to the following institutions for their support a Space Enterprise at Berkeley Logo + + Faraday Rocketry UPV Logo + ## Individual Contributors diff --git a/docs/reference/classes/aero_surfaces/GenericSurface.rst b/docs/reference/classes/aero_surfaces/GenericSurface.rst new file mode 100644 index 000000000..ed57c23af --- /dev/null +++ b/docs/reference/classes/aero_surfaces/GenericSurface.rst @@ -0,0 +1,5 @@ +Generic Surface Class +--------------------- + +.. autoclass:: rocketpy.GenericSurface + :members: \ No newline at end of file diff --git a/docs/reference/classes/aero_surfaces/LinearGenericSurface.rst b/docs/reference/classes/aero_surfaces/LinearGenericSurface.rst new file mode 100644 index 000000000..0c39469fb --- /dev/null +++ b/docs/reference/classes/aero_surfaces/LinearGenericSurface.rst @@ -0,0 +1,5 @@ +Linear Generic Surface Class +---------------------------- + +.. autoclass:: rocketpy.LinearGenericSurface + :members: \ No newline at end of file diff --git a/docs/reference/classes/aero_surfaces/index.rst b/docs/reference/classes/aero_surfaces/index.rst index 5cbb9eedb..c6e6a3efe 100644 --- a/docs/reference/classes/aero_surfaces/index.rst +++ b/docs/reference/classes/aero_surfaces/index.rst @@ -13,3 +13,5 @@ AeroSurface Classes EllipticalFins RailButtons AirBrakes + GenericSurface + LinearGenericSurface diff --git a/docs/static/institutional/faraday_team_logo.jpg b/docs/static/institutional/faraday_team_logo.jpg new file mode 100644 index 000000000..1ef1c2b85 Binary files /dev/null and b/docs/static/institutional/faraday_team_logo.jpg differ diff --git a/docs/static/rocket/aeroframe.png b/docs/static/rocket/aeroframe.png new file mode 100644 index 000000000..c48c86e1a Binary files /dev/null and b/docs/static/rocket/aeroframe.png differ diff --git a/docs/user/rocket/generic_surface.rst b/docs/user/rocket/generic_surface.rst new file mode 100644 index 000000000..f6568758e --- /dev/null +++ b/docs/user/rocket/generic_surface.rst @@ -0,0 +1,436 @@ +.. _genericsurfaces: + +Generic Surfaces and Custom Aerodynamic Coefficients +==================================================== + +Generic aerodynamic surfaces can be used to model aerodynamic forces based on +force and moment coefficients. The :class:`rocketpy.GenericSurface` receives the +coefficients as functions of the angle of attack, side slip angle, Mach number, +Reynolds number, pitch rate, yaw rate, and roll rate. + +The :class:`rocketpy.LinearGenericSurface` class model aerodynamic forces based +the force and moment coefficients derivatives. The coefficients are derivatives +of the force and moment coefficients with respect to the angle of attack, side +slip angle, Mach number, Reynolds number, pitch rate, yaw rate, and roll rate. + +These classes allows the user to be less dependent on the built-in aerodynamic +surfaces and to define their own aerodynamic coefficients. + +Both classes base their coefficient on the definition of the aerodynamic frame +of reference. + +Aerodynamic Frame +----------------- + +The aerodynamic frame of reference of the rocket is defined as follows: + +- The origin is at the rocket's center of dry mass (``center_of_dry_mass_position``). +- The ``z`` axis is defined along the rocket's centerline, pointing from the center of dry mass towards the nose. +- The ``x`` and ``y`` axes are perpendicular. +- The partial angle of attack (``alpha``) is defined as the angle, in the y-z + plane, from the velocity vector to the z axis. +- The partial side slip angle (``beta``) is defined as the angle, in the x-z + plane, from the velocity vector to the z axis. + +The following figure shows the aerodynamic frame of reference: + +.. figure:: ../../static/rocket/aeroframe.png + :align: center + :alt: Aerodynamic frame of reference + +In the figure we define: + +- :math:`\mathbf{\vec{V}}` as rocket velocity vector. +- :math:`x_B`, :math:`y_B`, and :math:`z_B` as the body axes. +- :math:`x_A`, :math:`y_A`, and :math:`z_A` as the aerodynamic axes. +- :math:`\alpha` as the partial angle of attack. +- :math:`\beta` as the side slip angle. +- :math:`L` as the lift force. +- :math:`D` as the drag force. +- :math:`Q` as the side force. + +Here we define the aerodynamic forces in the aerodynamic frame of reference as: + +.. math:: + \vec{\mathbf{F}}_A=\begin{bmatrix}X_A\\Y_A\\Z_A\end{bmatrix}_A=\begin{bmatrix}Q\\-L\\-D\end{bmatrix}_A + +The aerodynamic forces in the body axes coordinate system are defined as +:math:`\vec{\mathbf{F}}_B`. + +.. math:: + \vec{\mathbf{F}}_B=\begin{bmatrix}X_A\\Y_A\\Z_A\end{bmatrix}_B=\mathbf{M}_{BA}\cdot\begin{bmatrix}Q\\-L\\-D\end{bmatrix}_A + +Where the transformation matrix :math:`\mathbf{M}_{BA}`, which transforms the +aerodynamic forces from the aerodynamic frame of reference to the body axes +coordinate system, is defined as: + +.. math:: + \mathbf{M}_{BA} = \begin{bmatrix} + 1 & 0 & 0 \\ + 0 & \cos(\alpha) & -\sin(\alpha) \\ + 0 & \sin(\alpha) & \cos(\alpha) + \end{bmatrix} + \begin{bmatrix} + \cos(\beta) & 0 & -\sin(\beta) \\ + 0 & 1 & 0 \\ + \sin(\beta) & 0 & \cos(\beta) + \end{bmatrix} + + +The forces coefficients can finally be defined as: + +- :math:`C_L` as the lift coefficient. +- :math:`C_Q` as the side force coefficient (or cross stream force coefficient). +- :math:`C_D` as the drag coefficient. + +And the forces from the coefficients are defined as: + +.. math:: + \begin{bmatrix}X_A\\Y_A\\Z_A\end{bmatrix}_B =\mathbf{M}_{BA}\cdot\overline{q}\cdot A_{ref}\cdot\begin{bmatrix}C_Q\\-C_L\\-C_D\end{bmatrix}_A + +Where: + +- :math:`\bar{q}` is the dynamic pressure. +- :math:`A_{ref}` is the reference area used to calculate the coefficients. + Commonly the rocket's cross-sectional area is used as the reference area. + +The moment coefficients can be defined as: + +- :math:`C_l` as the rolling moment coefficient. +- :math:`C_m` as the pitching moment coefficient. +- :math:`C_n` as the yawing moment coefficient. + +And the moments from the coefficients are defined as: + +.. math:: + \vec{\mathbf{M}}_B=\begin{bmatrix}M_{x_A}\\M_{y_A}\\M_{z_A}\end{bmatrix}_B =\overline{q}\cdot A_{ref}\cdot L_{ref}\cdot\begin{bmatrix}C_m\\C_n\\C_l\end{bmatrix} + +Where: + +- :math:`L_{ref}` is the reference length used to calculate the coefficients. + Commonly the rocket's diameter is used as the reference length. + + +Aerodynamic angles +~~~~~~~~~~~~~~~~~~ + +The aerodynamic angles are defined in two different ways in RocketPy: + +- As the angle of attack (:math:`\alpha`) and the side slip \ + angle (:math:`\beta`), which are defined in the image above. These are used \ + in the calculation of the generic surface forces and moments. +- As the total angle of attack (:math:`\alpha_{\text{tot}}`), defined as the \ + angle between the total velocity vector and the rocket's centerline. This is \ + used in the calculation of the standard aerodynamic surface forces and moments. + +The partial angles are calculated as: + +.. math:: + \begin{aligned} + \alpha &= \arctan\left(\frac{V_y}{V_z}\right) \\ + \beta &= \arctan\left(\frac{V_x}{V_z}\right) + \end{aligned} + +The total angle of attack is calculated as: + +.. math:: + \alpha_{\text{tot}} = \arccos\left(\frac{\mathbf{\vec{V}}\cdot\mathbf{z_B}}{||\mathbf{\vec{V}}||\cdot||\mathbf{z_B}||}\right) + +.. note:: + When the simulation is done, the total angle of attack is accessed through + the :attr:`rocketpy.Flight.angle_of_attack` attribute. + The partial angles of attack and side slip are accessed through the + :attr:`rocketpy.Flight.partial_angle_of_attack` and + :attr:`rocketpy.Flight.angle_of_sideslip` attributes, respectively. + +.. _genericsurface: + +Generic Surface Class +--------------------- + +The :class:`rocketpy.GenericSurface` class is used to define an aerodynamic +surface based on force and moment coefficients. A generic surface is defined +as follows: + +.. seealso:: + For more information on class initialization, see + :class:`rocketpy.GenericSurface.__init__` + + +.. code-block:: python + + from rocketpy import GenericSurface + + radius = 0.0635 + + generic_surface = GenericSurface( + reference_area=np.pi * radius**2, + reference_length=2 * radius, + coefficients={ + "cL": "cL.csv", + "cQ": "cQ.csv", + "cD": "cD.csv", + "cm": "cm.csv", + "cn": "cn.csv", + "cl": "cl.csv", + }, + name="Generic Surface", + ) + +The ``coefficients`` argument is a dictionary containing the coefficients of the +generic surface. The keys of the dictionary are the coefficient names, and the +values are the coefficients. The possible coefficient names are: + +- ``cL``: Lift coefficient. +- ``cQ``: Side force coefficient. +- ``cD``: Drag coefficient. +- ``cm``: Pitching moment coefficient. +- ``cn``: Yawing moment coefficient. +- ``cl``: Rolling moment coefficient. + +Only one of the coefficients is required to be provided, but any combination of +the coefficients can be used. The coefficient values can be provided as a +single value, a callable function of seven arguments, or a path to a ``.csv`` +file containing the values. + +The coefficients are all functions of: + +- Angle of attack (:math:`\alpha`) in radians. +- Side slip angle (:math:`\beta`) in radians. +- Mach number (:math:`Ma`). +- Reynolds number (:math:`Re`). +- Pitch rate (:math:`q`) in radians per second. +- Yaw rate (:math:`r`) in radians per second. +- Roll rate (:math:`p`) in radians per second. + +.. math:: + \begin{aligned} + C_L &= f(\alpha, \beta, Ma, Re, q, r, p) \\ + C_Q &= f(\alpha, \beta, Ma, Re, q, r, p) \\ + C_D &= f(\alpha, \beta, Ma, Re, q, r, p) \\ + C_m &= f(\alpha, \beta, Ma, Re, q, r, p) \\ + C_n &= f(\alpha, \beta, Ma, Re, q, r, p) \\ + C_l &= f(\alpha, \beta, Ma, Re, q, r, p) + \end{aligned} + +From the coefficients, the forces and moments are calculated with + +.. math:: + \begin{aligned} + L &= \overline{q}\cdot A_{ref}\cdot C_L \\ + Q &= \overline{q}\cdot A_{ref}\cdot C_Q \\ + D &= \overline{q}\cdot A_{ref}\cdot C_D \\ + M_{m} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_m \\ + M_{n} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_n \\ + M_{l} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_l + \end{aligned} + +These coefficients can be defined as a callable such as: + +.. code-block:: python + + def coefficient(alpha, beta, Ma, Re, q, r, p): + ... + return value + +In which any algorithm can be implemented to calculate the coefficient values. + +Otherwise, the coefficients can be defined as a ``.csv`` file. The file must +contain a header with at least one of the following columns representing the +independent variables: + +- ``alpha``: Angle of attack. +- ``beta``: Side slip angle. +- ``mach``: Mach number. +- ``reynolds``: Reynolds number. +- ``q``: Pitch rate. +- ``r``: Yaw rate. +- ``p``: Roll rate. + +The last column must be the coefficient value, and must contain a header, +though the header name can be anything. + +.. important:: + Not all columns need to be present in the file, but the columns that are + present must be named, **and ordered**, as described above. + +An example of a ``.csv`` file is shown below: + +.. code-block:: + + "alpha", "mach", "coefficient" + -0.017, 0, -0.11 + -0.017, 1, -0.127 + -0.017, 2, -0.084 + -0.017, 3, -0.061 + 0.0, 0, 0.0 + 0.0, 1, 0.0 + 0.0, 2, 0.0 + 0.0, 3, 0.0 + 0.017, 0, 0.11 + 0.017, 1, 0.127 + 0.017, 2, 0.084 + 0.017, 3, 0.061 + +After the definition of the ``GenericSurface`` object, it must be added to the +rocket's configuration: + +.. seealso:: + For more information on how to add a generic surface to the rocket, see + :class:`rocketpy.Rocket.add_generic_surface` + +.. code-block:: python + :emphasize-lines: 5 + + from rocketpy import Rocket + rocket = Rocket( + ... + ) + rocket.add_surfaces(generic_surface, position=(0,0,0)) + +The position of the generic surface is defined in the User Defined coordinate +System, see :ref:`rocketaxes` for more information. + +.. tip:: + If defining the coefficients of the entire rocket is desired, only a single + generic surface can be added to the rocket, positioned at the center of dry + mass. This will be equivalent to defining the coefficients of the entire + rocket. + +.. attention:: + If there generic surface is positioned **not** at the center of dry mass, the + forces generated by the force coefficients (cL, cQ, cD) will generate a + moment around the center of dry mass. This moment will be calculated and + added to the moment generated by the moment coefficients (cm, cn, cl). + + +.. _lineargenericsurface: + +Linear Generic Surface Class +---------------------------- + +The :class:`rocketpy.LinearGenericSurface` class is used to define a aerodynamic +surface based on the forces and moments coefficient derivatives. A linear generic +surface will receive the derivatives of each coefficient with respect to the +independent variables. The derivatives are defined as: + +- :math:`C_{\alpha}=\frac{dC}{d\alpha}`: Coefficient derivative with respect to angle of attack. +- :math:`C_{\beta}=\frac{dC}{d\beta}`: Coefficient derivative with respect to side slip angle. +- :math:`C_{Ma}=\frac{dC}{dMa}`: Coefficient derivative with respect to Mach number. +- :math:`C_{Re}=\frac{dC}{dRe}`: Coefficient derivative with respect to Reynolds number. +- :math:`C_{q}=\frac{dC}{dq}`: Coefficient derivative with respect to pitch rate. +- :math:`C_{r}=\frac{dC}{dr}`: Coefficient derivative with respect to yaw rate. +- :math:`C_{p}=\frac{dC}{dp}`: Coefficient derivative with respect to roll rate. + +A non derivative coefficient :math:`C_{0}` is also included. + +Each coefficient derivative is defined as a function of all the seven +independent variables. + +The coefficients are then grouped into **forcing** coefficients: + +.. math:: + \begin{aligned} + C_{Lf} &= C_{L0} + C_{L\alpha}\cdot\alpha + C_{L\beta}\cdot\beta + C_{LMa}\cdot Ma + C_{LRe}\cdot Re \\ + C_{Qf} &= C_{Q0} + C_{Q\alpha}\cdot\alpha + C_{Q\beta}\cdot\beta + C_{QMa}\cdot Ma + C_{QRe}\cdot Re \\ + C_{Df} &= C_{D0} + C_{D\alpha}\cdot\alpha + C_{D\beta}\cdot\beta + C_{DMa}\cdot Ma + C_{DRe}\cdot Re \\ + C_{mf} &= C_{m0} + C_{m\alpha}\cdot\alpha + C_{m\beta}\cdot\beta + C_{mMa}\cdot Ma + C_{mRe}\cdot Re \\ + C_{nf} &= C_{n0} + C_{n\alpha}\cdot\alpha + C_{n\beta}\cdot\beta + C_{nMa}\cdot Ma + C_{nRe}\cdot Re \\ + C_{lf} &= C_{l0} + C_{l\alpha}\cdot\alpha + C_{l\beta}\cdot\beta + C_{lMa}\cdot Ma + C_{lRe}\cdot Re + \end{aligned} + +And **damping** coefficients: + +.. math:: + \begin{aligned} + C_{Ld} &= C_{L_{q}}\cdot q + C_{L_{r}}\cdot r + C_{L_{p}}\cdot p \\ + C_{Qd} &= C_{Q_{q}}\cdot q + C_{Q_{r}}\cdot r + C_{Q_{p}}\cdot p \\ + C_{Dd} &= C_{D_{q}}\cdot q + C_{D_{r}}\cdot r + C_{D_{p}}\cdot p \\ + C_{md} &= C_{m_{q}}\cdot q + C_{m_{r}}\cdot r + C_{m_{p}}\cdot p \\ + C_{nd} &= C_{n_{q}}\cdot q + C_{n_{r}}\cdot r + C_{n_{p}}\cdot p \\ + C_{ld} &= C_{l_{q}}\cdot q + C_{l_{r}}\cdot r + C_{l_{p}}\cdot p + \end{aligned} + +The forces and moments are then calculated as: + +.. math:: + \begin{aligned} + L &= \overline{q}\cdot A_{ref}\cdot C_{Lf} + \overline{q}\cdot A_{ref}\cdot \frac{L_{ref}}{2V} C_{Ld} \\ + Q &= \overline{q}\cdot A_{ref}\cdot C_{Qf} + \overline{q}\cdot A_{ref}\cdot \frac{L_{ref}}{2V} C_{Qd} \\ + D &= \overline{q}\cdot A_{ref}\cdot C_{Df} + \overline{q}\cdot A_{ref}\cdot \frac{L_{ref}}{2V} C_{Dd} \\ + M_{m} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_{mf} + \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot \frac{L_{ref}}{2V} C_{md} \\ + M_{n} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_{nf} + \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot \frac{L_{ref}}{2V} C_{nd} \\ + M_{l} &= \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot C_{lf} + \overline{q}\cdot A_{ref}\cdot L_{ref}\cdot \frac{L_{ref}}{2V} C_{ld} + \end{aligned} + +The linear generic surface is defined very similarly to the generic surface. +The coefficients are defined in the same way, but with the addition of the +derivative values. + +An example of a linear generic surface defined with **all** the coefficients is +shown below: + +.. seealso:: + For more information on class initialization, see + :class:`rocketpy.LinearGenericSurface.__init__` + +.. code-block:: python + + from rocketpy import LinearGenericSurface + linear_generic_surface = LinearGenericSurface( + reference_area=np.pi * 0.0635**2, + reference_length=2 * 0.0635, + coefficients={ + "cL_0": "cL_0.csv", + "cL_alpha": "cL_alpha.csv", + "cL_beta": "cL_beta.csv", + "cL_Ma": "cL_Ma.csv", + "cL_Re": "cL_Re.csv", + "cL_q": "cL_q.csv", + "cL_r": "cL_r.csv", + "cL_p": "cL_p.csv", + "cQ_0": "cQ_0.csv", + "cQ_alpha": "cQ_alpha.csv", + "cQ_beta": "cQ_beta.csv", + "cQ_Ma": "cQ_Ma.csv", + "cQ_Re": "cQ_Re.csv", + "cQ_q": "cQ_q.csv", + "cQ_r": "cQ_r.csv", + "cQ_p": "cQ_p.csv", + "cD_0": "cD_0.csv", + "cD_alpha": "cD_alpha.csv", + "cD_beta": "cD_beta.csv", + "cD_Ma": "cD_Ma.csv", + "cD_Re": "cD_Re.csv", + "cD_q": "cD_q.csv", + "cD_r": "cD_r.csv", + "cD_p": "cD_p.csv", + "cm_0": "cm_0.csv", + "cm_alpha": "cm_alpha.csv", + "cm_beta": "cm_beta.csv", + "cm_Ma": "cm_Ma.csv", + "cm_Re": "cm_Re.csv", + "cm_q": "cm_q.csv", + "cm_r": "cm_r.csv", + "cm_p": "cm_p.csv", + "cn_0": "cn_0.csv", + "cn_alpha": "cn_alpha.csv", + "cn_beta": "cn_beta.csv", + "cn_Ma": "cn_Ma.csv", + "cn_Re": "cn_Re.csv", + "cn_q": "cn_q.csv", + "cn_r": "cn_r.csv", + "cn_p": "cn_p.csv", + "cl_0": "cl_0.csv", + "cl_alpha": "cl_alpha.csv", + "cl_beta": "cl_beta.csv", + "cl_Ma": "cl_Ma.csv", + "cl_Re": "cl_Re.csv", + "cl_q": "cl_q.csv", + "cl_r": "cl_r.csv", + "cl_p": "cl_p.csv", + }, + ) + rocket.add_surfaces(linear_generic_surface, position=(0,0,0)) + + diff --git a/docs/user/rocket/rocket.rst b/docs/user/rocket/rocket.rst index 17ec63c1a..956175409 100644 --- a/docs/user/rocket/rocket.rst +++ b/docs/user/rocket/rocket.rst @@ -12,4 +12,10 @@ Rocket Usage :caption: Rocket Class Axes Definitions Rocket Class Axes Definitions + +.. toctree:: + :maxdepth: 3 + :caption: Generic Surfaces and Custom Aerodynamic Coefficients + + Generic Surfaces and Custom Aerodynamic Coefficients \ No newline at end of file diff --git a/rocketpy/__init__.py b/rocketpy/__init__.py index 90ffcf72a..539b8b2cb 100644 --- a/rocketpy/__init__.py +++ b/rocketpy/__init__.py @@ -30,6 +30,9 @@ Components, EllipticalFins, Fins, + FreeFormFins, + GenericSurface, + LinearGenericSurface, NoseCone, Parachute, RailButtons, diff --git a/rocketpy/mathutils/function.py b/rocketpy/mathutils/function.py index 095b82268..9e6e07b9d 100644 --- a/rocketpy/mathutils/function.py +++ b/rocketpy/mathutils/function.py @@ -16,6 +16,11 @@ import matplotlib.pyplot as plt import numpy as np from scipy import integrate, linalg, optimize +from scipy.interpolate import ( + LinearNDInterpolator, + NearestNDInterpolator, + RBFInterpolator, +) # Numpy 1.x compatibility, # TODO: remove these lines when all dependencies support numpy>=2.0.0 @@ -32,6 +37,7 @@ "akima": 2, "spline": 3, "shepard": 4, + "rbf": 5, } EXTRAPOLATION_TYPES = {"zero": 0, "natural": 1, "constant": 2} @@ -86,14 +92,20 @@ def __init__( interpolation : string, optional Interpolation method to be used if source type is ndarray. For 1-D functions, linear, polynomial, akima and spline are - supported. For N-D functions, only shepard is supported. - Default for 1-D functions is spline. + supported. For N-D functions, linear, shepard and rbf are + supported. + Default for 1-D functions is spline and for N-D functions is + shepard. extrapolation : string, optional Extrapolation method to be used if source type is ndarray. Options are 'natural', which keeps interpolation, 'constant', - which returns the value of the function at the edge of the interval, - and 'zero', which returns zero for all points outside of source - range. Default for 1-D functions is constant. + which returns the value of the function at the nearest edge of + the domain, and 'zero', which returns zero for all points outside + of source range. + Multidimensional 'natural' extrapolation for 'linear' interpolation + use a 'rbf' algorithm for smoother results. + Default for 1-D functions is constant and for N-D functions + is natural. title : string, optional Title to be displayed in the plots' figures. If none, the title will be constructed using the inputs and outputs arguments in the form @@ -224,6 +236,8 @@ def set_source(self, source): # pylint: disable=too-many-statements else: # Evaluate dimension self.__dom_dim__ = source.shape[1] - 1 + self._domain = source[:, :-1] + self._image = source[:, -1] # set x and y. If Function is 2D, also set z if self.__dom_dim__ == 1: @@ -277,8 +291,10 @@ def set_interpolation(self, method="spline"): method : string, optional Interpolation method to be used if source type is ndarray. For 1-D functions, linear, polynomial, akima and spline is - supported. For N-D functions, only shepard is supported. - Default is 'spline'. + supported. For N-D functions, linear, shepard and rbf are + supported. + Default for 1-D functions is spline and for N-D functions is + shepard. Returns ------- @@ -314,9 +330,13 @@ def set_extrapolation(self, method="constant"): extrapolation : string, optional Extrapolation method to be used if source type is ndarray. Options are 'natural', which keeps interpolation, 'constant', - which returns the value of the function at the edge of the interval, - and 'zero', which returns zero for all points outside of source - range. Default is 'constant'. + which returns the value of the function at the nearest edge of + the domain, and 'zero', which returns zero for all points outside + of source range. + Multidimensional 'natural' extrapolation for 'linear' interpolation + use a 'rbf' algorithm for smoother results. + Default for 1-D functions is constant and for N-D functions + is natural. Returns ------- @@ -330,22 +350,29 @@ def set_extrapolation(self, method="constant"): def __set_interpolation_func(self): # pylint: disable=too-many-statements """Defines interpolation function used by the Function. Each - interpolation method has its own function with exception of shepard, - which has its interpolation/extrapolation function defined in - ``Function.__interpolate_shepard__``. The function is stored in - the attribute _interpolation_func.""" + interpolation method has its own function`. + The function is stored in the attribute _interpolation_func.""" interpolation = INTERPOLATION_TYPES[self.__interpolation__] if interpolation == 0: # linear + if self.__dom_dim__ == 1: - def linear_interpolation( - x, x_min, x_max, x_data, y_data, coeffs - ): # pylint: disable=unused-argument - x_interval = bisect_left(x_data, x) - x_left = x_data[x_interval - 1] - y_left = y_data[x_interval - 1] - dx = float(x_data[x_interval] - x_left) - dy = float(y_data[x_interval] - y_left) - return (x - x_left) * (dy / dx) + y_left + def linear_interpolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + x_interval = bisect_left(x_data, x) + x_left = x_data[x_interval - 1] + y_left = y_data[x_interval - 1] + dx = float(x_data[x_interval] - x_left) + dy = float(y_data[x_interval] - y_left) + return (x - x_left) * (dy / dx) + y_left + + else: + interpolator = LinearNDInterpolator(self._domain, self._image) + + def linear_interpolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + return interpolator(x) self._interpolation_func = linear_interpolation @@ -383,8 +410,41 @@ def spline_interpolation( self._interpolation_func = spline_interpolation - elif interpolation == 4: # shepard does not use interpolation function - self._interpolation_func = None + elif interpolation == 4: # shepard + + # pylint: disable=unused-argument + def shepard_interpolation(x, x_min, x_max, x_data, y_data, _): + arg_qty, arg_dim = x.shape + result = np.empty(arg_qty) + x = x.reshape((arg_qty, 1, arg_dim)) + sub_matrix = x_data - x + distances_squared = np.sum(sub_matrix**2, axis=2) + + # Remove zero distances from further calculations + zero_distances = np.where(distances_squared == 0) + valid_indexes = np.ones(arg_qty, dtype=bool) + valid_indexes[zero_distances[0]] = False + + weights = distances_squared[valid_indexes] ** (-1.5) + numerator_sum = np.sum(y_data * weights, axis=1) + denominator_sum = np.sum(weights, axis=1) + result[valid_indexes] = numerator_sum / denominator_sum + result[~valid_indexes] = y_data[zero_distances[1]] + + return result + + self._interpolation_func = shepard_interpolation + + elif interpolation == 5: # RBF + + interpolator = RBFInterpolator(self._domain, self._image, neighbors=100) + + def rbf_interpolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + return interpolator(x) + + self._interpolation_func = rbf_interpolation def __set_extrapolation_func(self): # pylint: disable=too-many-statements """Defines extrapolation function used by the Function. Each @@ -393,10 +453,7 @@ def __set_extrapolation_func(self): # pylint: disable=too-many-statements interpolation = INTERPOLATION_TYPES[self.__interpolation__] extrapolation = EXTRAPOLATION_TYPES[self.__extrapolation__] - if interpolation == 4: # shepard does not use extrapolation function - self._extrapolation_func = None - - elif extrapolation == 0: # zero + if extrapolation == 0: # zero def zero_extrapolation( x, x_min, x_max, x_data, y_data, coeffs @@ -407,15 +464,27 @@ def zero_extrapolation( elif extrapolation == 1: # natural if interpolation == 0: # linear - def natural_extrapolation( - x, x_min, x_max, x_data, y_data, coeffs - ): # pylint: disable=unused-argument - x_interval = 1 if x < x_min else -1 - x_left = x_data[x_interval - 1] - y_left = y_data[x_interval - 1] - dx = float(x_data[x_interval] - x_left) - dy = float(y_data[x_interval] - y_left) - return (x - x_left) * (dy / dx) + y_left + if self.__dom_dim__ == 1: + + def natural_extrapolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + x_interval = 1 if x < x_min else -1 + x_left = x_data[x_interval - 1] + y_left = y_data[x_interval - 1] + dx = float(x_data[x_interval] - x_left) + dy = float(y_data[x_interval] - y_left) + return (x - x_left) * (dy / dx) + y_left + + else: + interpolator = RBFInterpolator( + self._domain, self._image, neighbors=100 + ) + + def natural_extrapolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + return interpolator(x) elif interpolation == 1: # polynomial @@ -445,13 +514,55 @@ def natural_extrapolation( x = x - x_data[-2] return a[3] * x**3 + a[2] * x**2 + a[1] * x + a[0] + elif interpolation == 4: # shepard + + # pylint: disable=unused-argument + def natural_extrapolation(x, x_min, x_max, x_data, y_data, _): + arg_qty, arg_dim = x.shape + result = np.empty(arg_qty) + x = x.reshape((arg_qty, 1, arg_dim)) + sub_matrix = x_data - x + distances_squared = np.sum(sub_matrix**2, axis=2) + + # Remove zero distances from further calculations + zero_distances = np.where(distances_squared == 0) + valid_indexes = np.ones(arg_qty, dtype=bool) + valid_indexes[zero_distances[0]] = False + + weights = distances_squared[valid_indexes] ** (-1.5) + numerator_sum = np.sum(y_data * weights, axis=1) + denominator_sum = np.sum(weights, axis=1) + result[valid_indexes] = numerator_sum / denominator_sum + result[~valid_indexes] = y_data[zero_distances[1]] + + return result + + elif interpolation == 5: # RBF + + interpolator = RBFInterpolator(self._domain, self._image, neighbors=100) + + def natural_extrapolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + return interpolator(x) + self._extrapolation_func = natural_extrapolation elif extrapolation == 2: # constant - def constant_extrapolation( - x, x_min, x_max, x_data, y_data, coeffs - ): # pylint: disable=unused-argument - return y_data[0] if x < x_min else y_data[-1] + if self.__dom_dim__ == 1: + + def constant_extrapolation( + x, x_min, x_max, x_data, y_data, coeffs + ): # pylint: disable=unused-argument + return y_data[0] if x < x_min else y_data[-1] + + else: + + extrapolator = NearestNDInterpolator(self._domain, self._image) + + def constant_extrapolation(x, x_min, x_max, x_data, y_data, coeffs): + # pylint: disable=unused-argument + return extrapolator(x) self._extrapolation_func = constant_extrapolation @@ -496,10 +607,41 @@ def __get_value_opt_1d(self, x): return y def __get_value_opt_nd(self, *args): - """Evaluate the Function at a single point (x, y, z). This method is - used when the Function is N-D.""" - # always use shepard for N-D functions - return self.__interpolate_shepard__(args) + """Evaluate the Function in a vectorized fashion for N-D domains. + + Parameters + ---------- + args : tuple + Values where the Function is to be evaluated. + + Returns + ------- + result : scalar, ndarray + Value of the Function at the specified points. + """ + args = np.column_stack(args) + arg_qty = len(args) + result = np.empty(arg_qty) + + min_domain = self._domain.T.min(axis=1) + max_domain = self._domain.T.max(axis=1) + + lower, upper = args < min_domain, args > max_domain + extrap = np.logical_or(lower.any(axis=1), upper.any(axis=1)) + + if extrap.any(): + result[extrap] = self._extrapolation_func( + args[extrap], min_domain, max_domain, self._domain, self._image, None + ) + if (~extrap).any(): + result[~extrap] = self._interpolation_func( + args[~extrap], min_domain, max_domain, self._domain, self._image, None + ) + + if arg_qty == 1: + return float(result[0]) + + return result def set_discrete( self, @@ -530,15 +672,18 @@ def set_discrete( Number of samples to be taken from inside range. Default is 200. interpolation : string Interpolation method to be used if source type is ndarray. - For 1-D functions, linear, polynomial, akima and spline is - supported. For N-D functions, only shepard is supported. - Default is 'spline'. + For 1-D functions, linear, polynomial, akima and spline are + supported. For N-D functions, linear, shepard and rbf are + supported. + Default for 1-D functions is spline and for N-D functions is + shepard. extrapolation : string, optional Extrapolation method to be used if source type is ndarray. Options are 'natural', which keeps interpolation, 'constant', - which returns the value of the function at the edge of the interval, - and 'zero', which returns zero for all points outside of source - range. Default is 'constant'. + which returns the value of the function at the nearest edge of + the domain, and 'zero', which returns zero for all points outside + of source range. Default for 1-D functions is constant and for + N-D functions is natural. one_by_one : boolean, optional If True, evaluate Function in each sample point separately. If False, evaluates Function in vectorized form. Default is True. @@ -865,13 +1010,13 @@ def get_value(self, *args): ... [(0, 0, 0), (1, 1, 1), (1, 2, 2), (2, 4, 8), (3, 9, 27)] ... ) >>> f4.get_value(1, 1) - np.float64(1.0) + 1.0 >>> f4.get_value(2, 4) - np.float64(8.0) + 8.0 >>> abs(f4.get_value(1, 1.5) - 1.5) < 1e-2 # the interpolation is not perfect - np.True_ + True >>> f4.get_value(3, 9) - np.float64(27.0) + 27.0 """ if len(args) != self.__dom_dim__: raise ValueError( @@ -898,7 +1043,7 @@ def get_value(self, *args): if all(isinstance(arg, Iterable) for arg in args): return [self.source(*arg) for arg in zip(*args)] - elif self.__dom_dim__ > 1: # deals with nd functions and shepard interp + elif self.__dom_dim__ > 1: # deals with nd functions return self.get_value_opt(*args) # Returns value for other interpolation type @@ -1765,47 +1910,6 @@ def __interpolate_akima__(self): coeffs[4 * i : 4 * i + 4] = np.linalg.solve(matrix, result) self.__akima_coefficients__ = coeffs - def __interpolate_shepard__(self, args): - """Calculates the shepard interpolation from the given arguments. - The shepard interpolation is computed by a inverse distance weighting - in a vectorized manner. - - Parameters - ---------- - args : scalar, list - Values where the Function is to be evaluated. - - Returns - ------- - result : scalar, list - The result of the interpolation. - """ - x_data = self.source[:, 0:-1] # Support for N-Dimensions - y_data = self.source[:, -1] - - arg_stack = np.column_stack(args) - arg_qty, arg_dim = arg_stack.shape - result = np.zeros(arg_qty) - - # Reshape to vectorize calculations - x = arg_stack.reshape(arg_qty, 1, arg_dim) - - sub_matrix = x_data - x - distances_squared = np.sum(sub_matrix**2, axis=2) - - # Remove zero distances from further calculations - zero_distances = np.where(distances_squared == 0) - valid_indexes = np.ones(arg_qty, dtype=bool) - valid_indexes[zero_distances[0]] = False - - weights = distances_squared[valid_indexes] ** (-1.5) - numerator_sum = np.sum(y_data * weights, axis=1) - denominator_sum = np.sum(weights, axis=1) - result[valid_indexes] = numerator_sum / denominator_sum - result[~valid_indexes] = y_data[zero_distances[1]] - - return result if len(result) > 1 else result[0] - def __neg__(self): """Negates the Function object. The result has the same effect as multiplying the Function by -1. @@ -3141,6 +3245,15 @@ def __validate_source(self, source): # pylint: disable=too-many-statements raise ValueError( "Source must be a 2D array in the form [[x1, x2 ..., xn, y], ...]." ) + + source_len, source_dim = source.shape + + if source_len < source_dim: + raise ValueError( + "Too few data points to define a domain. The number of rows " + "must be greater than or equal to the number of columns." + ) + return source if isinstance(source, NUMERICAL_TYPES): @@ -3238,14 +3351,17 @@ def __validate_interpolation(self, interpolation): interpolation = "spline" ## multiple dimensions elif self.__dom_dim__ > 1: - if interpolation not in [None, "shepard"]: + if interpolation is None: + interpolation = "shepard" + if interpolation.lower() not in ["shepard", "linear", "rbf"]: warnings.warn( ( - "Interpolation method set to 'shepard'. Only 'shepard' " - "interpolation is supported for multiple dimensions." + "Interpolation method set to 'shepard'. The methods " + "'linear', 'shepard' and 'rbf' are supported for " + "multiple dimensions." ), ) - interpolation = "shepard" + interpolation = "shepard" return interpolation def __validate_extrapolation(self, extrapolation): @@ -3261,12 +3377,14 @@ def __validate_extrapolation(self, extrapolation): ## multiple dimensions elif self.__dom_dim__ > 1: - if extrapolation not in [None, "natural"]: + if extrapolation is None: + extrapolation = "natural" + if extrapolation.lower() not in ["constant", "natural", "zero"]: warnings.warn( - "Extrapolation method set to 'natural'. Other methods " - "are not supported yet." + "Extrapolation method set to 'natural' because the " + f"{extrapolation} method is not supported." ) - extrapolation = "natural" + extrapolation = "natural" return extrapolation diff --git a/rocketpy/plots/aero_surface_plots.py b/rocketpy/plots/aero_surface_plots.py index c242973b3..2a696b1fc 100644 --- a/rocketpy/plots/aero_surface_plots.py +++ b/rocketpy/plots/aero_surface_plots.py @@ -380,6 +380,73 @@ def draw(self): plt.show() +class _FreeFormFinsPlots(_FinsPlots): + """Class that contains all free form fin plots.""" + + # pylint: disable=too-many-statements + def draw(self): + """Draw the fin shape along with some important information, including + the center line, the quarter line and the center of pressure position. + + Returns + ------- + None + """ + # Color cycle [#348ABD, #A60628, #7A68A6, #467821, #D55E00, #CC79A7, + # #56B4E9, #009E73, #F0E442, #0072B2] + + # Center of pressure + cp_point = [self.aero_surface.cpz, self.aero_surface.Yma] + + # Mean Aerodynamic Chord + yma_line = plt.Line2D( + ( + self.aero_surface.mac_lead, + self.aero_surface.mac_lead + self.aero_surface.mac_length, + ), + (self.aero_surface.Yma, self.aero_surface.Yma), + color="#467821", + linestyle="--", + label="Mean Aerodynamic Chord", + ) + + # Plotting + fig = plt.figure(figsize=(7, 4)) + with plt.style.context("bmh"): + ax = fig.add_subplot(111) + + # Fin + ax.scatter( + self.aero_surface.shape_vec[0], + self.aero_surface.shape_vec[1], + color="#A60628", + ) + ax.plot( + self.aero_surface.shape_vec[0], + self.aero_surface.shape_vec[1], + color="#A60628", + ) + # line from the last point to the first point + ax.plot( + [self.aero_surface.shape_vec[0][-1], self.aero_surface.shape_vec[0][0]], + [self.aero_surface.shape_vec[1][-1], self.aero_surface.shape_vec[1][0]], + color="#A60628", + ) + + ax.add_line(yma_line) + ax.scatter(*cp_point, label="Center of Pressure", color="red", s=100, zorder=10) + ax.scatter(*cp_point, facecolors="none", edgecolors="red", s=500, zorder=10) + + # Plot settings + ax.set_xlabel("Root chord (m)") + ax.set_ylabel("Span (m)") + ax.set_title("Free Form Fin Cross Section") + ax.legend(bbox_to_anchor=(1.05, 1.0), loc="upper left") + + plt.tight_layout() + plt.show() + + class _TailPlots(_AeroSurfacePlots): """Class that contains all tail plots.""" @@ -409,3 +476,17 @@ def all(self): None """ self.drag_coefficient_curve() + + +class _GenericSurfacePlots(_AeroSurfacePlots): + """Class that contains all generic surface plots.""" + + def draw(self): + pass + + +class _LinearGenericSurfacePlots(_AeroSurfacePlots): + """Class that contains all linear generic surface plots.""" + + def draw(self): + pass diff --git a/rocketpy/plots/flight_plots.py b/rocketpy/plots/flight_plots.py index 1e5c53449..0babaff4c 100644 --- a/rocketpy/plots/flight_plots.py +++ b/rocketpy/plots/flight_plots.py @@ -635,9 +635,9 @@ def fluid_mechanics_data(self): # pylint: disable=too-many-statements ------- None """ - plt.figure(figsize=(9, 12)) + plt.figure(figsize=(9, 16)) - ax1 = plt.subplot(411) + ax1 = plt.subplot(611) ax1.plot(self.flight.mach_number[:, 0], self.flight.mach_number[:, 1]) ax1.set_xlim(0, self.flight.t_final) ax1.set_title("Mach Number") @@ -645,7 +645,7 @@ def fluid_mechanics_data(self): # pylint: disable=too-many-statements ax1.set_ylabel("Mach Number") ax1.grid() - ax2 = plt.subplot(412) + ax2 = plt.subplot(612) ax2.plot(self.flight.reynolds_number[:, 0], self.flight.reynolds_number[:, 1]) ax2.set_xlim(0, self.flight.t_final) ax2.ticklabel_format(style="sci", axis="y", scilimits=(0, 0)) @@ -654,7 +654,7 @@ def fluid_mechanics_data(self): # pylint: disable=too-many-statements ax2.set_ylabel("Reynolds Number") ax2.grid() - ax3 = plt.subplot(413) + ax3 = plt.subplot(613) ax3.plot( self.flight.dynamic_pressure[:, 0], self.flight.dynamic_pressure[:, 1], @@ -678,7 +678,7 @@ def fluid_mechanics_data(self): # pylint: disable=too-many-statements ax3.set_ylabel("Pressure (Pa)") ax3.grid() - ax4 = plt.subplot(414) + ax4 = plt.subplot(614) ax4.plot(self.flight.angle_of_attack[:, 0], self.flight.angle_of_attack[:, 1]) ax4.set_title("Angle of Attack") ax4.set_xlabel("Time (s)") @@ -687,6 +687,33 @@ def fluid_mechanics_data(self): # pylint: disable=too-many-statements ax4.set_ylim(0, self.flight.angle_of_attack(self.flight.out_of_rail_time) + 15) ax4.grid() + ax5 = plt.subplot(615) + ax5.plot( + self.flight.partial_angle_of_attack[:, 0], + self.flight.partial_angle_of_attack[:, 1], + ) + ax5.set_title("Partial Angle of Attack") + ax5.set_xlabel("Time (s)") + ax5.set_ylabel("Partial Angle of Attack (°)") + ax5.set_xlim(self.flight.out_of_rail_time, self.first_event_time) + ax5.set_ylim( + 0, self.flight.partial_angle_of_attack(self.flight.out_of_rail_time) + 15 + ) + ax5.grid() + + ax6 = plt.subplot(616) + ax6.plot( + self.flight.angle_of_sideslip[:, 0], self.flight.angle_of_sideslip[:, 1] + ) + ax6.set_title("Angle of Sideslip") + ax6.set_xlabel("Time (s)") + ax6.set_ylabel("Angle of Sideslip (°)") + ax6.set_xlim(self.flight.out_of_rail_time, self.first_event_time) + ax6.set_ylim( + 0, self.flight.angle_of_sideslip(self.flight.out_of_rail_time) + 15 + ) + ax6.grid() + plt.subplots_adjust(hspace=0.5) plt.show() diff --git a/rocketpy/plots/monte_carlo_plots.py b/rocketpy/plots/monte_carlo_plots.py index 2264597da..7f8313d4b 100644 --- a/rocketpy/plots/monte_carlo_plots.py +++ b/rocketpy/plots/monte_carlo_plots.py @@ -54,9 +54,14 @@ def ellipses( "The image file was not found. Please check the path." ) from e - impact_ellipses, apogee_ellipses, apogee_x, apogee_y, impact_x, impact_y = ( - generate_monte_carlo_ellipses(self.monte_carlo.results) - ) + ( + impact_ellipses, + apogee_ellipses, + apogee_x, + apogee_y, + impact_x, + impact_y, + ) = generate_monte_carlo_ellipses(self.monte_carlo.results) # Create plot figure plt.figure(figsize=(8, 6), dpi=150) diff --git a/rocketpy/plots/rocket_plots.py b/rocketpy/plots/rocket_plots.py index 2add1e78f..4ae47cdc4 100644 --- a/rocketpy/plots/rocket_plots.py +++ b/rocketpy/plots/rocket_plots.py @@ -5,6 +5,7 @@ from rocketpy.motors import EmptyMotor, HybridMotor, LiquidMotor, SolidMotor from rocketpy.rocket.aero_surface import Fins, NoseCone, Tail +from rocketpy.rocket.aero_surface.generic_surface import GenericSurface class _RocketPlots: @@ -208,7 +209,7 @@ def draw(self, vis_args=None, plane="xz"): reverse = csys == 1 self.rocket.aerodynamic_surfaces.sort_by_position(reverse=reverse) - drawn_surfaces = self._draw_aerodynamic_surfaces(ax, vis_args) + drawn_surfaces = self._draw_aerodynamic_surfaces(ax, vis_args, plane) last_radius, last_x = self._draw_tubes(ax, drawn_surfaces, vis_args) self._draw_motor(last_radius, last_x, ax, vis_args) self._draw_rail_buttons(ax, vis_args) @@ -224,7 +225,7 @@ def draw(self, vis_args=None, plane="xz"): plt.tight_layout() plt.show() - def _draw_aerodynamic_surfaces(self, ax, vis_args): + def _draw_aerodynamic_surfaces(self, ax, vis_args, plane): """Draws the aerodynamic surfaces and saves the position of the points of interest for the tubes.""" # List of drawn surfaces with the position of points of interest @@ -239,11 +240,15 @@ def _draw_aerodynamic_surfaces(self, ax, vis_args): for surface, position in self.rocket.aerodynamic_surfaces: if isinstance(surface, NoseCone): - self._draw_nose_cone(ax, surface, position, drawn_surfaces, vis_args) + self._draw_nose_cone(ax, surface, position.z, drawn_surfaces, vis_args) elif isinstance(surface, Tail): - self._draw_tail(ax, surface, position, drawn_surfaces, vis_args) + self._draw_tail(ax, surface, position.z, drawn_surfaces, vis_args) elif isinstance(surface, Fins): - self._draw_fins(ax, surface, position, drawn_surfaces, vis_args) + self._draw_fins(ax, surface, position.z, drawn_surfaces, vis_args) + elif isinstance(surface, GenericSurface): + self._draw_generic_surface( + ax, surface, position, drawn_surfaces, vis_args, plane + ) return drawn_surfaces def _draw_nose_cone(self, ax, surface, position, drawn_surfaces, vis_args): @@ -336,6 +341,39 @@ def _draw_fins(self, ax, surface, position, drawn_surfaces, vis_args): drawn_surfaces.append((surface, position, surface.rocket_radius, x_rotated[-1])) + def _draw_generic_surface( + self, + ax, + surface, + position, + drawn_surfaces, + vis_args, # pylint: disable=unused-argument + plane, + ): + """Draws the generic surface and saves the position of the points of interest + for the tubes.""" + if plane == "xz": + # z position of the sensor is the x position in the plot + x_pos = position[2] + # x position of the surface is the y position in the plot + y_pos = position[0] + elif plane == "yz": + # z position of the surface is the x position in the plot + x_pos = position[2] + # y position of the surface is the y position in the plot + y_pos = position[1] + else: + raise ValueError("Plane must be 'xz' or 'yz'.") + + ax.scatter( + x_pos, + y_pos, + linewidth=2, + zorder=10, + label=surface.name, + ) + drawn_surfaces.append((surface, position.z, self.rocket.radius, x_pos)) + def _draw_tubes(self, ax, drawn_surfaces, vis_args): """Draws the tubes between the aerodynamic surfaces.""" for i, d_surface in enumerate(drawn_surfaces): @@ -528,8 +566,8 @@ def _draw_rail_buttons(self, ax, vis_args): """Draws the rail buttons of the rocket.""" try: buttons, pos = self.rocket.rail_buttons[0] - lower = pos - upper = pos + buttons.buttons_distance * self.rocket._csys + lower = pos.z + upper = lower + buttons.buttons_distance * self.rocket._csys ax.scatter( lower, -self.rocket.radius, marker="s", color=vis_args["buttons"], s=15 ) diff --git a/rocketpy/plots/sensitivity_plots.py b/rocketpy/plots/sensitivity_plots.py index 1f507709c..023dd8f8e 100644 --- a/rocketpy/plots/sensitivity_plots.py +++ b/rocketpy/plots/sensitivity_plots.py @@ -3,7 +3,6 @@ class _SensitivityModelPlots: - def __init__(self, model): self.model = model diff --git a/rocketpy/prints/aero_surface_prints.py b/rocketpy/prints/aero_surface_prints.py index 7cc87c28f..378433511 100644 --- a/rocketpy/prints/aero_surface_prints.py +++ b/rocketpy/prints/aero_surface_prints.py @@ -72,7 +72,6 @@ def geometry(self): class _FinsPrints(_AeroSurfacePrints): - def geometry(self): print("Geometric information of the fin set:") print("-------------------------------------") @@ -184,6 +183,10 @@ class _EllipticalFinsPrints(_FinsPrints): """Class that contains all elliptical fins prints.""" +class _FreeFormFinsPrints(_FinsPrints): + """Class that contains all free form fins prints.""" + + class _TailPrints(_AeroSurfacePrints): """Class that contains all tail prints.""" @@ -228,3 +231,45 @@ def geometry(self): def all(self): pass + + +class _GenericSurfacePrints(_AeroSurfacePrints): + """Class that contains all generic surface prints.""" + + def geometry(self): + print("Geometric information of the Surface:") + print("----------------------------------") + print(f"Reference Area: {self.generic_surface.reference_area:.3f} m") + print(f"Reference length: {2*self.generic_surface.rocket_radius:.3f} m") + + def all(self): + """Prints all information of the generic surface. + + Returns + ------- + None + """ + self.identity() + self.geometry() + self.lift() + + +class _LinearGenericSurfacePrints(_AeroSurfacePrints): + """Class that contains all linear generic surface prints.""" + + def geometry(self): + print("Geometric information of the Surface:") + print("----------------------------------") + print(f"Reference Area: {self.generic_surface.reference_area:.3f} m") + print(f"Reference length: {2*self.generic_surface.rocket_radius:.3f} m") + + def all(self): + """Prints all information of the linear generic surface. + + Returns + ------- + None + """ + self.identity() + self.geometry() + self.lift() diff --git a/rocketpy/prints/rocket_prints.py b/rocketpy/prints/rocket_prints.py index 7ad42fd7b..c9f5585ac 100644 --- a/rocketpy/prints/rocket_prints.py +++ b/rocketpy/prints/rocket_prints.py @@ -1,3 +1,6 @@ +from rocketpy.rocket.aero_surface.generic_surface import GenericSurface + + class _RocketPrints: """Class that holds prints methods for Rocket class. @@ -98,7 +101,9 @@ def rocket_aerodynamics_quantities(self): None """ print("\nAerodynamics Lift Coefficient Derivatives\n") - for surface, position in self.rocket.aerodynamic_surfaces: + for surface, _ in self.rocket.aerodynamic_surfaces: + if isinstance(surface, GenericSurface): + continue name = surface.name # ref_factor corrects lift for different reference areas ref_factor = (surface.rocket_radius / self.rocket.radius) ** 2 @@ -113,7 +118,7 @@ def rocket_aerodynamics_quantities(self): cpz = surface.cp[2] # relative to the user defined coordinate system print( f"{name} Center of Pressure position: " - f"{position - self.rocket._csys * cpz:.3f} m" + f"{position.z - self.rocket._csys * cpz:.3f} m" ) print("\nStability\n") print( diff --git a/rocketpy/prints/sensitivity_prints.py b/rocketpy/prints/sensitivity_prints.py index 014be586c..8a0301d79 100644 --- a/rocketpy/prints/sensitivity_prints.py +++ b/rocketpy/prints/sensitivity_prints.py @@ -4,7 +4,6 @@ class _SensitivityModelPrints: - def __init__(self, model): self.model = model diff --git a/rocketpy/prints/sensors_prints.py b/rocketpy/prints/sensors_prints.py index 73ab062f8..4bf844c5a 100644 --- a/rocketpy/prints/sensors_prints.py +++ b/rocketpy/prints/sensors_prints.py @@ -68,7 +68,6 @@ def all(self): class _InertialSensorPrints(_SensorPrints): - def orientation(self): """Prints the orientation of the sensor.""" print("\nOrientation of the Sensor:\n") diff --git a/rocketpy/rocket/__init__.py b/rocketpy/rocket/__init__.py index fb6ea2b2c..0687b5ee5 100644 --- a/rocketpy/rocket/__init__.py +++ b/rocketpy/rocket/__init__.py @@ -4,6 +4,9 @@ AirBrakes, EllipticalFins, Fins, + FreeFormFins, + GenericSurface, + LinearGenericSurface, NoseCone, RailButtons, Tail, diff --git a/rocketpy/rocket/aero_surface/__init__.py b/rocketpy/rocket/aero_surface/__init__.py index 9d9c68586..ad784f8d0 100644 --- a/rocketpy/rocket/aero_surface/__init__.py +++ b/rocketpy/rocket/aero_surface/__init__.py @@ -1,6 +1,13 @@ from rocketpy.rocket.aero_surface.aero_surface import AeroSurface from rocketpy.rocket.aero_surface.air_brakes import AirBrakes -from rocketpy.rocket.aero_surface.fins import EllipticalFins, Fins, TrapezoidalFins +from rocketpy.rocket.aero_surface.fins import ( + EllipticalFins, + Fins, + FreeFormFins, + TrapezoidalFins, +) +from rocketpy.rocket.aero_surface.generic_surface import GenericSurface +from rocketpy.rocket.aero_surface.linear_generic_surface import LinearGenericSurface from rocketpy.rocket.aero_surface.nose_cone import NoseCone from rocketpy.rocket.aero_surface.rail_buttons import RailButtons from rocketpy.rocket.aero_surface.tail import Tail diff --git a/rocketpy/rocket/aero_surface/aero_surface.py b/rocketpy/rocket/aero_surface/aero_surface.py index 81b5610e3..15ca14f1d 100644 --- a/rocketpy/rocket/aero_surface/aero_surface.py +++ b/rocketpy/rocket/aero_surface/aero_surface.py @@ -90,3 +90,60 @@ def all_info(self): ------- None """ + + def compute_forces_and_moments( + self, + stream_velocity, + stream_speed, + stream_mach, + rho, + cp, + *args, + ): # pylint: disable=unused-argument + """Computes the forces and moments acting on the aerodynamic surface. + Used in each time step of the simulation. This method is valid for + the barrowman aerodynamic models. + + Parameters + ---------- + stream_velocity : tuple + Tuple containing the stream velocity components in the body frame. + stream_speed : int, float + Speed of the stream in m/s. + stream_mach : int, float + Mach number of the stream. + rho : int, float + Density of the stream in kg/m^3. + cp : Vector + Center of pressure coordinates in the body frame. + args : tuple + Additional arguments. + kwargs : dict + Additional keyword arguments. + + Returns + ------- + tuple of float + The aerodynamic forces (lift, side_force, drag) and moments + (pitch, yaw, roll) in the body frame. + """ + R1, R2, R3, M1, M2, M3 = 0, 0, 0, 0, 0, 0 + cpz = cp[2] + stream_vx, stream_vy, stream_vz = stream_velocity + if stream_vx**2 + stream_vy**2 != 0: # TODO: maybe try/except + # Normalize component stream velocity in body frame + stream_vzn = stream_vz / stream_speed + if -1 * stream_vzn < 1: + attack_angle = np.arccos(-stream_vzn) + c_lift = self.cl.get_value_opt(attack_angle, stream_mach) + # Component lift force magnitude + lift = 0.5 * rho * (stream_speed**2) * self.reference_area * c_lift + # Component lift force components + lift_dir_norm = (stream_vx**2 + stream_vy**2) ** 0.5 + lift_xb = lift * (stream_vx / lift_dir_norm) + lift_yb = lift * (stream_vy / lift_dir_norm) + # Total lift force + R1, R2, R3 = lift_xb, lift_yb, 0 + # Total moment + M1, M2, M3 = -cpz * lift_yb, cpz * lift_xb, 0 + return R1, R2, R3, M1, M2, M3 diff --git a/rocketpy/rocket/aero_surface/fins/__init__.py b/rocketpy/rocket/aero_surface/fins/__init__.py index f1efc603a..941aa5465 100644 --- a/rocketpy/rocket/aero_surface/fins/__init__.py +++ b/rocketpy/rocket/aero_surface/fins/__init__.py @@ -1,3 +1,4 @@ from rocketpy.rocket.aero_surface.fins.elliptical_fins import EllipticalFins from rocketpy.rocket.aero_surface.fins.fins import Fins +from rocketpy.rocket.aero_surface.fins.free_form_fins import FreeFormFins from rocketpy.rocket.aero_surface.fins.trapezoidal_fins import TrapezoidalFins diff --git a/rocketpy/rocket/aero_surface/fins/elliptical_fins.py b/rocketpy/rocket/aero_surface/fins/elliptical_fins.py index d30cf9aa1..f0d5f926f 100644 --- a/rocketpy/rocket/aero_surface/fins/elliptical_fins.py +++ b/rocketpy/rocket/aero_surface/fins/elliptical_fins.py @@ -106,7 +106,7 @@ def __init__( Parameters ---------- n : int - Number of fins, from 2 to infinity. + Number of fins, must be larger than 2. root_chord : int, float Fin root chord in meters. span : int, float diff --git a/rocketpy/rocket/aero_surface/fins/fins.py b/rocketpy/rocket/aero_surface/fins/fins.py index 74f7922f7..29f4e8cff 100644 --- a/rocketpy/rocket/aero_surface/fins/fins.py +++ b/rocketpy/rocket/aero_surface/fins/fins.py @@ -106,7 +106,7 @@ def __init__( Parameters ---------- n : int - Number of fins, from 2 to infinity. + Number of fins, must be larger than 2. root_chord : int, float Fin root chord in meters. span : int, float @@ -366,6 +366,66 @@ def fin_num_correction(n): else: return n / 2 + def compute_forces_and_moments( + self, + stream_velocity, + stream_speed, + stream_mach, + rho, + cp, + omega, + *args, + ): # pylint: disable=arguments-differ + """Computes the forces and moments acting on the aerodynamic surface. + + Parameters + ---------- + stream_velocity : tuple of float + The velocity of the airflow relative to the surface. + stream_speed : float + The magnitude of the airflow speed. + stream_mach : float + The Mach number of the airflow. + rho : float + Air density. + cp : Vector + Center of pressure coordinates in the body frame. + omega: tuple[float, float, float] + Tuple containing angular velocities around the x, y, z axes. + + Returns + ------- + tuple of float + The aerodynamic forces (lift, side_force, drag) and moments + (pitch, yaw, roll) in the body frame. + """ + + R1, R2, R3, M1, M2, _ = super().compute_forces_and_moments( + stream_velocity, + stream_speed, + stream_mach, + rho, + cp, + ) + clf_delta, cld_omega, cant_angle_rad = self.roll_parameters + M3_forcing = ( + (1 / 2 * rho * stream_speed**2) + * self.reference_area + * self.reference_length + * clf_delta.get_value_opt(stream_mach) + * cant_angle_rad + ) + M3_damping = ( + (1 / 2 * rho * stream_speed) + * self.reference_area + * (self.reference_length) ** 2 + * cld_omega.get_value_opt(stream_mach) + * omega[2] + / 2 + ) + M3 = M3_forcing - M3_damping + return R1, R2, R3, M1, M2, M3 + def draw(self): """Draw the fin shape along with some important information, including the center line, the quarter line and the center of pressure position. diff --git a/rocketpy/rocket/aero_surface/fins/free_form_fins.py b/rocketpy/rocket/aero_surface/fins/free_form_fins.py new file mode 100644 index 000000000..3abdb84ad --- /dev/null +++ b/rocketpy/rocket/aero_surface/fins/free_form_fins.py @@ -0,0 +1,368 @@ +import warnings + +import numpy as np + +from rocketpy.plots.aero_surface_plots import _FreeFormFinsPlots +from rocketpy.prints.aero_surface_prints import _FreeFormFinsPrints + +from .fins import Fins + + +class FreeFormFins(Fins): + """Class that defines and holds information for a free form fin set. + + This class inherits from the Fins class. + + Note + ---- + Local coordinate system: + - Origin located at the top of the root chord. + - Z axis along the longitudinal axis of symmetry, positive downwards (top -> bottom). + - Y axis perpendicular to the Z axis, in the span direction, positive upwards. + - X axis completes the right-handed coordinate system. + + See Also + -------- + Fins + + Attributes + ---------- + FreeFormFins.n : int + Number of fins in fin set. + FreeFormFins.rocket_radius : float + The reference rocket radius used for lift coefficient normalization, in + meters. + FreeFormFins.airfoil : tuple + Tuple of two items. First is the airfoil lift curve. + Second is the unit of the curve (radians or degrees). + FreeFormFins.cant_angle : float + Fins cant angle with respect to the rocket centerline, in degrees. + FreeFormFins.cant_angle_rad : float + Fins cant angle with respect to the rocket centerline, in radians. + FreeFormFins.root_chord : float + Fin root chord in meters. + FreeFormFins.span : float + Fin span in meters. + FreeFormFins.name : string + Name of fin set. + FreeFormFins.d : float + Reference diameter of the rocket, in meters. + FreeFormFins.ref_area : float + Reference area of the rocket, in m². + FreeFormFins.Af : float + Area of the longitudinal section of each fin in the set. + FreeFormFins.AR : float + Aspect ratio of each fin in the set + FreeFormFins.gamma_c : float + Fin mid-chord sweep angle. + FreeFormFins.Yma : float + Span wise position of the mean aerodynamic chord. + FreeFormFins.roll_geometrical_constant : float + Geometrical constant used in roll calculations. + FreeFormFins.tau : float + Geometrical relation used to simplify lift and roll calculations. + FreeFormFins.lift_interference_factor : float + Factor of Fin-Body interference in the lift coefficient. + FreeFormFins.cp : tuple + Tuple with the x, y and z local coordinates of the fin set center of + pressure. Has units of length and is given in meters. + FreeFormFins.cpx : float + Fin set local center of pressure x coordinate. Has units of length and + is given in meters. + FreeFormFins.cpy : float + Fin set local center of pressure y coordinate. Has units of length and + is given in meters. + FreeFormFins.cpz : float + Fin set local center of pressure z coordinate. Has units of length and + is given in meters. + FreeFormFins.cl : Function + Function which defines the lift coefficient as a function of the angle + of attack and the Mach number. Takes as input the angle of attack in + radians and the Mach number. Returns the lift coefficient. + FreeFormFins.clalpha : float + Lift coefficient slope. Has units of 1/rad. + FreeFormFins.mac_length : float + Mean aerodynamic chord length of the fin set. + FreeFormFins.mac_lead : float + Mean aerodynamic chord leading edge x coordinate. + """ + + def __init__( + self, + n, + shape_points, + rocket_radius, + cant_angle=0, + airfoil=None, + name="Fins", + ): + """Initialize FreeFormFins class. + + Parameters + ---------- + n : int + Number of fins, must be larger than 2. + shape_points : list + List of tuples (x, y) containing the coordinates of the fin's + geometry defining points. The point (0, 0) is the root leading edge. + Positive x is rearwards, positive y is upwards (span direction). + The shape will be interpolated between the points, in the order + they are given. The last point connects to the first point, and + represents the trailing edge. + rocket_radius : int, float + Reference radius to calculate lift coefficient, in meters. + cant_angle : int, float, optional + Fins cant angle with respect to the rocket centerline. Must + be given in degrees. + airfoil : tuple, optional + Default is null, in which case fins will be treated as flat plates. + Otherwise, if tuple, fins will be considered as airfoils. The + tuple's first item specifies the airfoil's lift coefficient + by angle of attack and must be either a .csv, .txt, ndarray + or callable. The .csv and .txt files can contain a single line + header and the first column must specify the angle of attack, while + the second column must specify the lift coefficient. The + ndarray should be as [(x0, y0), (x1, y1), (x2, y2), ...] + where x0 is the angle of attack and y0 is the lift coefficient. + If callable, it should take an angle of attack as input and + return the lift coefficient at that angle of attack. + The tuple's second item is the unit of the angle of attack, + accepting either "radians" or "degrees". + name : str + Name of fin set. + + Returns + ------- + None + """ + self.shape_points = shape_points + + down = False + for i in range(1, len(shape_points)): + if shape_points[i][1] > shape_points[i - 1][1] and down: + warnings.warn( + "Jagged fin shape detected. This may cause small inaccuracies " + "center of pressure and pitch moment calculations." + ) + break + if shape_points[i][1] < shape_points[i - 1][1]: + down = True + i += 1 + + root_chord = abs(shape_points[0][0] - shape_points[-1][0]) + ys = [point[1] for point in shape_points] + span = max(ys) - min(ys) + + super().__init__( + n, + root_chord, + span, + rocket_radius, + cant_angle, + airfoil, + name, + ) + + self.evaluate_geometrical_parameters() + self.evaluate_center_of_pressure() + self.evaluate_lift_coefficient() + self.evaluate_roll_parameters() + + self.prints = _FreeFormFinsPrints(self) + self.plots = _FreeFormFinsPlots(self) + + def evaluate_center_of_pressure(self): + """Calculates and returns the center of pressure of the fin set in local + coordinates. The center of pressure position is saved and stored as a + tuple. + + Returns + ------- + None + """ + # Center of pressure position in local coordinates + cpz = self.mac_lead + 0.25 * self.mac_length + self.cpx = 0 + self.cpy = 0 + self.cpz = cpz + self.cp = (self.cpx, self.cpy, self.cpz) + + def evaluate_geometrical_parameters(self): # pylint: disable=too-many-statements + """ + Calculates and saves the fin set's geometrical parameters such as the + fin area, aspect ratio, and parameters related to roll movement. This + method uses the same calculations to those in OpenRocket for free-form + fin shapes. + + Returns + ------- + None + """ + # pylint: disable=invalid-name + # pylint: disable=too-many-locals + # Calculate the fin area (Af) using the Shoelace theorem (polygon area formula) + Af = 0 + for i in range(len(self.shape_points) - 1): + x1, y1 = self.shape_points[i] + x2, y2 = self.shape_points[i + 1] + Af += (y1 + y2) * (x1 - x2) + Af = abs(Af) / 2 + if Af < 1e-6: + raise ValueError("Fin area is too small. Check the shape_points.") + + # Calculate aspect ratio (AR) and lift interference factors + AR = 2 * self.span**2 / Af # Aspect ratio + tau = (self.span + self.rocket_radius) / self.rocket_radius + lift_interference_factor = 1 + 1 / tau + + # Calculate roll forcing interference factor using OpenRocket's approach + roll_forcing_interference_factor = (1 / np.pi**2) * ( + (np.pi**2 / 4) * ((tau + 1) ** 2 / tau**2) + + ((np.pi * (tau**2 + 1) ** 2) / (tau**2 * (tau - 1) ** 2)) + * np.arcsin((tau**2 - 1) / (tau**2 + 1)) + - (2 * np.pi * (tau + 1)) / (tau * (tau - 1)) + + ((tau**2 + 1) ** 2 / (tau**2 * (tau - 1) ** 2)) + * (np.arcsin((tau**2 - 1) / (tau**2 + 1))) ** 2 + - (4 * (tau + 1)) + / (tau * (tau - 1)) + * np.arcsin((tau**2 - 1) / (tau**2 + 1)) + + (8 / (tau - 1) ** 2) * np.log((tau**2 + 1) / (2 * tau)) + ) + + # Define number of interpolation points along the span of the fin + points_per_line = 40 # Same as OpenRocket + + # Initialize arrays for leading/trailing edge and chord lengths + chord_lead = np.ones(points_per_line) * np.inf # Leading edge x coordinates + chord_trail = np.ones(points_per_line) * -np.inf # Trailing edge x coordinates + chord_length = np.zeros( + points_per_line + ) # Chord length for each spanwise section + + # Discretize fin shape and calculate chord length, leading, and trailing edges + for p in range(1, len(self.shape_points)): + x1, y1 = self.shape_points[p - 1] + x2, y2 = self.shape_points[p] + + # Compute corresponding points along the fin span (clamp to valid range) + prev_idx = int(y1 / self.span * (points_per_line - 1)) + curr_idx = int(y2 / self.span * (points_per_line - 1)) + prev_idx = np.clip(prev_idx, 0, points_per_line - 1) + curr_idx = np.clip(curr_idx, 0, points_per_line - 1) + + if prev_idx > curr_idx: + prev_idx, curr_idx = curr_idx, prev_idx + + # Compute intersection of fin edge with each spanwise section + for i in range(prev_idx, curr_idx + 1): + y = i * self.span / (points_per_line - 1) + if y1 != y2: + x = np.clip( + (y - y2) / (y1 - y2) * x1 + (y1 - y) / (y1 - y2) * x2, + min(x1, x2), + max(x1, x2), + ) + else: + x = x1 # Handle horizontal segments + + # Update leading and trailing edge positions + chord_lead[i] = min(chord_lead[i], x) + chord_trail[i] = max(chord_trail[i], x) + + # Update chord length + if y1 < y2: + chord_length[i] -= x + else: + chord_length[i] += x + + # Replace infinities and handle invalid values in chord_lead and chord_trail + for i in range(points_per_line): + if ( + np.isinf(chord_lead[i]) + or np.isinf(chord_trail[i]) + or np.isnan(chord_lead[i]) + or np.isnan(chord_trail[i]) + ): + chord_lead[i] = 0 + chord_trail[i] = 0 + if chord_length[i] < 0 or np.isnan(chord_length[i]): + chord_length[i] = 0 + if chord_length[i] > chord_trail[i] - chord_lead[i]: + chord_length[i] = chord_trail[i] - chord_lead[i] + + # Initialize integration variables for various aerodynamic and roll properties + radius = self.rocket_radius + total_area = 0 + mac_length = 0 # Mean aerodynamic chord length + mac_lead = 0 # Mean aerodynamic chord leading edge + mac_span = 0 # Mean aerodynamic chord spanwise position (Yma) + cos_gamma_sum = 0 # Sum of cosine of the sweep angle + roll_geometrical_constant = 0 + roll_damping_numerator = 0 + roll_damping_denominator = 0 + + # Perform integration over spanwise sections + dy = self.span / (points_per_line - 1) + for i in range(points_per_line): + chord = chord_trail[i] - chord_lead[i] + y = i * dy + + # Update integration variables + mac_length += chord * chord + mac_span += y * chord + mac_lead += chord_lead[i] * chord + total_area += chord + roll_geometrical_constant += chord_length[i] * (radius + y) ** 2 + roll_damping_numerator += radius**3 * chord / (radius + y) ** 2 + roll_damping_denominator += (radius + y) * chord + + # Update cosine of sweep angle (cos_gamma) + if i > 0: + dx = (chord_trail[i] + chord_lead[i]) / 2 - ( + chord_trail[i - 1] + chord_lead[i - 1] + ) / 2 + cos_gamma_sum += dy / np.hypot(dx, dy) + + # Finalize mean aerodynamic chord properties + mac_length *= dy + mac_span *= dy + mac_lead *= dy + total_area *= dy + roll_geometrical_constant *= dy + roll_damping_numerator *= dy + roll_damping_denominator *= dy + + mac_length /= total_area + mac_span /= total_area + mac_lead /= total_area + cos_gamma = cos_gamma_sum / (points_per_line - 1) + + # Store computed values + self.Af = Af # Fin area + self.AR = AR # Aspect ratio + self.gamma_c = np.arccos(cos_gamma) # Sweep angle + self.Yma = mac_span # Mean aerodynamic chord spanwise position + self.mac_length = mac_length + self.mac_lead = mac_lead + self.tau = tau + self.roll_geometrical_constant = roll_geometrical_constant + self.lift_interference_factor = lift_interference_factor + self.roll_forcing_interference_factor = roll_forcing_interference_factor + self.roll_damping_interference_factor = 1 + ( + roll_damping_numerator / roll_damping_denominator + ) + + # Evaluate the shape and finalize geometry + self.evaluate_shape() + + def evaluate_shape(self): + x_array, y_array = zip(*self.shape_points) + self.shape_vec = [np.array(x_array), np.array(y_array)] + + def info(self): + self.prints.geometry() + self.prints.lift() + + def all_info(self): + self.prints.all() + self.plots.all() diff --git a/rocketpy/rocket/aero_surface/fins/trapezoidal_fins.py b/rocketpy/rocket/aero_surface/fins/trapezoidal_fins.py index 3040e21c9..7a915f3b1 100644 --- a/rocketpy/rocket/aero_surface/fins/trapezoidal_fins.py +++ b/rocketpy/rocket/aero_surface/fins/trapezoidal_fins.py @@ -111,7 +111,7 @@ def __init__( Parameters ---------- n : int - Number of fins, from 2 to infinity. + Number of fins, must be larger than 2. root_chord : int, float Fin root chord in meters. tip_chord : int, float diff --git a/rocketpy/rocket/aero_surface/generic_surface.py b/rocketpy/rocket/aero_surface/generic_surface.py new file mode 100644 index 000000000..1a0e1015d --- /dev/null +++ b/rocketpy/rocket/aero_surface/generic_surface.py @@ -0,0 +1,445 @@ +import copy +import csv +import math + +import numpy as np + +from rocketpy.mathutils import Function +from rocketpy.mathutils.vector_matrix import Matrix, Vector + + +class GenericSurface: + """Defines a generic aerodynamic surface with custom force and moment + coefficients. The coefficients can be nonlinear functions of the angle of + attack, sideslip angle, Mach number, Reynolds number, pitch rate, yaw rate + and roll rate.""" + + def __init__( + self, + reference_area, + reference_length, + coefficients, + center_of_pressure=(0, 0, 0), + name="Generic Surface", + ): + """Create a generic aerodynamic surface, defined by its aerodynamic + coefficients. This surface is used to model any aerodynamic surface + that does not fit the predefined classes. + + Important + --------- + All the aerodynamic coefficients can be input as callable functions of + angle of attack, angle of sideslip, Mach number, Reynolds number, + pitch rate, yaw rate and roll rate. For CSV files, the header must + contain at least one of the following: "alpha", "beta", "mach", + "reynolds", "pitch_rate", "yaw_rate" and "roll_rate". + + See Also + -------- + :ref:`genericsurfaces`. + + Parameters + ---------- + reference_area : int, float + Reference area of the aerodynamic surface. Has the unit of meters + squared. Commonly defined as the rocket's cross-sectional area. + reference_length : int, float + Reference length of the aerodynamic surface. Has the unit of meters. + Commonly defined as the rocket's diameter. + coefficients: dict + List of coefficients. If a coefficient is omitted, it is set to 0. + The valid coefficients are:\n + cL: str, callable, optional + Lift coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + cQ: str, callable, optional + Side force coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + cD: str, callable, optional + Drag coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + cm: str, callable, optional + Pitch moment coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + cn: str, callable, optional + Yaw moment coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + cl: str, callable, optional + Roll moment coefficient. Can be a path to a CSV file or a callable. + Default is 0.\n + center_of_pressure : tuple, list, optional + Application point of the aerodynamic forces and moments. The + center of pressure is defined in the local coordinate system of the + aerodynamic surface. The default value is (0, 0, 0). + name : str, optional + Name of the aerodynamic surface. Default is 'GenericSurface'. + """ + + self.reference_area = reference_area + self.reference_length = reference_length + self.center_of_pressure = center_of_pressure + self.cp = center_of_pressure + self.cpx = center_of_pressure[0] + self.cpy = center_of_pressure[1] + self.cpz = center_of_pressure[2] + self.name = name + + default_coefficients = self._get_default_coefficients() + self._check_coefficients(coefficients, default_coefficients) + coefficients = self._complete_coefficients(coefficients, default_coefficients) + for coeff, coeff_value in coefficients.items(): + value = self._process_input(coeff_value, coeff) + setattr(self, coeff, value) + + def _get_default_coefficients(self): + """Returns default coefficients + + Returns + ------- + default_coefficients: dict + Dictionary whose keys are the coefficients names and keys + are the default values. + """ + default_coefficients = { + "cL": 0, + "cQ": 0, + "cD": 0, + "cm": 0, + "cn": 0, + "cl": 0, + } + return default_coefficients + + def _complete_coefficients(self, input_coefficients, default_coefficients): + """Creates a copy of the input coefficients dict and fill it with missing + keys with default values + + Parameters + ---------- + input_coefficients : str, dict + Coefficients dictionary passed by the user. If the user only specifies some + of the coefficients, the remaining are completed with class default + values + default_coefficients : dict + Default coefficients of the class + + Returns + ------- + coefficients : dict + Coefficients dictionary used to setup coefficient attributes + """ + coefficients = copy.deepcopy(input_coefficients) + for coeff, value in default_coefficients.items(): + if coeff not in coefficients.keys(): + coefficients[coeff] = value + + return coefficients + + def _check_coefficients(self, input_coefficients, default_coefficients): + """Check if input coefficients have only valid keys + + Parameters + ---------- + input_coefficients : str, dict + Coefficients dictionary passed by the user. If the user only specifies some + of the coefficients, the remaining are completed with class default + values + default_coefficients : dict + Default coefficients of the class + + Raises + ------ + ValueError + Raises a value error if the input coefficient has an invalid key + """ + invalid_keys = set(input_coefficients) - set(default_coefficients) + if invalid_keys: + raise ValueError( + f"Invalid coefficient name(s) used in key(s): {', '.join(invalid_keys)}. " + "Check the documentation for valid names." + ) + + def _compute_from_coefficients( + self, + rho, + stream_speed, + alpha, + beta, + mach, + reynolds, + pitch_rate, + yaw_rate, + roll_rate, + ): + """Compute the aerodynamic forces and moments from the aerodynamic + coefficients. + + Parameters + ---------- + rho : float + Air density. + stream_speed : float + Magnitude of the airflow speed. + alpha : float + Angle of attack in radians. + beta : float + Sideslip angle in radians. + mach : float + Mach number. + reynolds : float + Reynolds number. + pitch_rate : float + Pitch rate in radians per second. + yaw_rate : float + Yaw rate in radians per second. + roll_rate : float + Roll rate in radians per second. + + Returns + ------- + tuple of float + The aerodynamic forces (lift, side_force, drag) and moments + (pitch, yaw, roll) in the body frame. + """ + # Precompute common values + dyn_pressure_area = 0.5 * rho * stream_speed**2 * self.reference_area + dyn_pressure_area_length = dyn_pressure_area * self.reference_length + + # Compute aerodynamic forces + lift = dyn_pressure_area * self.cL( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + side = dyn_pressure_area * self.cQ( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + drag = dyn_pressure_area * self.cD( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + # Compute aerodynamic moments + pitch = dyn_pressure_area_length * self.cm( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + yaw = dyn_pressure_area_length * self.cn( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + roll = dyn_pressure_area_length * self.cl( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + return lift, side, drag, pitch, yaw, roll + + def compute_forces_and_moments( + self, + stream_velocity, + stream_speed, + stream_mach, + rho, + cp, + omega, + reynolds, + ): + """Computes the forces and moments acting on the aerodynamic surface. + Used in each time step of the simulation. This method is valid for + both linear and nonlinear aerodynamic coefficients. + + Parameters + ---------- + stream_velocity : tuple of float + The velocity of the airflow relative to the surface. + stream_speed : float + The magnitude of the airflow speed. + stream_mach : float + The Mach number of the airflow. + rho : float + Air density. + cp : Vector + Center of pressure coordinates in the body frame. + omega: tuple[float, float, float] + Tuple containing angular velocities around the x, y, z axes. + reynolds : float + Reynolds number. + omega: tuple of float + Tuple containing angular velocities around the x, y, z axes. + + Returns + ------- + tuple of float + The aerodynamic forces (lift, side_force, drag) and moments + (pitch, yaw, roll) in the body frame. + """ + # Stream velocity in standard aerodynamic frame + stream_velocity = -stream_velocity + + # Angles of attack and sideslip + alpha = np.arctan2(stream_velocity[1], stream_velocity[2]) + beta = np.arctan2(-stream_velocity[0], stream_velocity[2]) + + # Compute aerodynamic forces and moments + lift, side, drag, pitch, yaw, roll = self._compute_from_coefficients( + rho, + stream_speed, + alpha, + beta, + stream_mach, + reynolds, + omega[0], + omega[1], + omega[2], + ) + + # Conversion from aerodynamic frame to body frame + rotation_matrix = Matrix( + [ + [1, 0, 0], + [0, math.cos(alpha), -math.sin(alpha)], + [0, math.sin(alpha), math.cos(alpha)], + ] + ) @ Matrix( + [ + [math.cos(beta), 0, -math.sin(beta)], + [0, 1, 0], + [math.sin(beta), 0, math.cos(beta)], + ] + ) + R1, R2, R3 = rotation_matrix @ Vector([side, -lift, -drag]) + + # Dislocation of the aerodynamic application point to CDM + M1, M2, M3 = Vector([pitch, yaw, roll]) + (cp ^ Vector([R1, R2, R3])) + + return R1, R2, R3, M1, M2, M3 + + def _process_input(self, input_data, coeff_name): + """Process the input data, either as a CSV file or a callable function. + + Parameters + ---------- + input_data : str or callable + Input data to be processed, either a path to a CSV or a callable. + coeff_name : str + Name of the coefficient being processed for error reporting. + + Returns + ------- + Function + Function object with 7 input arguments (alpha, beta, mach, reynolds, + pitch_rate, yaw_rate, roll_rate). + """ + if isinstance(input_data, str): + # Input is assumed to be a file path to a CSV + return self.__load_csv(input_data, coeff_name) + elif isinstance(input_data, Function): + if input_data.__dom_dim__ != 7: + raise ValueError( + f"{coeff_name} function must have 7 input arguments" + " (alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate)." + ) + return input_data + elif callable(input_data): + # Check if callable has 7 inputs (alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + if input_data.__code__.co_argcount != 7: + raise ValueError( + f"{coeff_name} function must have 7 input arguments" + " (alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate)." + ) + return input_data + elif input_data == 0: + return Function( + lambda alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate: 0, + [ + 'alpha', + 'beta', + 'mach', + 'reynolds', + 'pitch_rate', + 'yaw_rate', + 'roll_rate', + ], + [coeff_name], + interpolation='linear', + extrapolation='natural', + ) + else: + raise TypeError( + f"Invalid input for {coeff_name}: must be a CSV file path" + " or a callable." + ) + + def __load_csv(self, file_path, coeff_name): + """Load a CSV file and create a Function object with the correct number + of arguments. The CSV file must have a header that specifies the + independent variables that are used. + + Parameters + ---------- + file_path : str + Path to the CSV file. + coeff_name : str + Name of the coefficient being processed. + + Returns + ------- + Function + Function object with 7 input arguments (alpha, beta, mach, reynolds, + pitch_rate, yaw_rate, roll_rate). + """ + try: + with open(file_path, mode='r') as file: + reader = csv.reader(file) + header = next(reader) + except (FileNotFoundError, IOError) as e: + raise ValueError(f"Error reading {coeff_name} CSV file: {e}") from e + + if not header: + raise ValueError(f"Invalid or empty CSV file for {coeff_name}.") + + # TODO make header strings flexible (e.g. 'alpha', 'Alpha', 'ALPHA') + independent_vars = [ + 'alpha', + 'beta', + 'mach', + 'reynolds', + 'pitch_rate', + 'yaw_rate', + 'roll_rate', + ] + present_columns = [col for col in independent_vars if col in header] + + # Check that the last column is not an independent variable + if header[-1] in independent_vars: + raise ValueError( + f"Last column in {coeff_name} CSV must be the coefficient" + " value, not an independent variable." + ) + + # Ensure that at least one independent variable is present + if not present_columns: + raise ValueError(f"No independent variables found in {coeff_name} CSV.") + + # Initialize the CSV-based function + csv_func = Function( + file_path, + interpolation='linear', + extrapolation='natural', + ) + + # Create a mask for the presence of each independent variable + # save on self to avoid loss of scope + _mask = [1 if col in present_columns else 0 for col in independent_vars] + + # Generate a lambda that applies only the relevant arguments to csv_func + def wrapper(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate): + args = [alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate] + # Select arguments that correspond to present variables + selected_args = [arg for arg, m in zip(args, _mask) if m] + return csv_func(*selected_args) + + # Create the interpolation function + func = Function( + wrapper, + independent_vars, + [coeff_name], + interpolation='linear', + extrapolation='natural', + ) + return func diff --git a/rocketpy/rocket/aero_surface/linear_generic_surface.py b/rocketpy/rocket/aero_surface/linear_generic_surface.py new file mode 100644 index 000000000..fba2b1db8 --- /dev/null +++ b/rocketpy/rocket/aero_surface/linear_generic_surface.py @@ -0,0 +1,410 @@ +from rocketpy.mathutils import Function +from rocketpy.plots.aero_surface_plots import _LinearGenericSurfacePlots +from rocketpy.prints.aero_surface_prints import _LinearGenericSurfacePrints +from rocketpy.rocket.aero_surface.generic_surface import GenericSurface + + +class LinearGenericSurface(GenericSurface): + """Class that defines a generic linear aerodynamic surface. This class is + used to define aerodynamic surfaces that have aerodynamic coefficients + defined as linear functions of the coefficients derivatives.""" + + def __init__( + self, + reference_area, + reference_length, + coefficients, + center_of_pressure=(0, 0, 0), + name="Generic Linear Surface", + ): + """Create a generic linear aerodynamic surface, defined by its + aerodynamic coefficients derivatives. This surface is used to model any + aerodynamic surface that does not fit the predefined classes. + + Important + --------- + All the aerodynamic coefficients can be input as callable functions of + angle of attack, angle of sideslip, Mach number, Reynolds number, + pitch rate, yaw rate and roll rate. For CSV files, the header must + contain at least one of the following: "alpha", "beta", "mach", + "reynolds", "pitch_rate", "yaw_rate" and "roll_rate". + + See Also + -------- + :ref:`genericsurfaces`. + + Parameters + ---------- + reference_area : int, float + Reference area of the aerodynamic surface. Has the unit of meters + squared. Commonly defined as the rocket's cross-sectional area. + reference_length : int, float + Reference length of the aerodynamic surface. Has the unit of meters. + Commonly defined as the rocket's diameter. + coefficients: dict, optional + List of coefficients. If a coefficient is omitted, it is set to 0. + The valid coefficients are:\n + cL_0: callable, str, optional + Coefficient of lift at zero angle of attack. Default is 0.\n + cL_alpha: callable, str, optional + Coefficient of lift derivative with respect to angle of attack. + Default is 0.\n + cL_beta: callable, str, optional + Coefficient of lift derivative with respect to sideslip angle. + Default is 0.\n + cL_p: callable, str, optional + Coefficient of lift derivative with respect to roll rate. + Default is 0.\n + cL_q: callable, str, optional + Coefficient of lift derivative with respect to pitch rate. + Default is 0.\n + cL_r: callable, str, optional + Coefficient of lift derivative with respect to yaw rate. + Default is 0.\n + cQ_0: callable, str, optional + Coefficient of pitch moment at zero angle of attack. + Default is 0.\n + cQ_alpha: callable, str, optional + Coefficient of pitch moment derivative with respect to angle of + attack. Default is 0.\n + cQ_beta: callable, str, optional + Coefficient of pitch moment derivative with respect to sideslip + angle. Default is 0.\n + cQ_p: callable, str, optional + Coefficient of pitch moment derivative with respect to roll rate. + Default is 0.\n + cQ_q: callable, str, optional + Coefficient of pitch moment derivative with respect to pitch rate. + Default is 0.\n + cQ_r: callable, str, optional + Coefficient of pitch moment derivative with respect to yaw rate. + Default is 0.\n + cD_0: callable, str, optional + Coefficient of drag at zero angle of attack. Default is 0.\n + cD_alpha: callable, str, optional + Coefficient of drag derivative with respect to angle of attack. + Default is 0.\n + cD_beta: callable, str, optional + Coefficient of drag derivative with respect to sideslip angle. + Default is 0.\n + cD_p: callable, str, optional + Coefficient of drag derivative with respect to roll rate. + Default is 0.\n + cD_q: callable, str, optional + Coefficient of drag derivative with respect to pitch rate. + Default is 0.\n + cD_r: callable, str, optional + Coefficient of drag derivative with respect to yaw rate. + Default is 0.\n + cm_0: callable, str, optional + Coefficient of pitch moment at zero angle of attack. + Default is 0.\n + cm_alpha: callable, str, optional + Coefficient of pitch moment derivative with respect to angle of + attack. Default is 0.\n + cm_beta: callable, str, optional + Coefficient of pitch moment derivative with respect to sideslip + angle. Default is 0.\n + cm_p: callable, str, optional + Coefficient of pitch moment derivative with respect to roll rate. + Default is 0.\n + cm_q: callable, str, optional + Coefficient of pitch moment derivative with respect to pitch rate. + Default is 0.\n + cm_r: callable, str, optional + Coefficient of pitch moment derivative with respect to yaw rate. + Default is 0.\n + cn_0: callable, str, optional + Coefficient of yaw moment at zero angle of attack. + Default is 0.\n + cn_alpha: callable, str, optional + Coefficient of yaw moment derivative with respect to angle of + attack. Default is 0.\n + cn_beta: callable, str, optional + Coefficient of yaw moment derivative with respect to sideslip angle. + Default is 0.\n + cn_p: callable, str, optional + Coefficient of yaw moment derivative with respect to roll rate. + Default is 0.\n + cn_q: callable, str, optional + Coefficient of yaw moment derivative with respect to pitch rate. + Default is 0.\n + cn_r: callable, str, optional + Coefficient of yaw moment derivative with respect to yaw rate. + Default is 0.\n + cl_0: callable, str, optional + Coefficient of roll moment at zero angle of attack. + Default is 0.\n + cl_alpha: callable, str, optional + Coefficient of roll moment derivative with respect to angle of + attack. Default is 0.\n + cl_beta: callable, str, optional + Coefficient of roll moment derivative with respect to sideslip + angle. Default is 0.\n + cl_p: callable, str, optional + Coefficient of roll moment derivative with respect to roll rate. + Default is 0.\n + cl_q: callable, str, optional + Coefficient of roll moment derivative with respect to pitch rate. + Default is 0.\n + cl_r: callable, str, optional + Coefficient of roll moment derivative with respect to yaw rate. + Default is 0.\n + center_of_pressure : tuple, optional + Application point of the aerodynamic forces and moments. The + center of pressure is defined in the local coordinate system of the + aerodynamic surface. The default value is (0, 0, 0). + name : str + Name of the aerodynamic surface. Default is 'GenericSurface'. + """ + + super().__init__( + reference_area=reference_area, + reference_length=reference_length, + coefficients=coefficients, + center_of_pressure=center_of_pressure, + name=name, + ) + + self.compute_all_coefficients() + + self.prints = _LinearGenericSurfacePrints(self) + self.plots = _LinearGenericSurfacePlots(self) + + def _get_default_coefficients(self): + """Returns default coefficients + + Returns + ------- + default_coefficients: dict + Dictionary whose keys are the coefficients names and keys + are the default values. + """ + default_coefficients = { + "cL_0": 0, + "cL_alpha": 0, + "cL_beta": 0, + "cL_p": 0, + "cL_q": 0, + "cL_r": 0, + "cQ_0": 0, + "cQ_alpha": 0, + "cQ_beta": 0, + "cQ_p": 0, + "cQ_q": 0, + "cQ_r": 0, + "cD_0": 0, + "cD_alpha": 0, + "cD_beta": 0, + "cD_p": 0, + "cD_q": 0, + "cD_r": 0, + "cm_0": 0, + "cm_alpha": 0, + "cm_beta": 0, + "cm_p": 0, + "cm_q": 0, + "cm_r": 0, + "cn_0": 0, + "cn_alpha": 0, + "cn_beta": 0, + "cn_p": 0, + "cn_q": 0, + "cn_r": 0, + "cl_0": 0, + "cl_alpha": 0, + "cl_beta": 0, + "cl_p": 0, + "cl_q": 0, + "cl_r": 0, + } + return default_coefficients + + def compute_forcing_coefficient(self, c_0, c_alpha, c_beta): + """Compute the forcing coefficient from the derivatives of the + aerodynamic coefficients.""" + + def total_coefficient( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ): + return ( + c_0(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + + c_alpha(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + * alpha + + c_beta(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + * beta + ) + + return Function( + total_coefficient, + [ + 'alpha', + 'beta', + 'mach', + 'reynolds', + 'pitch_rate', + 'yaw_rate', + 'roll_rate', + ], + ['coefficient'], + ) + + def compute_damping_coefficient(self, c_p, c_q, c_r): + """Compute the damping coefficient from the derivatives of the + aerodynamic coefficients.""" + + def total_coefficient( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ): + return ( + c_p(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + * pitch_rate + + c_q(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + * yaw_rate + + c_r(alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate) + * roll_rate + ) + + return Function( + total_coefficient, + [ + 'alpha', + 'beta', + 'mach', + 'reynolds', + 'pitch_rate', + 'yaw_rate', + 'roll_rate', + ], + ['coefficient'], + ) + + def compute_all_coefficients(self): + """Compute all the aerodynamic coefficients from the derivatives.""" + # pylint: disable=invalid-name + self.cLf = self.compute_forcing_coefficient( + self.cL_0, self.cL_alpha, self.cL_beta + ) + self.cLd = self.compute_damping_coefficient(self.cL_p, self.cL_q, self.cL_r) + + self.cQf = self.compute_forcing_coefficient( + self.cQ_0, self.cQ_alpha, self.cQ_beta + ) + self.cQd = self.compute_damping_coefficient(self.cQ_p, self.cQ_q, self.cQ_r) + + self.cDf = self.compute_forcing_coefficient( + self.cD_0, self.cD_alpha, self.cD_beta + ) + self.cDd = self.compute_damping_coefficient(self.cD_p, self.cD_q, self.cD_r) + + self.cmf = self.compute_forcing_coefficient( + self.cm_0, self.cm_alpha, self.cm_beta + ) + self.cmd = self.compute_damping_coefficient(self.cm_p, self.cm_q, self.cm_r) + + self.cnf = self.compute_forcing_coefficient( + self.cn_0, self.cn_alpha, self.cn_beta + ) + self.cnd = self.compute_damping_coefficient(self.cn_p, self.cn_q, self.cn_r) + + self.clf = self.compute_forcing_coefficient( + self.cl_0, self.cl_alpha, self.cl_beta + ) + self.cld = self.compute_damping_coefficient(self.cl_p, self.cl_q, self.cl_r) + + def _compute_from_coefficients( + self, + rho, + stream_speed, + alpha, + beta, + mach, + reynolds, + pitch_rate, + yaw_rate, + roll_rate, + ): + """Compute the aerodynamic forces and moments from the aerodynamic + coefficients. + + Parameters + ---------- + rho : float + Air density. + stream_speed : float + Magnitude of the airflow speed. + alpha : float + Angle of attack in radians. + beta : float + Sideslip angle in radians. + mach : float + Mach number. + reynolds : float + Reynolds number. + pitch_rate : float + Pitch rate in radians per second. + yaw_rate : float + Yaw rate in radians per second. + roll_rate : float + Roll rate in radians per second. + + Returns + ------- + tuple of float + The aerodynamic forces (lift, side_force, drag) and moments + (pitch, yaw, roll) in the body frame. + """ + # Precompute common values + dyn_pressure_area = 0.5 * rho * stream_speed**2 * self.reference_area + dyn_pressure_area_damping = ( + 0.5 * rho * stream_speed * self.reference_area * self.reference_length / 2 + ) + dyn_pressure_area_length = dyn_pressure_area * self.reference_length + dyn_pressure_area_length_damping = ( + 0.5 + * rho + * stream_speed + * self.reference_area + * self.reference_length**2 + / 2 + ) + + # Compute aerodynamic forces + lift = dyn_pressure_area * self.cLf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_damping * self.cLd( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + side = dyn_pressure_area * self.cQf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_damping * self.cQd( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + drag = dyn_pressure_area * self.cDf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_damping * self.cDd( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + # Compute aerodynamic moments + pitch = dyn_pressure_area_length * self.cmf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_length_damping * self.cmd( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + yaw = dyn_pressure_area_length * self.cnf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_length_damping * self.cnd( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + roll = dyn_pressure_area_length * self.clf( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) - dyn_pressure_area_length_damping * self.cld( + alpha, beta, mach, reynolds, pitch_rate, yaw_rate, roll_rate + ) + + return lift, side, drag, pitch, yaw, roll diff --git a/rocketpy/rocket/aero_surface/nose_cone.py b/rocketpy/rocket/aero_surface/nose_cone.py index 8d98113ae..7d59473e3 100644 --- a/rocketpy/rocket/aero_surface/nose_cone.py +++ b/rocketpy/rocket/aero_surface/nose_cone.py @@ -446,10 +446,6 @@ def evaluate_lift_coefficient(self): None """ # Calculate clalpha - # clalpha is currently a constant, meaning it is independent of Mach - # number. This is only valid for subsonic speeds. - # It must be set as a Function because it will be called and treated - # as a function of mach in the simulation. self.clalpha = Function( lambda mach: 2 * self.radius_ratio**2, "Mach", diff --git a/rocketpy/rocket/aero_surface/tail.py b/rocketpy/rocket/aero_surface/tail.py index 5785232fd..4f2783e02 100644 --- a/rocketpy/rocket/aero_surface/tail.py +++ b/rocketpy/rocket/aero_surface/tail.py @@ -163,10 +163,6 @@ def evaluate_lift_coefficient(self): None """ # Calculate clalpha - # clalpha is currently a constant, meaning it is independent of Mach - # number. This is only valid for subsonic speeds. - # It must be set as a Function because it will be called and treated - # as a function of mach in the simulation. self.clalpha = Function( lambda mach: 2 * ( diff --git a/rocketpy/rocket/components.py b/rocketpy/rocket/components.py index 2d580de7e..91bd05fa0 100644 --- a/rocketpy/rocket/components.py +++ b/rocketpy/rocket/components.py @@ -180,7 +180,7 @@ def clear(self): self._components.clear() def sort_by_position(self, reverse=False): - """Sort the list of components by position. + """Sort the list of components by z axis position. Parameters ---------- @@ -192,4 +192,4 @@ def sort_by_position(self, reverse=False): ------- None """ - self._components.sort(key=lambda x: x.position, reverse=reverse) + self._components.sort(key=lambda x: x.position.z, reverse=reverse) diff --git a/rocketpy/rocket/rocket.py b/rocketpy/rocket/rocket.py index a5964808f..fe0c803b5 100644 --- a/rocketpy/rocket/rocket.py +++ b/rocketpy/rocket/rocket.py @@ -1,3 +1,4 @@ +import math import warnings import numpy as np @@ -17,6 +18,8 @@ Tail, TrapezoidalFins, ) +from rocketpy.rocket.aero_surface.fins.free_form_fins import FreeFormFins +from rocketpy.rocket.aero_surface.generic_surface import GenericSurface from rocketpy.rocket.components import Components from rocketpy.rocket.parachute import Parachute from rocketpy.tools import parallel_axis_theorem_from_com @@ -107,6 +110,11 @@ class Rocket: Rocket.aerodynamic_surfaces : list Collection of aerodynamic surfaces of the rocket. Holds Nose cones, Fin sets, and Tails. + Rocket.surfaces_cp_to_cdm : dict + Dictionary containing the relative position of each aerodynamic surface + center of pressure to the rocket's center of mass. The key is the + aerodynamic surface object and the value is the relative position Vector + in meters. Rocket.parachutes : list Collection of parachutes of the rocket. Rocket.air_brakes : list @@ -290,6 +298,8 @@ def __init__( # pylint: disable=too-many-statements self.area = np.pi * self.radius**2 # Eccentricity data initialization + self.cm_eccentricity_x = 0 + self.cm_eccentricity_y = 0 self.cp_eccentricity_x = 0 self.cp_eccentricity_y = 0 self.thrust_eccentricity_y = 0 @@ -301,6 +311,7 @@ def __init__( # pylint: disable=too-many-statements self.air_brakes = [] self.sensors = Components() self.aerodynamic_surfaces = Components() + self.surfaces_cp_to_cdm = {} self.rail_buttons = Components() self.cp_position = Function( @@ -525,18 +536,50 @@ def evaluate_center_of_pressure(self): # Calculate total lift coefficient derivative and center of pressure if len(self.aerodynamic_surfaces) > 0: for aero_surface, position in self.aerodynamic_surfaces: + if isinstance(aero_surface, GenericSurface): + continue # ref_factor corrects lift for different reference areas ref_factor = (aero_surface.rocket_radius / self.radius) ** 2 self.total_lift_coeff_der += ref_factor * aero_surface.clalpha self.cp_position += ( ref_factor * aero_surface.clalpha - * (position - self._csys * aero_surface.cpz) + * (position.z - self._csys * aero_surface.cpz) ) - self.cp_position /= self.total_lift_coeff_der - + # Avoid errors when only generic surfaces are added + if self.total_lift_coeff_der.get_value(0) != 0: + self.cp_position /= self.total_lift_coeff_der return self.cp_position + def evaluate_surfaces_cp_to_cdm(self): + """Calculates the relative position of each aerodynamic surface center + of pressure to the rocket's center of dry mass in Body Axes Coordinate + System. + + Returns + ------- + self.surfaces_cp_to_cdm : dict + Dictionary mapping the relative position of each aerodynamic + surface center of pressure to the rocket's center of mass. + """ + for surface, position in self.aerodynamic_surfaces: + self.__evaluate_single_surface_cp_to_cdm(surface, position) + return self.surfaces_cp_to_cdm + + def __evaluate_single_surface_cp_to_cdm(self, surface, position): + """Calculates the relative position of each aerodynamic surface + center of pressure to the rocket's center of dry mass in Body Axes + Coordinate System.""" + pos = Vector( + [ + (position.x - self.cm_eccentricity_x) * self._csys - surface.cpx, + (position.y - self.cm_eccentricity_y) - surface.cpy, + (position.z - self.center_of_dry_mass_position) * self._csys + - surface.cpz, + ] + ) + self.surfaces_cp_to_cdm[surface] = pos + def evaluate_stability_margin(self): """Calculates the stability margin of the rocket as a function of mach number and time. @@ -917,6 +960,7 @@ def add_motor(self, motor, position): # pylint: disable=too-many-statements self.evaluate_reduced_mass() self.evaluate_thrust_to_weight() self.evaluate_center_of_pressure() + self.evaluate_surfaces_cp_to_cdm() self.evaluate_stability_margin() self.evaluate_static_margin() self.evaluate_com_to_cdm_function() @@ -932,7 +976,7 @@ def add_surfaces(self, surfaces, positions): surfaces : list, AeroSurface, NoseCone, TrapezoidalFins, EllipticalFins, Tail Aerodynamic surface to be added to the rocket. Can be a list of AeroSurface if more than one surface is to be added. - positions : int, float, list + positions : int, float, list, tuple, Vector Position, in m, of the aerodynamic surface's center of pressure relative to the user defined rocket coordinate system. If a list is passed, it will correspond to the position of each item @@ -952,16 +996,22 @@ def add_surfaces(self, surfaces, positions): ------- None """ - if not isinstance(surfaces, list): - surfaces = [surfaces] - positions = [positions] - - for surface, position in zip(surfaces, positions): - if isinstance(surface, RailButtons): - surface.rocket_radius = surface.rocket_radius or self.radius - self.rail_buttons.add(surface, position) - else: + # TODO: separate this method into smaller methods: https://github.com/RocketPy-Team/RocketPy/pull/696#discussion_r1771978422 + try: + for surface, position in zip(surfaces, positions): + if not isinstance(position, (Vector, tuple, list)): + position = Vector([0, 0, position]) + else: + position = Vector(position) self.aerodynamic_surfaces.add(surface, position) + self.__evaluate_single_surface_cp_to_cdm(surface, position) + except TypeError: + if not isinstance(positions, (Vector, tuple, list)): + positions = Vector([0, 0, positions]) + else: + positions = Vector(positions) + self.aerodynamic_surfaces.add(surfaces, positions) + self.__evaluate_single_surface_cp_to_cdm(surfaces, positions) self.evaluate_center_of_pressure() self.evaluate_stability_margin() @@ -1126,7 +1176,7 @@ def add_trapezoidal_fins( Parameters ---------- n : int - Number of fins, from 2 to infinity. + Number of fins, must be greater than 2. span : int, float Fin span in meters. root_chord : int, float @@ -1224,7 +1274,7 @@ def add_elliptical_fins( Parameters ---------- n : int - Number of fins, from 2 to infinity. + Number of fins, must be greater than 2. root_chord : int, float Fin root chord in meters. span : int, float @@ -1274,6 +1324,84 @@ def add_elliptical_fins( self.add_surfaces(fin_set, position) return fin_set + def add_free_form_fins( + self, + n, + shape_points, + position, + cant_angle=0.0, + radius=None, + airfoil=None, + name="Fins", + ): + """Create a free form fin set, storing its parameters as part of the + aerodynamic_surfaces list. Its parameters are the axial position along + the rocket and its derivative of the coefficient of lift in respect to + angle of attack. + + Parameters + ---------- + n : int + Number of fins, must be greater than 2. + shape_points : list + List of tuples (x, y) containing the coordinates of the fin's + geometry defining points. The point (0, 0) is the root leading edge. + Positive x is rearwards, positive y is upwards (span direction). + The shape will be interpolated between the points, in the order + they are given. The last point connects to the first point. + position : int, float + Fin set position relative to the rocket's coordinate system. + By fin set position, understand the point belonging to the root + chord which is highest in the rocket coordinate system (i.e. + the point closest to the nose cone tip). + + See Also + -------- + :ref:`positions` + cant_angle : int, float, optional + Fins cant angle with respect to the rocket centerline. Must + be given in degrees. + radius : int, float, optional + Reference fuselage radius where the fins are located. This is used + to calculate lift coefficient and to draw the rocket. If None, + which is default, the rocket radius will be used. + airfoil : tuple, optional + Default is null, in which case fins will be treated as flat plates. + Otherwise, if tuple, fins will be considered as airfoils. The + tuple's first item specifies the airfoil's lift coefficient + by angle of attack and must be either a .csv, .txt, ndarray + or callable. The .csv and .txt files can contain a single line + header and the first column must specify the angle of attack, while + the second column must specify the lift coefficient. The + ndarray should be as [(x0, y0), (x1, y1), (x2, y2), ...] + where x0 is the angle of attack and y0 is the lift coefficient. + If callable, it should take an angle of attack as input and + return the lift coefficient at that angle of attack. + The tuple's second item is the unit of the angle of attack, + accepting either "radians" or "degrees". + + Returns + ------- + fin_set : FreeFormFins + Fin set object created. + """ + + # Modify radius if not given, use rocket radius, otherwise use given. + radius = radius if radius is not None else self.radius + + fin_set = FreeFormFins( + n, + shape_points, + radius, + cant_angle, + airfoil, + name, + ) + + # Add fin set to the list of aerodynamic surfaces + self.add_surfaces(fin_set, position) + return fin_set + def add_parachute( self, name, cd_s, trigger, sampling_rate=100, lag=0, noise=(0, 0, 0) ): @@ -1564,13 +1692,22 @@ def set_rail_buttons( rail_buttons : RailButtons RailButtons object created """ + radius = radius or self.radius buttons_distance = abs(upper_button_position - lower_button_position) rail_buttons = RailButtons( - buttons_distance=buttons_distance, angular_position=angular_position + buttons_distance=buttons_distance, + angular_position=angular_position, + rocket_radius=radius, ) self.rail_buttons = Components() - rail_buttons.rocket_radius = radius or self.radius - self.rail_buttons.add(rail_buttons, lower_button_position) + position = Vector( + [ + radius * -math.sin(math.radians(angular_position)), + radius * math.cos(math.radians(angular_position)), + lower_button_position, + ] + ) + self.rail_buttons.add(rail_buttons, position) return rail_buttons def add_cm_eccentricity(self, x, y): @@ -1583,25 +1720,31 @@ def add_cm_eccentricity(self, x, y): ---------- x : float Distance in meters by which the CM is to be translated in - the x direction relative to geometrical center line. + the x direction relative to geometrical center line. The x axis + is defined according to the body axes coordinate system. y : float Distance in meters by which the CM is to be translated in - the y direction relative to geometrical center line. + the y direction relative to geometrical center line. The y axis + is defined according to the body axes coordinate system. Returns ------- self : Rocket Object of the Rocket class. + See Also + -------- + :ref:`rocketaxes` + Notes ----- Should not be used together with add_cp_eccentricity and add_thrust_eccentricity. """ - self.cp_eccentricity_x = -x - self.cp_eccentricity_y = -y - self.thrust_eccentricity_y = -x - self.thrust_eccentricity_x = -y + self.cm_eccentricity_x = x + self.cm_eccentricity_y = y + self.add_cp_eccentricity(-x, -y) + self.add_thrust_eccentricity(-x, -y) return self def add_cp_eccentricity(self, x, y): @@ -1614,14 +1757,22 @@ def add_cp_eccentricity(self, x, y): x : float Distance in meters by which the CP is to be translated in the x direction relative to the center of mass axial line. + The x axis is defined according to the body axes coordinate + system. y : float Distance in meters by which the CP is to be translated in the y direction relative to the center of mass axial line. + The y axis is defined according to the body axes coordinate + system. Returns ------- self : Rocket Object of the Rocket class. + + See Also + -------- + :ref:`rocketaxes` """ self.cp_eccentricity_x = x self.cp_eccentricity_y = y @@ -1636,16 +1787,22 @@ def add_thrust_eccentricity(self, x, y): x : float Distance in meters by which the line of action of the thrust force is to be translated in the x direction - relative to the center of mass axial line. + relative to the center of mass axial line. The x axis + is defined according to the body axes coordinate system. y : float Distance in meters by which the line of action of the thrust force is to be translated in the x direction - relative to the center of mass axial line. + relative to the center of mass axial line. The y axis + is defined according to the body axes coordinate system. Returns ------- self : Rocket Object of the Rocket class. + + See Also + -------- + :ref:`rocketaxes` """ self.thrust_eccentricity_y = x self.thrust_eccentricity_x = y diff --git a/rocketpy/simulation/flight.py b/rocketpy/simulation/flight.py index e7b53754a..706bcf534 100644 --- a/rocketpy/simulation/flight.py +++ b/rocketpy/simulation/flight.py @@ -736,10 +736,11 @@ def __simulate(self, verbose): for parachute in node.parachutes: # Calculate and save pressure signal - noisy_pressure, height_above_ground_level = ( - self.__calculate_and_save_pressure_signals( - parachute, node.t, self.y_sol[2] - ) + ( + noisy_pressure, + height_above_ground_level, + ) = self.__calculate_and_save_pressure_signals( + parachute, node.t, self.y_sol[2] ) if parachute.triggerfunc( noisy_pressure, @@ -975,12 +976,13 @@ def __simulate(self, verbose): ) for parachute in overshootable_node.parachutes: # Calculate and save pressure signal - noisy_pressure, height_above_ground_level = ( - self.__calculate_and_save_pressure_signals( - parachute, - overshootable_node.t, - overshootable_node.y_sol[2], - ) + ( + noisy_pressure, + height_above_ground_level, + ) = self.__calculate_and_save_pressure_signals( + parachute, + overshootable_node.t, + overshootable_node.y_sol[2], ) # Check for parachute trigger @@ -1230,7 +1232,7 @@ def effective_1rl(self): rail_buttons = self.rocket.rail_buttons[0] upper_r_button = ( rail_buttons.component.buttons_distance * self.rocket._csys - + rail_buttons.position + + rail_buttons.position.z ) except IndexError: # No rail buttons defined upper_r_button = nozzle @@ -1245,7 +1247,7 @@ def effective_2rl(self): nozzle = self.rocket.nozzle_position try: rail_buttons = self.rocket.rail_buttons[0] - lower_r_button = rail_buttons.position + lower_r_button = rail_buttons.position.z except IndexError: # No rail buttons defined lower_r_button = nozzle effective_2rl = self.rail_length - abs(nozzle - lower_r_button) @@ -1464,16 +1466,18 @@ def u_dot( a32 = 2 * (e2 * e3 + e0 * e1) a33 = 1 - 2 * (e1**2 + e2**2) # Transformation matrix: (123) -> (XYZ) - K = [[a11, a12, a13], [a21, a22, a23], [a31, a32, a33]] + K = Matrix([[a11, a12, a13], [a21, a22, a23], [a31, a32, a33]]) + Kt = K.transpose # Calculate Forces and Moments # Get freestream speed wind_velocity_x = self.env.wind_velocity_x.get_value_opt(z) wind_velocity_y = self.env.wind_velocity_y.get_value_opt(z) + speed_of_sound = self.env.speed_of_sound.get_value_opt(z) free_stream_speed = ( (wind_velocity_x - vx) ** 2 + (wind_velocity_y - vy) ** 2 + (vz) ** 2 ) ** 0.5 - free_stream_mach = free_stream_speed / self.env.speed_of_sound.get_value_opt(z) + free_stream_mach = free_stream_speed / speed_of_sound # Determine aerodynamics forces # Determine Drag Force @@ -1507,76 +1511,47 @@ def u_dot( vy_b = a12 * vx + a22 * vy + a32 * vz vz_b = a13 * vx + a23 * vy + a33 * vz # Calculate lift and moment for each component of the rocket - for aero_surface, position in self.rocket.aerodynamic_surfaces: - comp_cp = ( - position - self.rocket.center_of_dry_mass_position - ) * self.rocket._csys - aero_surface.cpz - reference_area = aero_surface.reference_area - reference_length = aero_surface.reference_length + velocity_in_body_frame = Vector([vx_b, vy_b, vz_b]) + w = Vector([omega1, omega2, omega3]) + for aero_surface, _ in self.rocket.aerodynamic_surfaces: + # Component cp relative to CDM in body frame + comp_cp = self.rocket.surfaces_cp_to_cdm[aero_surface] # Component absolute velocity in body frame - comp_vx_b = vx_b + comp_cp * omega2 - comp_vy_b = vy_b - comp_cp * omega1 - comp_vz_b = vz_b - # Wind velocity at component - comp_z = z + comp_cp + comp_vb = velocity_in_body_frame + (w ^ comp_cp) + # Wind velocity at component altitude + comp_z = z + (K @ comp_cp).z comp_wind_vx = self.env.wind_velocity_x.get_value_opt(comp_z) comp_wind_vy = self.env.wind_velocity_y.get_value_opt(comp_z) # Component freestream velocity in body frame - comp_wind_vx_b = a11 * comp_wind_vx + a21 * comp_wind_vy - comp_wind_vy_b = a12 * comp_wind_vx + a22 * comp_wind_vy - comp_wind_vz_b = a13 * comp_wind_vx + a23 * comp_wind_vy - comp_stream_vx_b = comp_wind_vx_b - comp_vx_b - comp_stream_vy_b = comp_wind_vy_b - comp_vy_b - comp_stream_vz_b = comp_wind_vz_b - comp_vz_b - comp_stream_speed = ( - comp_stream_vx_b**2 + comp_stream_vy_b**2 + comp_stream_vz_b**2 - ) ** 0.5 - # Component attack angle and lift force - comp_attack_angle = 0 - comp_lift, comp_lift_xb, comp_lift_yb = 0, 0, 0 - if comp_stream_vx_b**2 + comp_stream_vy_b**2 != 0: - # normalize component stream velocity in body frame - comp_stream_vz_bn = comp_stream_vz_b / comp_stream_speed - if -1 * comp_stream_vz_bn < 1: - comp_attack_angle = np.arccos(-comp_stream_vz_bn) - c_lift = aero_surface.cl.get_value_opt( - comp_attack_angle, free_stream_mach - ) - # component lift force magnitude - comp_lift = ( - 0.5 * rho * (comp_stream_speed**2) * reference_area * c_lift - ) - # component lift force components - lift_dir_norm = (comp_stream_vx_b**2 + comp_stream_vy_b**2) ** 0.5 - comp_lift_xb = comp_lift * (comp_stream_vx_b / lift_dir_norm) - comp_lift_yb = comp_lift * (comp_stream_vy_b / lift_dir_norm) - # add to total lift force - R1 += comp_lift_xb - R2 += comp_lift_yb - # Add to total moment - M1 -= (comp_cp) * comp_lift_yb - M2 += (comp_cp) * comp_lift_xb - # Calculates Roll Moment - try: - clf_delta, cld_omega, cant_angle_rad = aero_surface.roll_parameters - M3_forcing = ( - (1 / 2 * rho * free_stream_speed**2) - * reference_area - * reference_length - * clf_delta.get_value_opt(free_stream_mach) - * cant_angle_rad - ) - M3_damping = ( - (1 / 2 * rho * free_stream_speed) - * reference_area - * (reference_length) ** 2 - * cld_omega.get_value_opt(free_stream_mach) - * omega3 - / 2 - ) - M3 += M3_forcing - M3_damping - except AttributeError: - pass + comp_wind_vb = Kt @ Vector([comp_wind_vx, comp_wind_vy, 0]) + comp_stream_velocity = comp_wind_vb - comp_vb + comp_stream_speed = abs(comp_stream_velocity) + comp_stream_mach = comp_stream_speed / speed_of_sound + # Reynolds at component altitude + # TODO: Reynolds is only used in generic surfaces. This calculation + # should be moved to the surface class for efficiency + comp_reynolds = ( + self.env.density.get_value_opt(comp_z) + * comp_stream_speed + * aero_surface.reference_length + / self.env.dynamic_viscosity.get_value_opt(comp_z) + ) + # Forces and moments + X, Y, Z, M, N, L = aero_surface.compute_forces_and_moments( + comp_stream_velocity, + comp_stream_speed, + comp_stream_mach, + rho, + comp_cp, + w, + comp_reynolds, + ) + R1 += X + R2 += Y + R3 += Z + M1 += M + M2 += N + M3 += L # Off center moment M3 += self.rocket.cp_eccentricity_x * R2 - self.rocket.cp_eccentricity_y * R1 @@ -1663,7 +1638,7 @@ def u_dot( (R3 - b * propellant_mass_at_t * (alpha2 - omega1 * omega3) + thrust) / total_mass_at_t, ] - ax, ay, az = np.dot(K, L) + ax, ay, az = K @ Vector(L) az -= self.env.gravity.get_value_opt(z) # Include gravity # Create u_dot @@ -1788,13 +1763,9 @@ def u_dot_generalized( # Get rocket velocity in body frame velocity_in_body_frame = Kt @ v # Calculate lift and moment for each component of the rocket - for aero_surface, position in self.rocket.aerodynamic_surfaces: - comp_cpz = ( - position - self.rocket.center_of_dry_mass_position - ) * self.rocket._csys - aero_surface.cpz - comp_cp = Vector([0, 0, comp_cpz]) - reference_area = aero_surface.reference_area - reference_length = aero_surface.reference_length + for aero_surface, _ in self.rocket.aerodynamic_surfaces: + # Component cp relative to CDM in body frame + comp_cp = self.rocket.surfaces_cp_to_cdm[aero_surface] # Component absolute velocity in body frame comp_vb = velocity_in_body_frame + (w ^ comp_cp) # Wind velocity at component altitude @@ -1804,55 +1775,33 @@ def u_dot_generalized( # Component freestream velocity in body frame comp_wind_vb = Kt @ Vector([comp_wind_vx, comp_wind_vy, 0]) comp_stream_velocity = comp_wind_vb - comp_vb - comp_stream_vx_b, comp_stream_vy_b, comp_stream_vz_b = comp_stream_velocity comp_stream_speed = abs(comp_stream_velocity) comp_stream_mach = comp_stream_speed / speed_of_sound - # Component attack angle and lift force - comp_attack_angle = 0 - comp_lift, comp_lift_xb, comp_lift_yb = 0, 0, 0 - if comp_stream_vx_b**2 + comp_stream_vy_b**2 != 0: # TODO: maybe try/except - # Normalize component stream velocity in body frame - comp_stream_vz_bn = comp_stream_vz_b / comp_stream_speed - if -1 * comp_stream_vz_bn < 1: - comp_attack_angle = np.arccos(-comp_stream_vz_bn) - c_lift = aero_surface.cl.get_value_opt( - comp_attack_angle, comp_stream_mach - ) - # Component lift force magnitude - comp_lift = ( - 0.5 * rho * (comp_stream_speed**2) * reference_area * c_lift - ) - # Component lift force components - lift_dir_norm = (comp_stream_vx_b**2 + comp_stream_vy_b**2) ** 0.5 - comp_lift_xb = comp_lift * (comp_stream_vx_b / lift_dir_norm) - comp_lift_yb = comp_lift * (comp_stream_vy_b / lift_dir_norm) - # Add to total lift force - R1 += comp_lift_xb - R2 += comp_lift_yb - # Add to total moment - M1 -= (comp_cpz) * comp_lift_yb - M2 += (comp_cpz) * comp_lift_xb - # Calculates Roll Moment - try: - clf_delta, cld_omega, cant_angle_rad = aero_surface.roll_parameters - M3_forcing = ( - (1 / 2 * rho * comp_stream_speed**2) - * reference_area - * reference_length - * clf_delta.get_value_opt(comp_stream_mach) - * cant_angle_rad - ) - M3_damping = ( - (1 / 2 * rho * comp_stream_speed) - * reference_area - * (reference_length) ** 2 - * cld_omega.get_value_opt(comp_stream_mach) - * omega3 - / 2 - ) - M3 += M3_forcing - M3_damping - except AttributeError: - pass + # Reynolds at component altitude + # TODO: Reynolds is only used in generic surfaces. This calculation + # should be moved to the surface class for efficiency + comp_reynolds = ( + self.env.density.get_value_opt(comp_z) + * comp_stream_speed + * aero_surface.reference_length + / self.env.dynamic_viscosity.get_value_opt(comp_z) + ) + # Forces and moments + X, Y, Z, M, N, L = aero_surface.compute_forces_and_moments( + comp_stream_velocity, + comp_stream_speed, + comp_stream_mach, + rho, + comp_cp, + w, + comp_reynolds, + ) + R1 += X + R2 += Y + R3 += Z + M1 += M + M2 += N + M3 += L # Off center moment thrust = self.rocket.motor.thrust.get_value_opt(t) @@ -2058,37 +2007,44 @@ def get_solution_at_time(self, t, atol=1e-3): # Transform solution array into Functions @funcify_method("Time (s)", "X (m)", "spline", "constant") def x(self): - """Rocket x position as a Function of time.""" + """Rocket x position relative to the launch pad as a Function of + time.""" return self.solution_array[:, [0, 1]] @funcify_method("Time (s)", "Y (m)", "spline", "constant") def y(self): - """Rocket y position as a Function of time.""" + """Rocket y position relative to the lauch pad as a Function of + time.""" return self.solution_array[:, [0, 2]] @funcify_method("Time (s)", "Z (m)", "spline", "constant") def z(self): - """Rocket z position as a Function of time.""" + """Rocket z position relative to the launch pad as a Function of + time.""" return self.solution_array[:, [0, 3]] @funcify_method("Time (s)", "Altitude AGL (m)", "spline", "constant") def altitude(self): - """Rocket altitude above ground level as a Function of time.""" + """Rocket altitude above ground level as a Function of time. Ground + level is defined by the environment elevation.""" return self.z - self.env.elevation @funcify_method("Time (s)", "Vx (m/s)", "spline", "zero") def vx(self): - """Rocket x velocity as a Function of time.""" + """Velocity of the rocket center of dry mass in the direction of + the rocket x-axis as a Function of time.""" return self.solution_array[:, [0, 4]] @funcify_method("Time (s)", "Vy (m/s)", "spline", "zero") def vy(self): - """Rocket y velocity as a Function of time.""" + """Velocity of the rocket center of dry mass in the direction of + the rocket y-axis as a Function of time.""" return self.solution_array[:, [0, 5]] @funcify_method("Time (s)", "Vz (m/s)", "spline", "zero") def vz(self): - """Rocket z velocity as a Function of time.""" + """Velocity of the rocket center of dry mass in the direction of + the rocket z-axis as a Function of time.""" return self.solution_array[:, [0, 6]] @funcify_method("Time (s)", "e0", "spline", "constant") @@ -2113,88 +2069,94 @@ def e3(self): @funcify_method("Time (s)", "ω1 (rad/s)", "spline", "zero") def w1(self): - """Rocket angular velocity ω1 as a Function of time.""" + """Angular velocity of the rocket in the x-axis as a Function of time. + Sometimes referred to as pitch rate (q).""" return self.solution_array[:, [0, 11]] @funcify_method("Time (s)", "ω2 (rad/s)", "spline", "zero") def w2(self): - """Rocket angular velocity ω2 as a Function of time.""" + """Angular velocity of the rocket in the y-axis as a Function of time. + Sometimes referred to as yaw rate (r).""" return self.solution_array[:, [0, 12]] @funcify_method("Time (s)", "ω3 (rad/s)", "spline", "zero") def w3(self): - """Rocket angular velocity ω3 as a Function of time.""" + """Angular velocity of the rocket in the z-axis as a Function of time. + Sometimes referred to as roll rate (p).""" return self.solution_array[:, [0, 13]] # Process second type of outputs - accelerations components @funcify_method("Time (s)", "Ax (m/s²)", "spline", "zero") def ax(self): - """Rocket x acceleration as a Function of time.""" + """Acceleration of the rocket center of dry mass in the direction of + the rocket x-axis as a Function of time.""" return self.__evaluate_post_process[:, [0, 1]] @funcify_method("Time (s)", "Ay (m/s²)", "spline", "zero") def ay(self): - """Rocket y acceleration as a Function of time.""" + """Acceleration of the rocket center of dry mass in the direction of + the rocket y-axis as a Function of time.""" return self.__evaluate_post_process[:, [0, 2]] @funcify_method("Time (s)", "Az (m/s²)", "spline", "zero") def az(self): - """Rocket z acceleration as a Function of time.""" + """Acceleration of the rocket center of dry mass in the direction of + the rocket z-axis as a Function of time.""" return self.__evaluate_post_process[:, [0, 3]] @funcify_method("Time (s)", "α1 (rad/s²)", "spline", "zero") def alpha1(self): - """Rocket angular acceleration α1 as a Function of time.""" + """Angular acceleration of the rocket in the x-axis as a Function of + time. Sometimes referred to as pitch acceleration.""" return self.__evaluate_post_process[:, [0, 4]] @funcify_method("Time (s)", "α2 (rad/s²)", "spline", "zero") def alpha2(self): - """Rocket angular acceleration α2 as a Function of time.""" + """Angular acceleration of the rocket in the y-axis as a Function of + time. Sometimes referred to as yaw acceleration.""" return self.__evaluate_post_process[:, [0, 5]] @funcify_method("Time (s)", "α3 (rad/s²)", "spline", "zero") def alpha3(self): - """Rocket angular acceleration α3 as a Function of time.""" + """Angular acceleration of the rocket in the z-axis as a Function of + time. Sometimes referred to as roll acceleration.""" return self.__evaluate_post_process[:, [0, 6]] # Process third type of outputs - Temporary values @funcify_method("Time (s)", "R1 (N)", "spline", "zero") def R1(self): - """Aerodynamic force along the first axis that is perpendicular to the - rocket's axis of symmetry as a Function of time.""" + """Aerodynamic force along the x-axis of the rocket as a Function of + time.""" return self.__evaluate_post_process[:, [0, 7]] @funcify_method("Time (s)", "R2 (N)", "spline", "zero") def R2(self): - """Aerodynamic force along the second axis that is perpendicular to the - rocket's axis of symmetry as a Function of time.""" + """Aerodynamic force along the y-axis of the rocket as a Function of + time.""" return self.__evaluate_post_process[:, [0, 8]] @funcify_method("Time (s)", "R3 (N)", "spline", "zero") def R3(self): - """Aerodynamic force along the rocket's axis of symmetry as a + """Aerodynamic force along the z-axis (rocket's axis of symmetry) as a Function of time.""" return self.__evaluate_post_process[:, [0, 9]] - @funcify_method("Time (s)", "M1 (Nm)", "spline", "zero") + @funcify_method("Time (s)", "M1 (Nm)", "linear", "zero") def M1(self): - """Aerodynamic bending moment in the same direction as the axis that is - perpendicular to the rocket's axis of symmetry as a Function of - time. - """ + """Aerodynamic moment in the rocket x-axis as a Function of time. + Sometimes referred to as pitch moment.""" return self.__evaluate_post_process[:, [0, 10]] - @funcify_method("Time (s)", "M2 (Nm)", "spline", "zero") + @funcify_method("Time (s)", "M2 (Nm)", "linear", "zero") def M2(self): - """Aerodynamic bending moment in the same direction as the axis that is - perpendicular to the rocket's axis of symmetry as a Function - of time.""" + """Aerodynamic moment in the rocket y-axis as a Function of time. + Sometimes referred to as yaw moment.""" return self.__evaluate_post_process[:, [0, 11]] - @funcify_method("Time (s)", "M3 (Nm)", "spline", "zero") + @funcify_method("Time (s)", "M3 (Nm)", "linear", "zero") def M3(self): - """Aerodynamic bending moment in the same direction as the rocket's - axis of symmetry as a Function of time.""" + """Aerodynamic moment in the rocket z-axis as a Function of time. + Sometimes referred to as roll moment.""" return self.__evaluate_post_process[:, [0, 12]] @funcify_method("Time (s)", "Pressure (Pa)", "spline", "constant") @@ -2225,7 +2187,7 @@ def wind_velocity_x(self): @funcify_method("Time (s)", "Wind Velocity Y (North) (m/s)", "spline", "constant") def wind_velocity_y(self): - """Wind velocity in the y direction (north) as a Function of time.""" + """Wind velocity in the Y direction (north) as a Function of time.""" return [(t, self.env.wind_velocity_y.get_value_opt(z)) for t, z in self.z] # Process fourth type of output - values calculated from previous outputs @@ -2325,17 +2287,26 @@ def path_angle(self): # Attitude Angle @funcify_method("Time (s)", "Attitude Vector X Component") def attitude_vector_x(self): - """Rocket attitude vector X component as a Function of time.""" + """Rocket attitude vector X component as a Function of time. + Same as row 1, column 3 of the rotation matrix that defines + the conversion from the body frame to the inertial frame + at each time step.""" return 2 * (self.e1 * self.e3 + self.e0 * self.e2) # a13 @funcify_method("Time (s)", "Attitude Vector Y Component") def attitude_vector_y(self): - """Rocket attitude vector Y component as a Function of time.""" + """Rocket attitude vector Y component as a Function of time. + Same as row 2, column 3 of the rotation matrix that defines + the conversion from the body frame to the inertial frame + at each time step.""" return 2 * (self.e2 * self.e3 - self.e0 * self.e1) # a23 @funcify_method("Time (s)", "Attitude Vector Z Component") def attitude_vector_z(self): - """Rocket attitude vector Z component as a Function of time.""" + """Rocket attitude vector Z component as a Function of time. + Same as row 3, column 3 of the rotation matrix that defines + the conversion from the body frame to the inertial frame + at each time step.""" return 1 - 2 * (self.e1**2 + self.e2**2) # a33 @funcify_method("Time (s)", "Attitude Angle (°)") @@ -2536,6 +2507,7 @@ def max_dynamic_pressure(self): # Total Pressure @funcify_method("Time (s)", "Total Pressure (Pa)", "spline", "zero") def total_pressure(self): + """Total pressure as a Function of time.""" return self.pressure * (1 + 0.2 * self.mach_number**2) ** (3.5) @cached_property @@ -2552,6 +2524,9 @@ def max_total_pressure(self): # Dynamics functions and variables # Aerodynamic Lift and Drag + # TODO: These are not lift and drag, they are the aerodynamic forces in + # the rocket frame, meaning they are normal and axial forces. They should + # be renamed. @funcify_method("Time (s)", "Aerodynamic Lift Force (N)", "spline", "zero") def aerodynamic_lift(self): """Aerodynamic lift force as a Function of time.""" @@ -2576,6 +2551,7 @@ def aerodynamic_spin_moment(self): # Kinetic Energy @funcify_method("Time (s)", "Rotational Kinetic Energy (J)") def rotational_energy(self): + """Rotational kinetic energy as a Function of time.""" rotational_energy = 0.5 * ( self.rocket.I_11 * self.w1**2 + self.rocket.I_22 * self.w2**2 @@ -2624,7 +2600,7 @@ def total_energy(self): # thrust Power @funcify_method("Time (s)", "thrust Power (W)", "spline", "zero") def thrust_power(self): - """thrust power as a Function of time.""" + """Thrust power as a Function of time.""" thrust = deepcopy(self.rocket.motor.thrust) thrust = thrust.set_discrete_based_on_model(self.speed) thrust_power = thrust * self.speed @@ -2639,35 +2615,95 @@ def drag_power(self): return drag_power # Angle of Attack + @cached_property + def direction_cosine_matrixes(self): + """Direction cosine matrix representing the attitude of the body frame, + relative to the inertial frame, at each time step.""" + # Stack the y_arrays from e0, e1, e2, and e3 along a new axis + stacked_arrays = np.stack( + [self.e0.y_array, self.e1.y_array, self.e2.y_array, self.e3.y_array], + axis=-1, + ) + + # Apply the transformation to the stacked arrays along the last axis + Kt = np.array([Matrix.transformation(row).transpose for row in stacked_arrays]) + + return Kt + + @cached_property + def stream_velocity_body_frame(self): + """Stream velocity array at the center of dry mass in the body frame at + each time step.""" + Kt = self.direction_cosine_matrixes + stream_velocity = np.array( + [ + self.stream_velocity_x.y_array, + self.stream_velocity_y.y_array, + self.stream_velocity_z.y_array, + ] + ).transpose() + stream_velocity_body = np.squeeze( + np.matmul(Kt, stream_velocity[:, :, np.newaxis]) + ) + return stream_velocity_body + @funcify_method("Time (s)", "Angle of Attack (°)", "spline", "constant") def angle_of_attack(self): """Angle of attack of the rocket with respect to the freestream - velocity vector.""" - dot_product = [ - -self.attitude_vector_x.get_value_opt(i) - * self.stream_velocity_x.get_value_opt(i) - - self.attitude_vector_y.get_value_opt(i) - * self.stream_velocity_y.get_value_opt(i) - - self.attitude_vector_z.get_value_opt(i) - * self.stream_velocity_z.get_value_opt(i) - for i in self.time - ] + velocity vector. Sometimes called total angle of attack. Defined as the + angle between the freestream velocity vector and the rocket's z-axis. + All in the Body Axes Coordinate System.""" + # Define stream velocity z component in body frame + stream_vz_body = ( + -self.attitude_vector_x.y_array * self.stream_velocity_x.y_array + - self.attitude_vector_y.y_array * self.stream_velocity_y.y_array + - self.attitude_vector_z.y_array * self.stream_velocity_z.y_array + ) # Define freestream speed list - free_stream_speed = [self.free_stream_speed.get_value_opt(i) for i in self.time] - free_stream_speed = np.nan_to_num(free_stream_speed) + free_stream_speed = self.free_stream_speed.y_array - # Normalize dot product - dot_product_normalized = [ - i / j if j > 1e-6 else 0 for i, j in zip(dot_product, free_stream_speed) - ] - dot_product_normalized = np.nan_to_num(dot_product_normalized) - dot_product_normalized = np.clip(dot_product_normalized, -1, 1) + stream_vz_body_normalized = np.divide( + stream_vz_body, + free_stream_speed, + out=np.zeros_like(stream_vz_body), + where=free_stream_speed > 1e-6, + ) + stream_vz_body_normalized = np.clip(stream_vz_body_normalized, -1, 1) # Calculate angle of attack and convert to degrees - angle_of_attack = np.rad2deg(np.arccos(dot_product_normalized)) + angle_of_attack = np.rad2deg(np.arccos(stream_vz_body_normalized)) + angle_of_attack = np.nan_to_num(angle_of_attack) return np.column_stack([self.time, angle_of_attack]) + @funcify_method("Time (s)", "Partial Angle of Attack (°)", "spline", "constant") + def partial_angle_of_attack(self): + """Partial angle of attack of the rocket with respect to the stream + velocity vector. By partial angle of attack, it is meant the angle + between the stream velocity vector in the y-z plane and the rocket's + z-axis. All in the Body Axes Coordinate System.""" + # Stream velocity in standard aerodynamic frame + stream_velocity = -self.stream_velocity_body_frame + alpha = np.arctan2( + stream_velocity[:, 1], + stream_velocity[:, 2], + ) # y-z plane + return np.column_stack([self.time, np.rad2deg(alpha)]) + + @funcify_method("Time (s)", "Beta (°)", "spline", "constant") + def angle_of_sideslip(self): + """Angle of sideslip of the rocket with respect to the stream + velocity vector. Defined as the angle between the stream velocity + vector in the x-z plane and the rocket's z-axis. All in the Body + Axes Coordinate System.""" + # Stream velocity in standard aerodynamic frame + stream_velocity = -self.stream_velocity_body_frame + beta = np.arctan2( + -stream_velocity[:, 0], + stream_velocity[:, 2], + ) # x-z plane + return np.column_stack([self.time, np.rad2deg(beta)]) + # Frequency response and stability variables @funcify_method("Frequency (Hz)", "ω1 Fourier Amplitude", "spline", "zero") def omega1_frequency_response(self): @@ -2886,9 +2922,10 @@ def __calculate_rail_button_forces(self): # TODO: complex method. # Distance from Rail Button 1 (upper) to CM rail_buttons_tuple = self.rocket.rail_buttons[0] upper_button_position = ( - rail_buttons_tuple.component.buttons_distance + rail_buttons_tuple.position + rail_buttons_tuple.component.buttons_distance + + rail_buttons_tuple.position.z ) - lower_button_position = rail_buttons_tuple.position + lower_button_position = rail_buttons_tuple.position.z angular_position_rad = rail_buttons_tuple.component.angular_position_rad D1 = ( upper_button_position - self.rocket.center_of_dry_mass_position diff --git a/rocketpy/stochastic/stochastic_rocket.py b/rocketpy/stochastic/stochastic_rocket.py index 714637d1f..d9333dfc7 100644 --- a/rocketpy/stochastic/stochastic_rocket.py +++ b/rocketpy/stochastic/stochastic_rocket.py @@ -3,6 +3,7 @@ import warnings from random import choice +from rocketpy.mathutils.vector_matrix import Vector from rocketpy.motors.motor import EmptyMotor, GenericMotor, Motor from rocketpy.motors.solid_motor import SolidMotor from rocketpy.rocket.aero_surface import ( @@ -449,6 +450,9 @@ def get_surface_position(self_object, _): def _randomize_position(self, position): """Randomize a position provided as a tuple or list.""" if isinstance(position, tuple): + if isinstance(position[0], Vector): + # TODO implement randomization for X and Y positions + return position[-1](position[0].z, position[1]) return position[-1](position[0], position[1]) elif isinstance(position, list): return choice(position) if position else position @@ -496,7 +500,9 @@ def _create_surface(self, component_stochastic_surface): self.last_rnd_dict["aerodynamic_surfaces"].append( stochastic_surface.last_rnd_dict ) - self.last_rnd_dict["aerodynamic_surfaces"][-1]["position"] = position_rnd + self.last_rnd_dict["aerodynamic_surfaces"][-1]["position"] = Vector( + [0, 0, position_rnd] + ) return surface, position_rnd def _create_rail_buttons(self, component_stochastic_rail_buttons): diff --git a/tests/conftest.py b/tests/conftest.py index 6c4171b66..a23f5fd89 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -18,6 +18,8 @@ "tests.fixtures.monte_carlo.stochastic_fixtures", "tests.fixtures.monte_carlo.stochastic_motors_fixtures", "tests.fixtures.sensors.sensors_fixtures", + "tests.fixtures.generic_surfaces.generic_surfaces_fixtures", + "tests.fixtures.generic_surfaces.linear_generic_surfaces_fixtures", ] diff --git a/tests/fixtures/generic_surfaces/__init__.py b/tests/fixtures/generic_surfaces/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/tests/fixtures/generic_surfaces/generic_surfaces_fixtures.py b/tests/fixtures/generic_surfaces/generic_surfaces_fixtures.py new file mode 100644 index 000000000..e3aa869ea --- /dev/null +++ b/tests/fixtures/generic_surfaces/generic_surfaces_fixtures.py @@ -0,0 +1,42 @@ +import pandas as pd +import pytest + + +@pytest.fixture(scope="session") +def filename_valid_coeff(tmpdir_factory): + """Creates temporary files used to test if generic surfaces + initializes correctly from CSV files""" + filename = tmpdir_factory.mktemp("aero_surface_data").join("valid_coefficients.csv") + pd.DataFrame( + { + "alpha": [0, 1, 2, 3, 0.1], + "mach": [3, 2, 1, 0, 0.2], + "cL": [4, 2, 2, 4, 5], + } + ).to_csv(filename, index=False) + + return filename + + +@pytest.fixture( + params=( + { + "alpha": [0, 1, 2, 3, 0.1], + "cL": [4, 2, 2, 4, 5], + "mach": [3, 2, 1, 0, 0.2], + }, + { + "a": [0, 1, 2, 3, 0.1], + "b": [4, 2, 2, 4, 5], + }, + ) +) +def filename_invalid_coeff(tmpdir_factory, request): + """Creates temporary CSV files used to test if generic surfaces + raises errors when initialized incorrectly from CSV files""" + filename = tmpdir_factory.mktemp("aero_surface_data").join( + "tmp_invalid_coefficients.csv" + ) + pd.DataFrame(request.param).to_csv(filename, index=False) + + return filename diff --git a/tests/fixtures/generic_surfaces/linear_generic_surfaces_fixtures.py b/tests/fixtures/generic_surfaces/linear_generic_surfaces_fixtures.py new file mode 100644 index 000000000..35ab9c1a2 --- /dev/null +++ b/tests/fixtures/generic_surfaces/linear_generic_surfaces_fixtures.py @@ -0,0 +1,44 @@ +import pandas as pd +import pytest + + +@pytest.fixture(scope="session") +def filename_valid_coeff_linear_generic_surface(tmpdir_factory): + """Creates temporary files used to test if a linear generic surface + initializes correctly from CSV files""" + filename = tmpdir_factory.mktemp("aero_surface_data").join( + "valid_coefficients_lgs.csv" + ) + pd.DataFrame( + { + "alpha": [0, 1, 2, 3, 0.1], + "mach": [3, 2, 1, 0, 0.2], + "cL_0": [4, 2, 2, 4, 5], + } + ).to_csv(filename, index=False) + + return filename + + +@pytest.fixture( + params=( + { + "alpha": [0, 1, 2, 3, 0.1], + "cL_0": [4, 2, 2, 4, 5], + "mach": [3, 2, 1, 0, 0.2], + }, + { + "a": [0, 1, 2, 3, 0.1], + "b": [4, 2, 2, 4, 5], + }, + ) +) +def filename_invalid_coeff_linear_generic_surface(tmpdir_factory, request): + """Creates temporary CSV files used to test if a linear generic surface + raises errors when initialized incorrectly from CSV files""" + filename = tmpdir_factory.mktemp("aero_surface_data").join( + "tmp_invalid_coefficients_lgs.csv" + ) + pd.DataFrame(request.param).to_csv(filename, index=False) + + return filename diff --git a/tests/fixtures/surfaces/surface_fixtures.py b/tests/fixtures/surfaces/surface_fixtures.py index 10595205f..396206bd7 100644 --- a/tests/fixtures/surfaces/surface_fixtures.py +++ b/tests/fixtures/surfaces/surface_fixtures.py @@ -1,6 +1,7 @@ import pytest from rocketpy import NoseCone, RailButtons, Tail, TrapezoidalFins +from rocketpy.rocket.aero_surface.fins.free_form_fins import FreeFormFins @pytest.fixture @@ -62,6 +63,23 @@ def calisto_trapezoidal_fins(): ) +@pytest.fixture +def calisto_free_form_fins(): + """The free form fins of the Calisto rocket. + + Returns + ------- + rocketpy.FreeFormFins + The free form fins of the Calisto rocket. + """ + return FreeFormFins( + n=4, + shape_points=[(0, 0), (0.08, 0.1), (0.12, 0.1), (0.12, 0)], + rocket_radius=0.0635, + name="calisto_free_form_fins", + ) + + @pytest.fixture def calisto_rail_buttons(): """The rail buttons of the Calisto rocket. diff --git a/tests/integration/test_flight.py b/tests/integration/test_flight.py index 422941c4d..fc1dd1956 100644 --- a/tests/integration/test_flight.py +++ b/tests/integration/test_flight.py @@ -335,8 +335,8 @@ def test_rolling_flight( # pylint: disable=unused-argument position=-1.04956, cant_angle=0.5, ) - calisto.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) - calisto.aerodynamic_surfaces.add(calisto_tail, -1.313) + calisto.add_surfaces(calisto_nose_cone, 1.160) + calisto.add_surfaces(calisto_tail, -1.313) calisto.parachutes.append(calisto_main_chute) calisto.parachutes.append(calisto_drogue_chute) @@ -365,9 +365,9 @@ def test_eccentricity_on_flight( # pylint: disable=unused-argument test_rocket.set_rail_buttons(0.082, -0.618) test_rocket.add_motor(cesaroni_m1670, position=-1.373) - calisto.aerodynamic_surfaces.add(calisto_trapezoidal_fins, position=-1.04956) - calisto.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) - calisto.aerodynamic_surfaces.add(calisto_tail, -1.313) + calisto.add_surfaces(calisto_trapezoidal_fins, -1.04956) + calisto.add_surfaces(calisto_nose_cone, 1.160) + calisto.add_surfaces(calisto_tail, -1.313) calisto.add_cm_eccentricity(x=-0.01, y=-0.01) test_flight = Flight( diff --git a/tests/integration/test_rocket.py b/tests/integration/test_rocket.py index 4d5daf7a6..e51fe925e 100644 --- a/tests/integration/test_rocket.py +++ b/tests/integration/test_rocket.py @@ -1,6 +1,7 @@ from unittest.mock import patch import numpy as np +import pytest @patch("matplotlib.pyplot.show") @@ -14,8 +15,8 @@ def test_airfoil( ): test_rocket = calisto test_rocket.set_rail_buttons(0.082, -0.618) - calisto.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) - calisto.aerodynamic_surfaces.add(calisto_tail, -1.313) + calisto.add_surfaces(calisto_nose_cone, 1.160) + calisto.add_surfaces(calisto_tail, -1.313) test_rocket.add_trapezoidal_fins( 2, @@ -126,9 +127,47 @@ def test_rocket(mock_show, calisto_robust): # pylint: disable=unused-argument @patch("matplotlib.pyplot.show") def test_aero_surfaces_infos( # pylint: disable=unused-argument - mock_show, calisto_nose_cone, calisto_tail, calisto_trapezoidal_fins + mock_show, + calisto_nose_cone, + calisto_tail, + calisto_trapezoidal_fins, + calisto_free_form_fins, ): assert calisto_nose_cone.all_info() is None assert calisto_trapezoidal_fins.all_info() is None assert calisto_tail.all_info() is None - assert calisto_trapezoidal_fins.draw() is None + assert calisto_trapezoidal_fins.all_info() is None + assert calisto_free_form_fins.all_info() is None + + +@pytest.mark.parametrize( + "attribute, tolerance", + [ + ("cpz", 1e-3), + ("clalpha", 1e-3), + ("Af", 1e-3), + ("gamma_c", 1e-3), + ("Yma", 1e-3), + ("tau", 1e-3), + ("roll_geometrical_constant", 1e-3), + ("lift_interference_factor", 1e-3), + ("roll_forcing_interference_factor", 1e-3), + ("roll_damping_interference_factor", 1e-2), + ], +) +def test_calisto_free_form_fins_equivalence( + calisto_free_form_fins, calisto_trapezoidal_fins, attribute, tolerance +): + """Test the equivalence of the free form fins with the same geometric + characteristics as the trapezoidal fins, comparing cp, cnalpha, and + geometrical parameters.""" + + # Handle the 'clalpha' method comparison differently as it's a callable + if attribute == "clalpha": + free_form_value = calisto_free_form_fins.clalpha(0) + trapezoidal_value = calisto_trapezoidal_fins.clalpha(0) + else: + free_form_value = getattr(calisto_free_form_fins, attribute) + trapezoidal_value = getattr(calisto_trapezoidal_fins, attribute) + + assert abs(free_form_value - trapezoidal_value) < tolerance diff --git a/tests/unit/test_environment.py b/tests/unit/test_environment.py index 7769f7b85..714470a40 100644 --- a/tests/unit/test_environment.py +++ b/tests/unit/test_environment.py @@ -71,13 +71,18 @@ def test_location_set_topographic_profile_computes_elevation( def test_geodesic_coordinate_geodesic_to_utm_converts_coordinate(): """Tests the conversion from geodesic to UTM coordinates.""" - x, y, utm_zone, utm_letter, north_south_hemis, east_west_hemis = ( - Environment.geodesic_to_utm( - lat=32.990254, - lon=-106.974998, - semi_major_axis=6378137.0, # WGS84 - flattening=1 / 298.257223563, # WGS84 - ) + ( + x, + y, + utm_zone, + utm_letter, + north_south_hemis, + east_west_hemis, + ) = Environment.geodesic_to_utm( + lat=32.990254, + lon=-106.974998, + semi_major_axis=6378137.0, # WGS84 + flattening=1 / 298.257223563, # WGS84 ) assert np.isclose(x, 315468.64, atol=1e-5) assert np.isclose(y, 3651938.65, atol=1e-5) diff --git a/tests/unit/test_function.py b/tests/unit/test_function.py index 9efb64c0c..df540c1ae 100644 --- a/tests/unit/test_function.py +++ b/tests/unit/test_function.py @@ -502,6 +502,110 @@ def test_3d_shepard_interpolation(x, y, z, w_expected): assert np.isclose(w_expected, w, atol=1e-8).all() +@pytest.mark.parametrize( + "x,y,z_expected", + [ + (1, 0, 0), + (0, 1, 0), + (0, 0, 1), + (0.5, 0.5, 0), + (0.25, 0.25, 0.5), + ([0, 0.5], [0, 0.5], [1, 0]), + ], +) +def test_2d_rbf_interpolation(x, y, z_expected): + """Test the rbf interpolation method of the Function class.""" + # Test plane x + y + z = 1 + source = [(1, 0, 0), (0, 1, 0), (0, 0, 1)] + func = Function( + source=source, inputs=["x", "y"], outputs=["z"], interpolation="rbf" + ) + z = func(x, y) + z_opt = func.get_value_opt(x, y) + assert np.isclose(z, z_opt, atol=1e-8).all() + assert np.isclose(z_expected, z, atol=1e-8).all() + + +@pytest.mark.parametrize( + "x,y,z,w_expected", + [ + (0, 0, 0, 1), + (1, 0, 0, 0), + (0, 1, 0, 0), + (0, 0, 1, 0), + (0.5, 0.5, 0.5, -0.5), + (0.25, 0.25, 0.25, 0.25), + ([0, 0.5], [0, 0.5], [0, 0.5], [1, -0.5]), + ], +) +def test_3d_rbf_interpolation(x, y, z, w_expected): + """Test the rbf interpolation method of the Function class.""" + # Test plane x + y + z + w = 1 + source = [(1, 0, 0, 0), (0, 1, 0, 0), (0, 0, 1, 0), (0, 0, 0, 1)] + func = Function( + source=source, inputs=["x", "y", "z"], outputs=["w"], interpolation="rbf" + ) + w = func(x, y, z) + w_opt = func.get_value_opt(x, y, z) + assert np.isclose(w, w_opt, atol=1e-8).all() + assert np.isclose(w_expected, w, atol=1e-8).all() + + +@pytest.mark.parametrize( + "x,y,z_expected", + [ + (1, 0, 0), + (0, 1, 0), + (0, 0, 1), + (0.5, 0.5, 0), + (0.25, 0.25, 0.5), + ([0, 0.5], [0, 0.5], [1, 0]), + ], +) +def test_2d_linear_interpolation(x, y, z_expected): + """Test the linear interpolation method of the Function class.""" + # Test plane x + y + z = 1 + source = [(1, 0, 0), (0, 1, 0), (0, 0, 1)] + func = Function( + source=source, inputs=["x", "y"], outputs=["z"], interpolation="linear" + ) + z = func(x, y) + z_opt = func.get_value_opt(x, y) + assert np.isclose(z, z_opt, atol=1e-8).all() + assert np.isclose(z_expected, z, atol=1e-8).all() + + +@pytest.mark.parametrize( + "x,y,z,w_expected", + [ + (1, 0, 0, 0), + (0, 1, 0, 0), + (0, 0, 1, 0), + (0, 0, 0, 1), + (0.5, 0.5, 0.5, -0.5), + (0.25, 0.25, 0.25, 0.25), + ([0, 0.25], [0, 0.25], [0, 0.25], [1, 0.25]), + ], +) +def test_3d_linear_interpolation(x, y, z, w_expected): + """Test the linear interpolation method of the Function class.""" + # Test plane x + y + z + w = 1 + source = [ + (1, 0, 0, 0), + (0, 1, 0, 0), + (0, 0, 1, 0), + (0, 0, 0, 1), + (0.5, 0.5, 0.5, -0.5), + ] + func = Function( + source=source, inputs=["x", "y", "z"], outputs=["w"], interpolation="linear" + ) + w = func(x, y, z) + w_opt = func.get_value_opt(x, y, z) + assert np.isclose(w, w_opt, atol=1e-8).all() + assert np.isclose(w_expected, w, atol=1e-8).all() + + @pytest.mark.parametrize("a", [-1, -0.5, 0, 0.5, 1]) @pytest.mark.parametrize("b", [-1, -0.5, 0, 0.5, 1]) def test_multivariate_function(a, b): @@ -565,26 +669,6 @@ def test_set_discrete_based_on_2d_model(func_2d_from_csv): assert discretized_func.__extrapolation__ == func_2d_from_csv.__extrapolation__ -@pytest.mark.parametrize( - "x,y,z_expected", - [ - (1, 0, 0), - (0, 1, 0), - (0, 0, 1), - (0.5, 0.5, 1 / 3), - (0.25, 0.25, 25 / (25 + 2 * 5**0.5)), - ([0, 0.5], [0, 0.5], [1, 1 / 3]), - ], -) -def test_shepard_interpolation(x, y, z_expected): - """Test the shepard interpolation method of the Function class.""" - # Test plane x + y + z = 1 - source = [(1, 0, 0), (0, 1, 0), (0, 0, 1)] - func = Function(source=source, inputs=["x", "y"], outputs=["z"]) - z = func(x, y) - assert np.isclose(z, z_expected, atol=1e-8).all() - - @pytest.mark.parametrize("other", [1, 0.1, np.int_(1), np.float64(0.1), np.array([1])]) def test_sum_arithmetic_priority(other): """Test the arithmetic priority of the add operation of the Function class, diff --git a/tests/unit/test_generic_surfaces.py b/tests/unit/test_generic_surfaces.py new file mode 100644 index 000000000..a04ed429d --- /dev/null +++ b/tests/unit/test_generic_surfaces.py @@ -0,0 +1,91 @@ +import pytest + +from rocketpy import Function, GenericSurface +from rocketpy.mathutils import Vector + +REFERENCE_AREA = 1 +REFERENCE_LENGTH = 1 + + +@pytest.mark.parametrize( + "coefficients", + [ + "cL", + {"invalid_name": 0}, + {"cL": "inexistent_file.csv"}, + {"cL": Function(lambda x1, x2, x3, x4, x5, x6: 0)}, + {"cL": lambda x1: 0}, + {"cL": {}}, + ], +) +def test_invalid_initialization(coefficients): + """Checks if generic surface raises errors in initialization + when coefficient argument is invalid""" + + with pytest.raises((ValueError, TypeError)): + GenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients=coefficients, + ) + + +def test_invalid_initialization_from_csv(filename_invalid_coeff): + """Checks if generic surfaces raises errors when initialized incorrectly + from a csv file""" + with pytest.raises(ValueError): + GenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients={"cL": str(filename_invalid_coeff)}, + ) + + +@pytest.mark.parametrize( + "coefficients", + [ + {}, + {"cL": 0}, + { + "cL": 0, + "cQ": Function(lambda x1, x2, x3, x4, x5, x6, x7: 0), + "cD": lambda x1, x2, x3, x4, x5, x6, x7: 0, + }, + ], +) +def test_valid_initialization(coefficients): + """Checks if generic surface initializes correctly when coefficient + argument is valid""" + + GenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients=coefficients, + ) + + +def test_valid_initialization_from_csv(filename_valid_coeff): + """Checks if generic surfaces initializes correctly when + coefficients is set from a csv file""" + GenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients={"cL": str(filename_valid_coeff)}, + ) + + +def test_compute_forces_and_moments(): + """Checks if there are not logical errors in + compute forces and moments""" + + gs_object = GenericSurface(REFERENCE_AREA, REFERENCE_LENGTH, {}) + forces_and_moments = gs_object.compute_forces_and_moments( + stream_velocity=Vector((0, 0, 0)), + stream_speed=0, + stream_mach=0, + rho=0, + cp=Vector((0, 0, 0)), + omega=(0, 0, 0), + reynolds=0, + ) + assert forces_and_moments == (0, 0, 0, 0, 0, 0) diff --git a/tests/unit/test_linear_generic_surfaces.py b/tests/unit/test_linear_generic_surfaces.py new file mode 100644 index 000000000..858226444 --- /dev/null +++ b/tests/unit/test_linear_generic_surfaces.py @@ -0,0 +1,91 @@ +import pytest + +from rocketpy import Function, LinearGenericSurface +from rocketpy.mathutils import Vector + +REFERENCE_AREA = 1 +REFERENCE_LENGTH = 1 + + +@pytest.mark.parametrize( + "coefficients", + [ + "cL_0", + {"invalid_name": 0}, + {"cL_0": "inexistent_file.csv"}, + {"cL_0": Function(lambda x1, x2, x3, x4, x5, x6: 0)}, + {"cL_0": lambda x1: 0}, + {"cL_0": {}}, + ], +) +def test_invalid_initialization(coefficients): + """Checks if linear generic surface raises errors in initialization + when coefficient argument is invalid""" + + with pytest.raises((ValueError, TypeError)): + LinearGenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients=coefficients, + ) + + +def test_invalid_initialization_from_csv(filename_invalid_coeff_linear_generic_surface): + """Checks if linear generic surfaces raises errors when initialized incorrectly + from a csv file""" + with pytest.raises(ValueError): + LinearGenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients={"cL_0": str(filename_invalid_coeff_linear_generic_surface)}, + ) + + +@pytest.mark.parametrize( + "coefficients", + [ + {}, + {"cL_0": 0}, + { + "cL_0": 0, + "cQ_0": Function(lambda x1, x2, x3, x4, x5, x6, x7: 0), + "cD_0": lambda x1, x2, x3, x4, x5, x6, x7: 0, + }, + ], +) +def test_valid_initialization(coefficients): + """Checks if linear generic surface initializes correctly when coefficient + argument is valid""" + + LinearGenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients=coefficients, + ) + + +def test_valid_initialization_from_csv(filename_valid_coeff_linear_generic_surface): + """Checks if linear generic surfaces initializes correctly when + coefficients is set from a csv file""" + LinearGenericSurface( + reference_area=REFERENCE_AREA, + reference_length=REFERENCE_LENGTH, + coefficients={"cL_0": str(filename_valid_coeff_linear_generic_surface)}, + ) + + +def test_compute_forces_and_moments(): + """Checks if there are not logical errors in + compute forces and moments""" + + lgs_object = LinearGenericSurface(REFERENCE_AREA, REFERENCE_LENGTH, {}) + forces_and_moments = lgs_object.compute_forces_and_moments( + stream_velocity=Vector((0, 0, 0)), + stream_speed=1, + stream_mach=0, + rho=0, + cp=Vector((0, 0, 0)), + omega=(0, 0, 0), + reynolds=0, + ) + assert forces_and_moments == (0, 0, 0, 0, 0, 0) diff --git a/tests/unit/test_rocket.py b/tests/unit/test_rocket.py index a0d9b79a3..f048f4b3c 100644 --- a/tests/unit/test_rocket.py +++ b/tests/unit/test_rocket.py @@ -4,6 +4,7 @@ import pytest from rocketpy import Function, NoseCone, Rocket, SolidMotor +from rocketpy.mathutils.vector_matrix import Vector from rocketpy.motors.motor import EmptyMotor, Motor @@ -124,7 +125,7 @@ def test_add_trapezoidal_fins_sweep_angle( calisto_nose_cone, ): # Reference values from OpenRocket - calisto.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) + calisto.add_surfaces(calisto_nose_cone, Vector([0, 0, 1.160])) fin_set = calisto.add_trapezoidal_fins( n=3, span=0.090, @@ -164,7 +165,7 @@ def test_add_trapezoidal_fins_sweep_length( calisto_nose_cone, ): # Reference values from OpenRocket - calisto.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) + calisto.add_surfaces(calisto_nose_cone, Vector([0, 0, 1.160])) fin_set = calisto.add_trapezoidal_fins( n=3, span=0.090, @@ -379,7 +380,7 @@ def test_set_rail_button(calisto): == pytest.approx(0.7, 1e-12) ) # assert buttons position on rocket - assert calisto.rail_buttons[0].position == -0.5 + assert calisto.rail_buttons[0].position.z == -0.5 # assert angular position assert ( rail_buttons.angular_position @@ -389,16 +390,7 @@ def test_set_rail_button(calisto): # assert upper button position assert calisto.rail_buttons[0].component.buttons_distance + calisto.rail_buttons[ 0 - ].position == pytest.approx(0.2, 1e-12) - - -def test_add_rail_button(calisto, calisto_rail_buttons): - calisto.add_surfaces(calisto_rail_buttons, -0.5) - assert calisto.rail_buttons[0].position == -0.5 - upper_position = ( - calisto_rail_buttons.buttons_distance + calisto.rail_buttons[0].position - ) - assert upper_position == pytest.approx(0.2, 1e-12) + ].position.z == pytest.approx(0.2, 1e-12) def test_evaluate_total_mass(calisto_motorless): @@ -641,8 +633,8 @@ def test_coordinate_system_orientation( rocket_tail_to_nose.add_motor(motor_nozzle_to_combustion_chamber, position=-1.373) - rocket_tail_to_nose.aerodynamic_surfaces.add(calisto_nose_cone, 1.160) - rocket_tail_to_nose.aerodynamic_surfaces.add(calisto_trapezoidal_fins, -1.168) + rocket_tail_to_nose.add_surfaces(calisto_nose_cone, 1.160) + rocket_tail_to_nose.add_surfaces(calisto_trapezoidal_fins, -1.168) static_margin_tail_to_nose = rocket_tail_to_nose.static_margin @@ -658,8 +650,8 @@ def test_coordinate_system_orientation( rocket_nose_to_tail.add_motor(motor_combustion_chamber_to_nozzle, position=1.373) - rocket_nose_to_tail.aerodynamic_surfaces.add(calisto_nose_cone, -1.160) - rocket_nose_to_tail.aerodynamic_surfaces.add(calisto_trapezoidal_fins, 1.168) + rocket_nose_to_tail.add_surfaces(calisto_nose_cone, -1.160) + rocket_nose_to_tail.add_surfaces(calisto_trapezoidal_fins, 1.168) static_margin_nose_to_tail = rocket_nose_to_tail.static_margin