osbot-utils 1.53.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.
@@ -2,14 +2,15 @@ import asyncio
2
2
  import logging
3
3
  import typing
4
4
 
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.Flow__Config import Flow__Config
8
- from osbot_utils.testing.Stdout import Stdout
9
- from osbot_utils.utils.Misc import random_id, lower
10
- from osbot_utils.utils.Python_Logger import Python_Logger
11
- from osbot_utils.utils.Str import ansis_to_texts
12
- from osbot_utils.utils.Threads import invoke_async, invoke_in_new_event_loop
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.debug(f"Created flow run '{self.f__flow_id()}' for flow '{self.f__flow_name()}'")
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.debug(f"Executing flow run '{self.f__flow_id()}''")
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.logger.error(self.cformat.red(f"Error executing flow: {error}"))
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.info(f_magenta(line))
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.debug(f"Finished flow run '{self.f__flow_id()}''")
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.debug(f"{f_dark_grey('Flow return value')}: {f_bold(self.flow_return_value)}")
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.utils.Dev import pprint
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 # todo add a random Id value to this
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.task_flow.logger.debug(f"Executing task '{f_blue(self.task_name)}'")
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.task_flow.logger.error(f_red(f"Error executing '{self.task_name}' task: {self.task_error}"))
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.task_flow.logger.debug(f"Finished task '{f_blue(self.task_name)}'")
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.task_flow.logger.debug(f"{f_dark_grey('Task return value')}: {f_bold(self.task_return_value)}")
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))
@@ -5,6 +5,7 @@ class Flow__Config(Type_Safe):
5
5
  add_task_to_self : bool = True
6
6
  log_to_console : bool = False
7
7
  log_to_memory : bool = True
8
+ logging_enabled : bool = True
8
9
  print_logs : bool = False
9
10
  print_none_return_value: bool = False
10
11
  print_finished_message : bool = False
@@ -0,0 +1,6 @@
1
+ from osbot_utils.base_classes.Type_Safe import Type_Safe
2
+
3
+ class Flow__Event(Type_Safe):
4
+ event_type : str
5
+ event_source: object
6
+ event_data : dict
@@ -0,0 +1,8 @@
1
+ from osbot_utils.base_classes.Type_Safe import Type_Safe
2
+
3
+
4
+ class Flow__Event_Type(Type_Safe):
5
+ FLOW_START : str = 'flow_start'
6
+ FLOW_STOP : str = 'flow_stop'
7
+ TASK_START : str = 'task_start'
8
+ TASK_STOP : str = 'task_stop'
File without changes
@@ -4,8 +4,9 @@ import pickle
4
4
  import sys
5
5
  import types
6
6
  import typing
7
- from typing import Union
8
7
 
8
+ from typing import Union
9
+ from types import SimpleNamespace
9
10
  from osbot_utils.utils.Misc import list_set
10
11
  from osbot_utils.utils.Str import str_unicode_escape, str_max_width
11
12
 
@@ -120,6 +121,32 @@ def dict_remove(data, target):
120
121
  del data[target]
121
122
  return data
122
123
 
124
+
125
+ def dict_to_obj(target, class_name="_"):
126
+ DynamicClass = type(class_name, (SimpleNamespace,), {})
127
+ if isinstance(target, dict):
128
+ new_dict = {}
129
+ for key, value in target.items():
130
+ new_dict[key] = dict_to_obj(value, class_name=class_name) # Recursively convert elements in the dict
131
+ return DynamicClass(**new_dict)
132
+ elif isinstance(target, list): # Recursively convert elements in the list
133
+ return [dict_to_obj(item, class_name=class_name) for item in target]
134
+ elif isinstance(target, tuple): # Recursively convert elements in the tuple
135
+ return tuple(dict_to_obj(item, class_name=class_name) for item in target)
136
+
137
+ return target
138
+
139
+ def obj_to_dict(target): # Recursively converts an object (SimpleNamespace) back into a dictionary."""
140
+ if isinstance(target, SimpleNamespace): # Convert SimpleNamespace attributes to a dictionary
141
+ return {key: obj_to_dict(value) for key, value in target.__dict__.items()}
142
+ elif isinstance(target, list): # Handle lists: convert each item in the list
143
+ return [obj_to_dict(item) for item in target]
144
+ elif isinstance(target, tuple): # Handle tuples: convert each item and return as a tuple
145
+ return tuple(obj_to_dict(item) for item in target)
146
+ elif isinstance(target, set): # Handle sets: convert each item and return as a set
147
+ return {obj_to_dict(item) for item in target}
148
+ return target # Return non-object types as is
149
+
123
150
  def enum_from_value(enum_type, value):
124
151
  try:
125
152
  return enum_type[value] # Attempt to convert the value to an Enum member by name
osbot_utils/version CHANGED
@@ -1 +1 @@
1
- v1.53.0
1
+ v1.55.0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: osbot_utils
3
- Version: 1.53.0
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
- ![Current Release](https://img.shields.io/badge/release-v1.53.0-blue)
25
+ ![Current Release](https://img.shields.io/badge/release-v1.55.0-blue)
26
26
  [![codecov](https://codecov.io/gh/owasp-sbot/OSBot-Utils/graph/badge.svg?token=GNVW0COX1N)](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=gGvbG2LQjzZ_Sf2wPneHmJco1pv9ejXes5Z0gmWQQxk,6541
158
- osbot_utils/helpers/flows/Flow__Config.py,sha256=PE8hH8KXDHz1Ex93cPMuR9nkv8AoXzXQwVxleSDkU7k,341
159
- osbot_utils/helpers/flows/Task.py,sha256=b-EKKM4ZEXQcNDOjWHLcxkC0orhIxCOYs7Ij2aCsE7Y,4265
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
@@ -286,7 +290,7 @@ osbot_utils/utils/Json.py,sha256=7COxBlZRnpxtpNqpmzMPYkcKTnCok-s686nT27oiKEQ,648
286
290
  osbot_utils/utils/Json_Cache.py,sha256=mLPkkDZN-3ZVJiDvV1KBJXILtKkTZ4OepzOsDoBPhWg,2006
287
291
  osbot_utils/utils/Lists.py,sha256=tPz5x5s3sRO97WZ_nsxREBPC5cwaHrhgaYBhsrffTT8,5599
288
292
  osbot_utils/utils/Misc.py,sha256=nODZT6p44B4xYiIiqfEeKYEErQiKR9SGthhGtZWGhkI,16804
289
- osbot_utils/utils/Objects.py,sha256=frOLTJsIhGye1j99StuIJSkOlPmPDNI3BaWWAr98MEc,15392
293
+ osbot_utils/utils/Objects.py,sha256=oiWF75BPahRtBRhFrQi_5Mv4wrqkjJnKAc_diw9CZdk,17321
290
294
  osbot_utils/utils/Png.py,sha256=V1juGp6wkpPigMJ8HcxrPDIP4bSwu51oNkLI8YqP76Y,1172
291
295
  osbot_utils/utils/Process.py,sha256=lr3CTiEkN3EiBx3ZmzYmTKlQoPdkgZBRjPulMxG-zdo,2357
292
296
  osbot_utils/utils/Python_Logger.py,sha256=tx8N6wRKL3RDHboDRKZn8SirSJdSAE9cACyJkxrThZ8,12792
@@ -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=cfLGcb-cU0atx9PN7uRd-7SjRkvlf2gUwCK4UUZZCt8,8
302
- osbot_utils-1.53.0.dist-info/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
303
- osbot_utils-1.53.0.dist-info/METADATA,sha256=PQqhK7mvfVoonWblVF3oPIdHKqP7CXisueyqZiBVCeE,1266
304
- osbot_utils-1.53.0.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
305
- osbot_utils-1.53.0.dist-info/RECORD,,
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,,