atomicshop 2.18.34__py3-none-any.whl → 2.19.1__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.

Potentially problematic release.


This version of atomicshop might be problematic. Click here for more details.

atomicshop/__init__.py CHANGED
@@ -1,4 +1,4 @@
1
1
  """Atomic Basic functions and classes to make developer life easier"""
2
2
 
3
3
  __author__ = "Den Kras"
4
- __version__ = '2.18.34'
4
+ __version__ = '2.19.1'
@@ -0,0 +1,52 @@
1
+ import sys
2
+ import subprocess
3
+ import tempfile
4
+
5
+ from atomicshop.print_api import print_api
6
+ from atomicshop.wrappers import githubw
7
+ from atomicshop.permissions import permissions
8
+
9
+
10
+ WINDOWS_TESSERACT_DEFAULT_INSTALLATION_DIRECTORY: str = r"C:\Program Files\Tesseract-OCR"
11
+
12
+
13
+ def main():
14
+ if not permissions.is_admin():
15
+ print_api("Please run this script as an Administrator.", color="red")
16
+ return 1
17
+ """
18
+ print_api("PIP Installing Robocorp.")
19
+ subprocess.check_call(["pip", "install", "--upgrade", "rpaframework"])
20
+
21
+ print_api("PIP Installing Robocorp-Browser.")
22
+ subprocess.check_call(["pip", "install", "--upgrade", "robotframework-browser"])
23
+
24
+ print_api("PIP Installing Robocorp-Recognition.")
25
+ subprocess.check_call(["pip", "install", "--upgrade", "rpaframework-recognition"])
26
+
27
+ print_api("Installing Playwright browsers.")
28
+ subprocess.check_call(["playwright", "install"])
29
+
30
+ print_api("Initializing Robocorp Browser.")
31
+ subprocess.check_call(["rfbrowser", "init"])
32
+
33
+ print_api("Installing Tesseract OCR.")
34
+ github_wrapper = githubw.GitHubWrapper(
35
+ user_name="tesseract-ocr",
36
+ repo_name="tesseract",
37
+ branch="main")
38
+ github_wrapper.build_links_from_user_and_repo()
39
+ temp_file_path: str = tempfile.gettempdir()
40
+ tesseract_installer = github_wrapper.download_latest_release(
41
+ target_directory=temp_file_path,
42
+ string_pattern="*tesseract*exe")
43
+
44
+ # The Admin needed to install Tesseract.
45
+ subprocess.check_call([tesseract_installer, "/S"])
46
+ """
47
+ # Add Tesseract to the PATH.
48
+ subprocess.check_call(["setx", "PATH", f"%PATH%;{WINDOWS_TESSERACT_DEFAULT_INSTALLATION_DIRECTORY}"])
49
+
50
+
51
+ if __name__ == '__main__':
52
+ sys.exit(main())
@@ -52,7 +52,7 @@ class MultiProcessorRecursive:
52
52
  :param process_function: function, function to execute on the input list.
53
53
  :param input_list: list, list of inputs to process.
54
54
  :param max_workers: integer, number of workers to execute functions in parallel. Default is None, which
55
- is the number of CPUs.
55
+ is the number of CPUs that will be counted automatically by the multiprocessing module.
56
56
  :param cpu_percent_max: integer, maximum CPU percentage. Above that usage, we will wait before starting new
57
57
  execution.
58
58
  :param memory_percent_max: integer, maximum memory percentage. Above that usage, we will wait, before starting
@@ -65,7 +65,7 @@ class MultiProcessorRecursive:
65
65
  If this is used, the system resources will be checked before starting each new execution from this
66
66
  shared dict instead of performing new checks.
67
67
 
68
- Usage:
68
+ Usage Examples:
69
69
  def unpack_file(file_path):
70
70
  # Process the file at file_path and unpack it.
71
71
  # Return a list of new file paths that were extracted from the provided path.
@@ -74,19 +74,68 @@ class MultiProcessorRecursive:
74
74
  # List of file paths to process
75
75
  file_paths = ["path1", "path2", "path3"]
76
76
 
77
- # Create an instance of MultiProcessor
78
- # Note: unpacking.unpack_file is passed without parentheses
79
- processor = MultiProcessor(
80
- process_function=unpack_file,
81
- input_list=file_paths,
82
- max_workers=4, # Number of parallel workers
83
- cpu_percent_max=80, # Max CPU usage percentage
84
- memory_percent_max=80, # Max memory usage percentage
85
- wait_time=5 # Time to wait if resources are overused
86
- )
87
-
88
- # Run the processing
89
- processor.run_process()
77
+ # Note: unpack_file Callable is passed to init without parentheses.
78
+
79
+ 1. Providing the list directly to process at once:
80
+ # Initialize the processor.
81
+ processor = MultiProcessor(
82
+ process_function=unpack_file,
83
+ input_list=file_paths,
84
+ max_workers=4, # Number of parallel workers
85
+ cpu_percent_max=80, # Max CPU usage percentage
86
+ memory_percent_max=80, # Max memory usage percentage
87
+ wait_time=5 # Time to wait if resources are overused
88
+ )
89
+
90
+ # Process the list of files at once.
91
+ processor.run_process()
92
+ # Shutdown the pool processes after processing.
93
+ processor.shutdown_pool()
94
+
95
+ 2. Processing each file in the list differently then adding to the list of the multiprocessing instance then executing.
96
+ # Initialize the processor once, before the loop, with empty input_list.
97
+ processor = MultiProcessor(
98
+ process_function=unpack_file,
99
+ input_list=[],
100
+ max_workers=4, # Number of parallel workers
101
+ cpu_percent_max=80, # Max CPU usage percentage
102
+ memory_percent_max=80, # Max memory usage percentage
103
+ wait_time=5 # Time to wait if resources are overused
104
+ )
105
+
106
+ for file_path in file_paths:
107
+ # <Process each file>.
108
+ # Add the result to the input_list of the processor.
109
+ processor.input_list.append(file_path)
110
+
111
+ # Process the list of files at once.
112
+ processor.run_process()
113
+ # Shutdown the pool processes after processing.
114
+ processor.shutdown_pool()
115
+
116
+ 3. Processing each file in the list separately, since we're using an unpacking function that
117
+ will create more files, but the context for this operation is different for extraction
118
+ of each main file inside the list:
119
+
120
+ # Initialize the processor once, before the loop, with empty input_list.
121
+ processor = MultiProcessor(
122
+ process_function=unpack_file,
123
+ input_list=[],
124
+ max_workers=4, # Number of parallel workers
125
+ cpu_percent_max=80, # Max CPU usage percentage
126
+ memory_percent_max=80, # Max memory usage percentage
127
+ wait_time=5 # Time to wait if resources are overused
128
+ )
129
+
130
+ for file_path in file_paths:
131
+ # <Process each file>.
132
+ # Add the result to the input_list of the processor.
133
+ processor.input_list.append(file_path)
134
+ # Process the added file path separately.
135
+ processor.run_process()
136
+
137
+ # Shutdown the pool processes after processing.
138
+ processor.shutdown_pool()
90
139
  """
91
140
 
92
141
  self.process_function = process_function
@@ -97,41 +146,57 @@ class MultiProcessorRecursive:
97
146
  self.wait_time: float = wait_time
98
147
  self.system_monitor_manager_dict: multiprocessing.managers.DictProxy = system_monitor_manager_dict
99
148
 
149
+ # Create the pool once and reuse it
150
+ self.pool: multiprocessing.Pool = multiprocessing.Pool(processes=self.max_workers)
151
+
152
+ # Keep track of outstanding async results across calls
153
+ self.async_results: list = []
154
+
100
155
  def run_process(self):
101
- with multiprocessing.Pool(processes=self.max_workers) as pool:
102
- # Keep track of the async results
103
- async_results = []
104
-
105
- while self.input_list:
106
- new_input_list = []
107
- for item in self.input_list:
108
- # Check system resources before processing each item
109
- system_resources.wait_for_resource_availability(
110
- cpu_percent_max=self.cpu_percent_max,
111
- memory_percent_max=self.memory_percent_max,
112
- wait_time=self.wait_time,
113
- system_monitor_manager_dict=self.system_monitor_manager_dict)
114
-
115
- # Process the item
116
- async_result = pool.apply_async(self.process_function, (item,))
117
- async_results.append(async_result)
118
-
119
- # Reset input_list for next round of processing
120
- self.input_list = []
121
-
122
- # Collect results as they complete
123
- for async_result in async_results:
124
- try:
125
- result = async_result.get()
126
- # Assuming process_function returns a list, extend new_input_list
127
- new_input_list.extend(result)
128
- except Exception:
129
- raise
130
-
131
- # Update the input_list for the next iteration
132
- self.input_list = new_input_list
133
- # Clear the async_results for the next iteration
134
- async_results.clear()
156
+ # This method can be called multiple times to add new tasks without creating a new pool
157
+ if not self.input_list:
158
+ return # Nothing to process
159
+
160
+ new_input_list = []
161
+
162
+ for item in self.input_list:
163
+ # Check system resources before scheduling each item
164
+ system_resources.wait_for_resource_availability(
165
+ cpu_percent_max=self.cpu_percent_max,
166
+ memory_percent_max=self.memory_percent_max,
167
+ wait_time=self.wait_time,
168
+ system_monitor_manager_dict=self.system_monitor_manager_dict
169
+ )
170
+
171
+ # Schedule the task on the existing pool
172
+ async_result = self.pool.apply_async(self.process_function, (item,))
173
+ self.async_results.append(async_result)
174
+
175
+ # Clear the input_list now that tasks are scheduled
176
+ self.input_list = []
177
+
178
+ # Collect results for the tasks that were scheduled
179
+ for async_result in self.async_results:
180
+ try:
181
+ result = async_result.get() # Blocking wait for result
182
+ # Assuming process_function returns a list, extend new_input_list
183
+ new_input_list.extend(result)
184
+ except Exception as e:
185
+ # Handle exceptions as needed
186
+ raise e
187
+
188
+ # Clear collected async results since they've been processed
189
+ self.async_results.clear()
190
+
191
+ # Update input_list with new files (if any) for further processing
192
+ self.input_list = new_input_list
193
+
194
+ def shutdown_pool(self):
195
+ """Shuts down the pool gracefully."""
196
+ if self.pool:
197
+ self.pool.close() # Stop accepting new tasks
198
+ self.pool.join() # Wait for all tasks to complete
199
+ self.pool = None
135
200
 
136
201
 
137
202
  class ConcurrentProcessorRecursive:
@@ -4,12 +4,6 @@ from dataclasses import dataclass
4
4
  from . import import_config
5
5
 
6
6
 
7
- SCRIPT_VERSION: str = '1.7.8'
8
- """
9
- added ca cert check and installation
10
- """
11
-
12
-
13
7
  # CONFIG = None
14
8
  LIST_OF_BOOLEANS: list = [
15
9
  ('dns', 'enable'),
@@ -59,7 +59,7 @@ def exit_cleanup():
59
59
  RECS_PROCESS_INSTANCE.join()
60
60
 
61
61
 
62
- def mitm_server(config_file_path: str):
62
+ def mitm_server(config_file_path: str, script_version: str):
63
63
  on_exit.register_exit_handler(exit_cleanup, at_exit=False)
64
64
 
65
65
  # Main function should return integer with error code, 0 is successful.
@@ -109,7 +109,7 @@ def mitm_server(config_file_path: str):
109
109
  system_logger.info("======================================")
110
110
  system_logger.info("Server Started.")
111
111
  system_logger.info(f"Python Version: {get_current_python_version_string()}")
112
- system_logger.info(f"Script Version: {config_static.SCRIPT_VERSION}")
112
+ system_logger.info(f"Script Version: {script_version}")
113
113
  system_logger.info(f"Atomic Workshop Version: {atomicshop.__version__}")
114
114
  system_logger.info(f"Log folder: {config_static.LogRec.logs_path}")
115
115
  if config_static.LogRec.enable_request_response_recordings_in_logs:
@@ -411,10 +411,10 @@ def _loop_at_midnight_recs_archive():
411
411
  time.sleep(60)
412
412
 
413
413
 
414
- def mitm_server_main(config_file_path: str):
414
+ def mitm_server_main(config_file_path: str, script_version: str):
415
415
  try:
416
416
  # Main function should return integer with error code, 0 is successful.
417
- return mitm_server(config_file_path)
417
+ return mitm_server(config_file_path, script_version)
418
418
  except KeyboardInterrupt:
419
419
  print_api.print_api("Server Stopped by [KeyboardInterrupt].", color='blue')
420
420
  exit_cleanup()
@@ -3,6 +3,14 @@ import fnmatch
3
3
 
4
4
  from .. import web, urls
5
5
  from ..print_api import print_api
6
+ from ..basics import strings
7
+
8
+
9
+ class MoreThanOneReleaseFoundError(Exception):
10
+ pass
11
+
12
+ class NoReleaseFoundError(Exception):
13
+ pass
6
14
 
7
15
 
8
16
  class GitHubWrapper:
@@ -174,16 +182,19 @@ class GitHubWrapper:
174
182
  download_urls.remove(download_url)
175
183
 
176
184
  # Find urls against 'string_pattern'.
177
- found_urls = fnmatch.filter(download_urls, string_pattern)
185
+ found_urls: list = fnmatch.filter(download_urls, string_pattern)
178
186
 
179
187
  # If more than 1 url answer the criteria, we can't download it. The user must be more specific in his input
180
188
  # strings.
181
189
  if len(found_urls) > 1:
182
190
  message = f'More than 1 result found in JSON response, try changing search string or extension.\n' \
183
191
  f'{found_urls}'
184
- print_api(message, color="red", error_type=True, **kwargs)
185
-
186
- return found_urls[0]
192
+ raise MoreThanOneReleaseFoundError(message)
193
+ elif len(found_urls) == 0:
194
+ message = f'No result found in JSON response, try changing search string or extension.'
195
+ raise NoReleaseFoundError(message)
196
+ else:
197
+ return found_urls[0]
187
198
 
188
199
  def download_latest_release(
189
200
  self,
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: atomicshop
3
- Version: 2.18.34
3
+ Version: 2.19.1
4
4
  Summary: Atomic functions and classes to make developer life easier
5
5
  Author: Denis Kras
6
6
  License: MIT License
@@ -1,4 +1,4 @@
1
- atomicshop/__init__.py,sha256=j1UndWBC7mEogg-U1kvD5l1DbldbQp6SbzJKKz6BwS8,124
1
+ atomicshop/__init__.py,sha256=YLnFX8FxOuGbWDCdCSt4RjxgiYy5Cc-U3A7YXjRxSd8,123
2
2
  atomicshop/_basics_temp.py,sha256=6cu2dd6r2dLrd1BRNcVDKTHlsHs_26Gpw8QS6v32lQ0,3699
3
3
  atomicshop/_create_pdf_demo.py,sha256=Yi-PGZuMg0RKvQmLqVeLIZYadqEZwUm-4A9JxBl_vYA,3713
4
4
  atomicshop/_patch_import.py,sha256=ENp55sKVJ0e6-4lBvZnpz9PQCt3Otbur7F6aXDlyje4,6334
@@ -55,6 +55,7 @@ atomicshop/a_installs/ubuntu/pycharm.py,sha256=Ld7YQBwPxrjuZcTG1K4kZqjbBdt8aooCV
55
55
  atomicshop/a_installs/win/fibratus.py,sha256=TU4e9gdZ_zI73C40uueJ59pD3qmN-UFGdX5GFoVf6cM,179
56
56
  atomicshop/a_installs/win/mongodb.py,sha256=AqyItXu19aaoe49pppDxtEkXey6PMy0PoT2Y_RmPpPE,179
57
57
  atomicshop/a_installs/win/pycharm.py,sha256=j_RSd7aDOyC3yDd-_GUTMLlQTmDrqtVFG--oUfGLiZk,140
58
+ atomicshop/a_installs/win/robocorp.py,sha256=sxSqncQIXr4rA8lCr_SwDU18IxWZlNznDE2WY3vPEMM,1789
58
59
  atomicshop/a_installs/win/wsl_ubuntu_lts.py,sha256=dZbPRLNKFeMd6MotjkE6UDY9cOiIaaclIdR1kGYWI50,139
59
60
  atomicshop/a_mains/dns_gateway_setting.py,sha256=ncc2rFQCChxlNP59UshwmTonLqC6MWblrVAzbbz-13M,149
60
61
  atomicshop/a_mains/msi_unpacker.py,sha256=5hrkqETYt9HIqR_3PMf32_q06kCrIcsdm_RJV9oY438,188
@@ -100,7 +101,7 @@ atomicshop/basics/isinstancing.py,sha256=fQ35xfqbguQz2BUn-3a4KVGskhTcIn8JjRtxV2r
100
101
  atomicshop/basics/list_of_classes.py,sha256=PJoE1VJdhhQ4gSFr88zW7IApXd4Ez7xLz-7vAM-7gug,978
101
102
  atomicshop/basics/list_of_dicts.py,sha256=tj0LNPf1ljNI_qpoO-PiOT4Ulmk1M-UpTGyn9twVcw8,8039
102
103
  atomicshop/basics/lists.py,sha256=I0C62vrDrNwCTNl0EjUZNa1Jsd8l0rTkp28GEx9QoEI,4258
103
- atomicshop/basics/multiprocesses.py,sha256=oU6LjcLLGBtPIGJzZBpDWoLU3HRmMoanITEOE2luAYw,18799
104
+ atomicshop/basics/multiprocesses.py,sha256=m8Vg6YDZX_Xz3r5uXvaTkFWQ-x6OvGOnTAVgWMFmvbU,21909
104
105
  atomicshop/basics/numbers.py,sha256=ESX0z_7o_ok3sOmCKAUBoZinATklgMy2v-4RndqXlVM,1837
105
106
  atomicshop/basics/package_module.py,sha256=fBd0uVgFce25ZCVtLq83iyowRlbwdWYFj_t4Ml7LU14,391
106
107
  atomicshop/basics/randoms.py,sha256=DmYLtnIhDK29tAQrGP1Nt-A-v8WC7WIEB8Edi-nk3N4,282
@@ -125,13 +126,13 @@ atomicshop/file_io/tomls.py,sha256=ol8EvQPf9sryTmZUf1v55BYSUQ6ml7HVVBHpNKbsIlA,9
125
126
  atomicshop/file_io/xlsxs.py,sha256=v_dyg9GD4LqgWi6wA1QuWRZ8zG4ZwB6Dz52ytdcmmmI,2184
126
127
  atomicshop/file_io/xmls.py,sha256=zh3SuK-dNaFq2NDNhx6ivcf4GYCfGM8M10PcEwDSpxk,2104
127
128
  atomicshop/mitm/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
128
- atomicshop/mitm/config_static.py,sha256=HIzxyMEj7DZksYvJsN5VuKpB-_HSVvuR6U59ztS9gi0,7871
129
+ atomicshop/mitm/config_static.py,sha256=DjSnDtMU8srdqca8s6Q-oFCWgjjiCjXRhyk-nafRAUk,7788
129
130
  atomicshop/mitm/config_toml_editor.py,sha256=2p1CMcktWRR_NW-SmyDwylu63ad5e0-w1QPMa8ZLDBw,1635
130
131
  atomicshop/mitm/connection_thread_worker.py,sha256=2180oQgD3qRS53zloeALs_vYt66RJWJtnaCqjl2jVs4,28187
131
132
  atomicshop/mitm/import_config.py,sha256=0Ij14aISTllTOiWYJpIUMOWobQqGofD6uafui5uWllE,9272
132
133
  atomicshop/mitm/initialize_engines.py,sha256=-3B8xkXyPOwrC_mDJm_uaW9s56VeV-f6VTu4-8dou2s,8258
133
134
  atomicshop/mitm/message.py,sha256=mNo4Lphr_Jo6IlNX5mPJzABpogWGkjOhwI4meAivwHw,2987
134
- atomicshop/mitm/mitm_main.py,sha256=Uko4lFG96ZeZ1yVJD5CT4c48NhfX_Hu1g0-THEiZfAc,23454
135
+ atomicshop/mitm/mitm_main.py,sha256=RqRlh5YIY2u817n_L1IUnki6JYXSmr7fTi3bjY5ygRA,23498
135
136
  atomicshop/mitm/recs_files.py,sha256=ZAAD0twun-FtmbSniXe3XQhIlawvANNB_HxwbHj7kwI,3151
136
137
  atomicshop/mitm/shared_functions.py,sha256=0lzeyINd44sVEfFbahJxQmz6KAMWbYrW5ou3UYfItvw,1777
137
138
  atomicshop/mitm/statistic_analyzer.py,sha256=5_sAYGX2Xunzo_pS2W5WijNCwr_BlGJbbOO462y_wN4,27533
@@ -186,7 +187,7 @@ atomicshop/wrappers/astw.py,sha256=VkYfkfyc_PJLIOxByT6L7B8uUmKY6-I8XGZl4t_z828,4
186
187
  atomicshop/wrappers/configparserw.py,sha256=JwDTPjZoSrv44YKwIRcjyUnpN-FjgXVfMqMK_tJuSgU,22800
187
188
  atomicshop/wrappers/cryptographyw.py,sha256=LfzTnwvJE03G6WZryOOf43VKhhnyMakzHpn8DPPCoy4,13252
188
189
  atomicshop/wrappers/ffmpegw.py,sha256=wcq0ZnAe0yajBOuTKZCCaKI7CDBjkq7FAgdW5IsKcVE,6031
189
- atomicshop/wrappers/githubw.py,sha256=gUUVlFnlwKxINXAoK0PTOditEEhe0sWe06aAZ4eQz64,12240
190
+ atomicshop/wrappers/githubw.py,sha256=Ft-QQ4sewzjEWEiW89A_nv9wcKVaQdq76TWvVS6r9mI,12576
190
191
  atomicshop/wrappers/msiw.py,sha256=GQLqud72nfex3kvO1bJSruNriCYTYX1_G1gSf1MPkIA,6118
191
192
  atomicshop/wrappers/numpyw.py,sha256=sBV4gSKyr23kXTalqAb1oqttzE_2XxBooCui66jbAqc,1025
192
193
  atomicshop/wrappers/olefilew.py,sha256=biD5m58rogifCYmYhJBrAFb9O_Bn_spLek_9HofLeYE,2051
@@ -321,8 +322,8 @@ atomicshop/wrappers/socketw/statistics_csv.py,sha256=fgMzDXI0cybwUEqAxprRmY3lqbh
321
322
  atomicshop/wrappers/winregw/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
322
323
  atomicshop/wrappers/winregw/winreg_installed_software.py,sha256=Qzmyktvob1qp6Tjk2DjLfAqr_yXV0sgWzdMW_9kwNjY,2345
323
324
  atomicshop/wrappers/winregw/winreg_network.py,sha256=AENV88H1qDidrcpyM9OwEZxX5svfi-Jb4N6FkS1xtqA,8851
324
- atomicshop-2.18.34.dist-info/LICENSE.txt,sha256=lLU7EYycfYcK2NR_1gfnhnRC8b8ccOTElACYplgZN88,1094
325
- atomicshop-2.18.34.dist-info/METADATA,sha256=IC6AQpsSMvtPkGi3wXttRIZb-2uDkNfDkbm-xpOPC8s,10631
326
- atomicshop-2.18.34.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
327
- atomicshop-2.18.34.dist-info/top_level.txt,sha256=EgKJB-7xcrAPeqTRF2laD_Np2gNGYkJkd4OyXqpJphA,11
328
- atomicshop-2.18.34.dist-info/RECORD,,
325
+ atomicshop-2.19.1.dist-info/LICENSE.txt,sha256=lLU7EYycfYcK2NR_1gfnhnRC8b8ccOTElACYplgZN88,1094
326
+ atomicshop-2.19.1.dist-info/METADATA,sha256=R3uYBPX5Re5tpvd0nokDfmh_F1z1rWPThXWTPLvoFE4,10630
327
+ atomicshop-2.19.1.dist-info/WHEEL,sha256=GJ7t_kWBFywbagK5eo9IoUwLW6oyOeTKmQ-9iHFVNxQ,92
328
+ atomicshop-2.19.1.dist-info/top_level.txt,sha256=EgKJB-7xcrAPeqTRF2laD_Np2gNGYkJkd4OyXqpJphA,11
329
+ atomicshop-2.19.1.dist-info/RECORD,,