From c0b4abafd9cb70b5cbc0d940226050afc3cbc130 Mon Sep 17 00:00:00 2001 From: Gabriel Barberini Date: Sat, 16 Nov 2024 01:20:51 -0300 Subject: [PATCH 1/5] updates pylint ignore rules --- Makefile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Makefile b/Makefile index fa15f27..b3a6294 100644 --- a/Makefile +++ b/Makefile @@ -10,7 +10,7 @@ flake8: pylint: pylint ./lib || true - pylint --disable=E0401,W0621 ./tests || true + pylint --disable=E0401,W0621,R0913,R0917 ./tests || true ruff: ruff check --fix ./lib || true From fdb0f7cd86ef21c6a65a70cf7dc05908fcaa55bf Mon Sep 17 00:00:00 2001 From: Gabriel Barberini Date: Sat, 16 Nov 2024 10:40:38 -0300 Subject: [PATCH 2/5] adds rocket route tests --- lib/controllers/flight.py | 2 +- lib/controllers/rocket.py | 2 +- lib/models/aerosurfaces.py | 11 +- lib/models/rocket.py | 10 +- lib/routes/flight.py | 4 +- lib/routes/rocket.py | 4 +- tests/test_routes/test_rockets_route.py | 670 ++++++++++++++++++++++++ 7 files changed, 687 insertions(+), 16 deletions(-) create mode 100644 tests/test_routes/test_rockets_route.py diff --git a/lib/controllers/flight.py b/lib/controllers/flight.py index 1d2aa8d..d112aa4 100644 --- a/lib/controllers/flight.py +++ b/lib/controllers/flight.py @@ -183,7 +183,7 @@ async def get_rocketpy_flight_binary( @classmethod async def update_flight_by_id( - cls, flight: Flight, flight_id: str + cls, flight_id: str, flight: Flight ) -> Union[FlightUpdated, HTTPException]: """ Update a models.Flight in the database. diff --git a/lib/controllers/rocket.py b/lib/controllers/rocket.py index 5b42b60..e3817d6 100644 --- a/lib/controllers/rocket.py +++ b/lib/controllers/rocket.py @@ -170,7 +170,7 @@ async def get_rocketpy_rocket_binary( @classmethod async def update_rocket_by_id( - cls, rocket: Rocket, rocket_id: str + cls, rocket_id: str, rocket: Rocket ) -> Union[RocketUpdated, HTTPException]: """ Update a models.Rocket in the database. diff --git a/lib/models/aerosurfaces.py b/lib/models/aerosurfaces.py index 53ae8c8..aec87f0 100644 --- a/lib/models/aerosurfaces.py +++ b/lib/models/aerosurfaces.py @@ -1,8 +1,17 @@ from enum import Enum -from typing import Optional, Tuple, List +from typing import Optional, Tuple, List, Union from pydantic import BaseModel +class Parachute(BaseModel): + name: str + cd_s: float + sampling_rate: int + lag: float + trigger: Union[str, float] + noise: Tuple[float, float, float] + + class RailButtons(BaseModel): name: str = "RailButtons" upper_button_position: float diff --git a/lib/models/rocket.py b/lib/models/rocket.py index 2b6b1a9..c1763ea 100644 --- a/lib/models/rocket.py +++ b/lib/models/rocket.py @@ -7,6 +7,7 @@ NoseCone, Tail, RailButtons, + Parachute, ) @@ -15,15 +16,6 @@ class CoordinateSystemOrientation(str, Enum): NOSE_TO_TAIL: str = "NOSE_TO_TAIL" -class Parachute(BaseModel): - name: str - cd_s: float - sampling_rate: int - lag: float - trigger: Union[str, float] - noise: Tuple[float, float, float] - - class Rocket(BaseModel): # Required parameters motor: Motor diff --git a/lib/routes/flight.py b/lib/routes/flight.py index cb921db..e88b87a 100644 --- a/lib/routes/flight.py +++ b/lib/routes/flight.py @@ -59,7 +59,7 @@ async def read_flight(flight_id: str) -> FlightView: @router.get( - "/rocketpy/{flight_id}", + "/{flight_id}/rocketpy", responses={ 203: { "description": "Binary file download", @@ -146,7 +146,7 @@ async def update_flight( """ with tracer.start_as_current_span("update_flight"): flight.rocket.motor.set_motor_kind(motor_kind) - return await FlightController.update_flight_by_id(flight, flight_id) + return await FlightController.update_flight_by_id(flight_id, flight) @router.get("/{flight_id}/summary") diff --git a/lib/routes/rocket.py b/lib/routes/rocket.py index cbbe93d..f847754 100644 --- a/lib/routes/rocket.py +++ b/lib/routes/rocket.py @@ -73,11 +73,11 @@ async def update_rocket( """ with tracer.start_as_current_span("update_rocket"): rocket.motor.set_motor_kind(motor_kind) - return await RocketController.update_rocket_by_id(rocket, rocket_id) + return await RocketController.update_rocket_by_id(rocket_id, rocket) @router.get( - "/rocketpy/{rocket_id}", + "/{rocket_id}/rocketpy", responses={ 203: { "description": "Binary file download", diff --git a/tests/test_routes/test_rockets_route.py b/tests/test_routes/test_rockets_route.py new file mode 100644 index 0000000..2a3afb6 --- /dev/null +++ b/tests/test_routes/test_rockets_route.py @@ -0,0 +1,670 @@ +from unittest.mock import patch +import json +import pytest +from fastapi.testclient import TestClient +from fastapi import HTTPException, status +from lib.models.aerosurfaces import ( + Fins, + NoseCone, + Tail, + RailButtons, + Parachute, +) +from lib.models.rocket import Rocket +from lib.models.motor import ( + Motor, + MotorKinds, + MotorTank, + TankFluids, + TankKinds, +) +from lib.controllers.rocket import RocketController +from lib.views.motor import MotorView +from lib.views.rocket import ( + RocketCreated, + RocketUpdated, + RocketDeleted, + RocketSummary, + RocketView, +) +from lib import app + +client = TestClient(app) + + +@pytest.fixture +def stub_motor(): + motor = Motor( + thrust_source=[[0, 0]], + burn_time=0, + nozzle_radius=0, + dry_mass=0, + dry_inertia=[0, 0, 0], + center_of_dry_mass_position=0, + ) + motor_json = motor.model_dump_json() + return json.loads(motor_json) + + +@pytest.fixture +def stub_tank(): + tank = MotorTank( + geometry=[[(0, 0), 0]], + gas=TankFluids(name='gas', density=0), + liquid=TankFluids(name='liquid', density=0), + flux_time=(0, 0), + position=0, + discretize=0, + name='tank', + ) + tank_json = tank.model_dump_json() + return json.loads(tank_json) + + +@pytest.fixture +def stub_level_tank(stub_tank): + stub_tank.update({'tank_kind': TankKinds.LEVEL, 'liquid_height': 0}) + return stub_tank + + +@pytest.fixture +def stub_mass_flow_tank(stub_tank): + stub_tank.update( + { + 'tank_kind': TankKinds.MASS_FLOW, + 'gas_mass_flow_rate_in': 0, + 'gas_mass_flow_rate_out': 0, + 'liquid_mass_flow_rate_in': 0, + 'liquid_mass_flow_rate_out': 0, + 'initial_liquid_mass': 0, + 'initial_gas_mass': 0, + } + ) + return stub_tank + + +@pytest.fixture +def stub_ullage_tank(stub_tank): + stub_tank.update({'tank_kind': TankKinds.ULLAGE, 'ullage': 0}) + return stub_tank + + +@pytest.fixture +def stub_mass_tank(stub_tank): + stub_tank.update( + {'tank_kind': TankKinds.MASS, 'liquid_mass': 0, 'gas_mass': 0} + ) + return stub_tank + + +@pytest.fixture +def stub_rocket_summary(): + rocket_summary = RocketSummary() + rocket_summary_json = rocket_summary.model_dump_json() + return json.loads(rocket_summary_json) + + +@pytest.fixture +def stub_nose_cone(): + nose_cone = NoseCone( + name='nose', + length=0, + kind='kind', + position=0, + base_radius=0, + rocket_radius=0, + ) + nose_cone_json = nose_cone.model_dump_json() + return json.loads(nose_cone_json) + + +@pytest.fixture +def stub_fins(): + fins = Fins( + fins_kind='TRAPEZOIDAL', + name='fins', + n=0, + root_chord=0, + span=0, + position=0, + ) + fins_json = fins.model_dump_json() + return json.loads(fins_json) + + +@pytest.fixture +def stub_tail(): + tail = Tail( + name='tail', + top_radius=0, + bottom_radius=0, + length=0, + position=0, + radius=0, + ) + tail_json = tail.model_dump_json() + return json.loads(tail_json) + + +@pytest.fixture +def stub_rail_buttons(): + rail_buttons = RailButtons( + upper_button_position=0, + lower_button_position=0, + angular_position=0, + ) + rail_buttons_json = rail_buttons.model_dump_json() + return json.loads(rail_buttons_json) + + +@pytest.fixture +def stub_parachute(): + parachute = Parachute( + name='parachute', + cd_s=0, + sampling_rate=0, + lag=0, + trigger='trigger', + noise=(0, 0, 0), + ) + parachute_json = parachute.model_dump_json() + return json.loads(parachute_json) + + +@pytest.fixture +def stub_rocket(stub_motor): + rocket = Rocket( + motor=stub_motor, + radius=0, + mass=0, + motor_position=0, + center_of_mass_without_motor=0, + inertia=[0, 0, 0], + power_off_drag=[(0, 0)], + power_on_drag=[(0, 0)], + coordinate_system_orientation='TAIL_TO_NOSE', + ) + rocket_json = rocket.model_dump_json() + return json.loads(rocket_json) + + +def test_create_rocket(stub_rocket): + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.HYBRID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_rocket_optional_params( + stub_rocket, + stub_nose_cone, + stub_fins, + stub_tail, + stub_rail_buttons, + stub_parachute, +): + stub_rocket.update( + { + 'parachutes': [stub_parachute], + 'rail_buttons': stub_rail_buttons, + 'nose': stub_nose_cone, + 'fins': [stub_fins], + 'tail': stub_tail, + } + ) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.HYBRID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_generic_motor_rocket(stub_rocket, stub_motor): + stub_motor.update( + { + 'chamber_radius': 0, + 'chamber_height': 0, + 'chamber_position': 0, + 'propellant_initial_mass': 0, + 'nozzle_position': 0, + } + ) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'GENERIC'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.GENERIC) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_liquid_motor_level_tank_rocket( + stub_rocket, stub_motor, stub_level_tank +): + stub_motor.update({'tanks': [stub_level_tank]}) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'LIQUID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.LIQUID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_liquid_motor_mass_flow_tank_rocket( + stub_rocket, stub_motor, stub_mass_flow_tank +): + stub_motor.update({'tanks': [stub_mass_flow_tank]}) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'LIQUID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.LIQUID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_liquid_motor_ullage_tank_rocket( + stub_rocket, stub_motor, stub_ullage_tank +): + stub_motor.update({'tanks': [stub_ullage_tank]}) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'LIQUID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.LIQUID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_liquid_motor_mass_tank_rocket( + stub_rocket, stub_motor, stub_mass_tank +): + stub_motor.update({'tanks': [stub_mass_tank]}) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'LIQUID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.LIQUID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_hybrid_motor_rocket(stub_rocket, stub_motor, stub_level_tank): + stub_motor.update( + { + 'grain_number': 0, + 'grain_density': 0, + 'grain_outer_radius': 0, + 'grain_initial_inner_radius': 0, + 'grain_initial_height': 0, + 'grains_center_of_mass_position': 0, + 'grain_separation': 0, + 'throat_radius': 0, + 'tanks': [stub_level_tank], + } + ) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.HYBRID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_solid_motor_rocket(stub_rocket, stub_motor): + stub_motor.update( + { + 'grain_number': 0, + 'grain_density': 0, + 'grain_outer_radius': 0, + 'grain_initial_inner_radius': 0, + 'grain_initial_height': 0, + 'grains_center_of_mass_position': 0, + 'grain_separation': 0, + } + ) + stub_rocket.update({'motor': stub_motor}) + with patch.object( + RocketController, + 'create_rocket', + return_value=RocketCreated(rocket_id='123'), + ) as mock_create_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'SOLID'} + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully created', + } + mock_set_motor_kind.assert_called_once_with(MotorKinds.SOLID) + mock_create_rocket.assert_called_once_with(Rocket(**stub_rocket)) + + +def test_create_rocket_invalid_input(): + response = client.post('/rockets/', json={'radius': 'foo', 'mass': 'bar'}) + assert response.status_code == 422 + + +def test_create_rocket_server_error(stub_rocket): + with patch.object( + RocketController, + 'create_rocket', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.post( + '/rockets/', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} + + +def test_read_rocket(stub_rocket, stub_motor): + motor_view = MotorView(**stub_motor, selected_motor_kind=MotorKinds.HYBRID) + stub_rocket.update(motor=motor_view) + rocket_view = RocketView(**stub_rocket) + with patch.object( + RocketController, + 'get_rocket_by_id', + return_value=rocket_view, + ) as mock_read_rocket: + response = client.get('/rockets/123') + assert response.status_code == 200 + assert response.json() == json.loads(rocket_view.model_dump_json()) + mock_read_rocket.assert_called_once_with('123') + + +def test_read_rocket_not_found(): + with patch.object( + RocketController, + 'get_rocket_by_id', + side_effect=HTTPException(status_code=status.HTTP_404_NOT_FOUND), + ) as mock_read_rocket: + response = client.get('/rockets/123') + assert response.status_code == 404 + assert response.json() == {'detail': 'Not Found'} + mock_read_rocket.assert_called_once_with('123') + + +def test_read_rocket_server_error(): + with patch.object( + RocketController, + 'get_rocket_by_id', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.get('/rockets/123') + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} + + +def test_update_rocket(stub_rocket): + with patch.object( + RocketController, + 'update_rocket_by_id', + return_value=RocketUpdated(rocket_id='123'), + ) as mock_update_rocket: + with patch.object( + Motor, 'set_motor_kind', side_effect=None + ) as mock_set_motor_kind: + response = client.put( + '/rockets/123', + json=stub_rocket, + params={'motor_kind': 'GENERIC'}, + ) + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully updated', + } + mock_update_rocket.assert_called_once_with( + '123', Rocket(**stub_rocket) + ) + mock_set_motor_kind.assert_called_once_with(MotorKinds.GENERIC) + + +def test_update_rocket_invalid_input(): + response = client.put( + '/rockets/123', + json={'mass': 'foo', 'radius': 'bar'}, + params={'motor_kind': 'GENERIC'}, + ) + assert response.status_code == 422 + + +def test_update_rocket_not_found(stub_rocket): + with patch.object( + RocketController, + 'update_rocket_by_id', + side_effect=HTTPException(status_code=status.HTTP_404_NOT_FOUND), + ): + response = client.put( + '/rockets/123', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 404 + assert response.json() == {'detail': 'Not Found'} + + +def test_update_rocket_server_error(stub_rocket): + with patch.object( + RocketController, + 'update_rocket_by_id', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.put( + '/rockets/123', json=stub_rocket, params={'motor_kind': 'HYBRID'} + ) + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} + + +def test_delete_rocket(): + with patch.object( + RocketController, + 'delete_rocket_by_id', + return_value=RocketDeleted(rocket_id='123'), + ) as mock_delete_rocket: + response = client.delete('/rockets/123') + assert response.status_code == 200 + assert response.json() == { + 'rocket_id': '123', + 'message': 'Rocket successfully deleted', + } + mock_delete_rocket.assert_called_once_with('123') + + +def test_delete_rocket_server_error(): + with patch.object( + RocketController, + 'delete_rocket_by_id', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.delete('/rockets/123') + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} + + +def test_simulate_rocket(stub_rocket_summary): + with patch.object( + RocketController, + 'simulate_rocket', + return_value=RocketSummary(**stub_rocket_summary), + ) as mock_simulate_rocket: + response = client.get('/rockets/123/summary') + assert response.status_code == 200 + assert response.json() == stub_rocket_summary + mock_simulate_rocket.assert_called_once_with('123') + + +def test_simulate_rocket_not_found(): + with patch.object( + RocketController, + 'simulate_rocket', + side_effect=HTTPException(status_code=status.HTTP_404_NOT_FOUND), + ) as mock_simulate_rocket: + response = client.get('/rockets/123/summary') + assert response.status_code == 404 + assert response.json() == {'detail': 'Not Found'} + mock_simulate_rocket.assert_called_once_with('123') + + +def test_simulate_rocket_server_error(): + with patch.object( + RocketController, + 'simulate_rocket', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.get('/rockets/123/summary') + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} + + +def test_read_rocketpy_rocket(): + with patch.object( + RocketController, + 'get_rocketpy_rocket_binary', + return_value=b'rocketpy', + ) as mock_read_rocketpy_rocket: + response = client.get('/rockets/123/rocketpy') + assert response.status_code == 203 + assert response.content == b'rocketpy' + assert response.headers['content-type'] == 'application/octet-stream' + mock_read_rocketpy_rocket.assert_called_once_with('123') + + +def test_read_rocketpy_rocket_not_found(): + with patch.object( + RocketController, + 'get_rocketpy_rocket_binary', + side_effect=HTTPException(status_code=status.HTTP_404_NOT_FOUND), + ) as mock_read_rocketpy_rocket: + response = client.get('/rockets/123/rocketpy') + assert response.status_code == 404 + assert response.json() == {'detail': 'Not Found'} + mock_read_rocketpy_rocket.assert_called_once_with('123') + + +def test_read_rocketpy_rocket_server_error(): + with patch.object( + RocketController, + 'get_rocketpy_rocket_binary', + side_effect=HTTPException( + status_code=status.HTTP_500_INTERNAL_SERVER_ERROR + ), + ): + response = client.get('/rockets/123/rocketpy') + assert response.status_code == 500 + assert response.json() == {'detail': 'Internal Server Error'} From 40659608f7e4a8ddc7588431ef4f54affc372d38 Mon Sep 17 00:00:00 2001 From: Gabriel Barberini Date: Sat, 16 Nov 2024 10:50:26 -0300 Subject: [PATCH 3/5] Update pylint.yml ignores R0913, R0917 --- .github/workflows/pylint.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/pylint.yml b/.github/workflows/pylint.yml index 1e76065..e0096f9 100644 --- a/.github/workflows/pylint.yml +++ b/.github/workflows/pylint.yml @@ -21,4 +21,4 @@ jobs: pip install -r requirements.txt - name: Analysing the code with pylint run: | - pylint -d C0200,C0301,C0114,R0903,C0115,W0246,R0914,C0209,E1121,C0103,C2801,R0801,E1101,E0401,E0611,R0911,C0116,W0212,W0719,W0601,W1203,W0123,W0511,W0621 $(git ls-files '*.py') + pylint -d C0200,C0301,C0114,R0903,C0115,W0246,R0914,C0209,E1121,C0103,C2801,R0801,E1101,E0401,E0611,R0911,C0116,W0212,W0719,W0601,W1203,W0123,W0511,W0621,R0913,R0917 $(git ls-files '*.py') From 0b79762b592515f9b0f39c4923c289253de644b7 Mon Sep 17 00:00:00 2001 From: Gabriel Barberini Date: Sat, 16 Nov 2024 10:54:08 -0300 Subject: [PATCH 4/5] updates mermaid flowchart --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index b68d421..8bd0556 100644 --- a/README.md +++ b/README.md @@ -174,7 +174,7 @@ sequenceDiagram Rocketpy lib -->> API: Simulation Results API -->> User: Simulation Results - User ->> API: POST /rocketpy/model/:id + User ->> API: POST /model/:id/rocketpy API -->> MongoDB: Retrieve Rocketpy Model MongoDB -->> API: Rocketpy Model API ->> Rocketpy lib: Rocketpy Model From dcb54b8dae6759e7aae383348e3b85f859149e00 Mon Sep 17 00:00:00 2001 From: Gabriel Barberini Date: Sat, 23 Nov 2024 11:43:07 -0300 Subject: [PATCH 5/5] addresses PR review --- Makefile | 5 +++-- lib/models/aerosurfaces.py | 8 ++++---- pyproject.toml | 5 +++++ tests/test_routes/test_rockets_route.py | 2 +- 4 files changed, 13 insertions(+), 7 deletions(-) diff --git a/Makefile b/Makefile index b3a6294..815840e 100644 --- a/Makefile +++ b/Makefile @@ -1,4 +1,5 @@ -format: black flake8 pylint ruff +format: black ruff +lint: flake8 pylint black: black ./lib || true @@ -10,7 +11,7 @@ flake8: pylint: pylint ./lib || true - pylint --disable=E0401,W0621,R0913,R0917 ./tests || true + pylint ./tests || true ruff: ruff check --fix ./lib || true diff --git a/lib/models/aerosurfaces.py b/lib/models/aerosurfaces.py index aec87f0..68d8e41 100644 --- a/lib/models/aerosurfaces.py +++ b/lib/models/aerosurfaces.py @@ -1,13 +1,13 @@ from enum import Enum from typing import Optional, Tuple, List, Union -from pydantic import BaseModel +from pydantic import BaseModel, Field class Parachute(BaseModel): name: str - cd_s: float - sampling_rate: int - lag: float + cd_s: float = Field(..., ge=0, description="Must be non-negative") + sampling_rate: float = Field(..., gt=0, description="Must be positive") + lag: float = Field(..., ge=0, description="Must be non-negative") trigger: Union[str, float] noise: Tuple[float, float, float] diff --git a/pyproject.toml b/pyproject.toml index 570738d..a341da2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -43,6 +43,11 @@ disable = """ broad-exception-caught, raise-missing-from, too-many-instance-attributes, + redefined-outer-name, + import-error, + too-many-arguments, + redefined-outer-name, + too-many-positional-arguments, """ [tool.ruff] diff --git a/tests/test_routes/test_rockets_route.py b/tests/test_routes/test_rockets_route.py index 2a3afb6..8583e4b 100644 --- a/tests/test_routes/test_rockets_route.py +++ b/tests/test_routes/test_rockets_route.py @@ -162,7 +162,7 @@ def stub_parachute(): parachute = Parachute( name='parachute', cd_s=0, - sampling_rate=0, + sampling_rate=1, lag=0, trigger='trigger', noise=(0, 0, 0),