aboutsummaryrefslogtreecommitdiff
path: root/mesonbuild/msubprojects.py
diff options
context:
space:
mode:
authorXavier Claessens <xavier.claessens@collabora.com>2021-06-22 13:42:57 -0400
committerXavier Claessens <xavier.claessens@collabora.com>2021-06-25 15:16:55 -0400
commit4aaccdb6ba20cdd4ffc5072393afe9a388587d34 (patch)
treea7118b0fcaf9f3a4c5aec1cf78c02c9e808a5bb0 /mesonbuild/msubprojects.py
parent27970b1d85f47a2865c439b11dcadb0f8add2678 (diff)
downloadmeson-4aaccdb6ba20cdd4ffc5072393afe9a388587d34.zip
meson-4aaccdb6ba20cdd4ffc5072393afe9a388587d34.tar.gz
meson-4aaccdb6ba20cdd4ffc5072393afe9a388587d34.tar.bz2
msubprojects: Display progress while tasks are running
Logs are printed only when the task is done to not interleave logs while running multiple tasks in parallel. That means that nothing is printed until the task is done and that could take time.
Diffstat (limited to 'mesonbuild/msubprojects.py')
-rwxr-xr-xmesonbuild/msubprojects.py58
1 files changed, 49 insertions, 9 deletions
diff --git a/mesonbuild/msubprojects.py b/mesonbuild/msubprojects.py
index 1e13b14..63ea98a 100755
--- a/mesonbuild/msubprojects.py
+++ b/mesonbuild/msubprojects.py
@@ -3,8 +3,10 @@ import argparse
import asyncio
import threading
import copy
+import shutil
from concurrent.futures.thread import ThreadPoolExecutor
from pathlib import Path
+import typing as T
from . import mlog
from .mesonlib import quiet_git, GitException, Popen_safe, MesonException, windows_proof_rmtree
@@ -13,10 +15,46 @@ from .wrap import wraptool
ALL_TYPES_STRING = ', '.join(ALL_TYPES)
-class Runner:
- lock = threading.Lock()
+class Logger:
+ def __init__(self, total_tasks: int) -> None:
+ self.lock = threading.Lock()
+ self.total_tasks = total_tasks
+ self.completed_tasks = 0
+ self.running_tasks = set()
+ self.should_erase_line = ''
+
+ def flush(self) -> None:
+ if self.should_erase_line:
+ print(self.should_erase_line, end='\r')
+ self.should_erase_line = ''
+
+ def print_progress(self) -> None:
+ line = f'Progress: {self.completed_tasks} / {self.total_tasks}'
+ max_len = shutil.get_terminal_size().columns - len(line)
+ running = ', '.join(self.running_tasks)
+ if len(running) + 3 > max_len:
+ running = running[:max_len - 6] + '...'
+ line = line + f' ({running})'
+ print(self.should_erase_line, line, sep='', end='\r')
+ self.should_erase_line = '\x1b[K'
+
+ def start(self, wrap_name: str) -> None:
+ with self.lock:
+ self.running_tasks.add(wrap_name)
+ self.print_progress()
+
+ def done(self, wrap_name: str, log_queue: T.List[T.Tuple[mlog.TV_LoggableList, T.Any]]) -> None:
+ with self.lock:
+ self.flush()
+ for args, kwargs in log_queue:
+ mlog.log(*args, **kwargs)
+ self.running_tasks.remove(wrap_name)
+ self.completed_tasks += 1
+ self.print_progress()
- def __init__(self, r: Resolver, wrap: PackageDefinition, repo_dir: str, options: argparse.Namespace) -> None:
+
+class Runner:
+ def __init__(self, logger: Logger, r: Resolver, wrap: PackageDefinition, repo_dir: str, options: argparse.Namespace) -> None:
# FIXME: Do a copy because Resolver.resolve() is stateful method that
# cannot be called from multiple threads.
self.wrap_resolver = copy.copy(r)
@@ -25,15 +63,15 @@ class Runner:
self.options = options
self.run_method = options.subprojects_func.__get__(self)
self.log_queue = []
+ self.logger = logger
def log(self, *args, **kwargs):
self.log_queue.append((args, kwargs))
def run(self):
+ self.logger.start(self.wrap.name)
result = self.run_method()
- with self.lock:
- for args, kwargs in self.log_queue:
- mlog.log(*args, **kwargs)
+ self.logger.done(self.wrap.name, self.log_queue)
return result
def update_wrapdb_file(self):
@@ -491,15 +529,17 @@ def run(options):
task_names = []
loop = asyncio.get_event_loop()
executor = ThreadPoolExecutor(options.num_processes)
+ if types:
+ wraps = [wrap for wrap in wraps if wrap.type in types]
+ logger = Logger(len(wraps))
for wrap in wraps:
- if types and wrap.type not in types:
- continue
dirname = Path(subprojects_dir, wrap.directory).as_posix()
- runner = Runner(r, wrap, dirname, options)
+ runner = Runner(logger, r, wrap, dirname, options)
task = loop.run_in_executor(executor, runner.run)
tasks.append(task)
task_names.append(wrap.name)
results = loop.run_until_complete(asyncio.gather(*tasks))
+ logger.flush()
post_func = getattr(options, 'post_func', None)
if post_func:
post_func(options)