bumblebee-status/bumblebee_status/util/cli.py

81 lines
2.9 KiB
Python
Raw Normal View History

2020-03-01 13:36:12 +00:00
import os
import shlex
import subprocess
import logging
2020-05-14 18:36:01 +00:00
def execute(
cmd,
wait=True,
ignore_errors=False,
include_stderr=False,
env=None,
return_exitcode=False,
shell=False,
2020-05-14 18:36:01 +00:00
):
2020-05-05 17:55:37 +00:00
"""Executes a commandline utility and returns its output
:param cmd: the command (as string) to execute, returns the program's output
:param wait: set to True to wait for command completion, False to return immediately, defaults to True
:param ignore_errors: set to True to return a string when an exception is thrown, otherwise might throw, defaults to False
:param include_stderr: set to True to include stderr output in the return value, defaults to False
:param env: provide a dict here to specify a custom execution environment, defaults to None
:param return_exitcode: set to True to return a pair, where the first member is the exit code and the message the second, defaults to False
:param shell: set to True to run command in a separate shell, defaults to False
2020-05-05 17:55:37 +00:00
:raises RuntimeError: the command either didn't exist or didn't exit cleanly, and ignore_errors was set to False
:return: output of cmd, or stderr, if ignore_errors is True and the command failed; or a tuple of exitcode and the previous, if return_exitcode is set to True
:rtype: string or tuple (if return_exitcode is set to True)
2020-05-05 17:55:37 +00:00
"""
args = cmd if shell else shlex.split(cmd)
logging.debug(cmd)
if not env:
env = os.environ.copy()
myenv = env.copy()
myenv["LC_ALL"] = "C"
if "WAYLAND_SOCKET" in myenv:
del myenv["WAYLAND_SOCKET"]
try:
proc = subprocess.Popen(
args,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT if include_stderr else subprocess.PIPE,
env=myenv,
shell=shell,
)
except FileNotFoundError as e:
raise RuntimeError("{} not found".format(cmd))
if wait:
timeout = 60
try:
out, _ = proc.communicate(timeout=timeout)
except subprocess.TimeoutExpired as e:
logging.warning(
f"""
Communication with process pid={proc.pid} hangs for more
than {timeout} seconds.
If this is not expected, the process is stale, or
you might have run in stdout / stderr deadlock.
"""
)
out, _ = proc.communicate()
2020-03-30 19:09:09 +00:00
if proc.returncode != 0:
err = "{} exited with code {}".format(cmd, proc.returncode)
logging.warning(err)
2020-03-30 19:09:09 +00:00
if ignore_errors:
return (proc.returncode, err) if return_exitcode else err
2020-03-30 19:09:09 +00:00
raise RuntimeError(err)
res = out.decode("utf-8")
logging.debug(res)
return (proc.returncode, res) if return_exitcode else res
return (0, "") if return_exitcode else ""
# vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4