Skip to content

Commit

Permalink
app: Update Forall command to allow multiple concurrent processes
Browse files Browse the repository at this point in the history
Demonstrate asynchronous behavior for the Forall command and add an
argument to select the number of jobs.

Signed-off-by: Pieter De Gendt <pieter.degendt@basalte.be>
  • Loading branch information
pdgendt committed Oct 16, 2024
1 parent c3aadf5 commit 06fd709
Showing 1 changed file with 26 additions and 14 deletions.
40 changes: 26 additions & 14 deletions src/west/app/project.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
'''West project commands'''

import argparse
import asyncio
from functools import partial
import logging
import os
Expand Down Expand Up @@ -1670,16 +1671,15 @@ def do_add_parser(self, parser_adder):
parser.add_argument('projects', metavar='PROJECT', nargs='*',
help='''projects (by name or path) to operate on;
defaults to active cloned projects''')
parser.add_argument('-j', '--jobs', nargs='?', const=-1,
default=1, type=int, action='store',
help='''Use multiple jobs to parallelize commands.
Pass no number or -1 to run commands on all cores.''')
return parser

def do_run(self, args, user_args):
failed = []
group_set = set(args.groups)
env = os.environ.copy()
for project in self._cloned_projects(args, only_active=not args.all):
if group_set and not group_set.intersection(set(project.groups)):
continue

async def run_for_project(self, project, args, sem):
async with sem:
env = os.environ.copy()
env["WEST_PROJECT_NAME"] = project.name
env["WEST_PROJECT_PATH"] = project.path
env["WEST_PROJECT_ABSPATH"] = project.abspath if project.abspath else ''
Expand All @@ -1689,12 +1689,24 @@ def do_run(self, args, user_args):

cwd = args.cwd if args.cwd else project.abspath

self.banner(
f'running "{args.subcommand}" in {project.name_and_path}:')
rc = subprocess.Popen(args.subcommand, shell=True, env=env,
cwd=cwd).wait()
if rc:
failed.append(project)
self.banner(f'running "{args.subcommand}" in {project.name_and_path}:')
proc = await asyncio.create_subprocess_shell(args.subcommand,
cwd=cwd, env=env, shell=True)
return await proc.wait()

def do_run(self, args, unknown):
group_set = set(args.groups)
projects = [p for p in self._cloned_projects(args, only_active=not args.all)
if not group_set or group_set.intersection(set(p.groups))]

asyncio.run(self.do_run_async(args, projects))

async def do_run_async(self, args, projects):
sem = asyncio.Semaphore(args.jobs if args.jobs > 0 else os.cpu_count() or sys.maxsize)

rcs = await asyncio.gather(*[self.run_for_project(p, args, sem) for p in projects])

failed = [p for (p, rc) in zip(projects, rcs) if rc]
self._handle_failed(args, failed)

GREP_EPILOG = '''
Expand Down

0 comments on commit 06fd709

Please sign in to comment.