2023-07-27 15:53:10 +02:00
|
|
|
import sys
|
|
|
|
from queue import Queue
|
|
|
|
from threading import Thread
|
2023-08-02 10:31:52 +02:00
|
|
|
from multiprocessing import Pool, Manager
|
|
|
|
|
|
|
|
####
|
|
|
|
# Define multi-processing internal global variables.
|
|
|
|
#########################################################################################
|
|
|
|
|
2023-08-02 11:32:34 +02:00
|
|
|
_MANAGER = Manager()
|
|
|
|
_SHARED_DICT = _MANAGER.dict()
|
|
|
|
_SHARED_DICT_LOCK = _MANAGER.Lock()
|
2023-07-27 15:53:10 +02:00
|
|
|
|
|
|
|
####
|
|
|
|
# Internal use only: Run a given function in a thread
|
|
|
|
#########################################################################################
|
|
|
|
def _single_thread(
|
|
|
|
q = None,
|
|
|
|
function: callable = None,
|
|
|
|
errors: list = []
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Internal use only: Run a given function in a thread
|
|
|
|
"""
|
|
|
|
params=q.get()
|
|
|
|
q.task_done()
|
|
|
|
try:
|
|
|
|
function(params)
|
|
|
|
except Exception as e:
|
|
|
|
errors.append("Error while runing single thread: "+str(e))
|
|
|
|
|
|
|
|
####
|
|
|
|
# Run a given function for given items list in a given number of threads
|
|
|
|
#########################################################################################
|
|
|
|
def run_threads(
|
|
|
|
max_threads: int = 1,
|
|
|
|
function: callable = None,
|
2023-08-02 10:31:52 +02:00
|
|
|
items: list = [],
|
|
|
|
print_errors: bool = False
|
2023-07-27 15:53:10 +02:00
|
|
|
) -> bool:
|
|
|
|
"""
|
|
|
|
Run a given function for given items list in a given number of threads
|
|
|
|
"""
|
2023-08-09 13:46:09 +02:00
|
|
|
from .output import print_stderr
|
2023-07-27 15:53:10 +02:00
|
|
|
|
|
|
|
# Assign threads
|
|
|
|
threads = max_threads
|
|
|
|
|
|
|
|
if threads > len(items):
|
|
|
|
threads = len(items)
|
|
|
|
|
|
|
|
if threads < 1:
|
|
|
|
threads = 1
|
|
|
|
|
|
|
|
# Distribute items per thread
|
|
|
|
items_per_thread = []
|
|
|
|
thread = 0
|
|
|
|
for item in items:
|
|
|
|
if not 0 <= thread < len(items_per_thread):
|
|
|
|
items_per_thread.append([])
|
|
|
|
|
|
|
|
items_per_thread[thread].append(item)
|
|
|
|
|
|
|
|
thread += 1
|
|
|
|
if thread >= threads:
|
|
|
|
thread=0
|
|
|
|
|
|
|
|
# Run threads
|
|
|
|
try:
|
|
|
|
q=Queue()
|
|
|
|
for n_thread in range(threads) :
|
|
|
|
q.put(items_per_thread[n_thread])
|
|
|
|
|
|
|
|
run_threads = []
|
|
|
|
errors = []
|
|
|
|
|
|
|
|
for n_thread in range(threads):
|
|
|
|
t = Thread(target=_single_thread, args=(q, function, errors))
|
|
|
|
t.daemon=True
|
|
|
|
t.start()
|
|
|
|
run_threads.append(t)
|
|
|
|
|
|
|
|
for t in run_threads:
|
|
|
|
t.join()
|
|
|
|
|
|
|
|
q.join()
|
|
|
|
|
2023-08-02 10:31:52 +02:00
|
|
|
if print_errors:
|
|
|
|
for error in errors:
|
2023-08-09 13:46:09 +02:00
|
|
|
print_stderr(str(error))
|
2023-07-27 15:53:10 +02:00
|
|
|
|
|
|
|
if len(errors) > 0:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
|
|
|
except Exception as e:
|
2023-08-02 10:31:52 +02:00
|
|
|
if print_errors:
|
2023-08-09 13:46:09 +02:00
|
|
|
print_stderr("Error while running threads: "+str(e))
|
2023-08-02 10:31:52 +02:00
|
|
|
return False
|
|
|
|
|
|
|
|
####
|
|
|
|
# Set a given value to a key in the internal shared dict.
|
|
|
|
# Used by all parallel processes.
|
|
|
|
#########################################################################################
|
|
|
|
def set_shared_dict_value(
|
|
|
|
key: str = None,
|
|
|
|
value = None
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Set a given value to a key in the internal shared dict.
|
|
|
|
Used by all parallel processes.
|
|
|
|
"""
|
2023-08-02 11:32:34 +02:00
|
|
|
global _SHARED_DICT
|
2023-08-02 10:31:52 +02:00
|
|
|
|
|
|
|
if key is not None:
|
2023-08-02 11:32:34 +02:00
|
|
|
with _SHARED_DICT_LOCK:
|
|
|
|
_SHARED_DICT[key] = value
|
2023-08-02 10:31:52 +02:00
|
|
|
|
|
|
|
####
|
|
|
|
# Add a given value to a key in the internal shared dict.
|
|
|
|
# Used by all parallel processes.
|
|
|
|
#########################################################################################
|
|
|
|
def add_shared_dict_value(
|
|
|
|
key: str = None,
|
|
|
|
value = None
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Add a given value to a key in the internal shared dict.
|
|
|
|
Used by all parallel processes.
|
|
|
|
"""
|
2023-08-02 11:32:34 +02:00
|
|
|
global _SHARED_DICT
|
2023-08-02 10:31:52 +02:00
|
|
|
|
|
|
|
if key is not None:
|
2023-08-02 11:32:34 +02:00
|
|
|
with _SHARED_DICT_LOCK:
|
|
|
|
if key in _SHARED_DICT:
|
|
|
|
_SHARED_DICT[key] += value
|
2023-08-02 10:31:52 +02:00
|
|
|
else:
|
|
|
|
set_shared_dict_value(key, value)
|
|
|
|
|
|
|
|
####
|
|
|
|
# Get the value of a key in the internal shared dict.
|
|
|
|
# Used by all parallel processes.
|
|
|
|
#########################################################################################
|
|
|
|
def get_shared_dict_value(
|
|
|
|
key: str = None
|
|
|
|
):
|
|
|
|
"""
|
|
|
|
Get the value of a key in the internal shared dict.
|
|
|
|
Used by all parallel processes.
|
|
|
|
"""
|
2023-08-02 11:32:34 +02:00
|
|
|
global _SHARED_DICT
|
2023-08-02 10:31:52 +02:00
|
|
|
|
2023-08-02 11:32:34 +02:00
|
|
|
with _SHARED_DICT_LOCK:
|
|
|
|
if key in _SHARED_DICT and key is not None:
|
|
|
|
return _SHARED_DICT[key]
|
2023-08-02 10:31:52 +02:00
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
####
|
|
|
|
# Run a given function for given items list in a given number of processes
|
|
|
|
# Given function receives each item as first parameter
|
|
|
|
#########################################################################################
|
|
|
|
def run_processes(
|
|
|
|
max_processes: int = 1,
|
|
|
|
function: callable = None,
|
|
|
|
items: list = [],
|
|
|
|
print_errors: bool = False
|
|
|
|
) -> bool:
|
|
|
|
"""
|
|
|
|
Run a given function for given items list in a given number of processes
|
|
|
|
"""
|
2023-08-09 13:46:09 +02:00
|
|
|
from .output import print_stderr
|
2023-08-02 10:31:52 +02:00
|
|
|
|
|
|
|
# Assign processes
|
|
|
|
processes = max_processes
|
|
|
|
|
|
|
|
if processes > len(items):
|
|
|
|
processes = len(items)
|
|
|
|
|
|
|
|
if processes < 1:
|
|
|
|
processes = 1
|
|
|
|
|
|
|
|
# Run processes
|
|
|
|
with Pool(processes) as pool:
|
|
|
|
try:
|
|
|
|
pool.map(function, items)
|
|
|
|
result = True
|
|
|
|
except Exception as error:
|
|
|
|
if print_errors:
|
2023-08-09 13:46:09 +02:00
|
|
|
print_stderr(str(error))
|
2023-08-02 10:31:52 +02:00
|
|
|
result = False
|
|
|
|
|
|
|
|
return result
|