Skip to content

Commit 02c1444

Browse files
authored
Merge pull request #19 from stfc/refactor-client-handlers
refactor client-side handlers
2 parents a2b01c3 + 4d42304 commit 02c1444

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

48 files changed

+1136
-2094
lines changed
+56-71
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,24 @@
1-
# Adding New Preset To An Existing Preset Group
1+
# Adding A New Preset
22

3-
## **1. Add the preset name to the corresponding enum class in `openstackquery/enums/query_presets.py`**
3+
## **1. Add the preset name to the QueryPresets enum class in `openstackquery/enums/query_presets.py`**
44

55
e.g.
66
```python
7-
class QueryPresetsGeneric(QueryPresets):
7+
class QueryPresets(EnumWithAliases):
88
"""
99
Enum class which holds generic query comparison operators
1010
"""
1111

1212
EQUAL_TO = auto()
1313
...
14-
NEW_PRESET = auto() # <- we add this line to repesent a new preset enum belonging to the 'Generic' group
14+
NEW_PRESET = auto() # <- we add this line to represent a new preset enum belonging to the 'Generic' group
1515
```
1616

1717
(Optional) Add alias mappings for the preset - see [Adding Aliases](ADDING_ALIASES.md)
1818

19-
## **2. Edit the corresponding handler class in `openstackquery/handlers/client_side_handler_<preset-group>.py`.**
19+
## **2. Create a function to act as the client-side filter function for your new query preset
2020

21-
Here you must:
22-
- add a 'client-side' filter function as a method
23-
- add the mapping between the enum and filter function in self._filter_functions.
21+
Client-side filter functions are located in `openstackquery/handlers/client_side_filters.py`
2422

2523
The filter function must:
2624
- **take as input at least one parameter - `prop`**:
@@ -30,32 +28,41 @@ The filter function must:
3028
- `True` if the prop passes filter
3129
- `False` if not
3230

33-
e.g. editing `client_side_handler_generic.py`
3431
```python
35-
class ClientSideHandlerGeneric(ClientSideHandler):
36-
...
32+
def prop_new_preset_filter_func(self, prop: Any, arg1, arg2):
33+
"""
34+
A new preset filter - takes a property value, performs some logic and returns a boolean if
35+
property passes a filter or not
36+
:param prop: property value to check
37+
:param arg1: some arg the filter uses
38+
:param arg2: some other arg the filter uses
39+
:returns: True or False
40+
"""
41+
# Define your filter logic here
42+
```
3743

38-
def __init__(self, filter_function_mappings: PresetPropMappings):
39-
super().__init__(filter_function_mappings)
44+
## **3. Edit the corresponding handler class in `openstackquery/handlers/client_side_handler.py`.**
4045

41-
self._filter_functions = {
42-
QueryPresetsGeneric.EQUAL_TO: self._prop_equal_to,
43-
...
44-
QueryPresetsGeneric.NEW_PRESET: self._new_preset_filter_func # <- 2) add the enum-to-function mapping
45-
}
46+
Here you must:
47+
- add a 'client-side' filter function as a method
48+
- add the mapping between the enum and filter function in self._filter_functions.
49+
50+
e.g. editing `client_side_handler_generic.py`
51+
```python
52+
from openstackquery.handlers.client_side_filters import (
53+
prop_new_preset_filter_func # newly made filter func
54+
)
4655

56+
class ClientSideHandler(HandlerBase):
4757
...
4858

49-
def _new_preset_filter_func(self, prop: Any, arg1, arg2):
50-
"""
51-
A new preset filter - takes a property value, performs some logic and returns a boolean if
52-
property passes a filter or not
53-
:param prop: property value to check
54-
:param arg1: some arg the filter uses
55-
:param arg2: some other arg the filter uses
56-
:returns: True or False
57-
"""
58-
... # Define your preset logic here
59+
def __init__(self, preset_prop_mappings: ClientSidePresetPropertyMappings):
60+
self._filter_functions = {
61+
QueryPresets.EQUAL_TO: self._prop_equal_to,
62+
...
63+
QueryPresets.NEW_PRESET: prop_new_preset_filter_func # <- 2) add the enum-to-function mapping
64+
}
65+
...
5966
```
6067

6168
## **3. Edit the query class mappings for each Query class you wish to use the preset in**
@@ -76,24 +83,22 @@ class ServerMapping(MappingInterface):
7683
...
7784

7885
@staticmethod
79-
def get_client_side_handlers() -> ServerSideHandler:
86+
def get_client_side_handler() -> ClientSideHandler:
8087
...
81-
return QueryClientSideHandlers(
82-
# set generic query preset mappings
83-
generic_handler=ClientSideHandlerGeneric(
84-
{
85-
# Line below maps EQUAL_TO preset on all available properties
86-
# ["*"] - represents all props
87-
QueryPresetsGeneric.EQUAL_TO: ["*"],
88-
...
89-
# Line below maps our 'new preset' to two properties which the preset can run on
90-
# Running the preset on any other property leads to an error
91-
QueryPresetsGeneric.NEW_PRESET: [
92-
ServerProperties.SERVER_ID,
93-
ServerProperties.SERVER_NAME
94-
]
95-
}
96-
),
88+
return ClientSideHandler(
89+
90+
{
91+
# Line below maps EQUAL_TO preset on all available properties
92+
# ["*"] - represents all props
93+
QueryPresets.EQUAL_TO: ["*"],
94+
# ...
95+
# Line below maps our 'new preset' to two properties which the preset can run on
96+
# Running the preset on any other property leads to an error
97+
QueryPresets.NEW_PRESET: [
98+
ServerProperties.SERVER_ID,
99+
ServerProperties.SERVER_NAME
100+
]
101+
}
97102
)
98103
...
99104
```
@@ -110,40 +115,20 @@ e.g. Adding server-side mapping for `QueryPresetsGeneric.NEW_PRESET` to `ServerQ
110115
```python
111116

112117
class ServerMapping(MappingInterface):
113-
...
118+
#...
114119

115120
@staticmethod
116121
def get_server_side_handler() -> ServerSideHandler:
117-
...
122+
#...
118123
return ServerSideHandler(
119124
{
120-
QueryPresetsGeneric.NEW_PRESET: {
125+
#...
126+
QueryPresets.NEW_PRESET: {
121127
# adding a server-side mapping for NEW_PRESET when given SERVER_ID
122128
ServerProperties.SERVER_ID: lambda value, arg1, arg2:
123129
{"server-side-kwarg": value, "server-side-arg1": arg1, "server-side-arg2": arg2}
124130
}
125131
}
126-
...
132+
)
133+
#...
127134
```
128-
129-
# **Adding a new preset group**
130-
131-
As stated above - presets are grouped based on the datatype of the property the act on. If you need another preset
132-
group - you can add one like so:
133-
134-
1. Create a new preset group class in `openstackquery/enums/query_presets.py`
135-
- it inherits from base class QueryPresets
136-
137-
138-
2. Create new client side handler in `openstackquery/handlers/client_side_handler_<preset>.py`
139-
- it inherits from base class `ClientSideHandler` - `openstackquery/handlers/client_side_handler.py`.
140-
141-
142-
3. Add your preset as a attribute in query_client_side_handlers dataclass
143-
- located in `openstackquery/structs/query_client_side_handlers.py`
144-
145-
146-
4. Follow steps mentioned above to add new presets to the new preset group class you've created
147-
148-
5. Edit `QueryPresets` type declaration at the bottom of the file `openstackquery/enums/query_presets.py` and add your new
149-
preset class to it

docs/developer_docs/ADDING_NEW_QUERIES.md

+31-38
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ class ResourceProperties(PropEnum):
1616

1717
# add enums for each property you found here like so
1818
PROP_1 = auto() # replace `PROP_1` with property name
19-
...
19+
# ...
2020

2121
@staticmethod
2222
def get_prop_mapping(prop):
@@ -62,18 +62,20 @@ class ResourceRunner(RunnerWrapper):
6262

6363
def _parse_meta_params(
6464
self,
65-
conn: OpenstackConnection
65+
conn: OpenstackConnection,
6666

6767
# define meta params that could be passed
6868
# they should all be optional, if no meta-params are passed, the query should still work with default values
6969
meta_param_1 = None,
7070
meta_param_2 = None,
71-
...
71+
# ...
7272
):
73+
pass
7374
# Define logic here that will alter the keyword arguments that will be passed to the openstacksdk command
7475
# based on the meta-parameter values passed in. This method should return a dictionary which will be merged with
7576
# server-side filters and be used when calling the specific openstacksdk query which gets the resources being queried for
76-
...
77+
78+
# ...
7779

7880
def _run_query(
7981
self,
@@ -153,55 +155,46 @@ we can add mapping like so:
153155
...
154156
return {
155157
# Here we map the prop_1 enum stored in ResourceProperties to the user_id enum stored in UserProperties
158+
# Values must be a list - because a query property may map to multiple other query properties
156159
# These two properties MUST store the same info
157-
ResourceProperties.PROP_1: UserProperties.USER_ID,
160+
ResourceProperties.PROP_1: [UserProperties.USER_ID],
158161
}
159162
...
160163
```
161164

162-
### 3d. Set the client_side_handlers
165+
### 3d. Set the client_side_handler
163166
We must define which preset-property pair can be used together when calling `where()` on this Query class.
164167

165-
The `get_client_side_handlers` class is where we define these mappings.
166-
This class creates a Dataclass called `QueryClientSideHandlers` from the mappings we define.
168+
The `get_client_side_handler` class is where we define these mappings.
169+
This class creates an object of `ClientSideHandler` from which mappings can be stored.
167170

168171
Here you must:
169172
1. Evaluate which presets you want to the query to accept and which properties they should work on
170-
2. You must add the presets like so:
173+
2. Add the presets like so:
171174

172175
```python
173-
174-
def get_client_side_handlers() -> ServerSideHandler:
176+
# This is required - at least one preset mapping must be defined here
177+
def get_client_side_handlers() -> ClientSideHandler:
175178
...
176-
return QueryClientSideHandlers(
177-
# generic_handler = set preset-property mappings that belong to generic presets
178-
# This is required - at least one preset mapping must be defined here
179-
generic_handler=ClientSideHandlerGeneric(
180-
{
181-
# Line below maps EQUAL_TO preset on all available properties
182-
# ["*"] - represents all props
183-
QueryPresetsGeneric.EQUAL_TO: ["*"],
184-
...
185-
}
186-
),
187-
# do the same for each of these (all optional)
188-
string_handler=ClientSideHandlerString(
189-
{
190-
# Line Below maps MATCHES_REGEX preset on PROP_1 only
191-
# All other properties are invalid
192-
QueryPresetsString.MATCHES_REGEX: [ResourceProperties.PROP_1]
193-
}
194-
),
195-
# we don't want any datetime presets to be valid - so set it to None
196-
datetime_handler=None,
197-
198-
# we don't want any integer presets to be valid - so set it to None
199-
integer_handler=None
179+
return ClientSideHandler(
180+
{
181+
# Line below maps EQUAL_TO preset on all available properties
182+
# ["*"] - represents all props
183+
QueryPresets.EQUAL_TO: ["*"],
184+
#...
185+
186+
# Line Below maps MATCHES_REGEX preset on PROP_1 only
187+
# All other properties are invalid
188+
QueryPresets.MATCHES_REGEX: [ResourceProperties.PROP_1]
189+
190+
#...
191+
# keep going until all presets you want to allow are mapped
192+
}
200193
)
201-
...
194+
# ...
202195
```
203196

204-
## 3e (Optional) Map client-side filters
197+
## 3e (Optional) Map server-side filters
205198

206199
To add a server-side filter you must:
207200
1. Read the Openstack API documentation for each Query the preset works on
@@ -234,7 +227,7 @@ will call `QueryFactory` with the `ResourceMapping` class we just created.
234227

235228
e.g. Add this function to `openstackquery/api/query_objects.py` (as usual, replace `Resource` with the name of the openstack resource your querying)
236229
```python
237-
def ResourceQuery() -> QueryAPI:
230+
def ResourceQuery() -> "QueryAPI":
238231
"""
239232
Simple helper function to setup a query using a factory
240233
"""

0 commit comments

Comments
 (0)