osbot-utils 1.54.0__py3-none-any.whl → 1.55.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- osbot_utils/helpers/flows/Flow.py +28 -24
- osbot_utils/helpers/flows/Flow__Events.py +32 -0
- osbot_utils/helpers/flows/Task.py +22 -7
- osbot_utils/helpers/flows/{Flow__Config.py → models/Flow__Config.py} +1 -0
- osbot_utils/helpers/flows/models/Flow__Event.py +6 -0
- osbot_utils/helpers/flows/models/Flow__Event_Type.py +8 -0
- osbot_utils/helpers/flows/models/__init__.py +0 -0
- osbot_utils/version +1 -1
- {osbot_utils-1.54.0.dist-info → osbot_utils-1.55.0.dist-info}/METADATA +2 -2
- {osbot_utils-1.54.0.dist-info → osbot_utils-1.55.0.dist-info}/RECORD +12 -8
- {osbot_utils-1.54.0.dist-info → osbot_utils-1.55.0.dist-info}/LICENSE +0 -0
- {osbot_utils-1.54.0.dist-info → osbot_utils-1.55.0.dist-info}/WHEEL +0 -0
@@ -2,14 +2,15 @@ import asyncio
|
|
2
2
|
import logging
|
3
3
|
import typing
|
4
4
|
|
5
|
-
from osbot_utils.base_classes.Type_Safe
|
6
|
-
from osbot_utils.helpers.CFormat
|
7
|
-
from osbot_utils.helpers.flows.Flow__Config
|
8
|
-
from osbot_utils.
|
9
|
-
from osbot_utils.
|
10
|
-
from osbot_utils.utils.
|
11
|
-
from osbot_utils.utils.
|
12
|
-
from osbot_utils.utils.
|
5
|
+
from osbot_utils.base_classes.Type_Safe import Type_Safe
|
6
|
+
from osbot_utils.helpers.CFormat import CFormat, f_dark_grey, f_magenta, f_bold
|
7
|
+
from osbot_utils.helpers.flows.models.Flow__Config import Flow__Config
|
8
|
+
from osbot_utils.helpers.flows.Flow__Events import flow_events
|
9
|
+
from osbot_utils.testing.Stdout import Stdout
|
10
|
+
from osbot_utils.utils.Misc import random_id, lower
|
11
|
+
from osbot_utils.utils.Python_Logger import Python_Logger
|
12
|
+
from osbot_utils.utils.Str import ansis_to_texts
|
13
|
+
from osbot_utils.utils.Threads import invoke_in_new_event_loop
|
13
14
|
|
14
15
|
FLOW__RANDOM_ID__PREFIX = 'flow_id__'
|
15
16
|
FLOW__RANDOM_NAME__PREFIX = 'flow_name__'
|
@@ -42,28 +43,25 @@ class Flow(Type_Safe):
|
|
42
43
|
if self.flow_config.log_to_console:
|
43
44
|
_.add_console_logger()
|
44
45
|
|
45
|
-
|
46
|
-
def debug(self, message):
|
47
|
-
self.logger.debug(message)
|
48
|
-
|
49
46
|
def create_flow(self):
|
50
47
|
self.set_flow_name()
|
51
|
-
self.
|
48
|
+
self.log_debug(f"Created flow run '{self.f__flow_id()}' for flow '{self.f__flow_name()}'")
|
52
49
|
|
53
50
|
def execute(self):
|
54
51
|
return self.execute_flow()
|
55
52
|
|
56
53
|
def execute_flow(self):
|
54
|
+
flow_events.on__flow__start(self)
|
57
55
|
if self.flow_config.log_to_memory:
|
58
56
|
self.logger.add_memory_logger() # todo: move to method that does pre-execute tasks
|
59
57
|
|
60
|
-
self.
|
58
|
+
self.log_debug(f"Executing flow run '{self.f__flow_id()}''")
|
61
59
|
try:
|
62
60
|
with Stdout() as stdout:
|
63
61
|
self.invoke_flow_target()
|
64
62
|
except Exception as error:
|
65
63
|
self.flow_error = error
|
66
|
-
self.
|
64
|
+
self.log_error(self.cformat.red(f"Error executing flow: {error}"))
|
67
65
|
|
68
66
|
self.log_captured_stdout (stdout)
|
69
67
|
self.print_flow_return_value ()
|
@@ -72,6 +70,7 @@ class Flow(Type_Safe):
|
|
72
70
|
if self.flow_config.log_to_memory:
|
73
71
|
self.captured_exec_logs = self.log_messages_with_colors()
|
74
72
|
self.logger.remove_memory_logger() # todo: move to method that does post-execute tasks
|
73
|
+
flow_events.on__flow__stop(self)
|
75
74
|
return self
|
76
75
|
|
77
76
|
def f__flow_id(self):
|
@@ -83,9 +82,6 @@ class Flow(Type_Safe):
|
|
83
82
|
def captured_logs(self):
|
84
83
|
return ansis_to_texts(self.captured_exec_logs)
|
85
84
|
|
86
|
-
def info(self, message):
|
87
|
-
self.logger.info(message)
|
88
|
-
|
89
85
|
|
90
86
|
async def invoke_flow_target__thread(self, flow): # this is a REALLY important method which is used to pin the flow object to the call stack
|
91
87
|
return await flow.flow_target(*flow.flow_args, **flow.flow_kwargs) # which is then used by the Task.find_flow method to find it
|
@@ -100,12 +96,23 @@ class Flow(Type_Safe):
|
|
100
96
|
def log_captured_stdout(self, stdout):
|
101
97
|
for line in stdout.value().splitlines():
|
102
98
|
if line:
|
103
|
-
self.
|
99
|
+
self.log_info(f_magenta(line))
|
104
100
|
if self.flow_config.print_logs:
|
105
101
|
print()
|
106
102
|
print()
|
107
103
|
self.print_log_messages()
|
108
104
|
|
105
|
+
def log_debug(self, message):
|
106
|
+
if self.flow_config.logging_enabled:
|
107
|
+
self.logger.debug(message)
|
108
|
+
|
109
|
+
def log_error(self, message):
|
110
|
+
if self.flow_config.logging_enabled:
|
111
|
+
self.logger.error(message)
|
112
|
+
|
113
|
+
def log_info(self, message):
|
114
|
+
if self.flow_config.logging_enabled:
|
115
|
+
self.logger.info(message)
|
109
116
|
|
110
117
|
def log_messages(self):
|
111
118
|
return ansis_to_texts(self.log_messages_with_colors())
|
@@ -128,12 +135,12 @@ class Flow(Type_Safe):
|
|
128
135
|
|
129
136
|
def print_flow_finished_message(self):
|
130
137
|
if self.flow_config.print_finished_message:
|
131
|
-
self.
|
138
|
+
self.log_debug(f"Finished flow run '{self.f__flow_id()}''")
|
132
139
|
|
133
140
|
def print_flow_return_value(self):
|
134
141
|
if self.flow_config.print_none_return_value is False and self.flow_return_value is None:
|
135
142
|
return
|
136
|
-
self.
|
143
|
+
self.log_debug(f"{f_dark_grey('Flow return value')}: {f_bold(self.flow_return_value)}")
|
137
144
|
|
138
145
|
|
139
146
|
|
@@ -170,6 +177,3 @@ class Flow(Type_Safe):
|
|
170
177
|
with self as _:
|
171
178
|
if not _.flow_id:
|
172
179
|
_.flow_id = self.random_flow_id()
|
173
|
-
#if not _.flow_name:
|
174
|
-
# _.flow_name = self.flow_target.__name__
|
175
|
-
#self.random_flow_name()
|
@@ -0,0 +1,32 @@
|
|
1
|
+
from osbot_utils.base_classes.Type_Safe import Type_Safe
|
2
|
+
from osbot_utils.helpers.flows.models.Flow__Event import Flow__Event
|
3
|
+
from osbot_utils.helpers.flows.models.Flow__Event_Type import Flow__Event_Type
|
4
|
+
|
5
|
+
|
6
|
+
class Flow_Events(Type_Safe):
|
7
|
+
event_listeners : list
|
8
|
+
|
9
|
+
def on__flow__start(self, flow):
|
10
|
+
flow_event = Flow__Event(event_type=Flow__Event_Type.FLOW_START, event_source=flow)
|
11
|
+
self.raise_event(flow_event)
|
12
|
+
|
13
|
+
def on__flow__stop(self, flow): # todo: see of flow_ended or flow_completed are better names
|
14
|
+
flow_event = Flow__Event(event_type=Flow__Event_Type.FLOW_STOP , event_source=flow)
|
15
|
+
self.raise_event(flow_event)
|
16
|
+
|
17
|
+
def on__task__start(self, task):
|
18
|
+
flow_event = Flow__Event(event_type=Flow__Event_Type.TASK_START, event_source=task)
|
19
|
+
self.raise_event(flow_event)
|
20
|
+
|
21
|
+
def on__task__stop(self, task): # todo: see of flow_ended or flow_completed are better names
|
22
|
+
flow_event = Flow__Event(event_type=Flow__Event_Type.TASK_STOP , event_source=task)
|
23
|
+
self.raise_event(flow_event)
|
24
|
+
|
25
|
+
def raise_event(self, flow_event):
|
26
|
+
for listener in self.event_listeners:
|
27
|
+
try:
|
28
|
+
listener(flow_event)
|
29
|
+
except Exception as error:
|
30
|
+
print(f"Error in listener: {error}")
|
31
|
+
|
32
|
+
flow_events = Flow_Events()
|
@@ -2,18 +2,19 @@ import asyncio
|
|
2
2
|
import inspect
|
3
3
|
import typing
|
4
4
|
|
5
|
+
from osbot_utils.utils.Misc import random_id, lower
|
5
6
|
from osbot_utils.helpers.Dependency_Manager import Dependency_Manager
|
6
|
-
from osbot_utils.
|
7
|
-
|
7
|
+
from osbot_utils.helpers.flows.Flow__Events import flow_events
|
8
8
|
from osbot_utils.testing.Stdout import Stdout
|
9
9
|
from osbot_utils.helpers.CFormat import CFormat, f_dark_grey, f_red, f_blue, f_bold
|
10
10
|
from osbot_utils.base_classes.Type_Safe import Type_Safe
|
11
11
|
from osbot_utils.helpers.flows.Flow import Flow
|
12
12
|
|
13
|
+
TASK__RANDOM_ID__PREFIX = 'task_id__'
|
13
14
|
|
14
15
|
class Task(Type_Safe):
|
15
16
|
data : dict # dict available to the task to add and collect data
|
16
|
-
task_id : str
|
17
|
+
task_id : str
|
17
18
|
task_name : str # make this the function mame
|
18
19
|
cformat : CFormat
|
19
20
|
resolved_args : tuple
|
@@ -26,6 +27,12 @@ class Task(Type_Safe):
|
|
26
27
|
task_error : Exception = None
|
27
28
|
raise_on_error : bool = True
|
28
29
|
|
30
|
+
def log_debug(self, message):
|
31
|
+
self.task_flow.log_debug(message)
|
32
|
+
|
33
|
+
def log_error(self, message):
|
34
|
+
self.task_flow.log_error(message)
|
35
|
+
|
29
36
|
def execute__sync(self):
|
30
37
|
self.execute__before()
|
31
38
|
self.execute__task_target__sync()
|
@@ -44,14 +51,18 @@ class Task(Type_Safe):
|
|
44
51
|
if not self.task_name and self.task_target:
|
45
52
|
self.task_name = self.task_target.__name__
|
46
53
|
|
54
|
+
if not self.task_id:
|
55
|
+
self.task_id = self.random_task_id()
|
56
|
+
|
47
57
|
self.task_flow.executed_tasks.append(self)
|
48
|
-
self.
|
58
|
+
self.log_debug(f"Executing task '{f_blue(self.task_name)}'")
|
49
59
|
dependency_manager = Dependency_Manager()
|
50
60
|
dependency_manager.add_dependency('this_task', self )
|
51
61
|
dependency_manager.add_dependency('this_flow', self.task_flow )
|
52
62
|
dependency_manager.add_dependency('task_data', self.data )
|
53
63
|
dependency_manager.add_dependency('flow_data', self.task_flow.data)
|
54
64
|
self.resolved_args, self.resolved_kwargs = dependency_manager.resolve_dependencies(self.task_target, *self.task_args, **self.task_kwargs)
|
65
|
+
flow_events.on__task__start(self)
|
55
66
|
|
56
67
|
def execute__task_target__sync(self):
|
57
68
|
try:
|
@@ -73,11 +84,13 @@ class Task(Type_Safe):
|
|
73
84
|
self.print_task_return_value()
|
74
85
|
|
75
86
|
if self.task_error:
|
76
|
-
self.
|
87
|
+
self.log_error(f_red(f"Error executing '{self.task_name}' task: {self.task_error}"))
|
77
88
|
if self.raise_on_error:
|
78
89
|
raise Exception(f"'{self.task_name}' failed and task raise_on_error was set to True. Stopping flow execution")
|
79
90
|
|
80
91
|
self.print_task_finished_message()
|
92
|
+
|
93
|
+
flow_events.on__task__stop(self)
|
81
94
|
return self.task_return_value
|
82
95
|
|
83
96
|
|
@@ -92,12 +105,14 @@ class Task(Type_Safe):
|
|
92
105
|
|
93
106
|
def print_task_finished_message(self):
|
94
107
|
if self.task_flow.flow_config.print_finished_message:
|
95
|
-
self.
|
108
|
+
self.log_debug(f"Finished task '{f_blue(self.task_name)}'")
|
96
109
|
|
97
110
|
def print_task_return_value(self):
|
98
111
|
flow_config = self.task_flow.flow_config
|
99
112
|
if flow_config.print_none_return_value is False and self.task_return_value is None:
|
100
113
|
return
|
101
|
-
self.
|
114
|
+
self.log_debug(f"{f_dark_grey('Task return value')}: {f_bold(self.task_return_value)}")
|
102
115
|
|
103
116
|
|
117
|
+
def random_task_id(self):
|
118
|
+
return lower(random_id(prefix=TASK__RANDOM_ID__PREFIX))
|
File without changes
|
osbot_utils/version
CHANGED
@@ -1 +1 @@
|
|
1
|
-
v1.
|
1
|
+
v1.55.0
|
@@ -1,6 +1,6 @@
|
|
1
1
|
Metadata-Version: 2.1
|
2
2
|
Name: osbot_utils
|
3
|
-
Version: 1.
|
3
|
+
Version: 1.55.0
|
4
4
|
Summary: OWASP Security Bot - Utils
|
5
5
|
Home-page: https://github.com/owasp-sbot/OSBot-Utils
|
6
6
|
License: MIT
|
@@ -22,7 +22,7 @@ Description-Content-Type: text/markdown
|
|
22
22
|
|
23
23
|
Powerful Python util methods and classes that simplify common apis and tasks.
|
24
24
|
|
25
|
-

|
26
26
|
[](https://codecov.io/gh/owasp-sbot/OSBot-Utils)
|
27
27
|
|
28
28
|
|
@@ -154,13 +154,17 @@ osbot_utils/helpers/cache_requests/Cache__Requests__Row.py,sha256=h-yc7NkpScbHww
|
|
154
154
|
osbot_utils/helpers/cache_requests/Cache__Requests__Table.py,sha256=RgxAYhm-FIrXXteQRtD91pOLq8JXhSzxb51Jb6MTUdY,391
|
155
155
|
osbot_utils/helpers/cache_requests/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
156
156
|
osbot_utils/helpers/cache_requests/flows/flow__Cache__Requests.py,sha256=xgx_oExxkcvRwQN1UCobimECIMUKGoIX5oGdCmp8Nyw,243
|
157
|
-
osbot_utils/helpers/flows/Flow.py,sha256=
|
158
|
-
osbot_utils/helpers/flows/
|
159
|
-
osbot_utils/helpers/flows/Task.py,sha256=
|
157
|
+
osbot_utils/helpers/flows/Flow.py,sha256=unjYDAsdGqWuEC6sODmcKCDZU8fHnH-CkbmANMGYpa0,6898
|
158
|
+
osbot_utils/helpers/flows/Flow__Events.py,sha256=CS93ohhQ_P7sKr0R4vI2-72uiDQa0pIUEz5Lxam3kfA,1448
|
159
|
+
osbot_utils/helpers/flows/Task.py,sha256=r0SthKYz919C6u-NG6Rt4zpE8yO5DdRFSPjGTnNeZbU,4690
|
160
160
|
osbot_utils/helpers/flows/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
161
161
|
osbot_utils/helpers/flows/decorators/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
162
162
|
osbot_utils/helpers/flows/decorators/flow.py,sha256=LUL7bHjZ_lC3QoTNh-KsDZHzjE2u4mU-5EpZnfsEGuc,738
|
163
163
|
osbot_utils/helpers/flows/decorators/task.py,sha256=9bhQBPJU1dO-J4FAsFkmxqQMBNtay4FT_b1BdpHJ9sA,734
|
164
|
+
osbot_utils/helpers/flows/models/Flow__Config.py,sha256=NLIfT0S8zxbig6Q1p9KeoyE2qRVWorHxk-JXCv9Yx-U,382
|
165
|
+
osbot_utils/helpers/flows/models/Flow__Event.py,sha256=O2SNmDG8FTsQQqm5tWmbomQsC_usA7EqMQhTWIFl4KM,157
|
166
|
+
osbot_utils/helpers/flows/models/Flow__Event_Type.py,sha256=dc5qwV0TfqgTfd10HzJnIZsnZrv4FPOC1EUCwKpI7AI,235
|
167
|
+
osbot_utils/helpers/flows/models/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
164
168
|
osbot_utils/helpers/html/Dict_To_Css.py,sha256=u6B4Mx7PXr-gDrTrs1hgknnvsZVK4Fic5LqedKjo-lk,1097
|
165
169
|
osbot_utils/helpers/html/Dict_To_Html.py,sha256=OlRSaDGOeseBNTxRB2ho5whqEacMXeAXWOfeVSEYqC4,3355
|
166
170
|
osbot_utils/helpers/html/Dict_To_Tags.py,sha256=L8O8c0RPzP92EfeACk3pjXJfnlz-Rg38o2Gf9tS2UfM,3745
|
@@ -298,8 +302,8 @@ osbot_utils/utils/Toml.py,sha256=SD6IA4-mrtoBXcI0dIGKV9POMQNd6WYKvmDQq7GQ6ZQ,143
|
|
298
302
|
osbot_utils/utils/Version.py,sha256=Ww6ChwTxqp1QAcxOnztkTicShlcx6fbNsWX5xausHrg,422
|
299
303
|
osbot_utils/utils/Zip.py,sha256=G6Hk_hDcm9yvWzhTKzhT0R_6f0NBIAchHqMxGb3kfh4,14037
|
300
304
|
osbot_utils/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
301
|
-
osbot_utils/version,sha256=
|
302
|
-
osbot_utils-1.
|
303
|
-
osbot_utils-1.
|
304
|
-
osbot_utils-1.
|
305
|
-
osbot_utils-1.
|
305
|
+
osbot_utils/version,sha256=DaSzptqoG77WG2ht6TyXJYRLX-ouKuUV5NzkxiXP8yM,8
|
306
|
+
osbot_utils-1.55.0.dist-info/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
|
307
|
+
osbot_utils-1.55.0.dist-info/METADATA,sha256=uJ8iV358KsMOdWks77C17F6YAIX3nVJ7eO4DXN32iH8,1266
|
308
|
+
osbot_utils-1.55.0.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
|
309
|
+
osbot_utils-1.55.0.dist-info/RECORD,,
|
File without changes
|
File without changes
|