From fbcedff2b3ad98e9d717842afc4d7c474e5cf19c Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 5 Jun 2024 16:27:18 -0400 Subject: [PATCH 01/21] Added configs --- onair/config/onair_pybullet_config.ini | 14 ++++++++ .../pybullet_tlm_config.json | 35 +++++++++++++++++++ 2 files changed, 49 insertions(+) create mode 100644 onair/config/onair_pybullet_config.ini create mode 100644 onair/data/telemetry_configs/pybullet_tlm_config.json diff --git a/onair/config/onair_pybullet_config.ini b/onair/config/onair_pybullet_config.ini new file mode 100644 index 00000000..0f9e8f68 --- /dev/null +++ b/onair/config/onair_pybullet_config.ini @@ -0,0 +1,14 @@ +[DEFAULT] +TelemetryDataFilePath = onair/data/raw_telemetry_data/data_physics_generation/Errors +TelemetryFile = 700_crash_to_earth_1.csv +TelemetryMetadataFilePath = OnAIR/onair/data/telemetry_configs/ +MetaFile = pybullet_tlm_config.json +ParserFileName = OnAIR/onair/data_handling/redis_adapter.py + +KnowledgeRepPluginDict = {'KAST':'KAST/__init__.py'} +LearnersPluginDict = {'learner':'OnAIR/plugins/generic/__init__.py'} +PlannersPluginDict = {'planner':'OnAIR/plugins/generic/__init__.py'} +ComplexPluginDict = {'complex':'OnAIR/plugins/drone_state_machine/__init__.py'} + +[RUN_FLAGS] +IO_Flag = true \ No newline at end of file diff --git a/onair/data/telemetry_configs/pybullet_tlm_config.json b/onair/data/telemetry_configs/pybullet_tlm_config.json new file mode 100644 index 00000000..71c5f39b --- /dev/null +++ b/onair/data/telemetry_configs/pybullet_tlm_config.json @@ -0,0 +1,35 @@ +{ + "subsystems": { + "STATES": { + "time": { + "description": "Pybullet drone 0 state" + }, + "sim_drone_0_state.pose": { + "description": "Pybullet drone 0 state" + }, + "sim_drone_0_state.rpms": { + "description": "Pybullet drone 1 state" + }, + "sim_drone_0_state.rpy": { + "description": "Pybullet drone 1 state" + }, + "sim_drone_0_state.vel": { + "description": "Pybullet drone 1 state" + }, + "sim_drone_0_state.ang_vel": { + "description": "Pybullet drone 1 state" + } + } + }, + "redis_subscriptions": [ + "sim_drone_0_state" + ], + "order": [ + "time", + "sim_drone_0_state.pose", + "sim_drone_0_state.rpms", + "sim_drone_0_state.rpy", + "sim_drone_0_state.vel", + "sim_drone_0_state.ang_vel" + ] +} From 99657aae33c0ed89eb2c583c52f9cffe5d69fbb1 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 5 Jun 2024 16:27:29 -0400 Subject: [PATCH 02/21] Added drone complex controller --- plugins/drone_state_machine/__init__.py | 0 .../drone_state_machine_plugin.py | 72 +++++++++++++++++++ 2 files changed, 72 insertions(+) create mode 100644 plugins/drone_state_machine/__init__.py create mode 100644 plugins/drone_state_machine/drone_state_machine_plugin.py diff --git a/plugins/drone_state_machine/__init__.py b/plugins/drone_state_machine/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/plugins/drone_state_machine/drone_state_machine_plugin.py b/plugins/drone_state_machine/drone_state_machine_plugin.py new file mode 100644 index 00000000..9d801fab --- /dev/null +++ b/plugins/drone_state_machine/drone_state_machine_plugin.py @@ -0,0 +1,72 @@ +# GSC-19165-1, "The On-Board Artificial Intelligence Research (OnAIR) Platform" +# +# Copyright © 2023 United States Government as represented by the Administrator of +# the National Aeronautics and Space Administration. No copyright is claimed in the +# United States under Title 17, U.S. Code. All Other Rights Reserved. +# +# Licensed under the NASA Open Source Agreement version 1.3 +# See "NOSA GSC-19165-1 OnAIR.pdf" + +import numpy as np +import redis +from time import sleep +import json + +from onair.src.ai_components.ai_plugin_abstract.ai_plugin import AIPlugin + +TARGET_ALT = 5 # What altitude should the drone maintain during operations? +POSITION_TOLERANCE = 0.2 # How close must the drone get to a target location for operations purposes? +HOME_COORDS = [0.0, 0.0] # Where is the drone's home coordinates? May be different than origin in some cases +DRONE_NUMBER = 1 # Which drone is this? Used to publish commands to the right channel + +def xy_position_difference(pos1, pos2): # Calculate the difference between XY positions, used for position tolerance comparisons + return np.sqrt((pos1[0] - pos2[0])**2 + (pos1[1] - pos2[1])**2) + +class Plugin(AIPlugin): + def __init__(self, name, headers): + """ + Initialize Redis connection and set target altitude, position tolerance, and home coordinates + """ + # Redis initialization + super().__init__(name, headers) + pool = redis.ConnectionPool(host="localhost", port=6379, password="") + self.r = redis.Redis(connection_pool=pool, charset="utf-8", decode_responses=True) + self.pub_channel = 'mavlink_cmd' + + + self.target_alt = TARGET_ALT * -1 + self.home = HOME_COORDS + + self.state = 'standby' # Initial drone operating state + + self.target_position = None + + def send_cmd(self, cmd): # Command publishing to Redis + serialized_cmd = json.dumps(cmd) + self.r.publish(f'edp_drone_{DRONE_NUMBER}_command', serialized_cmd) + + def update(self,low_level_data=[], high_level_data={}): + """ + Given streamed data point, system should update internally + """ + print(f'\nCOMPLEX REASONER sees high level data as: {high_level_data}') + self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] + pass + + def render_reasoning(self): + """ + Basic state machine implemented to take off, move, pause, return, and land. + """ + rnd_xy = np.random.uniform(-5,5, size=2) + + if self.target_position == None: + self.target_position = [rnd_xy[0], rnd_xy[1], 0.5] # XYZ + self.send_cmd(self.target_position) + else: + position_err = xy_position_difference(self.drone_position, self.target_position) + if position_err < POSITION_TOLERANCE: + self.target_position = [rnd_xy[0], rnd_xy[1], 1.0] + self.send_cmd(self.target_position) + else: + print(f'Traveling to new target position {self.target_position}') + From e5e4816bd318c95c4316f740faf45a8f7e582faf Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Fri, 14 Jun 2024 17:19:01 -0400 Subject: [PATCH 03/21] Removed KAST from pybullet_config.ini and removed the KAST usage from drone_state_machine_plugin.py on line 53 by replacing it with a coordinate (0,0,0) temporarily. --- onair/config/onair_pybullet_config.ini | 2 +- plugins/drone_state_machine/drone_state_machine_plugin.py | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/onair/config/onair_pybullet_config.ini b/onair/config/onair_pybullet_config.ini index 0f9e8f68..041b21c8 100644 --- a/onair/config/onair_pybullet_config.ini +++ b/onair/config/onair_pybullet_config.ini @@ -5,7 +5,7 @@ TelemetryMetadataFilePath = OnAIR/onair/data/telemetry_configs/ MetaFile = pybullet_tlm_config.json ParserFileName = OnAIR/onair/data_handling/redis_adapter.py -KnowledgeRepPluginDict = {'KAST':'KAST/__init__.py'} +KnowledgeRepPluginDict = {'knowledge':'OnAIR/plugins/generic/__init__.py'} LearnersPluginDict = {'learner':'OnAIR/plugins/generic/__init__.py'} PlannersPluginDict = {'planner':'OnAIR/plugins/generic/__init__.py'} ComplexPluginDict = {'complex':'OnAIR/plugins/drone_state_machine/__init__.py'} diff --git a/plugins/drone_state_machine/drone_state_machine_plugin.py b/plugins/drone_state_machine/drone_state_machine_plugin.py index 9d801fab..cc906c25 100644 --- a/plugins/drone_state_machine/drone_state_machine_plugin.py +++ b/plugins/drone_state_machine/drone_state_machine_plugin.py @@ -50,7 +50,8 @@ def update(self,low_level_data=[], high_level_data={}): Given streamed data point, system should update internally """ print(f'\nCOMPLEX REASONER sees high level data as: {high_level_data}') - self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] + # self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] + self.drone_position = (0, 0, 0) pass def render_reasoning(self): From 3acd4bea3f5304957b6a0b05a684e3fac835bfab Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Mon, 17 Jun 2024 14:51:19 -0400 Subject: [PATCH 04/21] Made changes to remove KAST and still have drone_state_machine_plugin.py working --- onair/config/onair_pybullet_config.ini | 2 +- plugins/drone_state_machine/drone_state_machine_plugin.py | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/onair/config/onair_pybullet_config.ini b/onair/config/onair_pybullet_config.ini index 0f9e8f68..041b21c8 100644 --- a/onair/config/onair_pybullet_config.ini +++ b/onair/config/onair_pybullet_config.ini @@ -5,7 +5,7 @@ TelemetryMetadataFilePath = OnAIR/onair/data/telemetry_configs/ MetaFile = pybullet_tlm_config.json ParserFileName = OnAIR/onair/data_handling/redis_adapter.py -KnowledgeRepPluginDict = {'KAST':'KAST/__init__.py'} +KnowledgeRepPluginDict = {'knowledge':'OnAIR/plugins/generic/__init__.py'} LearnersPluginDict = {'learner':'OnAIR/plugins/generic/__init__.py'} PlannersPluginDict = {'planner':'OnAIR/plugins/generic/__init__.py'} ComplexPluginDict = {'complex':'OnAIR/plugins/drone_state_machine/__init__.py'} diff --git a/plugins/drone_state_machine/drone_state_machine_plugin.py b/plugins/drone_state_machine/drone_state_machine_plugin.py index 9d801fab..28d73ece 100644 --- a/plugins/drone_state_machine/drone_state_machine_plugin.py +++ b/plugins/drone_state_machine/drone_state_machine_plugin.py @@ -50,7 +50,9 @@ def update(self,low_level_data=[], high_level_data={}): Given streamed data point, system should update internally """ print(f'\nCOMPLEX REASONER sees high level data as: {high_level_data}') - self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] + #self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] + #Temporary change to remove the use of KAST, during initial testing for SPAR intern project + self.drone_position = (0,0,0) pass def render_reasoning(self): From 06f6631a394ad896c997370bf5901aca39008db5 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Tue, 18 Jun 2024 10:40:07 -0400 Subject: [PATCH 05/21] Fixed default so it works --- onair/config/default_config.ini | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/onair/config/default_config.ini b/onair/config/default_config.ini index 81a4f518..e4d8dffd 100644 --- a/onair/config/default_config.ini +++ b/onair/config/default_config.ini @@ -1,27 +1,27 @@ # Required Section: DEFAULT section is the basic required elements for running OnAIR [DEFAULT] # Required Key: TelemetryDataFilePath is the path for TelemetryFile -TelemetryDataFilePath = onair/data/raw_telemetry_data/data_physics_generation/Errors +TelemetryDataFilePath = OnAIR/onair/data/raw_telemetry_data/data_physics_generation/Errors # Required Key: TelemetryDataFile is a file used by the selected parser # NOTE: TelemetryDataFile is required even when selected parser does not use it TelemetryFile = 700_crash_to_earth_1.csv # Required Key: TelemetryMetadataFilePath is the path for TelemetryMetadataFilePath -TelemetryMetadataFilePath = onair/data/telemetry_configs/ +TelemetryMetadataFilePath = OnAIR/onair/data/telemetry_configs/ # Required Key: MetaFile describes frame composition of data MetaFile = data_physics_generation_CONFIG.json # Required Key: ParserFileName is the name of the parser DataSource object to use -ParserFileName = onair/data_handling/csv_parser.py +ParserFileName = OnAIR/onair/data_handling/csv_parser.py # Plugins # NOTE: even though keys are required, they may be set to empty dicts # Required Key: KnowledgeRepPluginDict(s) are used by the VehicleRep -KnowledgeRepPluginDict = {'generic':'plugins/generic/__init__.py'} +KnowledgeRepPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} # Required Key: LearnersPluginDict(s) are used by Agent for learning -LearnersPluginDict = {'generic':'plugins/generic/__init__.py'} +LearnersPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} # Required Key: LearnersPluginDict(s) are used by Agent for planning -PlannersPluginDict = {'generic':'plugins/generic/__init__.py'} +PlannersPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} # Required Key: ComplexPluginDict(s) are used by Agent for complex reasoning -ComplexPluginDict = {'generic':'plugins/generic/__init__.py'} +ComplexPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} # Required Section: RUN_FLAGS are settable values to change running experience [RUN_FLAGS] From d8d152838ab2f6a8dd03aeaf144f423bef4ed8ea Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Thu, 20 Jun 2024 13:49:51 -0400 Subject: [PATCH 06/21] Confirmed you can add redis configurations to meta data file parsing with no breaking to local redis hosting defaults --- .../redis_example_CONFIG.json | 4 +++- onair/data_handling/redis_adapter.py | 19 ++++++++++++++++--- 2 files changed, 19 insertions(+), 4 deletions(-) diff --git a/onair/data/telemetry_configs/redis_example_CONFIG.json b/onair/data/telemetry_configs/redis_example_CONFIG.json index 1813570c..33adffa9 100644 --- a/onair/data/telemetry_configs/redis_example_CONFIG.json +++ b/onair/data/telemetry_configs/redis_example_CONFIG.json @@ -28,7 +28,9 @@ "state_0", "state_1", "state_2" - ], + ], + "address": "localhost", + "port": 6379, "order": [ "time", "state_0.x", diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index ba346004..ea8cc157 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -29,8 +29,6 @@ class DataSource(OnAirDataSource): def __init__(self, data_file, meta_file, ss_breakdown = False): super().__init__(data_file, meta_file, ss_breakdown) - self.address = 'localhost' - self.port = 6379 self.db = 0 self.server = None self.new_data_lock = threading.Lock() @@ -47,7 +45,7 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): def connect(self): """Establish connection to REDIS server.""" print_msg('Redis adapter connecting to server...') - self.server = redis.Redis(self.address, self.port, self.db) + self.server = redis.Redis(self.address, self.port, self.db, password=self.password) if self.server.ping(): print_msg('... connected!') @@ -72,6 +70,21 @@ def parse_meta_data_file(self, meta_data_file, ss_breakdown): meta = parseJson(meta_data_file) keys = meta.keys() + if 'address' in keys: + self.address = meta['address'] + else: + self.address = 'localhost' + + if 'port' in keys: + self.port = meta['port'] + else: + self.port = 6379 + + if 'password' in keys: + self.password = meta['password'] + else: + self.password = '' + if 'order' in keys: self.order = meta['order'] else: From bbbb0ebea28641d60de2e51491dc63b3ce163e69 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Thu, 20 Jun 2024 14:15:57 -0400 Subject: [PATCH 07/21] Add unit test --- .../onair/data_handling/test_redis_adapter.py | 46 ++++++++++++++++--- 1 file changed, 40 insertions(+), 6 deletions(-) diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index 777dfbb3..dfc74150 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -20,8 +20,6 @@ # __init__ tests def test_redis_adapter_DataSource__init__sets_redis_values_then_connects_and_subscribes_to_subscriptions(mocker): # Arrange - expected_address = 'localhost' - expected_port = 6379 expected_db = 0 expected_server = None expected_subscriptions = MagicMock() @@ -50,8 +48,6 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects_and_sub # Assert assert OnAirDataSource.__init__.call_count == 1 assert OnAirDataSource.__init__.call_args_list[0].args == (arg_data_file, arg_meta_file, arg_ss_breakdown) - assert cut.address == expected_address - assert cut.port == expected_port assert cut.db == expected_db assert cut.server == expected_server assert cut.new_data_lock == fake_new_data_lock @@ -72,6 +68,7 @@ def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_at expected_address = MagicMock() expected_port = MagicMock() expected_db = MagicMock() + expected_password = MagicMock() fake_server = MagicMock() cut = DataSource.__new__(DataSource) @@ -89,7 +86,7 @@ def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_at assert redis_adapter.print_msg.call_count == 2 assert redis_adapter.print_msg.call_args_list[0].args == ('Redis adapter connecting to server...',) assert redis.Redis.call_count == 1 - assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db) + assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password) assert fake_server.ping.call_count == 1 assert redis_adapter.print_msg.call_args_list[1].args == ('... connected!',) assert cut.server == fake_server @@ -99,6 +96,7 @@ def test_redis_adapter_DataSource_fails_to_connect_to_server(mocker): expected_address = MagicMock() expected_port = MagicMock() expected_db = MagicMock() + expected_password = MagicMock() fake_server = MagicMock() cut = DataSource.__new__(DataSource) @@ -117,7 +115,7 @@ def test_redis_adapter_DataSource_fails_to_connect_to_server(mocker): assert redis_adapter.print_msg.call_count == 1 assert redis_adapter.print_msg.call_args_list[0].args == ("Redis adapter connecting to server...",) assert redis.Redis.call_count == 1 - assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db) + assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password) assert fake_server.ping.call_count == 1 assert cut.server == fake_server @@ -628,6 +626,42 @@ def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_m assert cut.subscriptions == expected_subscriptions assert result == expected_extracted_configs +def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_for_redis_server_configurations(mocker): + # Arrange + cut = DataSource.__new__(DataSource) + arg_configFile = MagicMock() + arg_ss_breakdown = MagicMock() + + + expected_extracted_configs = MagicMock() + expected_subscriptions = [MagicMock()] * pytest.gen.randint(0, 10) # 0 to 10 arbitrary + expected_address = MagicMock() + expected_port = MagicMock() + expected_password = MagicMock() + fake_meta = {'fake_other_stuff': MagicMock(), + 'order': MagicMock(), + 'redis_subscriptions': expected_subscriptions, + 'address': expected_address, + 'port': expected_port, + 'password': expected_password} + + mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) + mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) + + # Act + result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) + + # Assert + assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 + assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) + assert redis_adapter.parseJson.call_count == 1 + assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) + assert cut.subscriptions == expected_subscriptions + assert result == expected_extracted_configs + assert cut.address == expected_address + assert cut.port == expected_port + assert cut.password == expected_password + def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown_and_sets_subscriptions_to_empty_when_none_given(mocker): # Arrange cut = DataSource.__new__(DataSource) From d7e7d1a8b536b599778163c78be911ae1b27ff14 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Thu, 20 Jun 2024 14:21:05 -0400 Subject: [PATCH 08/21] Add password to unit tests --- test/onair/data_handling/test_redis_adapter.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index dfc74150..80c484bc 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -75,6 +75,7 @@ def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_at cut.address = expected_address cut.port = expected_port cut.db = expected_db + cut.password = expected_password mocker.patch(redis_adapter.__name__ + '.print_msg') mocker.patch('redis.Redis', return_value=fake_server) @@ -103,6 +104,7 @@ def test_redis_adapter_DataSource_fails_to_connect_to_server(mocker): cut.address = expected_address cut.port = expected_port cut.db = expected_db + cut.password = expected_password mocker.patch(redis_adapter.__name__ + '.print_msg') mocker.patch('redis.Redis', return_value=fake_server) From 097369404153292dd77082163863d2399af79190 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Thu, 20 Jun 2024 14:30:40 -0400 Subject: [PATCH 09/21] Removed argument name --- onair/data_handling/redis_adapter.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index ea8cc157..e737d9aa 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -45,7 +45,7 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): def connect(self): """Establish connection to REDIS server.""" print_msg('Redis adapter connecting to server...') - self.server = redis.Redis(self.address, self.port, self.db, password=self.password) + self.server = redis.Redis(self.address, self.port, self.db, self.password) if self.server.ping(): print_msg('... connected!') From 35237b399c5b03b06ef4f510950e1d0811354799 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Fri, 21 Jun 2024 12:45:46 -0400 Subject: [PATCH 10/21] Updated config to have a neater dictionary of redis server configs + handle those configs --- .../redis_example_CONFIG.json | 6 ++-- onair/data_handling/redis_adapter.py | 29 ++++++++++--------- 2 files changed, 20 insertions(+), 15 deletions(-) diff --git a/onair/data/telemetry_configs/redis_example_CONFIG.json b/onair/data/telemetry_configs/redis_example_CONFIG.json index 33adffa9..6a87a7c9 100644 --- a/onair/data/telemetry_configs/redis_example_CONFIG.json +++ b/onair/data/telemetry_configs/redis_example_CONFIG.json @@ -29,8 +29,10 @@ "state_1", "state_2" ], - "address": "localhost", - "port": 6379, + "redis" : { + "address": "localhost", + "port": 6379 + }, "order": [ "time", "state_0.x", diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index e737d9aa..65773ae0 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -70,20 +70,23 @@ def parse_meta_data_file(self, meta_data_file, ss_breakdown): meta = parseJson(meta_data_file) keys = meta.keys() - if 'address' in keys: - self.address = meta['address'] - else: - self.address = 'localhost' - - if 'port' in keys: - self.port = meta['port'] - else: - self.port = 6379 + # Setup redis server configuration + if 'redis' in keys: + redis_config_keys = meta['redis'].keys() + if 'address' in redis_config_keys: + self.address = meta['redis']['address'] + else: + self.address = 'localhost' - if 'password' in keys: - self.password = meta['password'] - else: - self.password = '' + if 'port' in redis_config_keys: + self.port = meta['redis']['port'] + else: + self.port = 6379 + + if 'password' in redis_config_keys: + self.password = meta['redis']['password'] + else: + self.password = '' if 'order' in keys: self.order = meta['order'] From d4b5fed1cc469360de362d706d94950e471fa392 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Fri, 21 Jun 2024 12:45:55 -0400 Subject: [PATCH 11/21] Updated unit test --- test/onair/data_handling/test_redis_adapter.py | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index 80c484bc..ff249a83 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -640,12 +640,11 @@ def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_m expected_address = MagicMock() expected_port = MagicMock() expected_password = MagicMock() + expected_redis_configs = {'address': expected_address, 'port': expected_port, 'password': expected_password} fake_meta = {'fake_other_stuff': MagicMock(), 'order': MagicMock(), 'redis_subscriptions': expected_subscriptions, - 'address': expected_address, - 'port': expected_port, - 'password': expected_password} + 'redis': expected_redis_configs} mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) From a5f4eaa22881636c6a14ce9407d025f9fd33e1f8 Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Wed, 26 Jun 2024 11:10:37 -0400 Subject: [PATCH 12/21] Modified Redis adapter to work with multiple servers --- .../redis_example_CONFIG.json | 28 ++++-- onair/data_handling/redis_adapter.py | 98 +++++++++++-------- 2 files changed, 76 insertions(+), 50 deletions(-) diff --git a/onair/data/telemetry_configs/redis_example_CONFIG.json b/onair/data/telemetry_configs/redis_example_CONFIG.json index 6a87a7c9..7c253e9c 100644 --- a/onair/data/telemetry_configs/redis_example_CONFIG.json +++ b/onair/data/telemetry_configs/redis_example_CONFIG.json @@ -24,15 +24,25 @@ } } }, - "redis_subscriptions": [ - "state_0", - "state_1", - "state_2" - ], - "redis" : { - "address": "localhost", - "port": 6379 - }, + + "redis" : [ + { + "address": "localhost", + "port": 6379, + "subscriptions": [ + "state_0" + ] + }, + { + "address": "localhost", + "port": 6380, + "subscriptions": [ + "state_1", + "state_2" + ] + } + ], + "order": [ "time", "state_0.x", diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index 65773ae0..4d225a57 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -29,10 +29,9 @@ class DataSource(OnAirDataSource): def __init__(self, data_file, meta_file, ss_breakdown = False): super().__init__(data_file, meta_file, ss_breakdown) - self.db = 0 - self.server = None self.new_data_lock = threading.Lock() self.new_data = False + self.servers = [] self.currentData = [] self.currentData.append({'headers':self.order, 'data':list('-' * len(self.order))}) @@ -40,65 +39,82 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): 'data':list('-' * len(self.order))}) self.double_buffer_read_index = 0 self.connect() - self.subscribe(self.subscriptions) def connect(self): """Establish connection to REDIS server.""" print_msg('Redis adapter connecting to server...') - self.server = redis.Redis(self.address, self.port, self.db, self.password) + for idx, server_config in enumerate(self.server_configs): + server_config_keys = server_config.keys() + if 'address' in server_config_keys: + address = server_config['address'] + else: + address = 'localhost' + + if 'port' in server_config_keys: + port = server_config['port'] + else: + port = 6379 + + if 'db' in server_config_keys: + db = server_config['db'] + else: + db = 0 - if self.server.ping(): - print_msg('... connected!') + if 'password' in server_config_keys: + password = server_config['password'] + else: + password = '' - def subscribe(self, subscriptions): - """Subscribe to REDIS message channel(s) and launch listener thread.""" - if len(subscriptions) != 0 and self.server.ping(): - self.pubsub = self.server.pubsub() + self.servers.append(redis.Redis(address, port, db, password)) - for s in subscriptions: - self.pubsub.subscribe(s) - print_msg(f"Subscribing to channel: {s}") + if self.servers[-1].ping(): + print_msg(f'... connected to server # {idx}!') + else: + print_msg(f'Did not connect to server # {idx}', 'RED') + + #if there are subscriptions in this Redis server configuration's subscription key + if len(server_config['subscriptions']) !=0: + #Set up Redis pubsub function for the current server + pubsub = self.servers[-1].pubsub() + + for s in server_config['subscriptions']: + pubsub.subscribe(s) + print_msg(f"Subscribing to channel: {s} on server # {idx}") + + listen_thread = threading.Thread(target=self.message_listener, args=(pubsub,)) + listen_thread.start() + else: + print_msg(f"No subscriptions given!") - listen_thread = threading.Thread(target=self.message_listener) - listen_thread.start() - else: - print_msg(f"No subscriptions given!") def parse_meta_data_file(self, meta_data_file, ss_breakdown): + self.server_configs = [] configs = extract_meta_data_handle_ss_breakdown( meta_data_file, ss_breakdown) meta = parseJson(meta_data_file) keys = meta.keys() # Setup redis server configuration + #Checking in 'redis' exists if 'redis' in keys: - redis_config_keys = meta['redis'].keys() - if 'address' in redis_config_keys: - self.address = meta['redis']['address'] - else: - self.address = 'localhost' - - if 'port' in redis_config_keys: - self.port = meta['redis']['port'] - else: - self.port = 6379 - - if 'password' in redis_config_keys: - self.password = meta['redis']['password'] - else: - self.password = '' + count_server_config = 0 + #Checking if dictionaries within 'redis' key each have a 'subscription' key. Error will be thrown if not. + for server_config in meta['redis']: + redis_config_keys = server_config.keys() + if 'subscriptions' in redis_config_keys == False: + raise ConfigKeyError(f'Config file: \'{meta_data_file}\' ' \ + f'missing required key \'suscriptions\' from {count_server_config} in key \'redis\' ') + count_server_config +=1 + + #Saving all of Redis dictionaries from JSON file to self.server_configs + self.server_configs = meta['redis'] if 'order' in keys: self.order = meta['order'] else: raise ConfigKeyError(f'Config file: \'{meta_data_file}\' ' \ - 'missing required key \'order\'') - - if 'redis_subscriptions' in meta.keys(): - self.subscriptions = meta['redis_subscriptions'] - else: - self.subscriptions = [] - + 'missing required key \'order\'') + return configs def process_data_file(self, data_file): @@ -131,9 +147,9 @@ def has_more(self): """Live connection should always return True""" return True - def message_listener(self): + def message_listener(self, pubsub): """Loop for listening for messages on channels""" - for message in self.pubsub.listen(): + for message in pubsub.listen(): if message['type'] == 'message': channel_name = f"{message['channel'].decode()}" # Attempt to load message as json From bc285766c562f7d39b1e9011db587ed22a7d9a9e Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Thu, 27 Jun 2024 12:52:09 -0400 Subject: [PATCH 13/21] Adding multi-server experiment --- redis-experiment-publisher-multi-server.py | 70 ++++++++++++++++++++++ 1 file changed, 70 insertions(+) create mode 100644 redis-experiment-publisher-multi-server.py diff --git a/redis-experiment-publisher-multi-server.py b/redis-experiment-publisher-multi-server.py new file mode 100644 index 00000000..7c20576d --- /dev/null +++ b/redis-experiment-publisher-multi-server.py @@ -0,0 +1,70 @@ +import redis +import time +import random + +# Initialize the Redis connection for server #1 +redis_host = "localhost" +redis_port = 6379 +# When your Redis server requires a password, fill it in here +redis_password = "" +# Connect to Redis +r1 = redis.Redis(host=redis_host, + port=redis_port, + password=redis_password, + decode_responses=True) + +# Initialize the Redis connection for server #1 +redis_host = "localhost" +redis_port = 6380 +# When your Redis server requires a password, fill it in here +redis_password = "" +# Connect to Redis +r2 = redis.Redis(host=redis_host, + port=redis_port, + password=redis_password, + decode_responses=True) + +# List of channel names +server1_channels = ['state_0'] +server2_channels = ['state_1', 'state_2'] +# Publish messages on each channel in random order +def publish_messages(): + loop_count = 0 + inner_loop_count = 0 + max_loops = 9 + while loop_count < max_loops: + random.shuffle(server1_channels) + for channel in server1_channels: + r1.publish(channel, f'{{"time":{inner_loop_count}, ' \ + f'"x":{inner_loop_count+0.1}, ' \ + f'"y":{inner_loop_count+0.2}}}') + + print(f"Published data to {channel}, " \ + f"[{inner_loop_count}, " \ + f"{inner_loop_count+0.1}, " \ + f"{inner_loop_count+0.2}]") + + inner_loop_count += 1 + time.sleep(2) + + random.shuffle(server2_channels) + for channel in server2_channels: + r2.publish(channel, f'{{"time":{inner_loop_count}, ' \ + f'"x":{inner_loop_count+0.1}, ' \ + f'"y":{inner_loop_count+0.2}}}') + + print(f"Published data to {channel}, " \ + f"[{inner_loop_count}, " \ + f"{inner_loop_count+0.1}, " \ + f"{inner_loop_count+0.2}]") + + inner_loop_count += 1 + time.sleep(2) + + loop_count += 1 + print(f"Completed {loop_count} loops") + + + +if __name__ == "__main__": + publish_messages() From 93b35332ad23e6e05f64c67a249e381606831be3 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Tue, 16 Jul 2024 10:53:37 -0400 Subject: [PATCH 14/21] Allow access to OnAir from different directories --- __init__.py | 4 ++++ 1 file changed, 4 insertions(+) create mode 100644 __init__.py diff --git a/__init__.py b/__init__.py new file mode 100644 index 00000000..1a164f55 --- /dev/null +++ b/__init__.py @@ -0,0 +1,4 @@ +import sys +import os + +sys.path.append(os.path.abspath(os.path.dirname(__file__))) \ No newline at end of file From e6d8abbf5368aeee67f6e67eadd38a3db37cdbd2 Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Tue, 30 Jul 2024 15:31:18 -0400 Subject: [PATCH 15/21] Updates to redis_adapter unit test and to redis_adapter.py to have subscriptions only be set up if server ping was successful. --- onair/data_handling/redis_adapter.py | 49 ++++--- .../onair/data_handling/test_redis_adapter.py | 125 +++++++++++++----- 2 files changed, 120 insertions(+), 54 deletions(-) diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index 4d225a57..9af718e0 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -41,8 +41,11 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): self.connect() def connect(self): + num_redis_print_msg_calls = 0 """Establish connection to REDIS server.""" + print("Here") print_msg('Redis adapter connecting to server...') + num_redis_print_msg_calls += 1 for idx, server_config in enumerate(self.server_configs): server_config_keys = server_config.keys() if 'address' in server_config_keys: @@ -66,26 +69,34 @@ def connect(self): password = '' self.servers.append(redis.Redis(address, port, db, password)) - - if self.servers[-1].ping(): + print("Appended to servers") + + try: + print("In try") + self.servers[-1].ping() print_msg(f'... connected to server # {idx}!') - else: - print_msg(f'Did not connect to server # {idx}', 'RED') - - #if there are subscriptions in this Redis server configuration's subscription key - if len(server_config['subscriptions']) !=0: - #Set up Redis pubsub function for the current server - pubsub = self.servers[-1].pubsub() - - for s in server_config['subscriptions']: - pubsub.subscribe(s) - print_msg(f"Subscribing to channel: {s} on server # {idx}") - - listen_thread = threading.Thread(target=self.message_listener, args=(pubsub,)) - listen_thread.start() - else: - print_msg(f"No subscriptions given!") - + num_redis_print_msg_calls += 1 + + #if there are subscriptions in this Redis server configuration's subscription key + if len(server_config['subscriptions']) !=0: + #Set up Redis pubsub function for the current server + pubsub = self.servers[-1].pubsub() + + for s in server_config['subscriptions']: + pubsub.subscribe(s) + print_msg(f"Subscribing to channel: {s} on server # {idx}") + num_redis_print_msg_calls += 1 + listen_thread = threading.Thread(target=self.message_listener, args=(pubsub,)) + listen_thread.start() + else: + print_msg(f"No subscriptions given!") + num_redis_print_msg_calls += 1 + + except: + print_msg(f'Did not connect to server # {idx}. Not setting up subscriptions.', 'RED') + num_redis_print_msg_calls += 1 + + print("num_redis_print_msg_calls: ", num_redis_print_msg_calls) def parse_meta_data_file(self, meta_data_file, ss_breakdown): self.server_configs = [] diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index ff249a83..5b577795 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -18,10 +18,9 @@ import threading # __init__ tests -def test_redis_adapter_DataSource__init__sets_redis_values_then_connects_and_subscribes_to_subscriptions(mocker): +def test_redis_adapter_DataSource__init__sets_redis_values_then_connects(mocker): # Arrange - expected_db = 0 - expected_server = None + expected_server = [] expected_subscriptions = MagicMock() arg_data_file = MagicMock() @@ -40,17 +39,17 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects_and_sub mocker.patch.object(OnAirDataSource, '__init__', new=MagicMock()) mocker.patch('threading.Lock', return_value=fake_new_data_lock) mocker.patch.object(cut, 'connect') - mocker.patch.object(cut, 'subscribe') # Act cut.__init__(arg_data_file, arg_meta_file, arg_ss_breakdown) + #TO CHECK: Anything you mocked, anything you changed, and what your test name is looking for # Assert assert OnAirDataSource.__init__.call_count == 1 assert OnAirDataSource.__init__.call_args_list[0].args == (arg_data_file, arg_meta_file, arg_ss_breakdown) - assert cut.db == expected_db - assert cut.server == expected_server + assert cut.servers == expected_server assert cut.new_data_lock == fake_new_data_lock + assert threading.Lock.call_count == 1 assert cut.new_data == False assert cut.currentData == [{'headers':fake_order, 'data':list('-' * len(fake_order))}, @@ -59,67 +58,123 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects_and_sub assert cut.double_buffer_read_index == 0 assert cut.connect.call_count == 1 assert cut.connect.call_args_list[0].args == () - assert cut.subscribe.call_count == 1 - assert cut.subscribe.call_args_list[0].args == (expected_subscriptions, ) # connect tests def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_attributes(mocker): # Arrange - expected_address = MagicMock() - expected_port = MagicMock() - expected_db = MagicMock() - expected_password = MagicMock() + expected_db = 0 + expected_password = '' + #TODO: Run in a loop + fake_server_configs = [{"address": MagicMock(), "port": 1234,"db": 1, "password": 'test', "subscriptions": ["state_0", "state_1"]}, {"address": '000.000.000.222', "port": 5678, "db": 2, "password": 'test2', "subscriptions" : ["state_2", "state_3"]}] + fake_server = MagicMock() - cut = DataSource.__new__(DataSource) - cut.address = expected_address - cut.port = expected_port - cut.db = expected_db - cut.password = expected_password + fake_message_listener = MagicMock() + fake_listen_thread = MagicMock() + cut = DataSource.__new__(DataSource) + cut.server_configs = fake_server_configs + cut.servers = [] + cut.message_listener = fake_message_listener + + mocker.patch(redis_adapter.__name__ + '.print_msg') mocker.patch('redis.Redis', return_value=fake_server) + mocker.patch.object(fake_server, 'ping') + mocker.patch('threading.Thread', return_value=fake_listen_thread) + mocker.patch.object(fake_listen_thread, 'start') # Act cut.connect() # Assert - assert redis_adapter.print_msg.call_count == 2 + assert redis_adapter.print_msg.call_count == 7 assert redis_adapter.print_msg.call_args_list[0].args == ('Redis adapter connecting to server...',) - assert redis.Redis.call_count == 1 - assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password) - assert fake_server.ping.call_count == 1 - assert redis_adapter.print_msg.call_args_list[1].args == ('... connected!',) - assert cut.server == fake_server + assert redis_adapter.print_msg.call_args_list[1].args == ('... connected to server # 0!',) + assert redis_adapter.print_msg.call_args_list[2].args == ('Subscribing to channel: state_0 on server # 0',) + assert redis_adapter.print_msg.call_args_list[3].args == ('Subscribing to channel: state_1 on server # 0',) + assert redis_adapter.print_msg.call_args_list[4].args == ('... connected to server # 1!',) + assert redis_adapter.print_msg.call_args_list[5].args == ('Subscribing to channel: state_2 on server # 1',) + assert redis_adapter.print_msg.call_args_list[6].args == ('Subscribing to channel: state_3 on server # 1',) + + assert redis.Redis.call_count == 2 + assert redis.Redis.call_args_list[0].args == (fake_server_configs[0]["address"], fake_server_configs[0]["port"], fake_server_configs[0]["db"], fake_server_configs[0]["password"] ) + assert redis.Redis.call_args_list[1].args == (fake_server_configs[1]["address"], fake_server_configs[1]["port"], fake_server_configs[1]["db"], fake_server_configs[1]["password"] ) + + assert fake_server.ping.call_count == 2 + assert cut.servers == [fake_server, fake_server] + +#TODO: Need a test that shows that all of the default vals get used for server config +# connect tests +def test_redis_adapter_DataSource_connect_establishes_server_with_default_attributes(mocker): + # Arrange + expected_address = 'localhost' + expected_port = 6379 + expected_db = 0 + expected_password = '' + + fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions" : ["state_2", "state_3"]}] + + fake_server = MagicMock() + + fake_message_listener = MagicMock() + fake_listen_thread = MagicMock() + + cut = DataSource.__new__(DataSource) + cut.server_configs = fake_server_configs + cut.servers = [] + cut.message_listener = fake_message_listener + + + mocker.patch(redis_adapter.__name__ + '.print_msg') + mocker.patch('redis.Redis', return_value=fake_server) + mocker.patch.object(fake_server, 'ping') + mocker.patch('threading.Thread', return_value=fake_listen_thread) + mocker.patch.object(fake_listen_thread, 'start') + + # Act + cut.connect() + + # Assert + assert redis.Redis.call_count == 2 + assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password ) + assert redis.Redis.call_args_list[1].args == (expected_address, expected_port, expected_db, expected_password ) + def test_redis_adapter_DataSource_fails_to_connect_to_server(mocker): # Arrange - expected_address = MagicMock() - expected_port = MagicMock() - expected_db = MagicMock() - expected_password = MagicMock() + expected_address = 'localhost' + expected_port = 6379 + expected_db = 0 + expected_password = '' + + fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions" : ["state_2", "state_3"]}] fake_server = MagicMock() + fake_message_listener = MagicMock() + fake_listen_thread = MagicMock() + cut = DataSource.__new__(DataSource) - cut.address = expected_address - cut.port = expected_port - cut.db = expected_db - cut.password = expected_password + cut.server_configs = fake_server_configs + cut.servers = [] mocker.patch(redis_adapter.__name__ + '.print_msg') mocker.patch('redis.Redis', return_value=fake_server) mocker.patch.object(fake_server, 'ping', return_value=False) + mocker.patch('threading.Thread', return_value=fake_listen_thread) + mocker.patch.object(fake_listen_thread, 'start') + # Act cut.connect() # Assert - assert redis_adapter.print_msg.call_count == 1 + assert redis_adapter.print_msg.call_count == 7 assert redis_adapter.print_msg.call_args_list[0].args == ("Redis adapter connecting to server...",) - assert redis.Redis.call_count == 1 + assert redis.Redis.call_count == 2 assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password) - assert fake_server.ping.call_count == 1 - assert cut.server == fake_server + assert fake_server.ping.call_count == 2 + assert cut.servers == [fake_server, fake_server] # subscribe_message tests def test_redis_adapter_DataSource_subscribe_subscribes_to_each_given_subscription_and_starts_listening_when_server_available(mocker): From 259dc532599d22b5c8e67347555047c8f14daabe Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Fri, 2 Aug 2024 17:32:55 -0400 Subject: [PATCH 16/21] All unit tests passing and 100% code coverage! --- onair/data_handling/redis_adapter.py | 42 +- .../onair/data_handling/test_redis_adapter.py | 445 ++++++++++-------- 2 files changed, 281 insertions(+), 206 deletions(-) diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index 9af718e0..07715ce9 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -43,7 +43,6 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): def connect(self): num_redis_print_msg_calls = 0 """Establish connection to REDIS server.""" - print("Here") print_msg('Redis adapter connecting to server...') num_redis_print_msg_calls += 1 for idx, server_config in enumerate(self.server_configs): @@ -67,18 +66,18 @@ def connect(self): password = server_config['password'] else: password = '' + + #if there are subscriptions in this Redis server configuration's subscription key + if len(server_config['subscriptions']) != 0: + #Create the servers and append them to self.servers list + self.servers.append(redis.Redis(address, port, db, password)) + + try: + #Ping server to make sure we can connect + self.servers[-1].ping() + print_msg(f'... connected to server # {idx}!') + num_redis_print_msg_calls += 1 - self.servers.append(redis.Redis(address, port, db, password)) - print("Appended to servers") - - try: - print("In try") - self.servers[-1].ping() - print_msg(f'... connected to server # {idx}!') - num_redis_print_msg_calls += 1 - - #if there are subscriptions in this Redis server configuration's subscription key - if len(server_config['subscriptions']) !=0: #Set up Redis pubsub function for the current server pubsub = self.servers[-1].pubsub() @@ -88,13 +87,16 @@ def connect(self): num_redis_print_msg_calls += 1 listen_thread = threading.Thread(target=self.message_listener, args=(pubsub,)) listen_thread.start() - else: - print_msg(f"No subscriptions given!") + + #This except will be hit if self.servers[-1].ping() threw an exception (could not properly ping server) + except: + print_msg(f'Did not connect to server # {idx}. Not setting up subscriptions.', 'RED') num_redis_print_msg_calls += 1 - except: - print_msg(f'Did not connect to server # {idx}. Not setting up subscriptions.', 'RED') - num_redis_print_msg_calls += 1 + else: + print_msg("No subscriptions given! Redis server not created") + num_redis_print_msg_calls += 1 + print("num_redis_print_msg_calls: ", num_redis_print_msg_calls) @@ -106,15 +108,15 @@ def parse_meta_data_file(self, meta_data_file, ss_breakdown): keys = meta.keys() # Setup redis server configuration - #Checking in 'redis' exists + #Checking if 'redis' exists if 'redis' in keys: count_server_config = 0 #Checking if dictionaries within 'redis' key each have a 'subscription' key. Error will be thrown if not. for server_config in meta['redis']: redis_config_keys = server_config.keys() - if 'subscriptions' in redis_config_keys == False: + if ('subscriptions' in redis_config_keys) == False: raise ConfigKeyError(f'Config file: \'{meta_data_file}\' ' \ - f'missing required key \'suscriptions\' from {count_server_config} in key \'redis\' ') + f'missing required key \'subscriptions\' from {count_server_config} in key \'redis\'') count_server_config +=1 #Saving all of Redis dictionaries from JSON file to self.server_configs diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index 5b577795..a68b0e42 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -30,7 +30,6 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects(mocker) fake_new_data_lock = MagicMock() cut = DataSource.__new__(DataSource) - cut.subscriptions = expected_subscriptions fake_order = MagicMock() fake_order.__len__.return_value = \ pytest.gen.randint(1, 10) # from 1 to 10 arbitrary @@ -43,7 +42,6 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects(mocker) # Act cut.__init__(arg_data_file, arg_meta_file, arg_ss_breakdown) - #TO CHECK: Anything you mocked, anything you changed, and what your test name is looking for # Assert assert OnAirDataSource.__init__.call_count == 1 assert OnAirDataSource.__init__.call_args_list[0].args == (arg_data_file, arg_meta_file, arg_ss_breakdown) @@ -62,9 +60,6 @@ def test_redis_adapter_DataSource__init__sets_redis_values_then_connects(mocker) # connect tests def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_attributes(mocker): # Arrange - expected_db = 0 - expected_password = '' - #TODO: Run in a loop fake_server_configs = [{"address": MagicMock(), "port": 1234,"db": 1, "password": 'test', "subscriptions": ["state_0", "state_1"]}, {"address": '000.000.000.222', "port": 5678, "db": 2, "password": 'test2', "subscriptions" : ["state_2", "state_3"]}] fake_server = MagicMock() @@ -104,7 +99,6 @@ def test_redis_adapter_DataSource_connect_establishes_server_with_initialized_at assert fake_server.ping.call_count == 2 assert cut.servers == [fake_server, fake_server] -#TODO: Need a test that shows that all of the default vals get used for server config # connect tests def test_redis_adapter_DataSource_connect_establishes_server_with_default_attributes(mocker): # Arrange @@ -141,154 +135,170 @@ def test_redis_adapter_DataSource_connect_establishes_server_with_default_attrib assert redis.Redis.call_args_list[1].args == (expected_address, expected_port, expected_db, expected_password ) -def test_redis_adapter_DataSource_fails_to_connect_to_server(mocker): - # Arrange - expected_address = 'localhost' - expected_port = 6379 - expected_db = 0 - expected_password = '' - - fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions" : ["state_2", "state_3"]}] +def test_redis_adapter_DataSource_fails_to_connect_to_server_with_ping_and_states_no_subscriptions_(mocker): + fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] fake_server = MagicMock() - fake_message_listener = MagicMock() - fake_listen_thread = MagicMock() - cut = DataSource.__new__(DataSource) cut.server_configs = fake_server_configs cut.servers = [] mocker.patch(redis_adapter.__name__ + '.print_msg') mocker.patch('redis.Redis', return_value=fake_server) - mocker.patch.object(fake_server, 'ping', return_value=False) - mocker.patch('threading.Thread', return_value=fake_listen_thread) - mocker.patch.object(fake_listen_thread, 'start') - + mocker.patch.object(fake_server, 'ping', side_effect=ConnectionError) # Act - cut.connect() + cut.connect() # Assert - assert redis_adapter.print_msg.call_count == 7 + assert redis_adapter.print_msg.call_count == 3 assert redis_adapter.print_msg.call_args_list[0].args == ("Redis adapter connecting to server...",) assert redis.Redis.call_count == 2 - assert redis.Redis.call_args_list[0].args == (expected_address, expected_port, expected_db, expected_password) assert fake_server.ping.call_count == 2 assert cut.servers == [fake_server, fake_server] + + assert redis_adapter.print_msg.call_args_list[0].args == ('Redis adapter connecting to server...',) + assert redis_adapter.print_msg.call_args_list[1].args == ('Did not connect to server # 0. Not setting up subscriptions.', 'RED') + assert redis_adapter.print_msg.call_args_list[2].args == ('Did not connect to server # 1. Not setting up subscriptions.', 'RED') + # subscribe_message tests def test_redis_adapter_DataSource_subscribe_subscribes_to_each_given_subscription_and_starts_listening_when_server_available(mocker): # Arrange - arg_subscriptions = [MagicMock()] * pytest.gen.randint(1, 10) # 1 to 10 arbitrary - + fake_server = MagicMock() fake_pubsub = MagicMock() - fake_thread = MagicMock() + fake_message_listener = MagicMock() + fake_listen_thread = MagicMock() + + fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] cut = DataSource.__new__(DataSource) - cut.server = fake_server + cut.server_configs = fake_server_configs + cut.message_listener = fake_message_listener + cut.servers = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] + mocker.patch('redis.Redis', return_value=fake_server) mocker.patch.object(fake_server, 'ping', return_value=True) mocker.patch.object(fake_server, 'pubsub', return_value=fake_pubsub) mocker.patch.object(fake_pubsub, 'subscribe') mocker.patch(redis_adapter.__name__ + '.print_msg') - mocker.patch('threading.Thread', return_value=fake_thread) - mocker.patch.object(fake_thread, 'start') + mocker.patch('threading.Thread', return_value=fake_listen_thread) + mocker.patch.object(fake_listen_thread, 'start') # Act - cut.subscribe(arg_subscriptions) + cut.connect() # Assert - assert fake_server.ping.call_count == 1 - assert fake_server.pubsub.call_count == 1 - assert fake_pubsub.subscribe.call_count == len(arg_subscriptions) - for i in range(len(arg_subscriptions)): - assert fake_pubsub.subscribe.call_args_list[i].args == (arg_subscriptions[i],) - assert redis_adapter.print_msg.call_args_list[i].args == (f"Subscribing to channel: {arg_subscriptions[i]}",) - assert threading.Thread.call_count == 1 - assert threading.Thread.call_args_list[0].kwargs == ({'target': cut.message_listener}) - assert fake_thread.start.call_count == 1 - assert cut.pubsub == fake_pubsub + assert fake_server.ping.call_count == 2 + assert fake_server.pubsub.call_count == 2 + + #This is already checked in the first test. Should it be removed? Should the first test not check the subscription messages? + assert redis_adapter.print_msg.call_count == 7 + assert redis_adapter.print_msg.call_args_list[0].args == ('Redis adapter connecting to server...',) + assert redis_adapter.print_msg.call_args_list[1].args == ('... connected to server # 0!',) + assert redis_adapter.print_msg.call_args_list[2].args == ('Subscribing to channel: state_0 on server # 0',) + assert redis_adapter.print_msg.call_args_list[3].args == ('Subscribing to channel: state_1 on server # 0',) + assert redis_adapter.print_msg.call_args_list[4].args == ('... connected to server # 1!',) + assert redis_adapter.print_msg.call_args_list[5].args == ('Subscribing to channel: state_2 on server # 1',) + assert redis_adapter.print_msg.call_args_list[6].args == ('Subscribing to channel: state_3 on server # 1',) + + assert fake_pubsub.subscribe.call_count == 2 + + assert threading.Thread.call_count == 2 + assert threading.Thread.call_args_list[0].kwargs == ({'target': cut.message_listener, 'args': (fake_pubsub,)}) + assert fake_listen_thread.start.call_count == 2 def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_empty(mocker): # Arrange - arg_subscriptions = [] - fake_server = MagicMock() + fake_servers = MagicMock() initial_pubsub = MagicMock() - fake_subscription = MagicMock() - fake_thread = MagicMock() - cut = DataSource.__new__(DataSource) - cut.server = fake_server - cut.pubsub = initial_pubsub - - mocker.patch.object(fake_server, 'ping', return_value=False) - mocker.patch(redis_adapter.__name__ + '.print_msg') - mocker.patch.object(fake_server, 'pubsub') - mocker.patch('threading.Thread') - mocker.patch.object(fake_thread, 'start') + fake_message_listener = MagicMock() + fake_listen_thread = MagicMock() - # Act - cut.subscribe(arg_subscriptions) + fake_server_configs = [{'subscriptions': {}}] - # Assert - assert fake_server.ping.call_count == 0 - assert fake_server.pubsub.call_count == 0 - assert threading.Thread.call_count == 0 - assert fake_thread.start.call_count == 0 - assert cut.pubsub == initial_pubsub - assert redis_adapter.print_msg.call_args_list[0].args == ("No subscriptions given!",) - -# Note the self.server.ping during runtime will error, not actually return False, but that means code will never run -# this unit test is for completeness of coverage -def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_server_does_not_respond_to_ping(mocker): - # Arrange - arg_channel = [MagicMock()] - fake_server = MagicMock() - initial_pubsub = MagicMock() - fake_subscription = MagicMock() - fake_thread = MagicMock() cut = DataSource.__new__(DataSource) - cut.server = fake_server - cut.pubsub = initial_pubsub + cut.server_configs = fake_server_configs + cut.message_listener = fake_message_listener + cut.servers = [] + - mocker.patch.object(fake_server, 'ping', return_value=False) + mocker.patch('redis.Redis', return_value=fake_servers) + mocker.patch.object(fake_servers, 'ping', return_value=True) + mocker.patch.object(fake_servers, 'pubsub', return_value=initial_pubsub) + mocker.patch.object(initial_pubsub, 'subscribe') mocker.patch(redis_adapter.__name__ + '.print_msg') - mocker.patch.object(fake_server, 'pubsub') - mocker.patch('threading.Thread') - mocker.patch.object(fake_thread, 'start') + mocker.patch('threading.Thread', return_value=fake_listen_thread) + mocker.patch.object(fake_listen_thread, 'start') + # Act - cut.subscribe(arg_channel) + cut.connect() # Assert - assert fake_server.ping.call_count == 1 - assert fake_server.pubsub.call_count == 0 + assert fake_servers.ping.call_count == 0 + assert fake_servers.pubsub.call_count == 0 + assert fake_servers.pubsub.subscribe.call_count == 0 assert threading.Thread.call_count == 0 - assert fake_thread.start.call_count == 0 - assert cut.pubsub == initial_pubsub - assert redis_adapter.print_msg.call_args_list[0].args == ("No subscriptions given!",) + assert fake_listen_thread.start.call_count == 0 + assert redis_adapter.print_msg.call_args_list[1].args == ("No subscriptions given! Redis server not created",) -# get_next tests -def test_redis_adapter_DataSource_get_next_returns_expected_data_when_new_data_is_true_and_double_buffer_read_index_is_0(): - # Arrange - # Renew DataSource to ensure test independence - cut = DataSource.__new__(DataSource) - cut.new_data = True - cut.new_data_lock = MagicMock() - cut.double_buffer_read_index = 0 - pre_call_index = cut.double_buffer_read_index - expected_result = MagicMock() - cut.currentData = [] - cut.currentData.append({'data': MagicMock()}) - cut.currentData.append({'data': expected_result}) - # Act - result = cut.get_next() +# Note the self.server.ping during runtime will error, not actually return False, but that means code will never run +# this unit test is for completeness of coverage +# def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_server_does_not_respond_to_ping(mocker): +# # Arrange +# fake_server = MagicMock() +# initial_pubsub = MagicMock() +# fake_subscription = MagicMock() +# fake_listen_thread = MagicMock() +# fake_message_listener = MagicMock() +# fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] + +# cut = DataSource.__new__(DataSource) +# cut.servers = fake_server +# cut.pubsub = initial_pubsub + - # Assert - assert cut.new_data == False - assert cut.double_buffer_read_index == 1 - assert result == expected_result +# mocker.patch.object(fake_server, 'ping', return_value=False) +# mocker.patch(redis_adapter.__name__ + '.print_msg') +# mocker.patch.object(fake_server, 'pubsub') +# mocker.patch('threading.Thread') +# mocker.patch.object(fake_thread, 'start') + +# # Act +# cut.connect() + +# # Assert +# assert fake_server.ping.call_count == 1 +# assert fake_server.pubsub.call_count == 0 +# assert threading.Thread.call_count == 0 +# assert fake_thread.start.call_count == 0 +# assert cut.pubsub == initial_pubsub +# assert redis_adapter.print_msg.call_args_list[1].args == ("No subscriptions given!",) + +# # get_next tests +# def test_redis_adapter_DataSource_get_next_returns_expected_data_when_new_data_is_true_and_double_buffer_read_index_is_0(): +# # Arrange +# # Renew DataSource to ensure test independence +# cut = DataSource.__new__(DataSource) +# cut.new_data = True +# cut.new_data_lock = MagicMock() +# cut.double_buffer_read_index = 0 +# pre_call_index = cut.double_buffer_read_index +# expected_result = MagicMock() +# cut.currentData = [] +# cut.currentData.append({'data': MagicMock()}) +# cut.currentData.append({'data': expected_result}) + +# # Act +# result = cut.get_next() + +# # Assert +# assert cut.new_data == False +# assert cut.double_buffer_read_index == 1 +# assert result == expected_result def test_redis_adapter_DataSource_get_next_returns_expected_data_when_new_data_is_true_and_double_buffer_read_index_is_1(): # Arrange @@ -390,15 +400,16 @@ def test_redis_adapter_DataSource_message_listener_warns_of_exit_and_does_not_ru # Arrange cut = DataSource.__new__(DataSource) - cut.pubsub = MagicMock(name="cut.pubsub") + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() fake_listener = MagicMock(name='fake_listener') fake_listener.__next__.side_effect = StopIteration - mocker.patch.object(cut.pubsub, 'listen', side_effect=[fake_listener]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', side_effect=[fake_listener]) mocker.patch(redis_adapter.__name__ + '.json.loads') mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 0 @@ -409,17 +420,19 @@ def test_redis_adapter_DataSource_message_listener_prints_warning_when_receiving # Arrange cut = DataSource.__new__(DataSource) - cut.pubsub = MagicMock() + #cut.pubsub = MagicMock() + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() ignored_message_types = ['subscribe', 'unsubscribe', 'psubscribe', 'punsubscribe', 'pmessage'] fake_message = {} fake_message['type'] = pytest.gen.choice(ignored_message_types) fake_message['channel'] = str(MagicMock(name='fake_message')).encode('utf-8') - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) mocker.patch(redis_adapter.__name__ + '.json.loads') mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 0 @@ -433,19 +446,19 @@ def test_redis_adapter_DataSource_message_listener_prints_warning_when_receiving def test_redis_adapter_DataSource_message_listener_prints_warning_when_data_not_json_format_and_does_not_update_frame(mocker): # Arrange cut = DataSource.__new__(DataSource) - - cut.pubsub = MagicMock() + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() fake_message = {} fake_message['type'] = 'message' fake_message['channel'] = str( MagicMock(name='fake_message_channel')).encode('utf-8') fake_message['data'] = str(MagicMock(name='fake_message_data')) - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) mocker.patch(redis_adapter.__name__ + '.json.loads', side_effect=ValueError) mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 1 @@ -465,7 +478,8 @@ def test_redis_adapter_DataSource_message_listener_warns_user_when_processed_dat cut.double_buffer_read_index = pytest.gen.choice([0 , 1]) cut.currentData = {0: {'headers': [], 'data': []}, 1: {'headers': [], 'data': []}} - cut.pubsub = MagicMock() + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() cut.new_data_lock = MagicMock() cut.new_data = False @@ -474,12 +488,12 @@ def test_redis_adapter_DataSource_message_listener_warns_user_when_processed_dat fake_message['channel'] = str( MagicMock(name='fake_message_channel')).encode('utf-8') fake_message['data'] = '{}' # empty_message - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) mocker.patch(redis_adapter.__name__ + '.json.loads', return_value={}) mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 1 @@ -501,7 +515,8 @@ def test_redis_adapter_DataSource_message_listener_warns_of_received_key_that_do 'data': ['-']}, 1: {'headers': ['time'], 'data': ['-']}} - cut.pubsub = MagicMock() + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() cut.new_data_lock = MagicMock() cut.new_data = False @@ -510,12 +525,12 @@ def test_redis_adapter_DataSource_message_listener_warns_of_received_key_that_do fake_message['channel'] = str( MagicMock(name='fake_message_channel')).encode('utf-8') fake_message['data'] = '{"time":0, "unknown_key":0}' - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) mocker.patch(redis_adapter.__name__ + '.json.loads', return_value={"time":0, "unknown_key":0}) mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 1 @@ -531,8 +546,9 @@ def test_redis_adapter_DataSource_message_listener_warns_of_received_key_that_do def test_redis_adapter_DataSource_message_listener_warns_of_expected_keys_that_do_not_appear_in_message(mocker): # Arrange cut = DataSource.__new__(DataSource) - cut.double_buffer_read_index = pytest.gen.choice([0 , 1]) - cut.pubsub = MagicMock() + cut.double_buffer_read_index = pytest.gen.choice([0 , 1]) + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() cut.new_data_lock = MagicMock() cut.new_data = False @@ -549,12 +565,12 @@ def test_redis_adapter_DataSource_message_listener_warns_of_expected_keys_that_d '.missing_key'], 'data': ['-', '-']}} fake_message['data'] = '{}' - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) mocker.patch(redis_adapter.__name__ + '.json.loads', return_value={}) mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 1 @@ -577,7 +593,8 @@ def test_redis_adapter_DataSource_message_listener_updates_new_data_with_receive # Arrange cut = DataSource.__new__(DataSource) cut.double_buffer_read_index = pytest.gen.choice([0 , 1]) - cut.pubsub = MagicMock() + fake_server = MagicMock() + fake_server.fake_pubsub = MagicMock() cut.new_data_lock = MagicMock() cut.new_data = False @@ -594,7 +611,7 @@ def test_redis_adapter_DataSource_message_listener_updates_new_data_with_receive '.correct_key', 'fakeotherchannel.x'], 'data': ['-', '-', '0']}} fake_message['data'] = '{}' - mocker.patch.object(cut.pubsub, 'listen', return_value=[fake_message]) + mocker.patch.object(fake_server.fake_pubsub, 'listen', return_value=[fake_message]) fake_data = { 'time': pytest.gen.randint(1, 100), # from 1 to 100 arbitrary 'correct_key': pytest.gen.randint(1, 100), # from 1 to 100 arbitrary @@ -604,7 +621,7 @@ def test_redis_adapter_DataSource_message_listener_updates_new_data_with_receive mocker.patch(redis_adapter.__name__ + '.print_msg') # Act - cut.message_listener() + cut.message_listener(fake_server.fake_pubsub) # Assert assert redis_adapter.json.loads.call_count == 1 @@ -657,17 +674,15 @@ def test_redis_adapter_DataSource_parse_meta_data_file_raises_ConfigKeyError_whe assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) assert e_info.match(exception_message) -def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown_and_sets_subscriptions_when_redis_subscriptions_occupied(mocker): +def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown(mocker): # Arrange cut = DataSource.__new__(DataSource) arg_configFile = MagicMock() arg_ss_breakdown = MagicMock() expected_extracted_configs = MagicMock() - expected_subscriptions = [MagicMock()] * pytest.gen.randint(0, 10) # 0 to 10 arbitrary fake_meta = {'fake_other_stuff': MagicMock(), - 'order': MagicMock(), - 'redis_subscriptions':expected_subscriptions} + 'order': MagicMock()} mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) @@ -680,26 +695,114 @@ def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_m assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) assert redis_adapter.parseJson.call_count == 1 assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) - assert cut.subscriptions == expected_subscriptions assert result == expected_extracted_configs -def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_for_redis_server_configurations(mocker): +# def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_for_redis_server_configurations(mocker): +# # Arrange +# cut = DataSource.__new__(DataSource) +# arg_configFile = MagicMock() +# arg_ss_breakdown = MagicMock() + + +# expected_extracted_configs = MagicMock() +# expected_subscriptions = [MagicMock()] * pytest.gen.randint(0, 10) # 0 to 10 arbitrary +# expected_address = MagicMock() +# expected_port = MagicMock() +# expected_password = MagicMock() +# expected_redis_configs = {'address': expected_address, 'port': expected_port, 'password': expected_password} +# fake_meta = {'fake_other_stuff': MagicMock(), +# 'order': MagicMock(), +# 'redis_subscriptions': expected_subscriptions, +# 'redis': expected_redis_configs} + +# mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) +# mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) + +# # Act +# result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) + +# # Assert +# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 +# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) +# assert redis_adapter.parseJson.call_count == 1 +# assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) +# assert cut.subscriptions == expected_subscriptions +# assert result == expected_extracted_configs +# assert cut.address == expected_address +# assert cut.port == expected_port +# assert cut.password == expected_password + +# def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown_and_sets_subscriptions_to_empty_when_none_given(mocker): +# # Arrange +# cut = DataSource.__new__(DataSource) +# arg_configFile = MagicMock() +# arg_ss_breakdown = MagicMock() + +# fake_configs = {'fake_other_stuff': MagicMock()} +# fake_meta = {'order': MagicMock()} + +# mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=fake_configs) +# mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) + +# # Act +# result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) + +# # Assert +# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 +# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) +# assert redis_adapter.parseJson.call_count == 1 +# assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) +# assert cut.subscriptions == [] +# assert result == fake_configs + +# redis_adapter get_vehicle_metadata tests +def test_redis_adapter_DataSource_get_vehicle_metadata_returns_list_of_headers_and_list_of_test_assignments(): + # Arrange + cut = DataSource.__new__(DataSource) + fake_all_headers = MagicMock() + fake_test_assignments = MagicMock() + fake_binning_configs = {} + fake_binning_configs['test_assignments'] = fake_test_assignments + + expected_result = (fake_all_headers, fake_test_assignments) + + cut.all_headers = fake_all_headers + cut.binning_configs = fake_binning_configs + + # Act + result = cut.get_vehicle_metadata() + + # Assert + assert result == expected_result + +# redis_adapter process_data_file tests +def test_redis_adapter_DataSource_process_data_file_does_nothing(): + # Arrange + cut = DataSource.__new__(DataSource) + arg_data_file = MagicMock() + + expected_result = None + + # Act + result = cut.process_data_file(arg_data_file) + + # Assert + assert result == expected_result + + +def test_redis_adapter_DataSource_parse_meta_data_file_redis_in_keys_subscriptions_exist_and_adds_redis_to_server_configs(mocker): # Arrange cut = DataSource.__new__(DataSource) arg_configFile = MagicMock() arg_ss_breakdown = MagicMock() + fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] + cut.server_configs = MagicMock() expected_extracted_configs = MagicMock() - expected_subscriptions = [MagicMock()] * pytest.gen.randint(0, 10) # 0 to 10 arbitrary - expected_address = MagicMock() - expected_port = MagicMock() - expected_password = MagicMock() - expected_redis_configs = {'address': expected_address, 'port': expected_port, 'password': expected_password} fake_meta = {'fake_other_stuff': MagicMock(), - 'order': MagicMock(), - 'redis_subscriptions': expected_subscriptions, - 'redis': expected_redis_configs} + 'redis': fake_server_configs, + 'order': MagicMock()} mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) @@ -712,65 +815,35 @@ def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_m assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) assert redis_adapter.parseJson.call_count == 1 assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) - assert cut.subscriptions == expected_subscriptions assert result == expected_extracted_configs - assert cut.address == expected_address - assert cut.port == expected_port - assert cut.password == expected_password + assert cut.server_configs == fake_meta['redis'] -def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown_and_sets_subscriptions_to_empty_when_none_given(mocker): +def test_redis_adapter_DataSource_parse_meta_data_file_redis_in_keys_subscriptions_do_not_exist(mocker): # Arrange cut = DataSource.__new__(DataSource) arg_configFile = MagicMock() arg_ss_breakdown = MagicMock() - fake_configs = {'fake_other_stuff': MagicMock()} - fake_meta = {'order': MagicMock()} + fake_server_configs = [{"address": 1}] + cut.server_configs = MagicMock() + + expected_extracted_configs = MagicMock() + fake_meta = {'fake_other_stuff': MagicMock(), + 'redis': fake_server_configs, + 'order': MagicMock()} - mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=fake_configs) + mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) # Act - result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) + with pytest.raises(ConfigKeyError) as e_info: + cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) + # Assert assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) assert redis_adapter.parseJson.call_count == 1 assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) - assert cut.subscriptions == [] - assert result == fake_configs - -# redis_adapter get_vehicle_metadata tests -def test_redis_adapter_DataSource_get_vehicle_metadata_returns_list_of_headers_and_list_of_test_assignments(): - # Arrange - cut = DataSource.__new__(DataSource) - fake_all_headers = MagicMock() - fake_test_assignments = MagicMock() - fake_binning_configs = {} - fake_binning_configs['test_assignments'] = fake_test_assignments - - expected_result = (fake_all_headers, fake_test_assignments) - - cut.all_headers = fake_all_headers - cut.binning_configs = fake_binning_configs - - # Act - result = cut.get_vehicle_metadata() - - # Assert - assert result == expected_result - -# redis_adapter process_data_file tests -def test_redis_adapter_DataSource_process_data_file_does_nothing(): - # Arrange - cut = DataSource.__new__(DataSource) - arg_data_file = MagicMock() - - expected_result = None - - # Act - result = cut.process_data_file(arg_data_file) - - # Assert - assert result == expected_result + assert cut.server_configs == [] + assert e_info.match(f'Config file: \'{arg_configFile}\' missing required key \'subscriptions\' from 0 in key \'redis\'') \ No newline at end of file From 95c3a4700270671ccf47b2f79a33a70a1a79782f Mon Sep 17 00:00:00 2001 From: Rachael Chertok Date: Tue, 6 Aug 2024 11:56:06 -0400 Subject: [PATCH 17/21] Removed commented functions from test redis adapter code and removed debug prints from redis_adapter.py --- onair/data_handling/redis_adapter.py | 11 +- .../onair/data_handling/test_redis_adapter.py | 117 +----------------- 2 files changed, 2 insertions(+), 126 deletions(-) diff --git a/onair/data_handling/redis_adapter.py b/onair/data_handling/redis_adapter.py index 07715ce9..4a9d4f6f 100644 --- a/onair/data_handling/redis_adapter.py +++ b/onair/data_handling/redis_adapter.py @@ -41,10 +41,8 @@ def __init__(self, data_file, meta_file, ss_breakdown = False): self.connect() def connect(self): - num_redis_print_msg_calls = 0 """Establish connection to REDIS server.""" print_msg('Redis adapter connecting to server...') - num_redis_print_msg_calls += 1 for idx, server_config in enumerate(self.server_configs): server_config_keys = server_config.keys() if 'address' in server_config_keys: @@ -76,7 +74,6 @@ def connect(self): #Ping server to make sure we can connect self.servers[-1].ping() print_msg(f'... connected to server # {idx}!') - num_redis_print_msg_calls += 1 #Set up Redis pubsub function for the current server pubsub = self.servers[-1].pubsub() @@ -84,21 +81,15 @@ def connect(self): for s in server_config['subscriptions']: pubsub.subscribe(s) print_msg(f"Subscribing to channel: {s} on server # {idx}") - num_redis_print_msg_calls += 1 listen_thread = threading.Thread(target=self.message_listener, args=(pubsub,)) listen_thread.start() #This except will be hit if self.servers[-1].ping() threw an exception (could not properly ping server) except: print_msg(f'Did not connect to server # {idx}. Not setting up subscriptions.', 'RED') - num_redis_print_msg_calls += 1 else: - print_msg("No subscriptions given! Redis server not created") - num_redis_print_msg_calls += 1 - - - print("num_redis_print_msg_calls: ", num_redis_print_msg_calls) + print_msg("No subscriptions given! Redis server not created") def parse_meta_data_file(self, meta_data_file, ss_breakdown): self.server_configs = [] diff --git a/test/onair/data_handling/test_redis_adapter.py b/test/onair/data_handling/test_redis_adapter.py index a68b0e42..bd583609 100644 --- a/test/onair/data_handling/test_redis_adapter.py +++ b/test/onair/data_handling/test_redis_adapter.py @@ -203,7 +203,7 @@ def test_redis_adapter_DataSource_subscribe_subscribes_to_each_given_subscriptio assert redis_adapter.print_msg.call_args_list[5].args == ('Subscribing to channel: state_2 on server # 1',) assert redis_adapter.print_msg.call_args_list[6].args == ('Subscribing to channel: state_3 on server # 1',) - assert fake_pubsub.subscribe.call_count == 2 + assert fake_pubsub.subscribe.call_count == 4 assert threading.Thread.call_count == 2 assert threading.Thread.call_args_list[0].kwargs == ({'target': cut.message_listener, 'args': (fake_pubsub,)}) @@ -232,7 +232,6 @@ def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_e mocker.patch('threading.Thread', return_value=fake_listen_thread) mocker.patch.object(fake_listen_thread, 'start') - # Act cut.connect() @@ -244,62 +243,6 @@ def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_e assert fake_listen_thread.start.call_count == 0 assert redis_adapter.print_msg.call_args_list[1].args == ("No subscriptions given! Redis server not created",) - -# Note the self.server.ping during runtime will error, not actually return False, but that means code will never run -# this unit test is for completeness of coverage -# def test_redis_adapter_DataSource_subscribe_states_no_subscriptions_given_when_server_does_not_respond_to_ping(mocker): -# # Arrange -# fake_server = MagicMock() -# initial_pubsub = MagicMock() -# fake_subscription = MagicMock() -# fake_listen_thread = MagicMock() -# fake_message_listener = MagicMock() -# fake_server_configs = [{"subscriptions": ["state_0", "state_1"]}, {"subscriptions": ["state_2", "state_3"]}] - -# cut = DataSource.__new__(DataSource) -# cut.servers = fake_server -# cut.pubsub = initial_pubsub - - -# mocker.patch.object(fake_server, 'ping', return_value=False) -# mocker.patch(redis_adapter.__name__ + '.print_msg') -# mocker.patch.object(fake_server, 'pubsub') -# mocker.patch('threading.Thread') -# mocker.patch.object(fake_thread, 'start') - -# # Act -# cut.connect() - -# # Assert -# assert fake_server.ping.call_count == 1 -# assert fake_server.pubsub.call_count == 0 -# assert threading.Thread.call_count == 0 -# assert fake_thread.start.call_count == 0 -# assert cut.pubsub == initial_pubsub -# assert redis_adapter.print_msg.call_args_list[1].args == ("No subscriptions given!",) - -# # get_next tests -# def test_redis_adapter_DataSource_get_next_returns_expected_data_when_new_data_is_true_and_double_buffer_read_index_is_0(): -# # Arrange -# # Renew DataSource to ensure test independence -# cut = DataSource.__new__(DataSource) -# cut.new_data = True -# cut.new_data_lock = MagicMock() -# cut.double_buffer_read_index = 0 -# pre_call_index = cut.double_buffer_read_index -# expected_result = MagicMock() -# cut.currentData = [] -# cut.currentData.append({'data': MagicMock()}) -# cut.currentData.append({'data': expected_result}) - -# # Act -# result = cut.get_next() - -# # Assert -# assert cut.new_data == False -# assert cut.double_buffer_read_index == 1 -# assert result == expected_result - def test_redis_adapter_DataSource_get_next_returns_expected_data_when_new_data_is_true_and_double_buffer_read_index_is_1(): # Arrange # Renew DataSource to ensure test independence @@ -697,64 +640,6 @@ def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_m assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) assert result == expected_extracted_configs -# def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_for_redis_server_configurations(mocker): -# # Arrange -# cut = DataSource.__new__(DataSource) -# arg_configFile = MagicMock() -# arg_ss_breakdown = MagicMock() - - -# expected_extracted_configs = MagicMock() -# expected_subscriptions = [MagicMock()] * pytest.gen.randint(0, 10) # 0 to 10 arbitrary -# expected_address = MagicMock() -# expected_port = MagicMock() -# expected_password = MagicMock() -# expected_redis_configs = {'address': expected_address, 'port': expected_port, 'password': expected_password} -# fake_meta = {'fake_other_stuff': MagicMock(), -# 'order': MagicMock(), -# 'redis_subscriptions': expected_subscriptions, -# 'redis': expected_redis_configs} - -# mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=expected_extracted_configs) -# mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) - -# # Act -# result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) - -# # Assert -# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 -# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) -# assert redis_adapter.parseJson.call_count == 1 -# assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) -# assert cut.subscriptions == expected_subscriptions -# assert result == expected_extracted_configs -# assert cut.address == expected_address -# assert cut.port == expected_port -# assert cut.password == expected_password - -# def test_redis_adapter_DataSource_parse_meta_data_file_returns_call_to_extract_meta_data_handle_ss_breakdown_and_sets_subscriptions_to_empty_when_none_given(mocker): -# # Arrange -# cut = DataSource.__new__(DataSource) -# arg_configFile = MagicMock() -# arg_ss_breakdown = MagicMock() - -# fake_configs = {'fake_other_stuff': MagicMock()} -# fake_meta = {'order': MagicMock()} - -# mocker.patch(redis_adapter.__name__ + '.extract_meta_data_handle_ss_breakdown', return_value=fake_configs) -# mocker.patch(redis_adapter.__name__ + '.parseJson', return_value=fake_meta) - -# # Act -# result = cut.parse_meta_data_file(arg_configFile, arg_ss_breakdown, ) - -# # Assert -# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_count == 1 -# assert redis_adapter.extract_meta_data_handle_ss_breakdown.call_args_list[0].args == (arg_configFile, arg_ss_breakdown) -# assert redis_adapter.parseJson.call_count == 1 -# assert redis_adapter.parseJson.call_args_list[0].args == (arg_configFile, ) -# assert cut.subscriptions == [] -# assert result == fake_configs - # redis_adapter get_vehicle_metadata tests def test_redis_adapter_DataSource_get_vehicle_metadata_returns_list_of_headers_and_list_of_test_assignments(): # Arrange From 9013c462380de37e43811f5e41d0ecc309ae5836 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 7 Aug 2024 14:10:32 -0400 Subject: [PATCH 18/21] Removing things that seem problematic --- onair/config/default_config.ini | 8 +- plugins/drone_state_machine/__init__.py | 0 .../drone_state_machine_plugin.py | 75 ------------------- 3 files changed, 4 insertions(+), 79 deletions(-) delete mode 100644 plugins/drone_state_machine/__init__.py delete mode 100644 plugins/drone_state_machine/drone_state_machine_plugin.py diff --git a/onair/config/default_config.ini b/onair/config/default_config.ini index a90e5f5e..c5e4a922 100644 --- a/onair/config/default_config.ini +++ b/onair/config/default_config.ini @@ -19,13 +19,13 @@ DataSourceFile = onair/data_handling/csv_parser.py [PLUGINS] # NOTE: even though keys are required, they may be set to empty dicts # Required Key: KnowledgeRepPluginDict(s) are used by the VehicleRep -KnowledgeRepPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} +KnowledgeRepPluginDict = {'generic':'plugins/generic/__init__.py'} # Required Key: LearnersPluginDict(s) are used by Agent for learning -LearnersPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} +LearnersPluginDict = {'generic':'plugins/generic/__init__.py'} # Required Key: LearnersPluginDict(s) are used by Agent for planning -PlannersPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} +PlannersPluginDict = {'generic':'plugins/generic/__init__.py'} # Required Key: ComplexPluginDict(s) are used by Agent for complex reasoning -ComplexPluginDict = {'generic':'OnAIR/plugins/generic/__init__.py'} +ComplexPluginDict = {'generic':'plugins/generic/__init__.py'} # Optional Section: OPTIONS are settable values to change running experience [OPTIONS] diff --git a/plugins/drone_state_machine/__init__.py b/plugins/drone_state_machine/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/plugins/drone_state_machine/drone_state_machine_plugin.py b/plugins/drone_state_machine/drone_state_machine_plugin.py deleted file mode 100644 index a39428ae..00000000 --- a/plugins/drone_state_machine/drone_state_machine_plugin.py +++ /dev/null @@ -1,75 +0,0 @@ -# GSC-19165-1, "The On-Board Artificial Intelligence Research (OnAIR) Platform" -# -# Copyright © 2023 United States Government as represented by the Administrator of -# the National Aeronautics and Space Administration. No copyright is claimed in the -# United States under Title 17, U.S. Code. All Other Rights Reserved. -# -# Licensed under the NASA Open Source Agreement version 1.3 -# See "NOSA GSC-19165-1 OnAIR.pdf" - -import numpy as np -import redis -from time import sleep -import json - -from onair.src.ai_components.ai_plugin_abstract.ai_plugin import AIPlugin - -TARGET_ALT = 5 # What altitude should the drone maintain during operations? -POSITION_TOLERANCE = 0.2 # How close must the drone get to a target location for operations purposes? -HOME_COORDS = [0.0, 0.0] # Where is the drone's home coordinates? May be different than origin in some cases -DRONE_NUMBER = 1 # Which drone is this? Used to publish commands to the right channel - -def xy_position_difference(pos1, pos2): # Calculate the difference between XY positions, used for position tolerance comparisons - return np.sqrt((pos1[0] - pos2[0])**2 + (pos1[1] - pos2[1])**2) - -class Plugin(AIPlugin): - def __init__(self, name, headers): - """ - Initialize Redis connection and set target altitude, position tolerance, and home coordinates - """ - # Redis initialization - super().__init__(name, headers) - pool = redis.ConnectionPool(host="localhost", port=6379, password="") - self.r = redis.Redis(connection_pool=pool, charset="utf-8", decode_responses=True) - self.pub_channel = 'mavlink_cmd' - - - self.target_alt = TARGET_ALT * -1 - self.home = HOME_COORDS - - self.state = 'standby' # Initial drone operating state - - self.target_position = None - - def send_cmd(self, cmd): # Command publishing to Redis - serialized_cmd = json.dumps(cmd) - self.r.publish(f'edp_drone_{DRONE_NUMBER}_command', serialized_cmd) - - def update(self,low_level_data=[], high_level_data={}): - """ - Given streamed data point, system should update internally - """ - print(f'\nCOMPLEX REASONER sees high level data as: {high_level_data}') - #self.drone_position = high_level_data['vehicle_rep']['KAST']['pose'] - #Temporary change to remove the use of KAST, during initial testing for SPAR intern project - self.drone_position = (0,0,0) - - pass - - def render_reasoning(self): - """ - Basic state machine implemented to take off, move, pause, return, and land. - """ - rnd_xy = np.random.uniform(-5,5, size=2) - - if self.target_position == None: - self.target_position = [rnd_xy[0], rnd_xy[1], 0.5] # XYZ - self.send_cmd(self.target_position) - else: - position_err = xy_position_difference(self.drone_position, self.target_position) - if position_err < POSITION_TOLERANCE: - self.target_position = [rnd_xy[0], rnd_xy[1], 1.0] - self.send_cmd(self.target_position) - else: - print(f'Traveling to new target position {self.target_position}') - From 30dbdde0eb6385f03345ba972b7f677fcea94c9d Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 7 Aug 2024 14:15:30 -0400 Subject: [PATCH 19/21] Remove one config that we don't need --- onair/config/onair_pybullet_config.ini | 14 -------------- 1 file changed, 14 deletions(-) delete mode 100644 onair/config/onair_pybullet_config.ini diff --git a/onair/config/onair_pybullet_config.ini b/onair/config/onair_pybullet_config.ini deleted file mode 100644 index 041b21c8..00000000 --- a/onair/config/onair_pybullet_config.ini +++ /dev/null @@ -1,14 +0,0 @@ -[DEFAULT] -TelemetryDataFilePath = onair/data/raw_telemetry_data/data_physics_generation/Errors -TelemetryFile = 700_crash_to_earth_1.csv -TelemetryMetadataFilePath = OnAIR/onair/data/telemetry_configs/ -MetaFile = pybullet_tlm_config.json -ParserFileName = OnAIR/onair/data_handling/redis_adapter.py - -KnowledgeRepPluginDict = {'knowledge':'OnAIR/plugins/generic/__init__.py'} -LearnersPluginDict = {'learner':'OnAIR/plugins/generic/__init__.py'} -PlannersPluginDict = {'planner':'OnAIR/plugins/generic/__init__.py'} -ComplexPluginDict = {'complex':'OnAIR/plugins/drone_state_machine/__init__.py'} - -[RUN_FLAGS] -IO_Flag = true \ No newline at end of file From c268fb6a8b3041413f696314dfb3dcef8deef5c0 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 7 Aug 2024 14:54:28 -0400 Subject: [PATCH 20/21] Added comments --- redis-experiment-publisher-multi-server.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/redis-experiment-publisher-multi-server.py b/redis-experiment-publisher-multi-server.py index 7c20576d..056a0154 100644 --- a/redis-experiment-publisher-multi-server.py +++ b/redis-experiment-publisher-multi-server.py @@ -13,9 +13,9 @@ password=redis_password, decode_responses=True) -# Initialize the Redis connection for server #1 +# Initialize the Redis connection for server #2 redis_host = "localhost" -redis_port = 6380 +redis_port = 6380 # Make sure you start your redis server with a different port # When your Redis server requires a password, fill it in here redis_password = "" # Connect to Redis From e32538c379f0ee7b7b9c10569fadd8a9584b9355 Mon Sep 17 00:00:00 2001 From: Hayley Owens Date: Wed, 7 Aug 2024 14:59:39 -0400 Subject: [PATCH 21/21] Removing pybullet things --- .../pybullet_tlm_config.json | 35 ------------------- 1 file changed, 35 deletions(-) delete mode 100644 onair/data/telemetry_configs/pybullet_tlm_config.json diff --git a/onair/data/telemetry_configs/pybullet_tlm_config.json b/onair/data/telemetry_configs/pybullet_tlm_config.json deleted file mode 100644 index 71c5f39b..00000000 --- a/onair/data/telemetry_configs/pybullet_tlm_config.json +++ /dev/null @@ -1,35 +0,0 @@ -{ - "subsystems": { - "STATES": { - "time": { - "description": "Pybullet drone 0 state" - }, - "sim_drone_0_state.pose": { - "description": "Pybullet drone 0 state" - }, - "sim_drone_0_state.rpms": { - "description": "Pybullet drone 1 state" - }, - "sim_drone_0_state.rpy": { - "description": "Pybullet drone 1 state" - }, - "sim_drone_0_state.vel": { - "description": "Pybullet drone 1 state" - }, - "sim_drone_0_state.ang_vel": { - "description": "Pybullet drone 1 state" - } - } - }, - "redis_subscriptions": [ - "sim_drone_0_state" - ], - "order": [ - "time", - "sim_drone_0_state.pose", - "sim_drone_0_state.rpms", - "sim_drone_0_state.rpy", - "sim_drone_0_state.vel", - "sim_drone_0_state.ang_vel" - ] -}