osbot-utils 1.28.0__py3-none-any.whl → 1.30.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.
@@ -0,0 +1,101 @@
1
+ from osbot_utils.utils.Env import get_env
2
+ from osbot_utils.base_classes.Type_Safe import Type_Safe
3
+ from osbot_utils.utils.Lists import list_index_by
4
+ from osbot_utils.utils.Misc import list_set
5
+
6
+ ENV_VAR__HCP_ACCESS_TOKEN = 'HCP_ACCESS_TOKEN'
7
+ ENV_VAR__HCP_APP_NAME = 'HCP_APP_NAME'
8
+ ENV_VAR__HCP_CLIENT_ID = 'HCP_CLIENT_ID'
9
+ ENV_VAR__HCP_CLIENT_SECRET = 'HCP_CLIENT_SECRET'
10
+ ENV_VAR__HCP_ORGANIZATION_ID = 'HCP_ORGANIZATION_ID'
11
+ ENV_VAR__HCP_PROJECT_ID = 'HCP_PROJECT_ID'
12
+
13
+ class Hashicorp_Secrets(Type_Safe):
14
+
15
+ # helper methods
16
+ def hcp__auth_details(self):
17
+ client_id = get_env(ENV_VAR__HCP_CLIENT_ID)
18
+ client_secret = get_env(ENV_VAR__HCP_CLIENT_SECRET)
19
+ return client_id, client_secret
20
+
21
+ def hcp__access_token(self): # todo: refactor to remove dependency on requests package (which is not part of the OSBOt_utils project)
22
+ import requests
23
+
24
+ access_token = get_env(ENV_VAR__HCP_ACCESS_TOKEN) # todo: add better way to detect when the access token as expired
25
+ if not access_token:
26
+
27
+ client_id, client_secret = self.hcp__auth_details()
28
+ token_url = 'https://auth.idp.hashicorp.com/oauth2/token'
29
+ payload = { 'client_id' : client_id ,
30
+ 'client_secret' : client_secret ,
31
+ 'grant_type' : 'client_credentials' ,
32
+ 'audience' : 'https://api.hashicorp.cloud' }
33
+
34
+ response = requests.post(token_url, data=payload) # todo: refactor into requests_post method
35
+ if response.status_code == 200:
36
+ access_token = response.json().get('access_token')
37
+ return access_token
38
+
39
+ def hcp__enabled(self):
40
+ if self.hcp__organization_id():
41
+ if self.hcp__project_id():
42
+ return True
43
+ return False
44
+
45
+ def hcp__app_name(self):
46
+ return get_env(ENV_VAR__HCP_APP_NAME)
47
+
48
+ def hcp__organization_id(self):
49
+ return get_env(ENV_VAR__HCP_ORGANIZATION_ID)
50
+
51
+ def hcp__project_id(self):
52
+ return get_env(ENV_VAR__HCP_PROJECT_ID)
53
+
54
+ def requests_get(self, path, data_field=None):
55
+ import requests
56
+
57
+ organization_id = self.hcp__organization_id()
58
+ project_id = self.hcp__project_id()
59
+ headers = {'Authorization': f"Bearer { self.hcp__access_token()}" }
60
+ url = f"https://api.cloud.hashicorp.com/secrets/2023-06-13/organizations/{organization_id}/projects/{project_id}/{path}"
61
+ response = requests.get(url, headers=headers)
62
+ if response.status_code == 200:
63
+ json_data = response.json()
64
+ if data_field is None:
65
+ data_field = path
66
+ return json_data.get(data_field)
67
+ return {}
68
+
69
+ # API methods
70
+ def app_secrets(self, app_name=None):
71
+ if app_name is None:
72
+ app_name = self.hcp__app_name()
73
+ path_secrets = f'apps/{app_name}/secrets'
74
+ app_secrets = self.requests_get(path_secrets, 'secrets')
75
+ return app_secrets
76
+
77
+ def app_secrets_open(self, app_name=None):
78
+ if app_name is None:
79
+ app_name = self.hcp__app_name()
80
+ path_secrets = f'apps/{app_name}/open'
81
+ app_secrets = self.requests_get(path_secrets, 'secrets')
82
+ return app_secrets
83
+
84
+ def app_secrets_names(self, app_name=None):
85
+ app_secrets = self.app_secrets(app_name)
86
+ return list_set(list_index_by(app_secrets, 'name'))
87
+
88
+ def app_secrets_values(self, app_name=None):
89
+ secrets_values = {}
90
+ app_secrets = self.app_secrets_open(app_name)
91
+ for app_secret in app_secrets:
92
+ secret_name = app_secret.get('name' )
93
+ secret_value = app_secret.get('version', {}).get('value')
94
+ secrets_values[secret_name] = secret_value
95
+ return secrets_values
96
+
97
+
98
+
99
+ def apps(self):
100
+ return self.requests_get('apps')
101
+
@@ -1,16 +1,20 @@
1
1
  from osbot_utils.base_classes.Type_Safe import Type_Safe
2
2
  from osbot_utils.utils.Dev import pprint
3
3
  from osbot_utils.utils.Files import files_list, file_create_from_bytes, temp_file, parent_folder, parent_folder_create
4
+ from osbot_utils.utils.Regex import list__match_regex, list__match_regexes
4
5
  from osbot_utils.utils.Zip import zip_bytes_empty, zip_bytes__files, zip_bytes__add_file, zip_bytes__add_files, \
5
6
  zip_bytes__replace_files, zip_bytes__replace_file, zip_bytes__file_list, zip_bytes__file, \
6
- zip_bytes__add_file__from_disk, zip_bytes__add_files__from_disk, zip_files, zip_file__files
7
+ zip_bytes__add_file__from_disk, zip_bytes__add_files__from_disk, zip_files, zip_file__files, zip_bytes__remove_files
7
8
 
8
9
 
9
10
  class Zip_Bytes(Type_Safe):
10
- zip_bytes : bytes
11
+ zip_bytes : bytes = None
11
12
 
12
- def __enter__(self):
13
+ def __init__(self):
14
+ super().__init__()
13
15
  self.zip_bytes = zip_bytes_empty()
16
+
17
+ def __enter__(self):
14
18
  return self
15
19
 
16
20
  def __exit__(self, exc_type, exc_val, exc_tb):
@@ -20,7 +24,9 @@ class Zip_Bytes(Type_Safe):
20
24
  self.zip_bytes = zip_bytes__add_file(self.zip_bytes, file_path, file_contents)
21
25
  return self
22
26
 
23
- def add_file__from_disk(self, base_path, file_to_add):
27
+ def add_file__from_disk(self, file_to_add, base_path=None):
28
+ if base_path is None:
29
+ base_path = parent_folder(file_to_add)
24
30
  self.zip_bytes = zip_bytes__add_file__from_disk(self.zip_bytes, base_path, file_to_add)
25
31
  return self
26
32
 
@@ -28,14 +34,22 @@ class Zip_Bytes(Type_Safe):
28
34
  self.zip_bytes = zip_bytes__add_files(self.zip_bytes, files_to_add)
29
35
  return self
30
36
 
31
- def add_files__from_disk(self, base_path, files_to_add):
32
- self.zip_bytes = zip_bytes__add_files__from_disk(self.zip_bytes, base_path, files_to_add)
37
+ def add_files__from_disk(self, base_path, files_to_add, path_prefix=None):
38
+ self.zip_bytes = zip_bytes__add_files__from_disk(self.zip_bytes, base_path, files_to_add, path_prefix=path_prefix)
33
39
  return self
34
40
 
35
- def add_folder__from_disk(self, base_path, folder_to_add, pattern="*"):
36
- files_to_add = files_list(folder_to_add, pattern=pattern)
41
+ def add_folder__from_disk(self, folder_to_add, *patterns):
42
+ base_path = folder_to_add
43
+ all_files_in_folder = files_list(folder_to_add)
44
+ files_to_add = list__match_regexes(all_files_in_folder, *patterns)
37
45
  return self.add_files__from_disk(base_path, files_to_add)
38
46
 
47
+ def add_folder__from_disk__with_prefix(self, folder_to_add, path_prefix, *patterns):
48
+ base_path = folder_to_add
49
+ all_files_in_folder = files_list(folder_to_add)
50
+ files_to_add = list__match_regexes(all_files_in_folder, *patterns)
51
+ return self.add_files__from_disk(base_path, files_to_add, path_prefix=path_prefix)
52
+
39
53
  def add_from_zip_file(self, path_zip_file):
40
54
  files_to_add = zip_file__files(path_zip_file)
41
55
  self.add_files(files_to_add)
@@ -47,15 +61,27 @@ class Zip_Bytes(Type_Safe):
47
61
  def file(self, file_path):
48
62
  return zip_bytes__file(self.zip_bytes, file_path)
49
63
 
64
+ def file_paths(self):
65
+ return self.files_list()
66
+
50
67
  def files(self):
51
68
  return zip_bytes__files(self.zip_bytes)
52
69
 
53
70
  def files_list(self):
54
71
  return zip_bytes__file_list(self.zip_bytes)
55
72
 
73
+ def list(self):
74
+ return self.files_list()
75
+
56
76
  def print_files_list(self):
57
77
  pprint(self.files_list())
58
78
  return self
79
+
80
+ def remove_files(self, *patterns):
81
+ files_to_remove = list__match_regexes(self.files_list(), *patterns)
82
+ self.zip_bytes = zip_bytes__remove_files(self.zip_bytes, files_to_remove)
83
+ return self
84
+
59
85
  def replace_files(self, files_to_replace):
60
86
  self.zip_bytes = zip_bytes__replace_files(self.zip_bytes, files_to_replace)
61
87
  return self
@@ -11,8 +11,8 @@ from osbot_utils.helpers.trace.Trace_Call__View_Model import Trace_Call__V
11
11
 
12
12
 
13
13
  def trace_calls(title = None , print_traces = True , show_locals = False, source_code = False ,
14
- ignore = None , include = None , show_path = False, duration_bigger_than = 0 , trace_depth=0,
15
- max_string = None , show_types = False, show_duration = False ,# show_caller = False , # todo: add back when show_caller is working again
14
+ ignore = None , include = None , show_path = False, duration_bigger_than = 0 ,
15
+ trace_depth = 0 , max_string = None , show_types = False, show_duration = False , # show_caller = False , # todo: add back when show_caller is working again
16
16
  show_class = False, contains = None , show_internals = False, enabled = True ,
17
17
  extra_data = False, show_lines = False, print_lines = False, show_types_padding = None , duration_padding=None):
18
18
  def decorator(func):
@@ -96,6 +96,13 @@ class Temp_Folder:
96
96
  file_path = path_combine(self.full_path, target_file)
97
97
  return file_contents(file_path)
98
98
 
99
+ def files_contents(self):
100
+ result = {}
101
+ for file_path in self.files():
102
+ full_path = path_combine(self.full_path, file_path)
103
+ result[file_path] = file_contents(full_path)
104
+ return result
105
+
99
106
  def files(self, show_parent_folder=False, include_folders=False):
100
107
  all_files = files_recursive(self.path(), include_folders=include_folders)
101
108
  if show_parent_folder:
osbot_utils/utils/Env.py CHANGED
@@ -6,10 +6,10 @@ from osbot_utils.utils.Files import all_parent_folders, file_exists
6
6
  from osbot_utils.utils.Misc import list_set
7
7
  from osbot_utils.utils.Str import strip_quotes
8
8
 
9
- def env__home_root():
9
+ def env__home_root(): # todo: this should be refatored to be env__home__is__root
10
10
  return os.getenv('HOME') == '/root'
11
11
 
12
- def env__terminal_xterm():
12
+ def env__terminal_xterm(): # todo: this should be refatored to be env__terminal__is__xterm
13
13
  return os.getenv('TERM') == 'xterm'
14
14
 
15
15
  def env__not_terminal_xterm():
@@ -81,7 +81,7 @@ class Files:
81
81
  return glob.glob(path_pattern, recursive=recursive)
82
82
 
83
83
  @staticmethod
84
- def files(path, pattern= '*.*', only_files=True):
84
+ def files(path, pattern= '*', only_files=True):
85
85
  result = []
86
86
  for file in Path(path).rglob(pattern):
87
87
  if only_files and is_not_file(file):
@@ -0,0 +1,25 @@
1
+ import re
2
+
3
+ from osbot_utils.utils.Status import status_error
4
+
5
+
6
+ def list__match_regex(target, pattern):
7
+ return list__match_regexes(target, [pattern])
8
+
9
+ def list__match_regexes(target, *patterns):
10
+ if len (patterns) == 0:
11
+ return target
12
+ compiled_patterns = []
13
+ for pattern in patterns:
14
+ try:
15
+ compiled_patterns.append(re.compile(pattern)) # todo: find better way to handle regex errors
16
+ except Exception as error:
17
+ return status_error(message='Error compiling pattern: {pattern}', error=error)
18
+ matched_files = []
19
+ if target and type(target) is list:
20
+ for file in target:
21
+ for compiled_pattern in compiled_patterns:
22
+ if compiled_pattern.match(file):
23
+ matched_files.append(file)
24
+ break
25
+ return matched_files
osbot_utils/utils/Zip.py CHANGED
@@ -52,14 +52,17 @@ def zip_bytes__add_file(zip_bytes, zip_file_path, file_contents):
52
52
  def zip_bytes__add_file__from_disk(zip_bytes, base_path, file_to_add):
53
53
  return zip_bytes__add_files__from_disk(zip_bytes, base_path, files_to_add=[file_to_add])
54
54
 
55
- def zip_bytes__add_files__from_disk(zip_bytes, base_path, files_to_add, replace_files=True):
55
+ def zip_bytes__add_files__from_disk(zip_bytes, base_path, files_to_add, replace_files=True, path_prefix=None):
56
56
  zip_files_to_add = {}
57
57
  if base_path[:-1] != '/':
58
58
  base_path += "/"
59
+
59
60
  for file_to_add in files_to_add:
60
61
  if file_exists(file_to_add):
61
62
  file_contents = file_contents_as_bytes(file_to_add)
62
63
  zip_file_path = file_to_add.replace(base_path, '')
64
+ if path_prefix:
65
+ zip_file_path = path_prefix + zip_file_path
63
66
  zip_files_to_add[zip_file_path] = file_contents
64
67
 
65
68
  if replace_files:
osbot_utils/version CHANGED
@@ -1 +1 @@
1
- v1.28.0
1
+ v1.30.0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: osbot_utils
3
- Version: 1.28.0
3
+ Version: 1.30.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.28.0-blue)
25
+ ![Current Release](https://img.shields.io/badge/release-v1.30.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
 
@@ -57,13 +57,14 @@ osbot_utils/graphs/mgraph/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJW
57
57
  osbot_utils/helpers/CFormat.py,sha256=1_XvqGwgU6qC97MbzcKF0o7s9mCXpU5Kq9Yf-1ixUwY,6808
58
58
  osbot_utils/helpers/CPrint.py,sha256=ztKPNmT8BGxeyPXSQKRs63PqqbgxKDz_BiZmzFMup9g,1413
59
59
  osbot_utils/helpers/Dict_To_Attr.py,sha256=NdhXl5mJH7-NaBk213amzc5Nfy3tJgW-N_uYIRE4hoc,208
60
+ osbot_utils/helpers/Hashicorp_Secrets.py,sha256=zjXa_dQvfR9L1uoulWJ8nYYaDvznV6o_QPPS4zmb6mo,4235
60
61
  osbot_utils/helpers/Local_Cache.py,sha256=0JZZX3fFImcwtbBvxAQl-EbBegSNJRhRMYF6ovTH6zY,3141
61
62
  osbot_utils/helpers/Local_Caches.py,sha256=HvuP5CURyVm_fVvJX-S4dml2bhRauzgA3be237yTaeY,1814
62
63
  osbot_utils/helpers/Print_Table.py,sha256=LEXbyqGg_6WSraI4cob4bNNSu18ddqvALp1zGK7bPhs,19126
63
64
  osbot_utils/helpers/Python_Audit.py,sha256=shpZlluJwqJBAlad6xN01FkgC1TsQ48RLvR5ZjmrKa4,1539
64
65
  osbot_utils/helpers/Random_Seed.py,sha256=14btja8LDN9cMGWaz4fCNcMRU_eyx49gas-_PQvHgy4,634
65
66
  osbot_utils/helpers/Type_Registry.py,sha256=Ajk3SyMSKDi2g9SJYUtTgg7PZkAgydaHcpbGuEN3S94,311
66
- osbot_utils/helpers/Zip_Bytes.py,sha256=Ls_AW_-vCusZxL3-Jfl9bJfOpwl4e7nTzxrXSWJ6Ty8,2850
67
+ osbot_utils/helpers/Zip_Bytes.py,sha256=GQAwNEoMXdG7CU1RDhIJ51vkgfbJZnUxjlMUmFGHL1M,3937
67
68
  osbot_utils/helpers/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
68
69
  osbot_utils/helpers/ast/Ast.py,sha256=lcPQOSxXI6zgmMnIVF9WM6ISqViWX-sq4d_UC0CDG8s,1155
69
70
  osbot_utils/helpers/ast/Ast_Base.py,sha256=5rHMupBlN_n6lOC31UnSW_lWqxqxaE31v0gn-t32OgQ,3708
@@ -227,7 +228,7 @@ osbot_utils/helpers/ssh/SSH__Linux__Amazon.py,sha256=ZJFb7LFTvclAuhH5OoOtJ361NoX
227
228
  osbot_utils/helpers/ssh/SSH__Python.py,sha256=O2DAwkbXzwkis8lffoqIL2NPSfYcN44Mr8i9Ey2iMKk,2066
228
229
  osbot_utils/helpers/ssh/TestCase__SSH.py,sha256=MD8sq0_kI4f6pEmEO0cLq2mQOhIqbP45ZxFJNG44Jg4,1773
229
230
  osbot_utils/helpers/ssh/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
230
- osbot_utils/helpers/trace/Trace_Call.py,sha256=WXw1QyHxVrkKk24NlMbrsLQdbQTgN2ocwEFqkfB6n2E,6408
231
+ osbot_utils/helpers/trace/Trace_Call.py,sha256=nKds4xTyrF-6A3XMCxVOR37KDzB3nfCQLykrrEsqp2k,6424
231
232
  osbot_utils/helpers/trace/Trace_Call__Config.py,sha256=6nJplzrY1adz3WcwTIuwEhBCk1P0U0NbyWRzDyIox_Q,3250
232
233
  osbot_utils/helpers/trace/Trace_Call__Graph.py,sha256=HCrXRKQI42DIQxxyFLcaosWiOcUyoITbeV17ICdXcXM,1156
233
234
  osbot_utils/helpers/trace/Trace_Call__Handler.py,sha256=3gEX-DykEFixfBmD0jBdMhnTujxhNbXkjOhqxNTc9ac,11563
@@ -252,7 +253,7 @@ osbot_utils/testing/Stderr.py,sha256=wi1gfjpsxnBK3aOl2jzCTWI-0En1HtPgEin97148_MQ
252
253
  osbot_utils/testing/Stdout.py,sha256=XQ9OlOW1aHXY1TiNu8O5b75RoDnoaX5RyMHml3OjlKw,459
253
254
  osbot_utils/testing/Temp_Env_Vars.py,sha256=oFuaegBlaV0aySkPe1Nzf-mdIKN03oTUKNfKijGz__M,751
254
255
  osbot_utils/testing/Temp_File.py,sha256=yZBL9MmcNU4PCQ4xlF4rSss4GylKoX3T_AJF-BlQhdI,1693
255
- osbot_utils/testing/Temp_Folder.py,sha256=gewV0L3B4ScKyorkVOQaQ-vz4DbeVRnqxCMS9A0eVLo,4953
256
+ osbot_utils/testing/Temp_Folder.py,sha256=Dbcohr2ciex6w-kB79R41Nuoa0pgpDbKtPGnlMmJ73k,5194
256
257
  osbot_utils/testing/Temp_Sys_Path.py,sha256=gOMD-7dQYQlejoDYUqsrmuZQ9DLC07ymPZB3zYuNmG4,256
257
258
  osbot_utils/testing/Temp_Web_Server.py,sha256=0A-gZsd0_3wRj2YuBEOWyV2rhT6dcS2BlArngPXGTtk,3186
258
259
  osbot_utils/testing/Temp_Zip.py,sha256=gppbJchk4tw_bu-7Vt6iJS9mGxeCvNNMMDzeVKHqRv8,1489
@@ -264,9 +265,9 @@ osbot_utils/utils/Assert.py,sha256=u9XLgYn91QvNWZGyPi29SjPJSXRHlm9andIn3NJEVog,1
264
265
  osbot_utils/utils/Call_Stack.py,sha256=MAq_0vMxnbeLfCe9qQz7GwJYaOuXpt3qtQwN6wiXsU0,6595
265
266
  osbot_utils/utils/Csv.py,sha256=oHLVpjRJqrLMz9lubMCNEoThXWju5rNTprcwHc1zq2c,1012
266
267
  osbot_utils/utils/Dev.py,sha256=HibpQutYy_iG8gGV8g1GztxNN4l29E4Bi7UZaVL6-L8,1203
267
- osbot_utils/utils/Env.py,sha256=uYLhqVXqqgfh03Zmf9Vdy9zFdFf0rasW6lteh_VM1xQ,5078
268
+ osbot_utils/utils/Env.py,sha256=Pbel6npitij9zag6SsWdPVH2j1BTZjxnwOR1vLiIyMo,5248
268
269
  osbot_utils/utils/Exceptions.py,sha256=KyOUHkXQ_6jDTq04Xm261dbEZuRidtsM4dgzNwSG8-8,389
269
- osbot_utils/utils/Files.py,sha256=SLuu_RuugTKvXi0zOJh_mkr2FE6TUyQvW60vgZbkHEY,20836
270
+ osbot_utils/utils/Files.py,sha256=bZFvjyC4TAmFH6RSp3Xvj_xBxfybCT2cE7x2qNTQmeE,20834
270
271
  osbot_utils/utils/Functions.py,sha256=0E6alPJ0fJpBiJgFOWooCOi265wSRyxxXAJ5CELBnso,3498
271
272
  osbot_utils/utils/Http.py,sha256=Z8V149M2HDrKBoXkDD5EXgqTGx6vQoUqXugXK__wcuw,4572
272
273
  osbot_utils/utils/Int.py,sha256=PmlUdU4lSwf4gJdmTVdqclulkEp7KPCVUDO6AcISMF4,116
@@ -278,14 +279,15 @@ osbot_utils/utils/Objects.py,sha256=WFH3oeXR1CU03oyzXfcIlktcoXQuKw9cIJn8xTs02AE,
278
279
  osbot_utils/utils/Png.py,sha256=V1juGp6wkpPigMJ8HcxrPDIP4bSwu51oNkLI8YqP76Y,1172
279
280
  osbot_utils/utils/Process.py,sha256=lr3CTiEkN3EiBx3ZmzYmTKlQoPdkgZBRjPulMxG-zdo,2357
280
281
  osbot_utils/utils/Python_Logger.py,sha256=tx8N6wRKL3RDHboDRKZn8SirSJdSAE9cACyJkxrThZ8,12792
282
+ osbot_utils/utils/Regex.py,sha256=0ubgp8HKsS3PNe2H6XlzMIcUuV7jhga3VkQVDNOJWuA,866
281
283
  osbot_utils/utils/Status.py,sha256=Yq4s0TelXgn0i2QjCP9V8mP30GabXp_UL-jjM6Iwiw4,4305
282
284
  osbot_utils/utils/Str.py,sha256=kxdY8ROX4FdJtCaMTfOc8fK_xcDICprNkefHu2MMNU4,2585
283
285
  osbot_utils/utils/Toml.py,sha256=dqiegndCJF7V1YT1Tc-b0-Bl6QWyL5q30urmQwMXfMQ,1402
284
286
  osbot_utils/utils/Version.py,sha256=Ww6ChwTxqp1QAcxOnztkTicShlcx6fbNsWX5xausHrg,422
285
- osbot_utils/utils/Zip.py,sha256=nj3_PrvKRlyeSWcFlbvEwhmu87Gl-siUfIoXxDPE32c,12056
287
+ osbot_utils/utils/Zip.py,sha256=t9txUxJzLBEHot6WJwF0iTTUQ1Gf_V2pVwsWzAqw_NU,12163
286
288
  osbot_utils/utils/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
287
- osbot_utils/version,sha256=MNi3rJWAauptL9RTT15nAiO62YEIThbbzD7BICVSPoQ,8
288
- osbot_utils-1.28.0.dist-info/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
289
- osbot_utils-1.28.0.dist-info/METADATA,sha256=LvLJ-gezDkfRszvTQeU-WyLaVX-z0RJmohGZ1qdQkk8,1266
290
- osbot_utils-1.28.0.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
291
- osbot_utils-1.28.0.dist-info/RECORD,,
289
+ osbot_utils/version,sha256=rySeFYh2TX0iwzR0MwppiJ1yllnTK1JQxMRPUGe8DS0,8
290
+ osbot_utils-1.30.0.dist-info/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
291
+ osbot_utils-1.30.0.dist-info/METADATA,sha256=XspSKWOcbtP6-hGL3r9N7NmMZn7n91lK68mf8vPxetI,1266
292
+ osbot_utils-1.30.0.dist-info/WHEEL,sha256=sP946D7jFCHeNz5Iq4fL4Lu-PrWrFsgfLXbbkciIZwg,88
293
+ osbot_utils-1.30.0.dist-info/RECORD,,