import subprocess as _subprocess
import sys as _sys
+from .. import LOG as LOG
+from ..utils import ENCODING as _ENCODING
+
_MSWINDOWS = _sys.platform == 'win32'
_POSIX = not _MSWINDOWS
def invoke(args, stdin=None, stdout=_subprocess.PIPE, stderr=_subprocess.PIPE,
- expect=(0,)):
- """
- expect should be a tuple of allowed exit codes.
+ cwd=None, expect=(0,), unicode_output=False, encoding=None):
+ """Invoke an external program and return the results
+
+ ``expect`` should be a tuple of allowed exit codes.
+
+ When ``unicode_output`` is ``True``, convert stdout and stdin
+ strings to unicode before returing them.
"""
+ LOG.debug('{}$ {}'.format(cwd, args))
try :
if _POSIX:
q = _subprocess.Popen(args, stdin=_subprocess.PIPE,
- stdout=stdout, stderr=stderr)
+ stdout=stdout, stderr=stderr,
+ close_fds=True, cwd=cwd)
else:
assert _MSWINDOWS == True, 'invalid platform'
# win32 don't have os.execvp() so run the command in a shell
q = _subprocess.Popen(args, stdin=_subprocess.PIPE,
- stdout=stdout, stderr=stderr, shell=True)
+ stdout=stdout, stderr=stderr, shell=True,
+ cwd=cwd)
except OSError, e:
raise ValueError([args, e])
stdout,stderr = q.communicate(input=stdin)
status = q.wait()
+ if unicode_output == True:
+ if encoding is None:
+ encoding = _ENCODING
+ if stdout is not None:
+ stdout = unicode(stdout, encoding)
+ if stderr is not None:
+ stderr = unicode(stderr, encoding)
if status not in expect:
raise ValueError([args, status, stdout, stderr])
return status, stdout, stderr
while True:
dirname,basename = _os_path.split(path)
elements.insert(0,basename)
- if dirname in ['', '.']:
+ if dirname in ['/', '', '.']:
break
path = dirname
return tuple(elements)