diff --git a/README.md b/README.md index 484caf8a..fafb58b9 100644 --- a/README.md +++ b/README.md @@ -477,7 +477,6 @@ level setting of -25. "frequency", "reference_level" ], - "clock_rate_lookup_by_sample_rate": [], "calibration_data": { "14000000.0": { "3545000000.0": { diff --git a/compose.yaml b/compose.yaml index 8bce4c9b..f5123cc0 100644 --- a/compose.yaml +++ b/compose.yaml @@ -54,6 +54,8 @@ services: - ENCRYPTION_KEY - FQDN - GIT_BRANCH + - GPS_MODULE + - GPS_CLASS - GUNICORN_LOG_LEVEL - IN_DOCKER=1 - IPS diff --git a/docs/openapi.json b/docs/openapi.json index feee0bb5..2d860ebd 100644 --- a/docs/openapi.json +++ b/docs/openapi.json @@ -1644,10 +1644,9 @@ "description": "[Required] The name of the action to be scheduled", "type": "string", "enum": [ - "logger", + "test_SEA_CBRS_Measure_Baseline", "test_monitor_sigan", "test_multi_frequency_iq_action", - "test_nasctn_sea_data_product", "test_single_frequency_iq_action", "test_single_frequency_m4s_action", "test_survey_iq_action", diff --git a/env.template b/env.template index 35dacf17..cacf4bb6 100644 --- a/env.template +++ b/env.template @@ -46,6 +46,8 @@ ENCRYPTION_KEY="$(python3 -c 'import secrets; import base64; print(base64.b64enc FQDN="$(hostname -f)" GIT_BRANCH="git:$(git rev-parse --abbrev-ref HEAD)@$(git rev-parse --short HEAD)" +GPS_MODULE="" +GPS_CLASS="" IPS="$(hostname -I) 127.0.0.1" diff --git a/src/initialization/action_loader.py b/src/initialization/action_loader.py index f94fb5a1..6b06c966 100644 --- a/src/initialization/action_loader.py +++ b/src/initialization/action_loader.py @@ -95,14 +95,22 @@ def load_actions( } logger.debug(discovered_plugins) actions = {} - if mock_sigan or running_tests: + if running_tests: logger.debug(f"Loading {len(test_actions)} test actions.") actions.update(test_actions) else: + # load scos-actions test_actions + if mock_sigan: + logger.debug(f"Loading {len(test_actions)} test actions.") + actions.update(test_actions) + # load other plugin actions for name, module in discovered_plugins.items(): logger.debug("Looking for actions in " + name + ": " + str(module)) discover = importlib.import_module(name + ".discover") - if hasattr(discover, "actions"): + if mock_sigan and hasattr(discover, "test_actions"): + logger.debug(f"loading {len(discover.test_actions)} test actions.") + actions.update(discover.test_actions) + elif hasattr(discover, "actions"): logger.debug(f"loading {len(discover.actions)} actions.") actions.update(discover.actions) if ( @@ -111,10 +119,14 @@ def load_actions( ): action_classes.update(discover.action_classes) + # load scos-sensor configs/actions logger.debug(f"Loading actions in {action_dir}") yaml_actions, yaml_test_actions = init( action_classes=action_classes, yaml_dir=action_dir ) - actions.update(yaml_actions) - logger.debug("Finished loading and registering actions") + if mock_sigan: + actions.update(yaml_test_actions) + else: + actions.update(yaml_actions) + logger.debug("Finished loading and registering actions") return actions diff --git a/src/initialization/sensor_loader.py b/src/initialization/sensor_loader.py index 5a911743..c0eba4ce 100644 --- a/src/initialization/sensor_loader.py +++ b/src/initialization/sensor_loader.py @@ -57,6 +57,7 @@ def load_sensor( ) sigan = None + gps = None try: if not settings.RUNNING_MIGRATIONS: if get_usb_device_exists(): @@ -77,6 +78,20 @@ def load_sensor( logger.warning(f"unable to create signal analyzer: {ex}") set_container_unhealthy() + try: + if settings.GPS_MODULE and settings.GPS_CLASS: + gps_module_setting = settings.GPS_MODULE + gps_module = importlib.import_module(gps_module_setting) + logger.info( + "Creating " + settings.GPS_CLASS + " from " + settings.GPS_MODULE + ) + gps_constructor = getattr(gps_module, settings.GPS_CLASS) + gps = gps_constructor() + else: + logger.info("GPS_MODULE and/or GPS_CLASS not specified. Not loading GPS.") + except BaseException as ex: + logger.warning(f"unable to create GPS: {ex}") + # Create sensor before handling calibrations sensor = Sensor( signal_analyzer=sigan, @@ -85,6 +100,7 @@ def load_sensor( preselector=preselector, switches=switches, location=location, + gps=gps, sensor_cal=None, differential_cal=None, ) diff --git a/src/requirements-dev.txt b/src/requirements-dev.txt index 77d73007..069d7838 100644 --- a/src/requirements-dev.txt +++ b/src/requirements-dev.txt @@ -184,10 +184,6 @@ numpy==1.24.4 # tekrsa-api-wrap nvidia-ml-py==12.535.133 # via gpustat -oauthlib==3.2.2 - # via - # -r requirements.txt - # requests-oauthlib opencensus==0.11.3 # via ray opencensus-context==0.1.3 @@ -292,11 +288,8 @@ requests==2.31.0 # its-preselector # ray # requests-mock - # requests-oauthlib requests-mock==1.11.0 # via -r requirements.txt -requests-oauthlib==1.3.1 - # via -r requirements.txt rpds-py==0.13.2 # via # -r requirements.txt @@ -316,11 +309,11 @@ scipy==1.10.1 # via # -r requirements.txt # scos-actions -scos-actions @ git+https://github.com/NTIA/scos-actions@9.0.0 +scos-actions @ git+https://github.com/NTIA/scos-actions@10.0.2 # via # -r requirements.txt # scos-tekrsa -scos-tekrsa @ git+https://github.com/NTIA/scos-tekrsa@6.0.0 +scos-tekrsa @ git+https://github.com/NTIA/scos-tekrsa@7.0.1 # via -r requirements.txt sigmf @ git+https://github.com/NTIA/SigMF@multi-recording-archive # via @@ -340,7 +333,7 @@ sqlparse==0.5.0 # via # -r requirements.txt # django -tekrsa-api-wrap==1.3.2 +tekrsa-api-wrap==1.3.3 # via # -r requirements.txt # scos-tekrsa diff --git a/src/requirements.in b/src/requirements.in index 1ba3c444..cf9e9e7f 100644 --- a/src/requirements.in +++ b/src/requirements.in @@ -10,8 +10,7 @@ jsonfield>=3.0, <4.0 packaging>=23.0, <24.0 psycopg2-binary>=2.0, <3.0 requests-mock>=1.0, <2.0 -requests_oauthlib>=1.0, <2.0 -scos_tekrsa @ git+https://github.com/NTIA/scos-tekrsa@6.0.0 +scos_tekrsa @ git+https://github.com/NTIA/scos-tekrsa@7.0.1 # The following are sub-dependencies for which SCOS Sensor enforces a # higher minimum patch version than the dependencies which require them. diff --git a/src/requirements.txt b/src/requirements.txt index 239e8ebc..ef77ffc2 100644 --- a/src/requirements.txt +++ b/src/requirements.txt @@ -93,8 +93,6 @@ numpy==1.24.4 # scos-actions # sigmf # tekrsa-api-wrap -oauthlib==3.2.2 - # via requests-oauthlib packaging==23.2 # via # -r requirements.in @@ -137,11 +135,8 @@ requests==2.31.0 # its-preselector # ray # requests-mock - # requests-oauthlib requests-mock==1.11.0 # via -r requirements.in -requests-oauthlib==1.3.1 - # via -r requirements.in rpds-py==0.13.2 # via # jsonschema @@ -152,9 +147,9 @@ ruamel-yaml-clib==0.2.8 # via ruamel-yaml scipy==1.10.1 # via scos-actions -scos-actions @ git+https://github.com/NTIA/scos-actions@9.0.0 +scos-actions @ git+https://github.com/NTIA/scos-actions@10.0.2 # via scos-tekrsa -scos-tekrsa @ git+https://github.com/NTIA/scos-tekrsa@6.0.0 +scos-tekrsa @ git+https://github.com/NTIA/scos-tekrsa@7.0.1 # via -r requirements.in sigmf @ git+https://github.com/NTIA/SigMF@multi-recording-archive # via scos-actions @@ -168,7 +163,7 @@ sqlparse==0.5.0 # via # -r requirements.in # django -tekrsa-api-wrap==1.3.2 +tekrsa-api-wrap==1.3.3 # via scos-tekrsa typing-extensions==4.8.0 # via asgiref diff --git a/src/schedule/tests/test_models.py b/src/schedule/tests/test_models.py index 4a8ed861..a16631c3 100644 --- a/src/schedule/tests/test_models.py +++ b/src/schedule/tests/test_models.py @@ -17,7 +17,7 @@ def test_take_until(test_input, future_t, expected): start, stop, interval = test_input entry = ScheduleEntry( - name="t", start=start, stop=stop, interval=interval, action="logger" + name="t", start=start, stop=stop, interval=interval, action="test_monitor_sigan" ) initial_times = list(entry.get_remaining_times()) r = [] @@ -32,32 +32,36 @@ def test_take_until(test_input, future_t, expected): def test_undefined_start_is_now(): - entry = ScheduleEntry(name="t", action="logger") + entry = ScheduleEntry(name="t", action="test_monitor_sigan") now = utils.timefn() assert entry.start in (now - 1, now, now + 1) def test_undefined_stop_is_never(): - entry = ScheduleEntry(name="t", action="logger", interval=1) + entry = ScheduleEntry(name="t", action="test_monitor_sigan", interval=1) assert entry.stop is None assert type(entry.get_remaining_times()) is itertools.count def test_relative_stop_becomes_absolute(): - e = ScheduleEntry(name="t", start=20, relative_stop=10, interval=1, action="logger") + e = ScheduleEntry( + name="t", start=20, relative_stop=10, interval=1, action="test_monitor_sigan" + ) assert e.start == 20 assert e.stop == 30 assert list(e.get_remaining_times()) == list(range(20, 30, 1)) def test_stop_before_start(): - e = ScheduleEntry(name="t", start=20, stop=10, interval=1, action="logger") + e = ScheduleEntry( + name="t", start=20, stop=10, interval=1, action="test_monitor_sigan" + ) assert list(e.get_remaining_times()) == list(range(0)) def test_no_interval_is_one_shot(): """Leaving `interval` blank should indicate "one-shot" entry.""" - e = ScheduleEntry(name="t", action="logger") + e = ScheduleEntry(name="t", action="test_monitor_sigan") remaining_times = list(e.get_remaining_times()) assert len(remaining_times) == 1 @@ -72,7 +76,7 @@ def test_no_interval_is_one_shot(): def test_no_interval_with_start_is_one_shot(): """Specifying start should not affect number of times.""" - e = ScheduleEntry(name="t", action="logger", start=1) + e = ScheduleEntry(name="t", action="test_monitor_sigan", start=1) remaining_times = list(e.get_remaining_times()) assert len(remaining_times) == 1 @@ -88,17 +92,19 @@ def test_no_interval_with_start_is_one_shot(): def test_no_interval_future_start(testclock): """One-shot entry should wait for start.""" # recall current t=0 so start=1 is 1 second in the future - e = ScheduleEntry(name="t", action="logger", start=1) + e = ScheduleEntry(name="t", action="test_monitor_sigan", start=1) assert not e.take_pending() def test_bad_interval_raises(): with pytest.raises(ValidationError): - ScheduleEntry(name="t", interval=-1, action="logger").clean_fields() + ScheduleEntry(name="t", interval=-1, action="test_monitor_sigan").clean_fields() with pytest.raises(ValidationError): - ScheduleEntry(name="t", interval=0, action="logger").clean_fields() + ScheduleEntry(name="t", interval=0, action="test_monitor_sigan").clean_fields() with pytest.raises(ValidationError): - ScheduleEntry(name="t", interval=0.1, action="logger").clean_fields() + ScheduleEntry( + name="t", interval=0.1, action="test_monitor_sigan" + ).clean_fields() def test_bad_action_raises(): @@ -108,22 +114,22 @@ def test_bad_action_raises(): def test_bad_name_raises(): with pytest.raises(ValidationError): # whitespace - ScheduleEntry(name="test 1", action="logger").clean_fields() + ScheduleEntry(name="test 1", action="test_monitor_sigan").clean_fields() with pytest.raises(ValidationError): # punctuation other than "_-" - ScheduleEntry(name="test1!", action="logger").clean_fields() + ScheduleEntry(name="test1!", action="test_monitor_sigan").clean_fields() # ok - ScheduleEntry(name="_test-Stuff123", action="logger").clean_fields() + ScheduleEntry(name="_test-Stuff123", action="test_monitor_sigan").clean_fields() def test_non_unique_name_raises(user): - ScheduleEntry(name="t", action="logger", owner=user).save() + ScheduleEntry(name="t", action="test_monitor_sigan", owner=user).save() with pytest.raises(ValidationError): - ScheduleEntry(name="t", action="logger", owner=user).full_clean() + ScheduleEntry(name="t", action="test_monitor_sigan", owner=user).full_clean() def test_defaults(): - entry = ScheduleEntry(name="t", action="logger") + entry = ScheduleEntry(name="t", action="test_monitor_sigan") assert entry.priority == DEFAULT_PRIORITY assert entry.start is not None assert entry.stop is None @@ -132,4 +138,4 @@ def test_defaults(): def test_str(): - str(ScheduleEntry(name="t", action="logger")) + str(ScheduleEntry(name="t", action="test_monitor_sigan")) diff --git a/src/schedule/tests/test_serializers.py b/src/schedule/tests/test_serializers.py index e7d0aa25..ce26fecc 100644 --- a/src/schedule/tests/test_serializers.py +++ b/src/schedule/tests/test_serializers.py @@ -17,39 +17,51 @@ [ # A name and action should be the minimum acceptable entry # i.e., (one-shot, ASAP) - {"name": "test", "action": "logger"}, + {"name": "test", "action": "test_monitor_sigan"}, # Stop 10 seconds after starting, start ASAP - {"name": "test", "action": "logger", "relative_stop": 10}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 10}, # Min integer interval ok - {"name": "test", "action": "logger", "interval": 10}, + {"name": "test", "action": "test_monitor_sigan", "interval": 10}, # Max priority ok - {"name": "test", "action": "logger", "priority": 19}, + {"name": "test", "action": "test_monitor_sigan", "priority": 19}, # Min user priority ok - {"name": "test", "action": "logger", "priority": 0}, + {"name": "test", "action": "test_monitor_sigan", "priority": 0}, # Stop 10 seconds after starting; start at absolute time { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25Z", "relative_stop": 10, }, # Start and stop at absolute time; equivalent to above { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25Z", "absolute_stop": "2018-03-16T17:12:35Z", }, # 'stop' and 'absolute_stop' are synonyms - {"name": "test", "action": "logger", "stop": "2018-03-16T17:12:35.0Z"}, + { + "name": "test", + "action": "test_monitor_sigan", + "stop": "2018-03-16T17:12:35.0Z", + }, # Subseconds are optional - {"name": "test", "action": "logger", "start": "2018-03-16T17:12:35Z"}, + { + "name": "test", + "action": "test_monitor_sigan", + "start": "2018-03-16T17:12:35Z", + }, # Sensor is timezone-aware - {"name": "test", "action": "logger", "start": "2018-03-22T13:53:25-06:00"}, + { + "name": "test", + "action": "test_monitor_sigan", + "start": "2018-03-22T13:53:25-06:00", + }, # All non-boolean, non-required fields accepts null to mean not defined { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": None, "absolute_stop": None, "relative_stop": None, @@ -60,7 +72,7 @@ "callback_url": None, }, # Explicit validate_only is valid - {"name": "test", "action": "logger", "validate_only": False}, + {"name": "test", "action": "test_monitor_sigan", "validate_only": False}, ], ) def test_valid_user_entries(entry_json, user): @@ -76,39 +88,51 @@ def test_valid_user_entries(entry_json, user): [ # A name and action should be the minimum acceptable entry # i.e., (one-shot, ASAP) - {"name": "test", "action": "logger"}, + {"name": "test", "action": "test_monitor_sigan"}, # Stop 10 seconds after starting, start ASAP - {"name": "test", "action": "logger", "relative_stop": 10}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 10}, # Min integer interval ok - {"name": "test", "action": "logger", "interval": 10}, + {"name": "test", "action": "test_monitor_sigan", "interval": 10}, # Max priority ok - {"name": "test", "action": "logger", "priority": 19}, + {"name": "test", "action": "test_monitor_sigan", "priority": 19}, # Min admin priority ok - {"name": "test", "action": "logger", "priority": -20}, + {"name": "test", "action": "test_monitor_sigan", "priority": -20}, # Stop 10 seconds after starting; start at absolute time { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25Z", "relative_stop": 10, }, # Start and stop at absolute time; equivalent to above { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25Z", "absolute_stop": "2018-03-16T17:12:35Z", }, # 'stop' and 'absolute_stop' are synonyms - {"name": "test", "action": "logger", "stop": "2018-03-16T17:12:35.0Z"}, + { + "name": "test", + "action": "test_monitor_sigan", + "stop": "2018-03-16T17:12:35.0Z", + }, # Subseconds are optional - {"name": "test", "action": "logger", "start": "2018-03-16T17:12:35Z"}, + { + "name": "test", + "action": "test_monitor_sigan", + "start": "2018-03-16T17:12:35Z", + }, # Sensor is timezone-aware - {"name": "test", "action": "logger", "start": "2018-03-22T13:53:25-06:00"}, + { + "name": "test", + "action": "test_monitor_sigan", + "start": "2018-03-22T13:53:25-06:00", + }, # All non-boolean, non-required fields accepts null to mean not defined { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": None, "absolute_stop": None, "relative_stop": None, @@ -119,9 +143,9 @@ def test_valid_user_entries(entry_json, user): "callback_url": None, }, # Explicit validate_only is valid - {"name": "test", "action": "logger", "validate_only": False}, + {"name": "test", "action": "test_monitor_sigan", "validate_only": False}, # Admin can create private entries - {"name": "test", "action": "logger"}, + {"name": "test", "action": "test_monitor_sigan"}, ], ) def test_valid_admin_entries(entry_json, user): @@ -136,44 +160,44 @@ def test_valid_admin_entries(entry_json, user): "entry_json", [ # name is a required field - {"action": "logger"}, + {"action": "test_monitor_sigan"}, # action is a required field {"name": "test"}, # non-integer priority - {"name": "test", "action": "logger", "priority": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "priority": 3.14}, # priority greater than max (19) - {"name": "test", "action": "logger", "priority": 20}, + {"name": "test", "action": "test_monitor_sigan", "priority": 20}, # non-integer interval - {"name": "test", "action": "logger", "interval": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "interval": 3.14}, # zero interval - {"name": "test", "action": "logger", "interval": 0}, + {"name": "test", "action": "test_monitor_sigan", "interval": 0}, # negative interval - {"name": "test", "action": "logger", "interval": -1}, + {"name": "test", "action": "test_monitor_sigan", "interval": -1}, # can't interpret both absolute and relative stop { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25.0Z", "absolute_stop": "2018-03-16T17:12:35.0Z", "relative_stop": 10, }, # 0 relative_stop - {"name": "test", "action": "logger", "relative_stop": 0}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 0}, # negative relative_stop - {"name": "test", "action": "logger", "relative_stop": -10}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": -10}, # non-integer relative_stop - {"name": "test", "action": "logger", "relative_stop": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 3.14}, # stop is before start { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:35Z", "stop": "2018-03-16T17:12:30Z", }, # stop is same as start { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:35Z", "stop": "2018-03-16T17:12:35Z", }, @@ -190,46 +214,46 @@ def test_invalid_user_entries(entry_json): "entry_json", [ # name is a required field - {"action": "logger"}, + {"action": "test_monitor_sigan"}, # action is a required field {"name": "test"}, # non-integer priority - {"name": "test", "action": "logger", "priority": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "priority": 3.14}, # priority less than min (for admin) - {"name": "test", "action": "logger", "priority": -21}, + {"name": "test", "action": "test_monitor_sigan", "priority": -21}, # priority greater than max (19) - {"name": "test", "action": "logger", "priority": 20}, + {"name": "test", "action": "test_monitor_sigan", "priority": 20}, # non-integer interval - {"name": "test", "action": "logger", "interval": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "interval": 3.14}, # zero interval - {"name": "test", "action": "logger", "interval": 0}, + {"name": "test", "action": "test_monitor_sigan", "interval": 0}, # negative interval - {"name": "test", "action": "logger", "interval": -1}, + {"name": "test", "action": "test_monitor_sigan", "interval": -1}, # can't interpret both absolute and relative stop { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:25.0Z", "absolute_stop": "2018-03-16T17:12:35.0Z", "relative_stop": 10, }, # 0 relative_stop - {"name": "test", "action": "logger", "relative_stop": 0}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 0}, # negative relative_stop - {"name": "test", "action": "logger", "relative_stop": -10}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": -10}, # non-integer relative_stop - {"name": "test", "action": "logger", "relative_stop": 3.14}, + {"name": "test", "action": "test_monitor_sigan", "relative_stop": 3.14}, # stop is before start { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:35Z", "stop": "2018-03-16T17:12:30Z", }, # stop is same as start { "name": "test", - "action": "logger", + "action": "test_monitor_sigan", "start": "2018-03-16T17:12:35Z", "stop": "2018-03-16T17:12:35Z", }, @@ -247,7 +271,9 @@ def test_invalid_admin_entries(entry_json): def test_serialized_fields(admin_client): """Certain fields on the schedule entry model should be serialized.""" - rjson = post_schedule(admin_client, {"name": "test", "action": "logger"}) + rjson = post_schedule( + admin_client, {"name": "test", "action": "test_monitor_sigan"} + ) # nullable fields assert "interval" in rjson @@ -274,6 +300,8 @@ def test_serialized_fields(admin_client): def test_non_serialized_fields(admin_client): """Certain fields on the schedule entry model should not be serialized.""" - rjson = post_schedule(admin_client, {"name": "test", "action": "logger"}) + rjson = post_schedule( + admin_client, {"name": "test", "action": "test_monitor_sigan"} + ) assert "relative_stop" not in rjson diff --git a/src/schedule/tests/utils.py b/src/schedule/tests/utils.py index ac8304be..3aa6b015 100644 --- a/src/schedule/tests/utils.py +++ b/src/schedule/tests/utils.py @@ -8,15 +8,15 @@ EMPTY_SCHEDULE_RESPONSE = [] -TEST_SCHEDULE_ENTRY = {"name": "test", "action": "logger"} +TEST_SCHEDULE_ENTRY = {"name": "test", "action": "test_monitor_sigan"} TEST_ALTERNATE_SCHEDULE_ENTRY = { "name": "test_alternate", - "action": "logger", + "action": "test_monitor_sigan", "priority": 5, } -TEST_PRIVATE_SCHEDULE_ENTRY = {"name": "test_private", "action": "logger"} +TEST_PRIVATE_SCHEDULE_ENTRY = {"name": "test_private", "action": "test_monitor_sigan"} def post_schedule(client, entry, expected_status=status.HTTP_201_CREATED): diff --git a/src/scheduler/tests/test_scheduler.py b/src/scheduler/tests/test_scheduler.py index 0dbede07..26577cf5 100644 --- a/src/scheduler/tests/test_scheduler.py +++ b/src/scheduler/tests/test_scheduler.py @@ -21,7 +21,7 @@ @pytest.mark.django_db def test_populate_queue(test_scheduler): """An entry in the schedule should be added to a read-only task queue.""" - create_entry("test", 1, 0, 5, 1, "logger") + create_entry("test", 1, 0, 5, 1, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) # now=0, so task with time 0 is run assert [e.time for e in s.task_queue] == [1, 2, 3, 4] @@ -32,8 +32,8 @@ def test_priority(test_scheduler): """A task with lower priority number should sort higher in task queue.""" lopri = 20 hipri = 10 - create_entry("lopri", lopri, 0, 5, 1, "logger") - create_entry("hipri", hipri, 0, 5, 1, "logger") + create_entry("lopri", lopri, 0, 5, 1, "test_monitor_sigan") + create_entry("hipri", hipri, 0, 5, 1, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) q = s.task_queue.to_list() @@ -45,7 +45,7 @@ def test_priority(test_scheduler): @pytest.mark.django_db def test_future_start(test_scheduler): """An entry with start time in future should remain in schedule.""" - create_entry("t", 1, 50, 100, 1, "logger") + create_entry("t", 1, 50, 100, 1, "test_monitor_sigan") test_scheduler.run(blocking=False) s = test_scheduler assert len(s.task_queue) == 0 @@ -75,11 +75,11 @@ def test_calls_actions(test_scheduler): @pytest.mark.django_db def test_add_entry(test_scheduler): """Creating a new entry instance adds it to the current schedule.""" - create_entry("t1", 10, 1, 100, 5, "logger") + create_entry("t1", 10, 1, 100, 5, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) advance_testclock(s.timefn, 49) - create_entry("t2", 20, 50, 300, 5, "logger") + create_entry("t2", 20, 50, 300, 5, "test_monitor_sigan") s.run(blocking=False) assert len(s.task_queue) == 20 assert s.task_queue[0].priority == 20 @@ -88,8 +88,8 @@ def test_add_entry(test_scheduler): @pytest.mark.django_db def test_remove_entry_by_delete(test_scheduler): """An entry is removed from schedule if it's deleted.""" - e1 = create_entry("t1", 10, 1, 300, 5, "logger") - e2 = create_entry("t2", 20, 50, 300, 5, "logger") + e1 = create_entry("t1", 10, 1, 300, 5, "test_monitor_sigan") + e2 = create_entry("t2", 20, 50, 300, 5, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) advance_testclock(s.timefn, 10) @@ -102,8 +102,8 @@ def test_remove_entry_by_delete(test_scheduler): @pytest.mark.django_db def test_remove_entry_by_cancel(test_scheduler): """scheduler.cancel removes an entry from schedule without deleting it.""" - e1 = create_entry("t1", 10, 1, 300, 5, "logger") - e2 = create_entry("t2", 20, 50, 300, 5, "logger") + e1 = create_entry("t1", 10, 1, 300, 5, "test_monitor_sigan") + e2 = create_entry("t2", 20, 50, 300, 5, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) advance_testclock(s.timefn, 10) @@ -116,7 +116,7 @@ def test_remove_entry_by_cancel(test_scheduler): @pytest.mark.django_db def test_start_stop(test_scheduler): """Calling stop on started scheduler thread should cause thread exit.""" - create_entry("t", 1, 1, 100, 5, "logger") + create_entry("t", 1, 1, 100, 5, "test_monitor_sigan") s = test_scheduler s.start() time.sleep(0.02) # hit minimum_duration @@ -130,7 +130,7 @@ def test_start_stop(test_scheduler): @pytest.mark.django_db def test_run_completes(test_scheduler): """The scheduler should return to idle state after schedule completes.""" - create_entry("t", 1, None, None, None, "logger") + create_entry("t", 1, None, None, None, "test_monitor_sigan") s = test_scheduler s.start() time.sleep(0.1) # hit minimum_duration @@ -159,7 +159,7 @@ def test_survives_failed_action(test_scheduler): @pytest.mark.django_db def test_compress_past_times(test_scheduler): """Multiple task times in the past should be compressed to one.""" - create_entry("t", 1, -10, 5, 1, "logger") + create_entry("t", 1, -10, 5, 1, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) # past times -10 through 0 are compressed and a single task is run, @@ -170,7 +170,7 @@ def test_compress_past_times(test_scheduler): @pytest.mark.django_db def test_compress_past_times_offset(test_scheduler): """Multiple task times in the past should be compressed to one.""" - create_entry("t", 1, -2, 14, 4, "logger") + create_entry("t", 1, -2, 14, 4, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) # past time -2 is run, then 2, 6, and 10 are queued @@ -182,7 +182,7 @@ def test_compress_past_times_offset(test_scheduler): @pytest.mark.django_db def test_next_task_time_value_when_start_changes(test_scheduler): """When an entry's start value changes, update `next_task_time`.""" - entry = create_entry("t", 1, 1, 10, 1, "logger") + entry = create_entry("t", 1, 1, 10, 1, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) assert entry.next_task_time == 1 @@ -218,7 +218,7 @@ def test_next_task_time_value_when_start_changes(test_scheduler): @pytest.mark.django_db def test_next_task_time_value_when_interval_changes(test_scheduler): """When an entry's interval value changes, update `next_task_time`.""" - entry = create_entry("t", 1, 1, 100, 1, "logger") + entry = create_entry("t", 1, 1, 100, 1, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) assert entry.next_task_time == 1 @@ -246,7 +246,7 @@ def test_next_task_time_value_when_interval_changes(test_scheduler): @pytest.mark.django_db def test_one_shot(test_scheduler): """If no start or interval given, entry should be run once and removed.""" - create_entry("t", 1, None, None, None, "logger") + create_entry("t", 1, None, None, None, "test_monitor_sigan") s = test_scheduler advance_testclock(s.timefn, 1) s.run(blocking=False) @@ -257,7 +257,7 @@ def test_one_shot(test_scheduler): @pytest.mark.django_db def test_task_queue(test_scheduler): """The scheduler should maintain a queue of upcoming tasks.""" - e = create_entry("t", 1, 1, 100, 5, "logger") + e = create_entry("t", 1, 1, 100, 5, "test_monitor_sigan") s = test_scheduler # upcoming tasks are queued @@ -282,7 +282,7 @@ def test_task_queue(test_scheduler): @pytest.mark.django_db def test_clearing_schedule_clears_task_queue(test_scheduler): """The scheduler should empty task_queue when schedule is deleted.""" - create_entry("t", 1, 1, 100, 5, "logger") + create_entry("t", 1, 1, 100, 5, "test_monitor_sigan") s = test_scheduler s.run(blocking=False) # queue first 10 tasks assert len(s.task_queue) == 10 @@ -416,7 +416,7 @@ def cb_request_handler(sess, resp): def test_notification_failed_status_unknown_host(test_scheduler): with requests_mock.Mocker() as m: callback_url = "https://results" - entry = create_entry("t", 1, 1, 100, 5, "logger", callback_url) + entry = create_entry("t", 1, 1, 100, 5, "test_monitor_sigan", callback_url) entry.save() token = entry.owner.auth_token m.post( @@ -437,7 +437,7 @@ def test_notification_failed_status_unknown_host(test_scheduler): @pytest.mark.django_db def test_notification_failed_status_request_ok_false(test_scheduler): - entry = create_entry("t", 1, 1, 100, 5, "logger") + entry = create_entry("t", 1, 1, 100, 5, "test_monitor_sigan") entry.save() entry.refresh_from_db() print("entry = " + entry.name) diff --git a/src/sensor/settings.py b/src/sensor/settings.py index d1a46a9f..f8625a36 100644 --- a/src/sensor/settings.py +++ b/src/sensor/settings.py @@ -347,6 +347,8 @@ DEVICE_MODEL = env("DEVICE_MODEL", default="RSA507A") SIGAN_MODULE = "scos_actions.hardware.mocks.mock_sigan" SIGAN_CLASS = "MockSignalAnalyzer" + GPS_MODULE = "scos_actions.hardware.mocks.mock_gps" + GPS_CLASS = "MockGPS" else: DATABASES = { "default": { @@ -361,6 +363,8 @@ DEVICE_MODEL = env("DEVICE_MODEL", default=None) SIGAN_MODULE = env.str("SIGAN_MODULE", default=None) SIGAN_CLASS = env.str("SIGAN_CLASS", default=None) + GPS_MODULE = env.str("GPS_MODULE", default=None) + GPS_CLASS = env.str("GPS_CLASS", default=None) if not IN_DOCKER: DATABASES["default"]["HOST"] = "localhost" diff --git a/src/status/views.py b/src/status/views.py index c7d070d1..c64b2789 100644 --- a/src/status/views.py +++ b/src/status/views.py @@ -68,9 +68,10 @@ def get_software_version(): sensor_loader.sensor.signal_analyzer.api_version ) if sensor_loader.sensor.signal_analyzer.plugin_version is not None: - software_version["scos_sigan_plugin"] = ( - sensor_loader.sensor.signal_analyzer.plugin_version - ) + software_version["scos_sigan_plugin"] = { + "name": sensor_loader.sensor.signal_analyzer.plugin_name, + "version": sensor_loader.sensor.signal_analyzer.plugin_version, + } logger.debug(software_version) return software_version