kiri
kiri

Reputation: 2642

How to use a custom file-like object as subprocess stdout/stderr?

Consider this code, where a subprocess.Popen is spawned. I'd like writes to the subprocess' stdout and stderr to go to my custom file-object's .write() method, however this isn't the case.

import subprocess

class Printer:

    def __init__(self):
        pass

    def write(self, chunk):
        print('Writing:', chunk)

    def fileno(self):
        return 0

    def close(self):
        return

proc = subprocess.Popen(['bash', '-c', 'echo Testing'], 
                        stdout=Printer(),
                        stderr=subprocess.STDOUT)
proc.wait()

Why is the .write() method not used, and what is the use of specifying a stdout= parameter in this case?

Upvotes: 11

Views: 5883

Answers (2)

Beefster
Beefster

Reputation: 789

Not directly. Maybe some future version of Python will support converting file-like objects into some sort of auto-populating pipe, but the crux of it is that the subprocess needs to have access to a file handle it can read without calling Python code somehow. This means it needs to be something that works at an OS level, meaning it is limited to a few possibilities:

  • Inheriting stdin from the parent process (which happens when stdin=None)
  • Taking input from a file (which happens when stdin is a file or an integer file handle)
  • Taking input from a pipe (which happens when stdin=subprocess.PIPE)

Using a file-like object means you're going to have to read the data yourself and then feed it through a pipe.

For example:

proc = subprocess.Popen(['sha256sum', '-'], stdin=subprocess.PIPE)
while True:
    chunk = filelike.read(BLOCK_SIZE)
    proc.stdin.write(chunk)
    if len(chunk) < BLOCK_SIZE:
        break
proc.wait()

Upvotes: 0

falsetru
falsetru

Reputation: 369094

According to the documentation:

stdin, stdout and stderr specify the executed program’s standard input, standard output and standard error file handles, respectively. Valid values are PIPE, DEVNULL, an existing file descriptor (a positive integer), an existing file object, and None.

Using subprocess.PIPE:

proc = subprocess.Popen(['bash', '-c', 'echo Testing'], 
                        stdout=subprocess.PIPE,
                        stderr=subprocess.STDOUT)
print('Writing:', proc.stdout.read())
# OR  print('Writing:', proc.stdout.read().decode())

Upvotes: 5

Related Questions