-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathaction.py
136 lines (110 loc) · 4.56 KB
/
action.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
import logging
from abc import ABC, abstractmethod
from copy import deepcopy
from typing import Optional
from scos_actions.hardware.sensor import Sensor
from scos_actions.metadata.sigmf_builder import SigMFBuilder
from scos_actions.metadata.structs import ntia_scos, ntia_sensor
from scos_actions.utils import ParameterException, get_parameter
logger = logging.getLogger(__name__)
class Action(ABC):
"""The action base class.
To create an action, create a subclass of `Action` with a descriptive
docstring and override the `__call__` method.
The scheduler reports the 'success' or 'failure' of an action by the
following convention:
* If at any point or for any reason that `__call__` function raises an
exception, the task is marked a 'failure' and `str(err)` is provided
as a detail to the user, where `err` is the raised Exception object.
* If the `__call__` function returns normally, the task was a 'success',
and if the value returned to the scheduler is a string, it will be
added to the task result's detail field.
"""
PRESELECTOR_PATH_KEY = "rf_path"
def __init__(self, parameters: dict):
self._sensor = None
self.parameters = deepcopy(parameters)
self.sigmf_builder = None
def configure(self, params: dict):
self.configure_sigan(params)
self.configure_preselector(params)
@property
def sensor(self):
return self._sensor
@sensor.setter
def sensor(self, value: Sensor):
self._sensor = value
def configure_sigan(self, params: dict):
for key, value in params.items():
if hasattr(self.sensor.signal_analyzer, key):
logger.debug(f"Applying setting to sigan: {key}: {value}")
setattr(self.sensor.signal_analyzer, key, value)
else:
logger.debug(f"Sigan does not have attribute {key}")
def configure_preselector(self, params: dict):
preselector = self.sensor.preselector
if self.sensor.has_configurable_preselector:
if self.PRESELECTOR_PATH_KEY in params:
path = params[self.PRESELECTOR_PATH_KEY]
logger.debug(f"Setting preselector RF path: {path}")
preselector.set_state(path)
else:
# Require the RF path to be specified if the sensor has a preselector.
raise ParameterException(
f"No {self.PRESELECTOR_PATH_KEY} value specified in the YAML config."
)
else:
# No preselector in use, so do not require an RF path
pass
def get_sigmf_builder(self, schedule_entry: dict) -> None:
"""
Set the `sigmf_builder` instance variable to an initialized SigMFBuilder.
Schedule entry and action information will be populated using `ntia-scos`
fields, and sensor metadata will be filled using `ntia-sensor` fields.
"""
sigmf_builder = SigMFBuilder()
schedule_entry_cleaned = {
k: v
for k, v in schedule_entry.items()
if k in ["id", "name", "start", "stop", "interval", "priority", "roles"]
}
if "id" not in schedule_entry_cleaned:
# If there is no ID, reuse the "name" as the ID as well
schedule_entry_cleaned["id"] = schedule_entry_cleaned["name"]
schedule_entry_obj = ntia_scos.ScheduleEntry(**schedule_entry_cleaned)
sigmf_builder.set_schedule(schedule_entry_obj)
action_obj = ntia_scos.Action(
name=self.name,
description=self.description,
summary=self.summary,
)
sigmf_builder.set_action(action_obj)
if self.sensor.location is not None:
sigmf_builder.set_geolocation(self.sensor.location)
if self.sensor.capabilities is not None and hasattr(
self.sensor.capabilities, "sensor"
):
sigmf_builder.set_sensor(
ntia_sensor.Sensor(**self.sensor.capabilities["sensor"])
)
self.sigmf_builder = sigmf_builder
@property
def summary(self):
try:
return self.description.splitlines()[0]
except IndexError:
return "Summary not provided."
@property
def description(self):
return self.__doc__
@property
def name(self):
return get_parameter("name", self.parameters)
@abstractmethod
def __call__(
self,
sensor: Sensor = None,
schedule_entry: Optional[dict] = None,
task_id: Optional[int] = None,
):
pass