Oct. 11, 2013

Subprocess and Shell Commands in Python

Subprocess Overview

For a long time I have been using os.system() when dealing with system
administration tasks in Python. 

The main reason for that, was that I thought that was the simplest way of
running Linux commands.
In the official python documentation we can read that subprocess should be used
for accessing system commands.

The subprocess module allows us to spawn processes, connect to their
input/output/error pipes, and obtain their return codes. 

Subprocess intends to replace several other, older modules and functions,
like: os.system, os.spawn*, os.popen*, popen2.* commands. [Source]
Let's start looking into the different functions of subprocess.


Run the command described by "args".

We can run the command line with the arguments passed as a list of strings
(example 1)  or by setting the shell argument to a True value (example 2) 

Note, the default value of the shell argument is False. 
Let's look at two examples where we show the summary of disk usage using
subprocess.call(['df', '-h'])
This time we set the shell argument to True
subprocess.call('du -hs $HOME', shell=True)
Note, the official Python documentation states a warning about using the
shell=True argument.

"Invoking the system shell with shell=True can be a security hazard if combined
with untrusted input" [source]
Now, let's move on and look at the Input / Output.

Input and Output

With subprocess you can suppress the output, which is very handy when you
want to run a system call but are not interested about the standard output. 

It also gives you a way to cleanly integrate shell commands into your scripts
while managing input/output in a standard way.

Return Codes

You can use subprocess.call return codes to determine the success of the command. 

Every process will return an exit code and you can do something with your script
based on that code. 

If the return code is anything else than zero, it means that an error occurred. 

If you want to do system administration in Python, I recommend reading
Python for Unix and Linux System Administration

stdin, stdout and stderr

One of the trickiest part I had with subprocess was how to work with pipes
and to pipe commands together. 

PIPE indicates that a new pipe to the child should be created.

The default setting is "None", which means that no redirection will occur.

The standard error (or stderr) can be STDOUT, which indicates that the stderr
data from the child process should be captured into the same file handle as
for stdout.


The underlying process creation and management in the subprocess module is
handled by the Popen class. subprocess.popen is replacing os.popen.
Let's get started with some real examples. 

subprocess.Popen takes a list of arguments
import subprocess

p = subprocess.Popen(["echo", "hello world"], stdout=subprocess.PIPE)

print p.communicate()

>>>('hello world
', None)
Note, even though you could have used "shell=True", it is not the recommended
way of doing it. 
If you know that you will only work with specific subprocess functions,
such as Popen and PIPE, then it is enough to only import those.
from subprocess import Popen, PIPE

p1 = Popen(["dmesg"], stdout=PIPE)

print p1.communicate()


The communicate() method returns a tuple (stdoutdata, stderrdata). 

Popen.communicate() interacts with process: Send data to stdin.

Read data from stdout and stderr, until end-of-file is reached.

Wait for process to terminate.

The optional input argument should be a string to be sent to the
child process, or None, if no data should be sent to the child.

Basically, when you use communicate() it means that you want to
execute the command

Ping program using subprocess

In the "More Reading" section below, you can find links to read more
about the subprocess module, but also examples. 
Let's write our own ping program where we first ask the user for input,
and then perform the ping request to that host.
# Import the module
import subprocess

# Ask the user for input
host = raw_input("Enter a host to ping: ")	

# Set up the echo command and direct the output to a pipe
p1 = subprocess.Popen(['ping', '-c 2', host], stdout=subprocess.PIPE)

# Run the command
output = p1.communicate()[0]

print output
Let's show one more example. This time we use the host command. 
target = raw_input("Enter an IP or Host to ping:

host = subprocess.Popen(['host', target], stdout = subprocess.PIPE).communicate()[0]

print host
I recommend that you read the links below to gain more knowledge about the
subprocess module in Python.

If you have any questions or comments, please use the comment field below. 
More Reading
The ever useful and neat subprocess module

Recommended Python Training – DataCamp

For Python training, our top recommendation is DataCamp.

Datacamp provides online interactive courses that combine interactive coding challenges with videos from top instructors in the field.

Datacamp has beginner to advanced Python training that programmers of all levels benefit from.


Read more about:
Disclosure of Material Connection: Some of the links in the post above are “affiliate links.” This means if you click on the link and purchase the item, I will receive an affiliate commission. Regardless, PythonForBeginners.com only recommend products or services that we try personally and believe will add value to our readers.