KEMBAR78
Python 101 | PDF | Python (Programming Language) | Integrated Development Environment
100% found this document useful (4 votes)
2K views295 pages

Python 101

Book to learning about Python

Uploaded by

Rodolfo Silva
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
100% found this document useful (4 votes)
2K views295 pages

Python 101

Book to learning about Python

Uploaded by

Rodolfo Silva
Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as PDF, TXT or read online on Scribd
You are on page 1/ 295

Python 101

Michael Driscoll
This book is for sale at http://leanpub.com/python_101
This version was published on 2016-10-27

This is a Leanpub book. Leanpub empowers authors and publishers with the Lean Publishing
process. Lean Publishing is the act of publishing an in-progress ebook using lightweight tools and
many iterations to get reader feedback, pivot until you have the right book and build traction once
you do.
2014 Michael Driscoll

Also By Michael Driscoll


Python 201
wxPython Cookbook

Contents
Introduction . . . . . . . . .
A Brief History of Python
About the Author . . . .
Conventions . . . . . . .
Requirements . . . . . .
Reader Feedback . . . . .
Errata . . . . . . . . . . .

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

1
3
3
4
4
4
5

Part I - Learning the Basics . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .

Chapter 1 - IDLE Programming


Using IDLE . . . . . . . . . .
Other Tips . . . . . . . . . .
Wrapping Up . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

8
8
12
13

Chapter 2 - All About Strings


How to Create a String . .
String Concatenation . . .
String Methods . . . . . . .
String Slicing . . . . . . . .
String Formatting . . . . .
Wrapping Up . . . . . . . .

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

14
14
16
16
18
19
23

Chapter 3 - Lists, Tuples and Dictionaries


Lists . . . . . . . . . . . . . . . . . . . .
Tuples . . . . . . . . . . . . . . . . . .
Dictionaries . . . . . . . . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

24
24
26
26
28

Chapter 4 - Conditional Statements


The if statement . . . . . . . . .
Boolean Operations . . . . . . .
Checking for Nothing . . . . . .
Special Characters . . . . . . . .
if __name__ == __main__ . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

29
29
31
32
34
35

.
.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

CONTENTS

Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
Chapter 5 - Loops . . . . .
The for Loop . . . . . .
The while Loop . . . .
What else is for in loops
Wrapping Up . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

36
36
38
40
41

Chapter 6 - Python Comprehensions


List Comprehensions . . . . . . .
Dictionary Comprehensions . . . .
Set Comprehensions . . . . . . . .
Wrapping Up . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

42
42
43
44
44

Chapter 7 - Exception Handling


Common Exceptions . . . . .
How to Handle Exceptions .
The finally Statement . . . .
try, except, or else! . . . . . .
Wrapping Up . . . . . . . . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

45
45
46
48
49
50

Chapter 8 - Working with Files . .


How to Read a File . . . . . . . .
How To Read Files Piece by Piece
How to Read a Binary File . . . .
Writing Files in Python . . . . .
Using the with Operator . . . . .
Catching Errors . . . . . . . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

51
51
52
53
53
54
54
55

Chapter 9 - Importing . .
import this . . . . . .
Using from to import
Importing Everything!
Wrapping Up . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

56
56
57
57
58

Chapter 10 - Functions . . . . . . .
An Empty Function (the stub) . .
Passing Arguments to a Function
Keyword Arguments . . . . . . .
*args and **kwargs . . . . . . . .
A Note on Scope and Globals . .
Coding Tips . . . . . . . . . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

59
59
59
60
62
62
63
64

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.

35

.
.
.
.
.

CONTENTS

Chapter 11 - Classes
Creating a Class
What is self? . .
Subclasses . . .
Wrapping Up . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

65
65
67
69
70

Part II - Learning from the Library . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .

71

Chapter 12 - Introspection
The Python Type . . . .
The Python Dir . . . .
Python Help! . . . . . .
Wrapping Up . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

73
73
74
75
75

Chapter 13 - The csv Module


Reading a CSV File . . . .
Writing a CSV File . . . .
Wrapping Up . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

76
76
77
80

Chapter 14 - configparser . . . . . . . . . . .
Creating a Config File . . . . . . . . . . .
How to Read, Update and Delete Options
How to Use Interpolation . . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

81
81
82
85
85

Chapter 15 - Logging . . . . . . . . . . . . . . . . . . . . . .
Creating a Simple Logger . . . . . . . . . . . . . . . . . .
How to log From Multiple Modules (and Formatting too!)
Configuring Logs for Work and Pleasure . . . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

86
86
87
90
93

Chapter 16 - The os Module . . . . . . .


os.name . . . . . . . . . . . . . . . . .
os.environ, os.getenv() and os.putenv()
os.chdir() and os.getcwd() . . . . . . .
os.mkdir() and os.makedirs() . . . . .
os.remove() and os.rmdir() . . . . . .
os.rename(src, dst) . . . . . . . . . . .
os.startfile() . . . . . . . . . . . . . . .
os.walk() . . . . . . . . . . . . . . . .
os.path . . . . . . . . . . . . . . . . .
os.path.basename . . . . . . . . . . .
os.path.dirname . . . . . . . . . . . .
os.path.exists . . . . . . . . . . . . . .

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

94
94
95
97
97
98
98
99
99
100
100
100
101

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.
.
.
.
.

CONTENTS

os.path.isdir / os.path.isfile
os.path.join . . . . . . . . .
os.path.split . . . . . . . .
Wrapping Up . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

101
102
102
103

Chapter 17 - The email / smtplib Module . . . . . . .


Email Basics - How to Send an Email with smtplib
Sending Multiple Emails at Once . . . . . . . . . .
Send email using the TO, CC and BCC lines . . . .
Add an attachment / body using the email module
Wrapping Up . . . . . . . . . . . . . . . . . . . . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

104
104
107
109
110
112

Chapter 18 - The sqlite Module . . . . . . . . . . . .


How to Create a Database and INSERT Some Data
Updating and Deleting Records . . . . . . . . . . .
Basic SQLite Queries . . . . . . . . . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

113
113
114
115
116

Chapter 19 - The subprocess Module


The call function . . . . . . . . . .
The Popen Class . . . . . . . . . .
Learning to Communicate . . . . .
Wrapping Up . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

117
117
119
119
121

Chapter 20 - The sys Module


sys.argv . . . . . . . . . .
sys.executable . . . . . .
sys.exit . . . . . . . . . .
sys.path . . . . . . . . . .
sys.platform . . . . . . .
sys.stdin / stdout / stderr
Wrapping Up . . . . . . .

.
.
.
.
.
.
.
.

.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

122
122
123
123
125
125
126
126

Chapter 21 - The threading module


Using Threads . . . . . . . . . .
Writing a Threaded Downloader
Using Queues . . . . . . . . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

127
127
128
131
133

Chapter 22 - Working with Dates and Time


The datetime Module . . . . . . . . . . .
datetime.date . . . . . . . . . . . . . . . .
datetime.datetime . . . . . . . . . . . . .
datetime.timedelta . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

135
135
135
136
137

CONTENTS

The time Module


time.ctime . . .
time.sleep . . . .
time.strftime . .
time.time . . . .
Wrapping Up . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

138
138
139
139
140
140

Chapter 23 - The xml module . . . . . .


Working with minidom . . . . . . . .
Parsing with ElementTree . . . . . . .
How to Create XML with ElementTree
How to Edit XML with ElementTree .
How to Parse XML with ElementTree
Wrapping Up . . . . . . . . . . . . . .

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

.
.
.
.
.
.
.

141
141
146
146
148
150
152

Part III - Intermediate Odds and Ends . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 153


Chapter 24 - The Python Debugger
How to Start the Debugger . . .
Stepping Through the Code . . .
Setting breakpoints . . . . . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

155
155
157
158
158

Chapter 25 - Decorators . . . . . . . . . . . . . . . . .
A Simple Function . . . . . . . . . . . . . . . . . . .
Creating a Logging Decorator . . . . . . . . . . . . .
Built-in Decorators . . . . . . . . . . . . . . . . . .
@classmethod and @staticmethod . . . . . . . . . .
Python Properties . . . . . . . . . . . . . . . . . . .
Replacing Setters and Getters with a Python property
Wrapping Up . . . . . . . . . . . . . . . . . . . . . .

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.

159
159
161
162
163
164
165
169

Chapter 26 - The lambda . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 170


Tkinter + lambda . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 170
Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 171
Chapter 27 - Code Profiling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 173
Profiling Your Code with cProfile . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 173
Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 176
Chapter 28 - An Intro to Testing . . . . .
Testing with doctest . . . . . . . . . .
Running doctest via the Terminal . . .
Running doctest Inside a Module . . .
Running doctest From a Separate File

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

177
177
177
179
180

CONTENTS

Test Driven Development with unittest .


The First Test . . . . . . . . . . . . . . .
The Second Test . . . . . . . . . . . . .
The Third (and Final) Test . . . . . . . .
Other Notes . . . . . . . . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

182
183
185
187
189
189

Part IV - Tips, Tricks and Tutorials . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 190


Chapter 29 - Installing Modules
Installing from Source . . . .
Using easy_install . . . . . .
Using pip . . . . . . . . . . .
A Note on Dependencies . .
Wrapping Up . . . . . . . . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

192
192
193
193
194
194

Chapter 30 - ConfigObj
Getting Started . . .
Using a configspec .
Wrapping Up . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

195
195
196
198

Chapter 31 - Parsing XML with lxml


Parsing XML with lxml . . . . . .
Parsing the Book Example . . . . .
Parsing XML with lxml.objectify .
Creating XML with lxml.objectify
Wrapping Up . . . . . . . . . . . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

199
200
201
202
205
208

Chapter 32 - Python Code Analysis


Getting Started with pylint . . .
Analyzing Your Code . . . . . .
Getting Started with pyflakes . .
Wrapping Up . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

209
209
209
212
213

Chapter 33 - The requests package


Using requests . . . . . . . . . .
How to Submit a Web Form . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

214
214
215
216

Chapter 34 - SQLAlchemy . . . . . . . . . . .
How to Create a Database . . . . . . . . . .
How to Insert / Add Data to Your Tables . .
How to Modify Records with SQLAlchemy
How to Delete Records in SQLAlchemy . .
The Basic SQL Queries of SQLAlchemy . .

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

.
.
.
.
.
.

217
217
220
222
223
223

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

CONTENTS

Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 225
Chapter 35 - virtualenv . . . . . . .
Installation . . . . . . . . . . . .
Creating a Virtual Environment .
Wrapping Up . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

226
226
226
227

Part V - Packaging and Distribution . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 228


Chapter 36 - Creating Modules and Packages
How to Create a Python Module . . . . . .
How to Create a Python Package . . . . . .
Wrapping Up . . . . . . . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

230
230
232
234

Chapter 37 - How to Add Your Code to PyPI


Creating a setup.py File . . . . . . . . . .
Registering Packages . . . . . . . . . . . .
Uploading Packages to PyPI . . . . . . . .
Wrapping Up . . . . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

235
235
236
237
238

.
.
.
.
.

Chapter 38 - The Python egg . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 239


Creating an egg . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 239
Wrapping Up . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 240
Chapter 39 - Python wheels
Getting Started . . . . . .
Creating a wheel . . . . .
Installing a Python wheel
Wrapping Up . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

241
241
241
242
243

Chapter 40 - py2exe . . . . . . . . . . .
Creating a Simple GUI . . . . . . . .
The py2exe setup.py file . . . . . . .
Creating an Advanced setup.py File
Wrapping Up . . . . . . . . . . . . .

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

.
.
.
.
.

244
244
246
248
249

Chapter 41 - bbfreeze . . . . . . . . . . . . .
Getting Started with bbfreeze . . . . . . .
Using bbfreezes Advanced Configuration
Wrapping Up . . . . . . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

250
250
252
255

Chapter 42 - cx_Freeze . . . . . . . . . . . . .
Getting Started with cx_Freeze . . . . . . .
Advanced cx_Freeze - Using a setup.py File
Wrapping Up . . . . . . . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

256
256
258
260

CONTENTS

Chapter 43 - PyInstaller . . . . . . .
Getting Started with PyInstaller
PyInstaller and wxPython . . . .
Wrapping Up . . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

261
261
263
266

Chapter 44 - Creating an Installer


Getting Started with GUI2Exe
Lets Make an Installer! . . . .
Wrapping Up . . . . . . . . . .

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

.
.
.
.

267
268
270
273

Appendix A: Putting It All Together .


The Background . . . . . . . . . .
The Specification . . . . . . . . . .
Breaking the Specification Down .
Turning Chapters Into a Book . . .
Reading the Customer CSV File . .
Emailing the PDF . . . . . . . . .
Putting it all Together . . . . . . .
Wrapping Up . . . . . . . . . . . .

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

.
.
.
.
.
.
.
.
.

275
275
275
276
276
278
279
281
284

.
.
.
.

Introduction
Welcome to Python 101! I wrote this book to help you learn Python 3. It is not meant to be an
exhaustive reference book. Instead, the object is to get you acquainted with the building blocks of
Python so that you can actually write something useful yourself. A lot of programming textbooks
only teach you the language, but do not go much beyond that. I will endeavour to not only get you
up to speed on the basics, but also to show you how to create useful programs. Now you may be
wondering why just learning the basics isnt enough. In my experience, when I get finished reading
an introductory text, I want to then create something, but I dont know how! Ive got the learning,
but not the glue to get from point A to point B. I think its important to not only teach you the basics,
but also cover intermediate material.
Thus, this book will be split into five parts:

Part one will cover Pythons basics


Part two will be on a small subset of Pythons Standard Library
Part three will be intermediate material
Part four will be a series of small tutorials
Part five will cover Python packaging and distribution

|
Let me spend a few moments explaining what each part has to offer. In part one, we will cover the
following:

Python types (strings, lists, dicts, etc)


Conditional statements
Loops
List and dictionary comprehensions
Exception Handling
File I/O
Functions and Classes

|
Part two will talk about some of Pythons standard library. The standard library is what comes prepackaged with Python. It is made up of modules that you can import to get added functionality. For
example, you can import the math module to gain some high level math functions. I will be cherry
picking the modules I use the most as a day-to-day professional and explaining how they work.
The reason I think this is a good idea is that they are common, every day modules that I think you
will benefit knowing about at the beginning of your Python education. This section will also cover

Introduction

various ways to install 3rd party modules. Finally, I will cover how to create your own modules and
packages and why youd want to do that in the first place. Here are some of the modules we will be
covering:

csv
ConfigParser
logging
os
smtplib / email
subprocess
sys
thread / queues
time / datetime

|
Part three will cover intermediate odds and ends. These are topics that are handy to know, but not
necessarily required to be able to program in Python. The topics covered are:

the Python debugger (pdb)


decorators
the lambda function
code profiling
a testing introduction

|
Part four will be made up of small tutorials that will help you to learn how to use Python in a practical
way. In this way, you will learn how to create Python programs that can actually do something
useful! You can take the knowledge in these tutorials to create your own scripts. Ideas for further
enhancements to these mini-applications will be provided at the end of each tutorial so you will
have something that you can try out on your own. Here are a few of the 3rd party packages that
well be covering:

pip and easy_install


configobj
lxml
requests
virtualenv
pylint / pychecker
SQLAlchemy

|
Part five is going to cover how to take your code and give it to your friends, family and the world!
You will learn the following:

Introduction

How to turn your reusable scripts into Python eggs, wheels and more
How to upload your creation to the Python Package Index (PyPI)
How to create binary executables so you can run your application without Python
How to create an installer for your application

|
The chapters and sections may not all be the same length. While every topic will be covered well,
not every topic will require the same page count.

A Brief History of Python


I think it helps to know the background of the Python programming language. Python was created in
the late 1980s. Everyone agrees that its creator is Guido van Rossum when he wrote it as a successor
to the ABC programming language that he was using. Guido named the language after one of his
favorite comedy acts: Monty Python. The language wasnt released until 1991 and it has grown a
lot in terms of the number of included modules and packages included. At the time of this writing,
there are two major versions of Python: the 2.x series and the 3.x (sometimes known as Python 3000)
. The 3.x series is not backwards compatible with 2.x because the idea when creating 3.x was to get
rid of some of the idiosyncrasies in the original. The current versions are 2.7.12 and 3.5.2. Most of the
features in 3.x have been backported to 2.x; however, 3.x is getting the majority of Pythons current
development, so it is the version of the future.
Some people think Python is just for writing little scripts to glue together real code, like C++
or Haskell. However you will find Python to be useful in almost any situation. Python is used
by lots of big name companies such as Google, NASA, LinkedIn, Industrial Light & Magic, and
many others. Python is used not only on the backend, but also on the front. In case youre new to
the computer science field, backend programming is the stuff thats behind the scenes; things like
database processing, document generation, etc. Frontend processing is the pretty stuff most users are
familiar with, such as web pages or desktop user interfaces. For example, there are some really nice
Python GUI toolkits such as wxPython, PySide, and Kivy. There are also several web frameworks like
Django, Pyramid, and Flask. You might find it surprising to know that Django is used for Instagram
and Pinterest. If you have used these or many other websites, then you have used something thats
powered by Python without even realizing it!

About the Author


You may be wondering about who I am and why I might be knowledgeable enough about Python to
write about it, so I thought Id give you a little information about myself. I started programming in
Python in the Spring of 2006 for a job. My first assignment was to port Windows login scripts from
http://www.artima.com/intv/pythonP.html

Introduction

Kixtart to Python. My second project was to port VBA code (basically a GUI on top of Microsoft
Office products) to Python, which is how I first got started in wxPython. Ive been using Python
ever since, doing a variation of backend programming and desktop front end user interfaces.
I realized that one way for me to remember how to do certain things in Python was to write
about them and thats how my Python blog came about: http://www.blog.pythonlibrary.org/. As
I wrote, I would receive feedback from my readers and I ended up expanding the blog to include
tips, tutorials, Python news, and Python book reviews. I work regularly with Packt Publishing as
a technical reviewer, which means that I get to try to check for errors in the books before theyre
published. I also have written for the Developer Zone (DZone) and i-programmer websites as well as
the Python Software Foundation. In November 2013, DZone published The Essential Core Python
Cheat Sheet that I co-authored.

Conventions
As with most technical books, this one includes a few conventions that you need to be aware of. New
topics and terminology will be in bold. You will also see some examples that look like the following:
1

>>> myString = "Welcome to Python!"

The >>> is a Python prompt symbol. You will see this in the Python interpreter and in IDLE. You
will learn more about each of these in the first chapter. Other code examples will be shown in a
similar manner, but without the >>>.

Requirements
You will need a working Python 3 installation. The examples should work in either Python 2.x or
3.x unless specifically marked otherwise. Most Linux and Mac machines come with Python already
installed. However, if you happen to find yourself without Python, you can go download a copy
from http://python.org/download/. There are up-to-date installation instructions on their website,
so I wont include any installation instructions in this book. Any additional requirements will be
explained later on in the book.

Reader Feedback
I welcome feedback about my writings. If youd like to let me know what you thought of the book,
you can send comments to the following address:
comments@pythonlibrary.org
http://python.org/download/

Introduction

Errata
I try my best not to publish errors in my writings, but it happens from time to time. If you happen
to see an error in this book, feel free to let me know by emailing me at the following:
errata@pythonlibrary.org

Part I - Learning the Basics


In Part I, we will learn the basics of the Python programming language. This section of the book
should get you ready to use all the building blocks of Python so that you will be ready to tackle the
following sections confidently.
Lets go over what well be covering:

IDLE
Strings
Lists, Dictionaries and Tuples
Conditional statements
Loops
Comprehensions
Exception Handling
File I/O
Importing modules and packages
Functions
Classes

The first chapter in this section will familiarize you with Pythons built-in development environment
that is known as IDLE. The next couple of chapters will go over some of Pythons types, such as
strings, lists, and dictionaries. After that well look at conditional statements in Python and looping
using Pythons for and while loops.
In the second half of this section, we will go into comprehensions, such as list and dictionary
comprehensions. Then well look at Pythons exception handling capabilities and how Pythons
file operations work. Next up is learning how to import pre-made modules and packages. The last
two chapters cover Python functions and classes.
Lets get started!

Part I - Learning the Basics

image

Chapter 1 - IDLE Programming


Using IDLE
Python comes with its own code editor: IDLE. There is some unconfirmed lore that the name for
IDLE comes from Eric Idle, an actor in Monty Python. I have no idea if thats true or not, but it would
make sense in this context as it appears to be a pun on the acronym IDE or Integrated Development
Environment. An IDE is an editor for programmers that provides color highlighting of key words
in the language, auto-complete, a debugger and lots of other fun things. You can find an IDE for
most popular languages and a number of IDEs will work with multiple languages. IDLE is kind of
a lite IDE, but it does have all those items mentioned. It allows the programmer to write Python
and debug their code quite easily. The reason I call it lite is the debugger is very basic and its
missing other features that programmers who have a background using products like Visual Studio
will miss. You might also like to know that IDLE was created using Tkinter, a Python GUI toolkit
that comes with Python.
To open up IDLE, you will need to find it and youll see something like this:

image

Yes, its a Python shell where you can type short scripts and see their output immediately and
even interact with code in real time. There is no compiling of the code as Python is an interpretive
language and runs in the Python interpreter. Lets write your first program now. Type the following
after the command prompt (>>>) in IDLE:

Chapter 1 - IDLE Programming

print("Hello from Python!")

You have just written your first program! All your program does is write a string to the screen, but
youll find that very helpful later on. Please note that the print statement has changed in Python
3.x. In Python 2.x, you would have written the above like this:
1

print "Hello from Python!"

In Python 3, the print statement was turned into a print function, which is why parentheses are
required. You will learn what functions are in chapter 10.
If you want to save your code into a file, go to the File menu and choose New Window (or press
CTRL+N). Now you can type in your program and save it here. The primary benefit of using the
Python shell is that you can experiment with small snippets to see how your code will behave before
you put the code into a real program. The code editor screen looks a little different than the IDLE
screenshot above:

image

Now well spend a little time looking at IDLEs other useful features.
Python comes with lots of modules and packages that you can import to add new features. For
example, you can import the math module for all kinds of good math functions, like square roots,
cosines, etcetera. In the File menu, youll find a Path Browser which is useful for figuring out
where Python looks for module imports. You see, Python first looks in the same directory as the
script that is running to see if the file it needs to import is there. Then it checks a predefined list of
other locations. You can actually add and remove locations as well. The Path Browser will show you
where these files are located on your hard drive, if you have imported anything. My Path Browser
looks like this:

10

Chapter 1 - IDLE Programming

image

Next theres a Class Browser that will help you navigate your code. This is actually something that
wont be very useful to you right now, but will be in the future. Youll find it helpful when you have
lots of lines of code in a single file as it will give you a tree-like interface for your code. Note that
you wont be able to load the Class Browser unless you have actually saved your program.
The Edit menu has your typical features, such as Copy, Cut, Paste, Undo, Redo and Select All. It also
contains various ways to search your code and do a search and replace. Finally, the Edit menu has
some menu items that will Show you various things, such as highlighting parentheses or displaying
the auto-complete list.
The Format menu has lots of useful functionality. It has some helpful items for indenting and
dedenting your code, as well as commenting out your code. I find that pretty helpful when Im
testing my code. Commenting out your code can be very helpful. One way it can be helpful is when
you have a lot of code and you need to find out why its not working correctly. Commenting out
portions of it and re-running the script can help you figure out where you went wrong. You just
go along slowly uncommenting out stuff until you hit your bug. Which reminds me; you may have
noticed that the main IDLE screen has a Debugger menu. That is nice for debugging, but only in
the Shell window. Sadly you cannot use the debugger in your main editing menu. If you need a
more versatile debugger, you should either find a different IDE or try Pythons debugger found in
the pdb library.

What are Comments?

Chapter 1 - IDLE Programming

11

A comment is a way to leave un-runnable code that documents what you are doing in
your code. Every programming language uses a different symbol to demarcate where a
comment starts and ends. What do comments look like in Python though? A comment
is anything that begins with an octothorpe (i.e. a hash or pound sign). The following is
an example of some comments in action:
1
2
3

# This is a comment before some code


print("Hello from Python!")
print("Winter is coming") # this is an in-line comment

You can write comments on a line all by themselves or following a statement, like the
second print statement above. The Python interpreter ignores comments, so you can
write anything you want in them. Most programmers I have met dont use comments
very much. However, I highly recommend using comments liberally not just for
yourself, but for anyone else who might have to maintain or enhance your code in the
future. I have found my own comments useful when I come back to a script that I wrote
6 months ago and I have found myself working with code that didnt have comments
and wishing that it did so I could figure it out faster.
Examples of good comments would include explanations about complex code statements, or adding an explanation for acronyms in your code. Sometimes youll need to
leave a comment to explain why you did something a certain way because its just not
obvious.

Now we need to get back to going over the menu options of IDLE:
The Run menu has a couple of handy options. You can use it to bring up the Python Shell, check your
code for errors, or run your code. The Options menu doesnt have very many items. It does have a
Configure option that allows you to change the code highlighting colors, fonts and key shortcuts.
Other than that, you get a Code Context option that is helpful in that it puts an overlay in the editing
window which will show you which class or function youre currently in. We will be explaining
functions and classes near the end of Part I. You will find this feature is useful whenever you have
a lot of code in a function and the name has scrolled off the top of the screen. With this option
enabled, that doesnt happen. Of course, if the function is too large to fit on one screen, then it may
be getting too long and it could be time to break that function down into multiple functions.
The Windows menu shows you a list of currently open Windows and allows you to switch between
them.
Last but not least is the Help menu where you can learn about IDLE, get help with IDLE itself or
load up a local copy of the Python documentation. The documentation will explain how each piece
of Python works and is pretty exhaustive in its coverage. The Help menu is probably the most helpful
in that you can get access to the docs even when youre not connected to the internet. You can search

12

Chapter 1 - IDLE Programming

the documentation, find HOWTOs, read about any of the builtin libraries, and learn so much your
head will probably start spinning.

Other Tips
When you see code examples in the following chapters, you can write and run them in IDLE. I wrote
all my programs in IDLE for the first couple of years of my Python programming life and I was pretty
happy with it. There are lots of free Python IDEs out there though and several IDEs that you have
to pay for. If you want to go cheap, you might want to take a look at Eclipse+PyDev, Editra or even
Notepad++. For a paid IDE, I would recommend WingWares IDE or possibly PyCharm. They have
many more features such as integration with code repositories, better debuggers, refactoring help,
etc.
In this book, we will be using IDLE in our examples because it comes with Python and will provide
a common test bed. I still think IDLE has the best, most consistent code highlighting of any IDE I
have used. Code highlighting is important in my mind in that it helps prevent me from using one
of Pythons keywords (or built-ins) for a variable name. In case youre wondering, here is a list of
those key words:
1
2
3
4
5
6
7

and
as
assert
break
class
continue
def

del
elif
else
except
exec
finally
for

from
global
if
import
in
is
lambda

not
or
pass
print
raise
return
try

while
with
yield

Lets see what happens as we type out a few things in Python:

image

Chapter 1 - IDLE Programming

13

As you can see, IDLE color coded everything. A key word is magenta, a string of text is in green, a
comment is in red and the output from the print function is in blue.

Wrapping Up
In this chapter we learned how to use Pythons integrated development environment, IDLE. We also
learned what comments are and how to use them. At this point, you should be familiar enough with
IDLE to use it in the rest of this book. There are many other integrated development environments
(IDEs) for Python. There are free ones like PyDev and Editra, and there are some others that you
have to pay for, such as WingWare and PyCharm. There are also plug-ins for regular text editors
that allow you to code in Python too. I think IDLE is a good place to start, but if you already have
a favorite editor, feel free to continue using that.
At this point, we are ready to move on and start learning about Pythons various data types. We will
start with Strings in the following chapter.

Chapter 2 - All About Strings


There are several data types in Python. The main data types that youll probably see the most are
string, integer, float, list, dict and tuple. In this chapter, well cover the string data type. Youll be
surprised how many things you can do with strings in Python right out of the box. Theres also a
string module that you can import to access even more functionality, but we wont be looking at
that in this chapter. Instead, we will be covering the following topics:

How to create strings


String concatenation
String methods
String slicing
String substitution

How to Create a String


Strings are usually created in one of three ways. You can use single, double or triple quotes. Lets
take a look!
1
2
3
4
5

>>> my_string = "Welcome to Python!"


>>> another_string = 'The bright red fox jumped the fence.'
>>> a_long_string = '''This is a
multi-line string. It covers more than
one line'''

The triple quoted line can be done with three single quotes or three double quotes. Either way, they
allow the programmer to write strings over multiple lines. If you print it out, you will notice that
the output retains the line breaks. If you need to use single quotes in your string, then wrap it in
double quotes. See the following example.
1
2
3

>>> my_string = "I'm a Python programmer!"


>>> otherString = 'The word "python" usually refers to a snake'
>>> tripleString = """Here's another way to embed "quotes" in a string"""

The code above demonstrates how you could put single quotes or double quotes into a string. Theres
actually one other way to create a string and that is by using the str method. Heres how it works:

Chapter 2 - All About Strings

1
2

15

>>> my_number = 123


>>> my_string = str(my_number)

If you type the code above into your interpreter, youll find that you have transformed the integer
value into a string and assigned the string to the variable my_string. This is known as casting. You
can cast some data types into other data types, like numbers into strings. But youll also find that
you cant always do the reverse, such as casting a string like ABC into an integer. If you do that,
youll end up with an error like the one in the following example:
1
2
3
4

>>> int('ABC')
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
ValueError: invalid literal for int() with base 10: 'ABC'

We will look at exception handling in a later chapter, but as you may have guessed from the message,
this means that you cannot convert a literal into an integer. However, if you had done
1

>>> x = int("123")

then that would have worked fine.


It should be noted that a string is one of Python immutable types. What this means is that you
cannot change a strings content after creation. Lets try to change one to see what happens:
1
2
3
4
5

>>> my_string = "abc"


>>> my_string[0] = "d"
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: 'str' object does not support item assignment

Here we try to change the first character from an a to a d; however this raises a TypeError that
stops us from doing so. Now you may think that by assigning a new string to the same variable that
youve changed the string. Lets see if thats true:

Chapter 2 - All About Strings

1
2
3
4
5
6
7
8
9

16

>>> my_string = "abc"


>>> id(my_string)
19397208
>>> my_string = "def"
>>> id(my_string)
25558288
>>> my_string = my_string + "ghi"
>>> id(my_string)
31345312

By checking the id of the object, we can determine that any time we assign a new value to the
variable, its identity changes.
Note that in Python 2.x, strings can only contain ASCII characters. If you require unicode in Python
2.x, then you will need to precede your string with a u. Heres an example:
1

my_unicode_string = u"This is unicode!"

The example above doesnt actually contain any unicode, but it should give you the general idea. In
Python 3.x, all strings are unicode.

String Concatenation
Concatenation is a big word that means to combine or add two things together. In this case, we
want to know how to add two strings together. As you might suspect, this operation is very easy in
Python:
1
2
3

>>> string_one = "My dog ate "


>>> string_two = "my homework!"
>>> string_three = string_one + string_two

The + operator concatenates the two strings into one.

String Methods
A string is an object in Python. In fact, everything in Python is an object. However, youre not really
ready for that. If you want to know more about how Python is an object oriented programming
language, then youll need to skip to that chapter. In the meantime, its enough to know that strings
have their very own methods built into them. For example, lets say you have the following string:

Chapter 2 - All About Strings

17

>>> my_string = "This is a string!"

Now you want to cause this string to be entirely in uppercase. To do that, all you need to do is call
its upper() method, like this:
1

>>> my_string.upper()

If you have your interpreter open, you can also do the same thing like this:
1

>>> "This is a string!".upper()

There are many other string methods. For example, if you wanted everything to be lowercase, you
would use the lower() method. If you wanted to remove all the leading and trailing white space,
you would use strip(). To get a list of all the string methods, type the following command into your
interpreter:
1

>>> dir(my_string)

You should end up seeing something like the following:


[__add__, __class__, __contains__, __delattr__, __doc__, __eq__, __format__,
__ge__, __getattribute__, __getitem__, __getnewargs__, __getslice__, __gt__,
__hash__, __init__, __le__, __len__, __lt__, __mod__, __mul__, __ne__,
__new__, __reduce__, __reduce_ex__, __repr__, __rmod__, __rmul__, __setattr__, __sizeof__, __str__, __subclasshook__, _formatter_field_name_split,
_formatter_parser, capitalize, center, count, decode, encode, endswith, expandtabs, find, format, index, isalnum, isalpha, isdigit, islower, isspace,
istitle, isupper, join, ljust, lower, lstrip, partition, replace, rfind, rindex,
rjust, rpartition, rsplit, rstrip, split, splitlines, startswith, strip, swapcase,
title, translate, upper, zfill]
You can safely ignore the methods that begin and end with double-underscores, such as __add__.
They are not used in every day Python coding. Focus on the other ones instead. If youd like to know
what one of them does, just ask for help. For example, say you want to learn what capitalize is for.
To find out, you would type
1

>>> help(my_string.capitalize)

This would return the following information:

Chapter 2 - All About Strings

18

Help on built-in function capitalize:


capitalize()
S.capitalize() -> string
Return a copy of the string S with only its first character capitalized.
You have just learned a little bit about a topic called introspection. Python allows easy introspection
of all its objects, which makes it very easy to use. Basically, introspection allows you to ask Python
about itself. In an earlier section, you learned about casting. You may have wondered how to tell
what type the variable was (i.e. an int or a string). You can ask Python to tell you that!
1
2

>>> type(my_string)
<type 'str'>

As you can see, the my_string variable is of type str!

String Slicing
One subject that youll find yourself doing a lot of in the real world is string slicing. I have been
surprised how often I have needed to know how to do this in my day-to-day job. Lets take a look
at how slicing works with the following string:
1

>>> my_string = "I like Python!"

Each character in a string can be accessed using slicing. For example, if I want to grab just the first
character, I could do this:
1

>>> my_string[0:1]

This grabs the first character in the string up to, but not including, the 2nd character. Yes, Python is
zero-based. Its a little easier to understand if we map out each characters position in a table:
0 1 2 3 4 5 6 7 8 9 10 11 12 13 - - - - - - - - - - I l i k e P y t h o n !
Thus we have a string that is 14 characters long, starting at zero and going through thirteen. Lets
do a few more examples to get these concepts into our heads better.

Chapter 2 - All About Strings

1
2
3
4
5
6
7
8
9
10
11
12
13
14

19

>>> my_string[:1]
'I'
>>> my_string[0:12]
'I like Pytho'
>>> my_string[0:13]
'I like Python'
>>> my_string[0:14]
'I like Python!'
>>> my_string[0:-5]
'I like Py'
>>> my_string[:]
'I like Python!'
>>> my_string[2:]
'like Python!'

As you can see from these examples, we can do a slice by just specifying the beginning of the slice (i.e.
my_string[2:]), the ending of the slice (i.e. my_string[:1]) or both (i.e. my_string[0:13]). We can even
use negative values that start at the end of the string. So the example where we did my_string[0:-5]
starts at zero, but ends 5 characters before the end of the string.
You may be wondering where you would use this. I find myself using it for parsing fixed width
records in files or occasionally for parsing complicated file names that follow a very specific naming
convention. I have also used it in parsing out values from binary-type files. Any job where you need
to do text file processing will be made easier if you understand slicing and how to use it effectively.
You can also access individual characters in a string via indexing. Here is an example:
1

>>> print(my_string[0])

The code above will print out the first character in the string.

String Formatting
String formatting (AKA substitution) is the topic of substituting values into a base string. Most of
the time, you will be inserting strings within strings; however you will also find yourself inserting
integers and floats into strings quite often as well. There are two different ways to accomplish this
task. Well start with the old way of doing things and then move on to the new.
Ye Olde Way of Substituting Strings
The easiest way to learn how to do this is to see a few examples. So here we go:

Chapter 2 - All About Strings

1
2
3
4
5
6
7
8
9
10

20

>>> my_string = "I like %s" % "Python"


>>> my_string
'I like Python'
>>> var = "cookies"
>>> newString = "I like %s" % var
>>> newString
'I like cookies'
>>> another_string = "I like %s and %s" % ("Python", var)
>>> another_string
'I like Python and cookies'

As youve probably guessed, the %s is the important piece in the code above. It tells Python that
you may be inserting text soon. If you follow the string with a percent sign and another string or
variable, then Python will attempt to insert it into the string. You can insert multiple strings by
putting multiple instances of %s inside your string. Youll see that in the last example. Just note that
when you insert more than one string, you have to enclose the strings that youre going to insert
with parentheses.
Now lets see what happens if we dont insert enough strings:
1
2
3
4

>>> another_string = "I like %s and %s" % "Python"


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: not enough arguments for format string

Oops! We didnt pass enough arguments to format the string! If you look carefully at the example
above, youll notice it has two instances of %s, so to insert strings into it, you have to pass it the
same number of strings! Now were ready to learn about inserting integers and floats. Lets take a
look!
1
2
3
4
5
6
7
8
9
10
11
12

>>> my_string = "%i + %i = %i" % (1,2,3)


>>> my_string
'1 + 2 = 3'
>>> float_string = "%f" % (1.23)
>>> float_string
'1.230000'
>>> float_string2 = "%.2f" % (1.23)
>>> float_string2
'1.23'
>>> float_string3 = "%.2f" % (1.237)
>>> float_string3
'1.24'

Chapter 2 - All About Strings

21

The first example above is pretty obvious. We create a string that accept three arguments and we
pass them in. Just in case you hadnt figured it out yet, no, Python isnt actually doing any addition
in that first example. For the second example, we pass in a float. Note that the output includes a lot
of extra zeroes. We dont want that, so we tell Python to limit it to two decimal places in the 3rd
example (%.2f). The last example shows you that Python will do some rounding for you if you
pass it a float thats greater than two decimal places.
Now lets see what happens if we pass it bad data:
1
2
3
4

>>> int_float_err = "%i + %f" % ("1", "2.00")


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: %d format: a number is required, not str

In this example, we pass it two strings instead of an integer and a float. This raises a TypeError and
tells us that Python was expecting a number. This refers to not passing an integer, so lets fix that
and see if that fixes the issue:
1
2
3
4

>>> int_float_err = "%i + %f" % (1, "2.00")


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: float argument required, not str

Nope. We get the same error, but a different message that tells us we should have passed a float. As
you can see, Python gives us pretty good information about what went wrong and how to fix it. If
you fix the inputs appropriately, then you should be able to get this example to run.
Lets move on to the new method of string formatting!
Templates and the New String Formatting Methodology
This new method was actually added back in Python 2.4 as string templates, but was added as a
regular string method via the format method in Python 2.6. So its not really a new method, just
newer. Anyway, lets start with templates!
1
2

>>> print("%(lang)s is fun!" % {"lang":"Python"})


Python is fun!

This probably looks pretty weird, but basically we just changed our %s into %(lang)s, which is
basically the %s with a variable inside it. The second part is actually called a Python dictionary that
we will be studying in the next section. Basically its a key:value pair, so when Python sees the key
lang in the string AND in the key of the dictionary that is passed in, it replaces that key with its
value. Lets look at some more samples:

Chapter 2 - All About Strings

1
2
3
4
5
6
7
8

22

>>> print("%(value)s %(value)s %(value)s !" % {"value":"SPAM"})


SPAM SPAM SPAM !
>>> print("%(x)i + %(y)i = %(z)i" % {"x":1, "y":2})
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
KeyError: 'z'
>>> print("%(x)i + %(y)i = %(z)i" % {"x":1, "y":2, "z":3})
1 + 2 = 3

In the first example, youll notice that we only passed in one value, but it was inserted 3 times! This
is one of the advantages of using templates. The second example has an issue in that we forgot to
pass in a key, namely the z key. The third example rectifies this issue and shows the result. Now
lets look at how we can do something similar with the strings format method!
1
2
3
4
5
6
7

>>> "Python is as simple as {0}, {1}, {2}".format("a", "b", "c")


'Python is as simple as a, b, c'
>>> "Python is as simple as {1}, {0}, {2}".format("a", "b", "c")
'Python is as simple as b, a, c'
>>> xy = {"x":0, "y":10}
>>> print("Graph a point at where x={x} and y={y}".format(**xy))
Graph a point at where x=0 and y=10

In the first two examples, you can see how we can pass items positionally. If we rearrange the order,
we get a slightly different output. The last example uses a dictionary like we were using in the
templates above. However, we have to extract the dictionary using the double asterisk to get it to
work correctly here.
There are lots of other things you can do with strings, such as specifying a width, aligning the text,
converting to different bases and much more. Be sure to take a look at some of the references below
for more information.

Pythons official documentation on the str type


String Formatting
More on String Formatting
Python 2.x documentation on unicode

https://docs.python.org/3/library/functions.html#func-str
https://docs.python.org/3/library/string.html#string-formatting
https://docs.python.org/3/library/string.html#formatexamples
http://docs.python.org/2/library/functions.html#unicode

Chapter 2 - All About Strings

23

Wrapping Up
We have covered a lot in this chapter. Lets review:
First we learned how to create strings themselves, then we moved on to the topic of string
concatenation. After that we looked at some of the methods that the string object gives us. Next
we looked at string slicing and we finished up by learning about string substitution.
In the next chapter, we will look at three more of Pythons built-in data types: lists, tuples and
dictionaries. Lets get to it!

Chapter 3 - Lists, Tuples and


Dictionaries
Python has several other important data types that youll probably use every day. They are called
lists, tuples and dictionaries. This chapters aim is to get you acquainted with each of these data
types. They are not particularly complicated, so I expect that you will find learning how to use them
very straight forward. Once you have mastered these three data types plus the string data type from
the previous chapter, you will be quite a ways along in your education of Python. Youll be using
these four building blocks in 99% of all the applications you will write.

Lists
A Python list is similar to an array in other languages. In Python, an empty list can be created in
the following ways.
1
2

>>> my_list = []
>>> my_list = list()

As you can see, you can create the list using square brackets or by using the Python built-in, list.
A list contains a list of elements, such as strings, integers, objects or a mixture of types. Lets take a
look at some examples:
1
2
3

>>> my_list = [1, 2, 3]


>>> my_list2 = ["a", "b", "c"]
>>> my_list3 = ["a", 1, "Python", 5]

The first list has 3 integers, the second has 3 strings and the third has a mixture. You can also create
lists of lists like this:
1
2
3

>>> my_nested_list = [my_list, my_list2]


>>> my_nested_list
[[1, 2, 3], ['a', 'b', 'c']]

Occasionally, youll want to combine two lists together. The first way is to use the extend method:

Chapter 3 - Lists, Tuples and Dictionaries

1
2
3
4
5

>>>
>>>
>>>
>>>
[4,

25

combo_list = []
one_list = [4, 5]
combo_list.extend(one_list)
combo_list
5]

A slightly easier way is to just add two lists together.


1
2
3
4
5

>>>
>>>
>>>
>>>
[1,

my_list = [1, 2, 3]
my_list2 = ["a", "b", "c"]
combo_list = my_list + my_list2
combo_list
2, 3, 'a', 'b', 'c']

Yes, it really is that easy. You can also sort a list. Lets spend a moment to see how to do that:
1
2
3
4

>>>
>>>
>>>
[2,

alpha_list = [34, 23, 67, 100, 88, 2]


alpha_list.sort()
alpha_list
23, 34, 67, 88, 100]

Now there is a got-cha above. Can you see it? Lets do one more example to make it obvious:
1
2
3
4
5

>>> alpha_list = [34, 23, 67, 100, 88, 2]


>>> sorted_list = alpha_list.sort()
>>> sorted_list
>>> print(sorted_list)
None

In this example, we try to assign the sorted list to a variable. However, when you call the sort()
method on a list, it sorts the list in-place. So if you try to assign the result to another variable, then
youll find out that youll get a None object, which is like a Null in other languages. Thus when you
want to sort something, just remember that you sort them in-place and you cannot assign it to a
different variable.
You can slice a list just like you do with a string:
1
2

>>> alpha_list[0:3]
[2, 23, 34]

This code returns a list of just the first 3 elements.

Chapter 3 - Lists, Tuples and Dictionaries

26

Tuples
A tuple is similar to a list, but you create them with parentheses instead of square brackets. You can
also use the tuple built-in. The main difference is that a tuple is immutable while the list is mutable.
Lets take a look at a few examples:
1
2
3
4
5

>>>
>>>
(1,
>>>
>>>

my_tuple = (1, 2, 3, 4, 5)
my_tuple[0:3]
2, 3)
another_tuple = tuple()
abc = tuple([1, 2, 3])

The code above demonstrates one way to create a tuple with five elements. It also shows that you can
do tuple slicing. However, you cannot sort a tuple! The last two examples shows how to create tuples
using the tuple keyword. The first one just creates an empty tuple whereas the second example has
three elements inside it. Notice that it has a list inside it. This is an example of casting. We can
change or cast an item from one data type to another. In this case, we cast a list into a tuple. If you
want to turn the abc tuple back into a list, you can do the following:
1

>>> abc_list = list(abc)

To reiterate, the code above casts the tuple (abc) into a list using the list function.

Dictionaries
A Python dictionary is basically a hash table or a hash mapping. In some languages, they might be
referred to as associative memories or associative arrays. They are indexed with keys, which can
be any immutable type. For example, a string or number can be a key. You need to be aware that a
dictionary is an unordered set of key:value pairs and the keys must be unique. You can get a list of
keys by calling a dictionary instances keys method. To check if a dictionary has a key, you can use
Pythons in keyword. In some of the older versions of Python (2.3 and older to be specific), you will
see the has_key keyword used for testing if a key is in a dictionary. This keyword is deprecated in
Python 2.x and removed entirely from Python 3.x.
Lets take a moment to see how we create a dictionary.

27

Chapter 3 - Lists, Tuples and Dictionaries

1
2
3
4
5

>>> my_dict = {}
>>> another_dict = dict()
>>> my_other_dict = {"one":1, "two":2, "three":3}
>>> my_other_dict
{'three': 3, 'two': 2, 'one': 1}

The first two examples show how to create an empty dictionary. All dictionaries are enclosed with
curly braces. The last line is printed out so you can see how unordered a dictionary is. Now its time
to find out how to access a value in a dictionary.
1
2
3
4
5

>>> my_other_dict["one"]
1
>>> my_dict = {"name":"Mike", "address":"123 Happy Way"}
>>> my_dict["name"]
'Mike'

In the first example, we use the dictionary from the previous example and pull out the value
associated with the key called one. The second example shows how to acquire the value for the
name key. Now lets see how to tell if a key is in a dictionary or not:
1
2
3
4

>>> "name" in my_dict


True
>>> "state" in my_dict
False

So, if the key is in the dictionary, Python returns a Boolean True. Otherwise it returns a Boolean
False. If you need to get a listing of all the keys in a dictionary, then you do this:
1
2

>>> my_dict.keys()
dict_keys(['name', 'address'])

In Python 2, the keys method returns a list. But in Python 3, it returns a view object. This gives the
developer the ability to update the dictionary and the view will automatically update too. Also note
that when using the in keyword for dictionary membership testing, it is better to do it against the
dictionary instead of the list returned from the keys method. See below:
1
2

>>> "name" in my_dict


>>> "name" in my_dict.keys()

# this is good
# this works too, but is slower

While this probably wont matter much to you right now, in a real job situation, seconds matter.
When you have thousands of files to process, these little tricks can save you a lot of time in the long
run!

Chapter 3 - Lists, Tuples and Dictionaries

28

Wrapping Up
In this chapter you just learned how to construct a Python list, tuple and dictionary. Make sure you
understand everything in this section before moving on. These concepts will assist you in designing
your programs. You will be building complex data structures using these building blocks every day
if you choose to pursue employment as a Python programmer. Each of these data types can be nested
inside the others. For example, you can have a nested dictionary, a dictionary of tuples, a tuple made
up of several dictionaries, and on and on.
When you are ready to move on, we will learn about Pythons support for conditional statements.

Chapter 4 - Conditional Statements


Every computer language I have ever used has had at least one conditional statement. Most of the
time that statement is the if/elif/else structure. This is what Python has. Other languages also include
the case/switch statement which I personally enjoy, however Python does not include it. You can
make your own if you really want to, but this book is focused on learning Python fundamentals, so
were going to be only focusing on whats included with Python in this chapter.
The conditional statement checks to see if a statement is True or False. Thats really all it does.
However we will also be looking at the following Boolean operations: and, or, and not. These
operations can change the behaviour of the conditional in simple and complex ways, depending
on your project.

The if statement
Pythons if statement is pretty easy to use. Lets spend a few minutes looking at some examples to
better acquaint ourselves with this construct.
1
2
3

>>> if 2 > 1:
print("This is a True statement!")
This is a True Statement!

This conditional tests the truthfulness of the following statement: 2 > 1. Since this statement
evaluates to True, it will cause the last line in the example to print to the screen or standard out
(stdout).

Python Cares About Space


The Python language cares a lot about space. You will notice that in our conditional
statement above, we indented the code inside the if statement four spaces. This is very
important! If you do not indent your blocks of code properly, the code will not execute
properly. It may not even run at all.
Also, do not mix tabs and spaces. IDLE will complain that there is an issue with your
file and you will have trouble figuring out what the issue is. The recommended number
of spaces to indent a block of code is four. You can actually indent your code any
number of spaces as long as you are consistent. However, the 4-space rule is one that is
recommended by the Python Style Guide and is the rule that is followed by the Python
code developers.

Chapter 4 - Conditional Statements

30

Lets look at another example:


1
2
3
4

>>> var1 = 1
>>> var2 = 3
>>> if var1 > var2:
print("This is also True")

In this one, we compare two variables that translate to the question: Is 1 > 3? Obviously one is not
greater than three, so it doesnt print anything. But what is we wanted it to print something? Thats
where the else statement comes in. Lets modify the conditional to add that piece:
1
2
3
4

if var1 > var2:


print("This is also True")
else:
print("That was False!")

If you run this code, it will print the string that follows the else statement. Lets change gears here
and get some information from the user to make this more interesting. In Python 2.x, you can get
information using a built-in called raw_input. If you are using Python 3.x, then youll find that
raw_input no longer exists. It was renamed to just input. They function in the same way though. To
confuse matters, Python 2.x actually has a built-in called input as well; however it tries to execute
what is entered as a Python expression whereas raw_input returns a string. Anyway, well be using
Python 2.xs raw_input for this example to get the users age.
1
2
3
4
5
6
7
8
9
10

# Python 2.x code


value = raw_input("How much is that doggy in the window? ")
value = int(value)
if value < 10:
print("That's a great deal!")
elif 10 <= value <= 20:
print("I'd still pay that...")
else:
print("Wow! That's too much!")

|
Lets break this down a bit. The first line asks the user for an amount. In the next line, it converts
the users input into an integer. So if you happen to type a floating point number like 1.23, it will
get truncated to just 1. If you happen to type something other than a number, then youll receive an

Chapter 4 - Conditional Statements

31

exception. Well be looking at how to handle exceptions in a later chapter, so for now, just enter an
integer.
In the next few lines, you can see how we check for 3 different cases: less than 10, greater than or
equal to 10 but less than or equal to 20 or something else. For each of these cases, a different string
is printed out. Try putting this code into IDLE and save it. Then run it a few times with different
inputs to see how it works.
You can add multiple elif statements to your entire conditional. The else is optional, but makes a
good default.

Boolean Operations
Now were ready to learn about Boolean operations (and, or, not). According to the Python
documentation, their order of priority is first or, then and, then not. Heres how they work:
or means that if any conditional that is ored together is True, then the following
statement runs
and means that all statements must be True for the following statement to run
not means that if the conditional evaluates to False, it is True. This is the most
confusing, in my opinion.
Lets take a look at some examples of each of these. We will start with or.
1
2
3
4
5

x = 10
y = 20
if x < 10 or y > 15:
print("This statement was True!")

Here we create a couple of variables and test if one is less than ten or if the other is greater than 15.
Because the latter is greater than 15, the print statement executes. As you can see, if one or both of
the statements are True, it will execute the statement. Lets take a look at how and works:
1
2
3
4
5
6

x = 10
y = 10
if x == 10 and y == 15:
print("This statement was True")
else:
print("The statement was False!")

Chapter 4 - Conditional Statements

32

If you run the code above, you will see that first statement does not get run. Instead, the statement
under the else is executed. Why is that? Well, it is because what we are testing is both x and y are
10 and 15 respectively. In this case, they are not, so we drop to the else. Thus, when you and two
statements together, both statements have to evaluate to True for it to execute the following code.
Also note that to test equality in Python, you have to use a double equal sign. A single equals sign is
known as the assignment operator and is only for assigning a value to a variable. If you had tried
to run the code above with one of those statement only having one equals sign, you would have
received a message about invalid syntax.
Note that you can also or and and more than two statements together. However, I do not recommend
that as the more statements that you do that too, the harder it can be to understand and debug.
Now were ready to take a look at the not operation.
1
2
3
4

my_list = [1, 2, 3, 4]
x = 10
if x not in my_list:
print("'x' is not in the list, so this is True!")

In this example, we create a list that contains four integers. Then we write a test that asks if x is
not in that list. Because x equals 10, the statement evaluates to True and the message is printed to
the screen. Another way to test for not is by using the exclamation point, like this:
1
2
3

x = 10
if x != 11:
print("x is not equal to 11!")

If you want to, you can combine the not operation with the other two to create more complex
conditional statements. Here is a simple example:
1
2
3
4
5

my_list = [1, 2, 3, 4]
x = 10
z = 11
if x not in my_list and z != 10:
print("This is True!")

Checking for Nothing


Because we are talking about statements that evaluate to True, we probably need to cover what
evaluates to False. Python has the keyword False which Ive mentioned a few times. However
an empty string, tuple or list also evaluates to False. There is also another keyword that basically
evaluates to False which is called None. The None value is used to represent the absence of value.
Its kind of analogous to Null, which you find in databases. Lets take a look at some code to help us
better understand how this all works:

Chapter 4 - Conditional Statements

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

33

empty_list = []
empty_tuple = ()
empty_string = ""
nothing = None
if empty_list == []:
print("It's an empty list!")
if empty_tuple:
print("It's not an empty tuple!")
if not empty_string:
print("This is an empty string!")
if not nothing:
print("Then it's nothing!")

The first four lines set up four variables. Next we create four conditionals to test them. The first
one checks to see if the empty_list is really empty. The second conditional checks to see if the
empty_tuple has something in it. Yes, you read that right, The second conditional only evaluates to
True if the tuple is not empty! The last two conditionals are doing the opposite of the second. The
third is checking if the string is empty and the fourth is checking if the nothing variable is really
None.
The not operator means that we are checking for the opposite meaning. In other words, we are
checking if the value is NOT True. So in the third example, we check if the empty string is REALLY
empty. Heres another way to write the same thing:
1
2

if empty_string == "":
print("This is an empty string!")

To really nail this down, lets set the empty_string variable to actually contain something:
1
2
3

>>> empty_string = "something"


>>> if empty_string == "":
print("This is an empty string!")

If you run this, you will find that nothing is printed as we will only print something if the variable
is an empty string.
Please note that none of these variables equals the other. They just evaluate the same way. To prove
this, well take a look at a couple of quick examples:

Chapter 4 - Conditional Statements

1
2
3
4

34

>>> empty_list == empty_string


False
>>> empty_string == nothing
False

As you can see, they do not equal each other. You will find yourself checking your data structures
for data a lot in the real world. Some programmers actually like to just wrap their structures in an
exception handler and if they happen to be empty, theyll catch the exception. Others like to use the
strategy mentioned above where you actually test the data structure to see if it has something in it.
Both strategies are valid.
Personally, I find the not operator a little confusing and dont use it that much. But you will find it
useful from time to time.

Special Characters
Strings can contain special characters, like tabs or new lines. We need to be aware of those as they
can sometimes crop up and cause problems. For example, the new line character is defined as n,
while the tab character is defined as t. Lets see a couple of examples so you will better understand
what these do:
1
2
3
4
5

>>> print("I have a \n new line in the middle")


I have a
new line in the middle
>>> print("This sentence is \ttabbed!")
This sentence is
tabbed!

Was the output as you expected? In the first example, we have a n in the middle of the sentence,
which forces it to print out a new line. Because we have a space after the new line character, the
second line is indented by a space. The second example shows what happens when we have a tab
character inside of a sentence.
Sometimes you will want to use escape characters in a string, such as a backslash. To use escape
characters, you have to actually use a backslash, so in the case of a backslash, you would actually
type two backslashes. Lets take a look:

Chapter 4 - Conditional Statements

1
2
3
4
5
6

35

>>> print("This is a backslash \")


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
EOL while scanning string literal: <string>, line 1, pos 30
>>> print("This is a backslash \\")
This is a backslash \

You will notice that the first example didnt work so well. Python thought we were escaping the
double-quote, so it couldnt tell where the end of the line (EOL) was and it threw an error. The
second example has the backslash appropriately escaped.

if __name__ == __main__
You will see a very common conditional statement used in many Python examples. This is what it
looks like:
1
2

if __name__ == "__main__":
# do something!

You will see this at the end of a file. This tells Python that you only want to run the following code
if this program is executed as a standalone file. I use this construct a lot to test that my code works
in the way I expect it to. We will be discussing this later in the book, but whenever you create a
Python script, you create a Python module. If you write it well, you might want to import it into
another module. When you do import a module, it will not run the code thats under the conditional
because __name__ will no longer equal __main__. We will look at this again in Chapter 11 when
we talk about classes.

Wrapping Up
Weve covered a fair bit of ground in this chapter. You have learned how to use conditional
statements in several different ways. We have also spent some time getting acquainted with Boolean
operators. As you have probably guessed, each of these chapters will get slightly more complex as
well be using each building block that we learn to build more complicated pieces of code. In the
next chapter, we will continue that tradition by learning about Pythons support of loops!

Chapter 5 - Loops
Every programming language I have tried has some kind of looping construct. Most have more than
one. The Python world has two types of loops:
the for loop and
the while loop
|
You will find that the for loop is by far the most popular of the two. Loops are used when you want
to do something many times. Usually you will find that you need to do some operation or a set of
operations on a piece of data over and over. This is where loops come in. They make it really easy
to apply this sort of logic to your data.
Lets get started learning how these fun structures work!

The for Loop


As mentioned above, you use a loop when you want to iterate over something n number of times.
Its a little easier to understand if we see an example. Lets use Pythons builtin range function. The
range function will create a list that is n in length. In Python 2.x, there is actually another function
called xrange that is a number generator and isnt as resource intensive as range. They basically
changed xrange into range in Python 3. Here is an example:
1
2

>>> range(5)
range(0, 5)

As you can see, the range function above took an integer and returned a range object. The range
function also accepts a beginning value, an end value and a step value. Here are two more examples:
1
2
3
4

>>> range(5,10)
range(5, 10)
>>> list(range(1, 10, 2))
[1, 3, 5, 7, 9]

The first example demonstrates that you can pass a beginning and end value and the range function
will return the numbers from the beginning value up to but not including the end value. So in the
case of 5-10, we get 5-9. The second example shows how to use the list function to cause the range
function to return every second element between 1 and 10. So it starts with one, skips two, etc. Now
youre probably wondering what this has to do with loops. Well one easy way to show how a loop
works is if we use the range function! Take a look:

Chapter 5 - Loops

1
2
3
4
5
6
7
8

37

>>> for number in range(5):


print(number)
0
1
2
3
4

What happened here? Lets read it from left to right to figure it out. For each number in a range of 5,
print the number. We know that if we call range with a value of 5, it will return a list of 5 elements.
So each time through the loop, it prints out each of the elements. The for loop above would be the
equivalent of the following:
1
2

>>> for number in [0, 1, 2, 3, 4]:


print(number)

The range function just makes it a little bit smaller. The for loop can loop over any kind of Python
iterator. Weve already seen how it can iterate over a list. Lets see if it can also iterate over a
dictionary.
1
2
3
4
5
6
7

>>> a_dict = {"one":1, "two":2, "three":3}


>>> for key in a_dict:
print(key)
three
two
one

When you use a for loop with a dictionary, youll see that it automatically loops over the keys. We
didnt have to say for key in a_dict.keys() (although that would have worked too). Python just did
the right thing for us. You may be wondering why the keys printed in a different order than they
were defined in the dictionary. As you may recall from chapter 3, dictionaries are unordered, so
when we iterate over it, the keys could be in any order.
Now if you know that the keys can be sorted, then you can do that before you iterate over them.
Lets change the dictionary slightly to see how that works.

Chapter 5 - Loops

1
2
3
4
5
6
7
8
9

>>>
>>>
>>>
>>>

38

a_dict = {1:"one", 2:"two", 3:"three"}


keys = a_dict.keys()
keys = sorted(keys)
for key in keys:
print(key)

1
2
3

Lets take a moment to figure out what this code does. First off, we create a dictionary that has
integers for keys instead of strings. Then we extract the keys from the dictionary. Whenever you
call the keys() method, it will return an unordered list of the keys. If you print them out and find
them to be in ascending order, then thats just happenstance. Now we have a view of the dictionarys
keys that are stored in a variable called keys. We sort it and then we use the for loop to loop over it.
Now were ready to make things a little bit more interesting. We are going to loop over a range, but
we want to print out only the even numbers. To do this, we want to use a conditional statement
instead of using the ranges step parameter. Heres one way you could do this:
1
2
3
4
5
6
7
8
9

>>> for number in range(10):


if number % 2 == 0:
print(number)
0
2
4
6
8

Youre probably wondering whats going on here. Whats up with the percent sign? In Python, the
% is called a modulus operator. When you use the modulus operator, it will return the remainder.
There is no remainder when you divide an even number by two, so we print those numbers out. You
probably wont use the modulus operator a lot in the wild, but I have found it useful from time to
time.
Now were ready to learn about the while loop.

The while Loop


The while loop is also used to repeat sections of code, but instead of looping n number of times, it
will only loop until a specific condition is met. Lets look at a very simple example:

Chapter 5 - Loops

1
2
3
4

39

>>> i = 0
>>> while i < 10:
print(i)
i = i + 1

The while loop is kind of like a conditional statement. Heres what this code means: while the variable
i is less than ten, print it out. Then at the end, we increase is value by one. If you run this code, it
should print out 0-9, each on its own line and then stop. If you remove the piece where we increment
is value, then youll end up with an infinite loop. This is usually a bad thing. Infinite loops are to
be avoided and are known as logic errors.
There is another way to break out of a loop. It is by using the break builtin. Lets see how that works:
1
2
3
4
5
6
7
8
9
10
11
12

>>> while i < 10:


print(i)
if i == 5:
break
i += 1
0
1
2
3
4
5

In this piece of code, we add a conditional to check if the variable i ever equals 5. If it does, then we
break out of the loop. As you can see from the sample output, as soon as it reaches 5, the code stops
even though we told the while loop to keep looping until it reached 10. You will also note that we
changed how we increment the value by using +=. This is a handy shortcut that you can also use
with other math operations, like subtraction (-=) and multiplication (*=).
The break builtin is known as a flow control tool. There is another one called continue that is used
to basically skip an iteration or continue with the next iteration. Heres one way to use it:

Chapter 5 - Loops

1
2
3
4
5
6
7
8
9
10
11
12

40

i = 0
while i < 10:
if i == 3:
i += 1
continue
print(i)
if i == 5:
break
i += 1

This is a little confusing, no? Basically we added a second conditional that checks if i equals 3. If it
does, we increment the variable and continue with the next loop, which effectively skips printing
the value 3 to the screen. As before, when we reach a value of 5, we break out of the loop.
Theres one more topic we need to cover regarding loops and thats the else statement.

What else is for in loops


The else statement in loops only executes if the loop completes successfully. The primary use of the
else statement is for searching for items:
1
2
3
4
5
6
7
8
9

my_list = [1, 2, 3, 4, 5]
for i in my_list:
if i == 3:
print("Item found!")
break
print(i)
else:
print("Item not found!")

In this code, we break out of the loop when i equals 3. This causes the else statement to be skipped.
If you want to experiment, you can change the conditional to look for a value thats not in the list,
which will cause the else statement to execute. To be honest, I have never seen anyone use this
structure in all my years as a programmer. Most of the examples I have seen are bloggers trying to
explain what it is used for. I have seen several who use it to raise an error if an item is not found
in the iterable that you were searching. You can read a fairly in depth article by one of the Python
core developers here.
https://ncoghlan_devs-python-notes.readthedocs.org/en/latest/python_concepts/break_else.html

Chapter 5 - Loops

41

Wrapping Up
Hopefully at this point you can see the value in Python loops. They make repetition easier and
pretty easy to understand. You will likely see the for loop much more often than the while loop.
In fact, we are going to look at another way for loops are used in the next chapter when we learn
about comprehensions! If youre still not quite sure how all this works, you may want to re-read this
chapter before continuing.

Chapter 6 - Python Comprehensions


The Python language has a couple of methods for creating lists and dictionaries that are known
as comprehensions. There is also a third type of comprehension for creating a Python set. In this
chapter we will learn how to use each type of comprehension. You will find that the comprehension
constructs build on the knowledge you have acquired from the previous chapters as they contain
loops and conditionals themselves.

List Comprehensions
List comprehensions in Python are very handy. They can also be a little hard to understand when
and why you would use them. List comprehensions tend to be harder to read than just using a simple
for loop as well. You may want to review the looping chapter before you continue.
If you are ready, then well spend some time looking at how to construct list comprehensions and
learn how they can be used. A list comprehension is basically a one line for loop that produces a
Python list data structure. Heres a simple example:
1

>>> x = [i for i in range(5)]

Lets break this down a bit. Python comes with a range function that can return a list of numbers.
By default, it returns integers starting at 0 and going up to but not including the number you pass it.
So in this case, it returns a list containing the integers 0-4. This can be useful if you need to create a
list very quickly. For example, say youre parsing a file and looking for something in particular. You
could use a list comprehension as a kind of filter:
1
2

if [i for i in line if "SOME TERM" in i]:


# do something

I have used code similar to this to look through a file quickly to parse out specific lines or sections
of the file. When you throw functions into the mix, you can start doing some really cool stuff. Say
you want to apply a function to every element in a list, such as when you need to cast a bunch of
strings into integers:

Chapter 6 - Python Comprehensions

1
2
3
4

>>>
>>>
>>>
[1,

43

x = ['1', '2', '3', '4', '5']


y = [int(i) for i in x]
y
2, 3, 4, 5]

This sort of thing comes up more often than youd think. I have also had to loop over a list of strings
and call a string method, such as strip on them because they had all kinds of leading or ending white
space:
1

>>> myStrings = [s.strip() for s in myStringList]

There are also occasions where one needs to create a nested list comprehension. One reason to do
that is to flatten multiple lists into one. This example comes from the Python documentation:
1
2
3

>>> vec = [[1,2,3], [4,5,6], [7,8,9]]


>>> [num for elem in vec for num in elem]
[1, 2, 3, 4, 5, 6, 7, 8, 9]

The documentation shows several other interesting examples for nested list comprehensions as well.
I highly recommend taking a look at it! At this point, you should now be capable of using list
comprehensions in your own code and use them well. Just use your imagination and youll start
seeing lots of good places where you too can use them.
Now were ready to move on to Pythons dictionary comprehensions!

Dictionary Comprehensions
Dictionary comprehensions started life in Python 3.0, but were backported to Python 2.7. They were
originally proposed in the Python Enhancement Proposal 274 (PEP 274) back in 2001. They are
pretty similar to a list comprehension in the way that they are organized.
The best way to understand is to just do one!
1
2

>>> print( {i: str(i) for i in range(5)} )


{0: '0', 1: '1', 2: '2', 3: '3', 4: '4'}

This is a pretty straightforward comprehension. Basically it is creating an integer key and string
value for each item in the range. Now you may be wondering how you could use a dictionary
comprehension in real life. Mark Pilgrim mentioned that you could use a dictionary comprehension
for swapping the dictionarys keys and values. Heres how you would do that:
http://www.python.org/dev/peps/pep-0274/
http://www.diveintopython3.net/comprehensions.html

Chapter 6 - Python Comprehensions

1
2
3

44

>>> my_dict = {1:"dog", 2:"cat", 3:"hamster"}


>>> print( {value:key for key, value in my_dict.items()} )
{'hamster': 3, 'dog': 1, 'cat': 2}

This will only work if the dictionary values are of a non-mutable type, such as a string. Otherwise
you will end up causing an exception to be raised.
I could also see a dictionary comprehension being useful for creating a table out of class variables
and their values. However, we havent covered classes at this point, so I wont confuse you with that
here.

Set Comprehensions
Set comprehensions are created in much the same way as dictionary comprehensions. Now a Python
set is much like a mathematical set in that it doesnt have any repeated elements. You can create a
normal set like this:
1
2
3
4

>>> my_list = [1, 2, 2, 3, 4, 5, 5, 7, 8]


>>> my_set = set(my_list)
>>> my_set
set([1, 2, 3, 4, 5, 7, 8])

As you can see from the example above, the call to set has removed the duplicates from the list. Now
lets rewrite this code to use a set comprehension:
1
2
3
4

>>> my_list = [1, 2, 2, 3, 4, 5, 5, 7, 8]


>>> my_set = {x for x in my_list}
>>> my_set
set([1, 2, 3, 4, 5, 7, 8])

You will notice that to create a set comprehension, we basically changed the square brackets that a
list comprehension uses to the curly braces that the dictionary comprehension has.

Wrapping Up
Now you know how to use the various Python comprehensions. You will probably find the list
comprehension the most useful at first and also the most popular. If you start using your imagination,
I am sure you will be able to find uses for all three types of comprehensions. Now were ready to
move on and learn about exception handling!

Chapter 7 - Exception Handling


What do you do when something bad happens in your program? Lets say you try to open a file, but
you typed in the wrong path or you ask the user for information and they type in some garbage. You
dont want your program to crash, so you implement exception handling. In Python, the construct
is usually wrapped in what is know as a try/except. We will be looking at the following topics in
this chapter:

Common exception types


Handling exceptions with try/except
Learn how try/except/finally works
Discover how the else statement works in conjunction with the try/except

Lets start out by learning about some of the most common exceptions that youll see in Python.
Note: an error and an exception are just different words that describe the same thing when we are
talking about exception handling.

Common Exceptions
You have seen a few exceptions already. Here is a list of the most common built-in exceptions
(definitions from the Python documentation):
Exception (this is what almost all the others are built off of)
AttributeError - Raised when an attribute reference or assignment fails.
IOError - Raised when an I/O operation (such as a print statement, the built-in
open() function or a method of a file object) fails for an I/O-related reason, e.g.,
file not found or disk full.
ImportError - Raised when an import statement fails to find the module definition
or when a from import fails to find a name that is to be imported.
IndexError - Raised when a sequence subscript is out of range.
KeyError - Raised when a mapping (dictionary) key is not found in the set of
existing keys.
KeyboardInterrupt - Raised when the user hits the interrupt key (normally
Control-C or Delete).
NameError - Raised when a local or global name is not found.
OSError - Raised when a function returns a system-related error.
SyntaxError - Raised when the parser encounters a syntax error.
http://docs.python.org/2/library/exceptions.html

Chapter 7 - Exception Handling

46

TypeError - Raised when an operation or function is applied to an object of


inappropriate type. The associated value is a string giving details about the type
mismatch.
ValueError - Raised when a built-in operation or function receives an argument
that has the right type but an inappropriate value, and the situation is not described
by a more precise exception such as IndexError.
ZeroDivisionError - Raised when the second argument of a division or modulo
operation is zero.
There are a lot of other exceptions as well, but you probably wont see them all that often. However,
if you are interested, you can go and read all about them in the Python documentation.

How to Handle Exceptions


Handling exceptions in Python is really easy. Lets spend some time writing some examples that will
cause exceptions. We will start with one of the most common computer science problems: division
by zero.
1
2
3
4
5
6
7
8
9
10
11

>>> 1 / 0
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
ZeroDivisionError: integer division or modulo by zero
>>> try:
1 / 0
except ZeroDivisionError:
print("You cannot divide by zero!")
You cannot divide by zero!

If you think back to elementary math class, you will recall that you cannot divide by zero. In Python,
this operation will cause an error, as you can see in the first half of the example. To catch the error,
we wrap the operation with a try/except statement.

Bare Excepts
Heres another way to catch the error:
http://docs.python.org/2/library/exceptions.html

Chapter 7 - Exception Handling

1
2
3
4

47

>>> try:
1 / 0
except:
print("You cannot divide by zero!")

This is not recommended! In Python, this is known as a bare except, which means it will catch any
and all exceptions. The reason this is not recommended is that you dont know which exception you
are catching. When you have something like except ZeroDivisionError, you are obviously trying
to catch a division by zero error. In the code above, you cannot tell what you are trying to catch.

Lets take a look at a couple of other examples.


1
2
3
4
5
6
7
8
9
10
11
12
13
14

>>> my_dict = {"a":1, "b":2, "c":3}


>>> try:
value = my_dict["d"]
except KeyError:
print("That key does not exist!")
That key does not exist!
>>> my_list = [1, 2, 3, 4, 5]
>>> try:
my_list[6]
except IndexError:
print("That index is not in the list!")
That index is not in the list!

In the first example, we create a 3-element dictionary. Then we try to access a key that is not in the
dictionary. Because the key is not in the dictionary, it raises a KeyError, which we catch. The second
example shows a list that is 5 items in length. We try to grab the 7th item from the index. Remember,
Python lists are zero-based, so when you say [6], youre asking for the 7th item. Anyway, because
there are only 5 items, it raises an IndexError, which we also catch.
You can also catch multiple exceptions with a single statement. There are a couple of different ways
to do this. Lets take a look:

Chapter 7 - Exception Handling

1
2
3
4
5
6
7
8
9

48

my_dict = {"a":1, "b":2, "c":3}


try:
value = my_dict["d"]
except IndexError:
print("This index does not exist!")
except KeyError:
print("This key is not in the dictionary!")
except:
print("Some other error occurred!")

This is a fairly standard way to catch multiple exceptions. First we try to access a key that doesnt
exist in the dictionary. The try/except checks to see if you are catching a KeyError, which you are in
the second except statement. You will also note that we have a bare except at the end. This is usually
not recommended, but youll probably see it from time to time, so its good to know about it. Also
note that most of the time, you wont need to wrap a block of code in multiple except handlers. You
normally just need to wrap it in one.
Heres another way to catch multiple exceptions:
1
2
3
4

try:
value = my_dict["d"]
except (IndexError, KeyError):
print("An IndexError or KeyError occurred!")

Notice that in this example, we are putting the errors that we want to catch inside of parentheses.
The problem with this method is that its hard to tell which error has actually occurred, so the
previous example is recommended.
Most of the time when an exception occurs, you will need to alert the user by printing to the screen
or logging the message. Depending on the severity of the error, you may need to exit your program.
Sometimes you will need to clean up before you exit the program. For example, if you have opened
a database connection, you will want to close it before you exit your program or you may end up
leaving connections open. Another example is closing a file handle that you have been writing to.
You will learn more about file handling in the next chapter. But first, we need to learn how to clean
up after ourselves. This is facilitated with the finally statement.

The finally Statement


The finally statement is really easy to use. Lets take a look at a silly example:

Chapter 7 - Exception Handling

1
2
3
4
5
6
7
8

49

my_dict = {"a":1, "b":2, "c":3}


try:
value = my_dict["d"]
except KeyError:
print("A KeyError occurred!")
finally:
print("The finally statement has executed!")

If you run the code above, it will print the statement in the except and the finally. This is pretty
simple, right? Now you can use the finally statement to clean up after yourself. You would also put
the exit code at the end of the finally statement.

try, except, or else!


The try/except statement also has an else clause. The else will only run if there are no errors raised.
We will spend a few moments looking at a couple examples:
1
2
3
4
5
6
7
8

my_dict = {"a":1, "b":2, "c":3}


try:
value = my_dict["a"]
except KeyError:
print("A KeyError occurred!")
else:
print("No error occurred!")

Here we have a dictionary with 3 elements and in the try/except we access a key that exists. This
works, so the KeyError is not raised. Because there is no error, the else executes and No error
occurred! is printed to the screen. Now lets add in the finally statement:
1
2
3
4
5
6
7
8
9
10

my_dict = {"a":1, "b":2, "c":3}


try:
value = my_dict["a"]
except KeyError:
print("A KeyError occurred!")
else:
print("No error occurred!")
finally:
print("The finally statement ran!")

Chapter 7 - Exception Handling

50

If you run this example, it will execute the else and finally statements. Most of the time, you wont
see the else statement used as any code that follows a try/except will be executed if no errors were
raised. The only good usage of the else statement that Ive seen mentioned is where you want to
execute a second piece of code that can also raise an error. Of course, if an error is raised in the
else, then it wont get caught.

Wrapping Up
Now you should be able to handle exceptions in your code. If you find your code raising an exception,
you will know how to wrap it in such a way that you can catch the error and exit gracefully or
continue without interruption.
Now were ready to move on and learn about how to work with files in Python.

Chapter 8 - Working with Files


This chapter introduces the topic of reading and writing data to files on your hard drive. You will
find that reading and writing files in Python is very easy to do. Lets get started!

How to Read a File


Python has a builtin function called open that we can use to open a file for reading. Create a text
file name test.txt with the following contents:
1
2
3
4

This
line
line
this

is a test file
2
3
line intentionally left blank

Here are a couple of examples that show how to use open for reading:
1
2

handle = open("test.txt")
handle = open(r"C:\Users\mike\py101book\data\test.txt", "r")

The first example will open a file named test.txt in read-only mode. This is the default mode of the
open function. Note that we didnt pass a fully qualified path to the file that we wanted to open
in the first example. Python will automatically look in the folder that the script is running in for
test.txt. If it doesnt find it, then you will receive an IOError.
The second example does show a fully qualified path to the file, but youll notice that it begins with
an r. This means that we want Python to treat the string as a raw string. Lets take a moment to
see the difference between specifying a raw string versus a regular string:
1
2
3
4

>>> print("C:\Users\mike\py101book\data\test.txt")
C:\Users\mike\py101book\data
est.txt
>>> print(r"C:\Users\mike\py101book\data\test.txt")
C:\Users\mike\py101book\data\test.txt

As you can see, when we dont specify it as a raw string, we get an invalid path. Why does this
happen? Well, as you might recall from the strings chapter, there are certain special characters that

Chapter 8 - Working with Files

52

need to be escaped, such as n or t. In this case, we see theres a t (i.e. a tab), so the string
obediently adds a tab to our path and screws it up for us.
The second argument in the second example is also an r. This tells open that we want to open
the file in read-only mode. In other words, it does the same thing as the first example, but its more
explicit. Now lets actually read the file!
Put the following lines into a Python script and save it in the same location as your test.txt file:
1
2
3
4

handle = open("test.txt", "r")


data = handle.read()
print(data)
handle.close()

If you run this, it will open the file and read the entire file as a string into the data variable. Then
we print that data and close the file handle. You should always close a file handle as you never
know when another program will want to access it. Closing the file will also help save memory and
prevent strange bugs in your programs. You can tell Python to just read a line at a time, to read all
the lines into a Python list or to read the file in chunks. The last option is very handy when you are
dealing with really large files and you dont want to read the whole thing in, which might fill up
the PCs memory.
Lets spend some time looking at different ways to read files.
1
2
3
4

handle = open("test.txt", "r")


data = handle.readline() # read just one line
print(data)
handle.close()

If you run this example, it will only read the first line of your text file and print it out. Thats not
too useful, so lets try the file handles readlines() method:
1
2
3
4

handle = open("test.txt", "r")


data = handle.readlines() # read ALL the lines!
print(data)
handle.close()

After running this code, you will see a Python list printed to the screen because thats what the
readlines method returns: a list! Lets take a moment to learn how to read a file in smaller chunks.

How To Read Files Piece by Piece


The easiest way to read a file in chunks is to use a loop. First we will learn how to read a file line by
line and then we will learn how to read it a kilobyte at a time. We will use a for loop for our first
example:

Chapter 8 - Working with Files

1
2
3
4
5
6

53

handle = open("test.txt", "r")


for line in handle:
print(line)
handle.close()

Here we open up a read-only file handle and then we use a for loop to iterate over it. You will find
that you can iterate over all kinds of objects in Python (strings, lists, tuples, keys in a dictionary,
etc). That was pretty simple, right? Now lets do it in chunks!
1
2
3
4
5
6
7

handle = open("test.txt", "r")


while True:
data = handle.read(1024)
print(data)
if not data:
break

In this example, we use Pythons while loop to read a kilobyte of the file at a time. As you probably
know, a kilobyte is 1024 bytes or characters. Now lets pretend that we want to read a binary file,
like a PDF.

How to Read a Binary File


Reading a binary file is very easy. All you need to do is change the file mode:
1

handle = open("test.pdf", "rb")

So this time we changed the file mode to rb, which means read-binary. You will find that you may
need to read binary files when you download PDFs from the internet or transfer files from PC to
PC.

Writing Files in Python


If you have been following along, you can probably guess what the file-mode flag is for writing files:
w and wb for write-mode and write-binary-mode. Lets take a look at a simple example, shall
we?
CAUTION: When using w or wb modes, if the file already exists, it will be overwritten with
no warning! You can check if a file exists before you open it by using Pythons os module. See the
os.path.exists section in Chapter 16.

Chapter 8 - Working with Files

1
2
3

54

handle = open("output.txt", "w")


handle.write("This is a test!")
handle.close()

That was easy! All we did here was change the file mode to w and we called the file handles write
method to write some text to the file. The file handle also has a writelines method that will accept
a list of strings that the handle will then write to disk in order.

Using the with Operator


Python has a neat little builtin called with which you can use to simplify reading and writing files.
The with operator creates what is known as a context manager in Python that will automatically
close the file for you when you are done processing it. Lets see how this works:
1
2
3

with open("test.txt") as file_handler:


for line in file_handler:
print(line)

The syntax for the with operator is a little strange, but youll pick it up pretty quickly. Basically
what were doing is replacing:
1

handle = open("test.txt")

with this:
1

with open("test.txt") as file_handler:

You can do all the usual file I/O operations that you would normally do as long as you are within
the with code block. Once you leave that code block, the file handle will close and you wont be able
to use it any more. Yes, you read that correctly. You no longer have to close the file handle explicitly
as the with operator does it automatically! See if you can change some of the earlier examples from
this chapter so that they use the with method too.

Catching Errors
Sometimes when you are working with files, bad things happen. The file is locked because some
other process is using it or you have some kind of permission error. When this happens, an IOError
will probably occur. In this section, we will look at how to catch errors the normal way and how to
catch them using the with operator. Hint: the idea is basically the same in both!

Chapter 8 - Working with Files

1
2
3
4
5
6
7
8

55

try:
file_handler = open("test.txt")
for line in file_handler:
print(line)
except IOError:
print("An IOError has occurred!")
finally:
file_handler.close()

In the example above, we wrap the usual code inside of a try/except construct. If an error occurs,
we print out a message to the screen. Note that we also make sure we close the file using the finally
statement. Now were ready to look at how we would do this same thing using with:
1
2
3
4
5
6

try:
with open("test.txt") as file_handler:
for line in file_handler:
print(line)
except IOError:
print("An IOError has occurred!")

As you might have guessed, we just wrapped the with block in the same way as we did in the
previous example. The difference here is that we do not need the finally statement as the context
manager handles that for us.

Wrapping Up
At this point you should be pretty well versed in dealing with files in Python. Now you know how
to read and write files using the older style and the newer with style. You will most likely see both
styles in the wild. In the next chapter, we will learn how to import other modules that come with
Python. This will allow us to create programs using pre-built modules. Lets get started!

Chapter 9 - Importing
Python comes with lots of pre-made code baked in. These pieces of code are known as modules and
packages. A module is a single importable Python file whereas a package is made up of two or more
modules. A package can be imported the same way a module is. Whenever you save a Python script
of your own, you have created a module. It may not be a very useful module, but thats what it is. In
this chapter, we will learn how to import modules using several different methods. Lets get started!

import this
Python provides the import keyword for importing modules. Lets give it a try:
1

import this

If you run this code in your interpreter, you should see something like the following as your output:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

The Zen of Python, by Tim Peters


Beautiful is better than ugly.
Explicit is better than implicit.
Simple is better than complex.
Complex is better than complicated.
Flat is better than nested.
Sparse is better than dense.
Readability counts.
Special cases aren't special enough to break the rules.
Although practicality beats purity.
Errors should never pass silently.
Unless explicitly silenced.
In the face of ambiguity, refuse the temptation to guess.
There should be one-- and preferably only one --obvious way to do it.
Although that way may not be obvious at first unless you're Dutch.
Now is better than never.
Although never is often better than *right* now.
If the implementation is hard to explain, it's a bad idea.
If the implementation is easy to explain, it may be a good idea.
Namespaces are one honking great idea -- let's do more of those!

Chapter 9 - Importing

57

You have found an Easter egg in Python known as the Zen of Python. Its actually a sort of an
unofficial best practices for Python. The this module doesnt actually do anything, but it provided
a fun little way to show how to import something. Lets actually import something we can use, like
the math module:
1
2
3

>>> import math


>>> math.sqrt(4)
2.0

Here we imported the math module and then we did something kind of new. We called one of
its functions, sqrt (i.e. square root). To call a method of an imported module, we have to use the
following syntax: module_name.method_name(argument). In this example, we found the square
root of 4. The math module has many other functions that we can use, such as cos (cosine), factorial,
log (logarithm), etc. You can call these functions in much the same way you did sqrt. The only thing
youll need to check is if they accept more arguments or not. Now lets look at another way to import.

Using from to import


Some people dont like having to preface everything they type with the module name. Python has a
solution for that! You can actually import just the functions you want from a module. Lets pretend
that we want to just import the sqrt function:
1
2
3

>>> from math import sqrt


>>> sqrt(16)
4.0

This works pretty much exactly how it is read: from the math module, import the sqrt function.
Let me explain it another way. We use Pythons from keyword to import the sqrt function from the
math module. You can also use this method to import multiple functions from the math function:
1

>>> from math import pi, sqrt

In this example, we import both pi and sqrt. If you tried to access pi you may have noticed that its
actually a value and not a function that you can call. It just returns the value of pi. When you do an
import, you may end up importing a value, a function or even another module! Theres one more
way to import stuff that we need to cover. Lets find out how to import everything!

Importing Everything!
Python provides a way to import all the functions and values from a module as well. This is actually
a bad idea as it can contaminate your namespace. A namespace is where all your variables live
during the life of the program. So lets say you have your own variable named sqrt, like this:

Chapter 9 - Importing

1
2

58

>>> from math import sqrt


>>> sqrt = 5

Now you have just changed the sqrt function into a variable that holds the value of 5. This is known
as shadowing. This becomes especially tricky when you import everything from a module. Lets
take a look:
>>> from math import * >>> sqrt = 5 >>> sqrt(16) Traceback (most recent call last): File
<string>, line 1, in <fragment> TypeError: int object is not callable
To import everything, instead of specifying a list of items, we just use the wildcard which means
we want to import everything. If we dont know whats in themath* module, we wont realize that
weve just clobbered one of the functions we imported. When we try to call the sqrt function after
reassigning it to an integer, we find out that it no longer works.
Thus it is recommended that in most cases, you should import items from modules using one of the
previous methods mentioned in this chapter. There are a few exceptions to this rule. Some modules
are made to be imported using the * method. One prominent example is Tkinter, a toolkit included
with Python that allows you to create desktop user interfaces. The reason that it is supposedly okay
to import Tkinter in this way is that the modules are named so that it is unlikely you would reuse
one yourself.

Wrapping Up
Now you know all about Python imports. There are dozens of modules included with Python that
you can use to give extra functionality to your programs. You can use the builtin modules to query
your OS, get information from the Windows Registry, set up logging utilities, parse XML, and much,
much more. We will be covering a few of these modules in Part II of this book.
In the next chapter, we will be looking at building our own functions. I think youll find this next
topic to be very helpful.

Chapter 10 - Functions
A function is a structure that you define. You get to decide if they have arguments or not. You can
add keyword arguments and default arguments too. A function is a block of code that starts with
the def keyword, a name for the function and a colon. Heres a simple example:
1
2

>>> def a_function():


print("You just created a function!")

This function doesnt do anything except print out some text. To call a function, you need to type
out the name of the function followed by an open and close parentheses:
1
2

>>> a_function()
You just created a function!

Simple, eh?

An Empty Function (the stub)


Sometimes when you are writing out some code, you just want to write the function definitions
without putting any code in them. Ive done this as kind of an outline. It helps you to see how your
application is going to be laid out. Heres an example:
1
2

>>> def empty_function():


pass

Heres something new: the pass statement. It is basically a null operation, which means that when
pass is executed, nothing happens.

Passing Arguments to a Function


Now were ready to learn about how to create a function that can accept arguments and also learn
how to pass said arguments to the function. Lets create a simple function that can add two numbers
together:

Chapter 10 - Functions

1
2
3
4
5

60

>>> def add(a, b):


return a + b
>>> add(1, 2)
3

All functions return something. If you dont tell it to return something, then it will return None. In
this case, we tell it to return a + b. As you can see, we can call the function by passing in two values.
If you dont pass enough or you pass too many arguments, then youll get an error:
1
2
3
4

>>> add(1)
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: add() takes exactly 2 arguments (1 given)

You can also call the function by passing the name of the arguments:
1
2
3
4
5

>>> add(a=2, b=3)


5
>>> total = add(b=4, a=5)
>>> print(total)
9

Youll notice that it doesnt matter what order you pass them to the function as long as they are
named correctly. In the second example, you can see that we assign the result of the function to a
variable named total. This is the usual way of calling a function as youll want to do something
with the result. You are probably wondering what would happen if we passed in arguments with
the wrong names attached. Would it work? Lets find out:
1
2
3
4

>>> add(c=5, d=2)


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: add() got an unexpected keyword argument 'c'

Whoops! We received an error. This means that we passed in a keyword argument that the function
didnt recognize. Coincidentally, keyword arguments are our next topic!

Keyword Arguments
Functions can also accept keyword arguments! They can actually accept both regular arguments and
keyword arguments. What this means is that you can specify which keywords are which and pass
them in. You saw this behavior in a previous example.

Chapter 10 - Functions

1
2
3
4
5

61

>>> def keyword_function(a=1, b=2):


return a+b
>>> keyword_function(b=4, a=5)
9

You could have also called this function without specifying the keywords. This function also
demonstrates the concept of default arguments. How? Well, try calling the function without any
arguments at all!
1
2

>>> keyword_function()
3

The function returned the number 3! Why? The reason is that a and b have default values of 1 and
2 respectively. Now lets create a function that has both a regular argument and a couple keyword
arguments:
1
2
3
4
5
6
7
8
9
10
11

>>> def mixed_function(a, b=2, c=3):


return a+b+c
>>> mixed_function(b=4, c=5)
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
TypeError: mixed_function() takes at least 1 argument (2 given)
>>> mixed_function(1, b=4, c=5)
10
>>> mixed_function(1)
6

There are 3 example cases in the above code. Lets go over each of them. In the first example, we
try calling our function using just the keyword arguments. This will give us a confusing error. The
Traceback says that our function accepts at least one argument, but that two were given. Whats
going on here? The fact is that the first argument is required because its not set to anything, so if
you only call the function with the keyword arguments, that causes an error.
For the second example, we call the mixed function with 3 values, naming two of them. This works
and gives us the expected result, which was 1+4+5=10. The third example shows what happens if
we only call the function by passing in just one valuethe one that didnt have a default. This also
works by taking the 1 and adding it to the two default values of 2 and 3 to get a result of 6!
Isnt that cool?

Chapter 10 - Functions

62

*args and **kwargs


You can also set up functions to accept any number of arguments or keyword arguments by using a
special syntax. To get infinite arguments, use *args and for infinite keyword arguments, use *kwargs.
The args and kwargs words are not important. Thats just convention. You could have called them
*bill and *ted and it would work the same way. The key here is in the number of asterisks.
Note: in addition to the convention of *args and *kwargs, you will also see a andkw from time to
time.
Lets take a look at a quick example:
1
2
3
4
5
6
7

>>> def many(*args, **kwargs):


print(args)
print(kwargs)
>>> many(1, 2, 3, name="Mike", job="programmer")
(1, 2, 3)
{'job': 'programmer', 'name': 'Mike'}

First we create our function using the new syntax and then we call it with three regular arguments
and two keyword arguments. The function itself will print out both types of arguments. As you can
see, the args parameter turns into a tuple and kwargs turns into a dictionary. You will see this type
of coding used in the Python source and in many 3rd party Python packages.

A Note on Scope and Globals


Python has the concept of scope just like most programming languages. Scope will tell us when a
variable is available to use and where. If we define the variables inside of a function, those variables
can only be used inside that function. Once that function ends, they can no longer be used because
they are out of scope. Lets take a look at an example:
1
2
3
4
5
6
7
8
9
10
11

def function_a():
a = 1
b = 2
return a+b
def function_b():
c = 3
return a+c
print( function_a() )
print( function_b() )

Chapter 10 - Functions

63

If you run this code, you will receive the following error:
1

NameError: global name 'a' is not defined

This is caused because the variable a is only defined in the first function and is not available in the
second. You can get around this by telling Python that a is a global variable. Lets take a look at how
thats done:
1
2
3
4
5
6
7
8
9
10
11
12

def function_a():
global a
a = 1
b = 2
return a+b
def function_b():
c = 3
return a+c
print(function_a())
print(function_b())

This code will work because we told Python to make a global, which means that that variable is
available everywhere in our program. This is usually a bad idea and not recommended. The reason
it is not recommended is that it makes it difficult to tell when the variable is defined. Another
problem is that when we define a global in one place, we may accidentally redefine its value in
another which may cause logic errors later that are difficult to debug.

Coding Tips
One of the biggest problems that new programmers need to learn is the idea of Dont Repeat Yourself
(DRY). This concept is that you should avoid writing the same code more than once. When you find
yourself doing that, then you know that chunk of code should go into a function. One great reason
to do this is that you will almost certainly need to change that piece of code again in the future and
if its in multiple places, then you will need to remember where all those locations are AND change
them.
Copying and pasting the same chunk of code all over is an example of spaghetti code. Try to avoid
this as much as possible. You will regret it at some point either because youll be the one having
to fix it or because youll find someone elses code that you have to maintain with these sorts of
problems.

Chapter 10 - Functions

64

Wrapping Up
You now have the foundational knowledge necessary to use functions effectively. You should practice
creating some simple functions and try calling them in different ways. Once youve played around
with functions a bit or you just think you thoroughly understand the concepts involved, you can
turn to the next chapter on classes.

Chapter 11 - Classes
Everything in Python is an object. Thats a very vague statement unless youve taken a computer
programming class or two. What this means is that every thing in Python has methods and values.
The reason is that everything is based on a class. A class is the blueprint of an object. Lets take a
look at what I mean:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

>>> x = "Mike"
>>> dir(x)
['__add__', '__class__', '__contains__', '__delattr__', '__doc__', '__eq__',
'__format__', '__ge__', '__getattribute__', '__getitem__', '__getnewargs__',
'__getslice__', '__gt__', '__hash__', '__init__', '__le__', '__len__', '__lt__',
'__mod__', '__mul__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__rep\
r__',
'__rmod__', '__rmul__', '__setattr__', '__sizeof__', '__str__', '__subclasshook_\
_',
'_formatter_field_name_split', '_formatter_parser', 'capitalize', 'center', 'cou\
nt',
'decode', 'encode', 'endswith', 'expandtabs', 'find', 'format', 'index', 'isalnu\
m',
'isalpha', 'isdigit', 'islower', 'isspace', 'istitle', 'isupper', 'join', 'ljust\
',
'lower', 'lstrip', 'partition', 'replace', 'rfind', 'rindex', 'rjust', 'rpartiti\
on',
'rsplit', 'rstrip', 'split', 'splitlines', 'startswith', 'strip', 'swapcase', 't\
itle',
'translate', 'upper', 'zfill']

Here we have a string assigned to the variable x. It might not look like much, but that string has a
lot of methods. If you use Pythons dir keyword, you can get a list of all the methods you can call
on your string. There are 71 methods here! Technically were not supposed to call the methods that
start with underscores directly, so that reduces the total to 38, but thats still a lot of methods! What
does this mean though? It means that a string is based on a class and x is an instance of that class!
In Python we can create our own classes. Lets get started!

Creating a Class
Creating a class in Python is very easy. Here is a very simple example:

Chapter 11 - Classes

1
2
3
4
5
6
7

66

# Python 2.x syntax


class Vehicle(object):
"""docstring"""
def __init__(self):
"""Constructor"""
pass

This class doesnt do anything in particular, however it is a very good learning tool. For example, to
create a class, we need to use Pythons class keyword, followed by the name of the class. In Python,
convention says that the class name should have the first letter capitalized. Next we have an open
parentheses followed by the word object and a closed parentheses. The object is what the class is
based on or inheriting from. This is known as the base class or parent class. Most classes in Python
are based on object. Classes have a special method called __init__ (for initialization). This method
is called whenever you create (or instantiate) an object based on this class. The __init__ method
is only called once and is not to be called again inside the program. Another term for __init__ is
constructor, although this term isnt used that much in Python.
You may be wondering why I keep saying method instead of function. A function changes its name
to method when it is within a class. You will also notice that every method has to have at least
one argument (i.e. self), which is not true with a regular function.
In Python 3, we dont need to explicitly say were inheriting from object. Instead, we could have
written the above like this:
1
2
3
4
5
6
7

# Python 3.x syntax


class Vehicle:
"""docstring"""
def __init__(self):
"""Constructor"""
pass

You will notice that the only difference is that we no longer need the parentheses if were basing
our class on object. Lets expand our class definition a bit and give it some attributes and methods.

Chapter 11 - Classes

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

67

class Vehicle(object):
"""docstring"""
def __init__(self, color, doors, tires):
"""Constructor"""
self.color = color
self.doors = doors
self.tires = tires
def brake(self):
"""
Stop the car
"""
return "Braking"
def drive(self):
"""
Drive the car
"""
return "I'm driving!"

The code above added three attributes and two methods. The three attributes are:
1
2
3

self.color = color
self.doors = doors
self.tires = tires

Attributes describe the vehicle. So the vehicle has a color, some number of doors and some number
of tires. It also has two methods. A method describes what a class does. So in this case, a vehicle can
brake and drive. You may have noticed that all of the methods, including the first one have a funny
argument called self. Lets talk about that!

What is self?
Python classes need a way to refer to themselves. This isnt some kind of narcissistic navel-gazing
on the part of the class. Instead, its a way to tell one instance from another. The word self is a way
to describe itself, literally. Lets take a look at an example as I always find that helpful when Im
learning something new and strange:
Add the following code to the end of that class you wrote above and save it:

Chapter 11 - Classes

1
2
3
4
5
6

68

if __name__ == "__main__":
car = Vehicle("blue", 5, 4)
print(car.color)
truck = Vehicle("red", 3, 6)
print(truck.color)

The conditional statement above is a common way of telling Python that you only want to run the
following code if this code is executed as a standalone file. If you had imported your module into
another script, then the code underneath the conditional would not run. Anyway, if you run this
code, you will create two instances of the Vehicle class: a car instance and a truck instance. Each
instance will have its own attributes and methods. This is why when we print out the color of each
instance, they are different. The reason is that the class is using that self argument to tell itself which
is which. Lets change the class a bit to make the methods more unique:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

class Vehicle(object):
"""docstring"""
def __init__(self, color, doors, tires, vtype):
"""Constructor"""
self.color = color
self.doors = doors
self.tires = tires
self.vtype = vtype
def brake(self):
"""
Stop the car
"""
return "%s braking" % self.vtype
def drive(self):
"""
Drive the car
"""
return "I'm driving a %s %s!" % (self.color, self.vtype)
if __name__ == "__main__":
car = Vehicle("blue", 5, 4, "car")
print(car.brake())
print(car.drive())

Chapter 11 - Classes

28
29
30

69

truck = Vehicle("red", 3, 6, "truck")


print(truck.drive())
print(truck.brake())

In this example, we pass in another parameter to tell the class which vehicle type were creating.
Then we call each method for each instance. If you run this code, you should see the following
output:
1
2
3
4

car braking
I'm driving a blue car!
I'm driving a red truck!
truck braking

This demonstrates how the instance keeps track of its self. You will also notice that we are able
to get the values of the attributes from the __init__ method into the other methods. The reason is
because all those attributes are prepended with self.. If we hadnt done that, the variables would
have gone out of scope at the end of the __init__ method.

Subclasses
The real power of classes becomes apparent when you get into subclasses. You may not have realized
it, but weve already created a subclass when we created a class based on object. In other words,
we subclassed object. Now because object isnt very interesting, the previous examples dont really
demonstrate the power of subclassing. So lets subclass our Vehicle class and find out how all this
works.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

class Car(Vehicle):
"""
The Car class
"""
#---------------------------------------------------------------------def brake(self):
"""
Override brake method
"""
return "The car class is breaking slowly!"
if __name__ == "__main__":
car = Car("yellow", 2, 4, "car")
car.brake()

Chapter 11 - Classes

16
17
18

70

'The car class is breaking slowly!'


car.drive()
"I'm driving a yellow car!"

For this example, we subclassed our Vehicle class. You will notice that we didnt include an __init__ method or a drive method. The reason is that when you subclass Vehicle, you get all its attributes
and methods unless you override them. Thus you will notice that we did override the brake method
and made it say something different from the default. The other methods we left the same. So when
you tell the car to brake, it uses the original method and we learn that were driving a yellow car.
Wasnt that neat?
Using the default values of the parent class is known as inheriting or inheritance. This is a big
topic in Object Oriented Programming (OOP). This is also a simple example of polymorphism.
Polymorphic classes typically have the same interfaces (i.e. methods, attributes), but they are not
aware of each other. In Python land, polymorphism isnt very rigid about making sure the interfaces
are exactly the same. Instead, it follows the concept of duck typing. The idea of duck typing is that
if it walks like a duck and talks like a duck, it must be a duck. So in Python, as long as the class
has method names that are the same, it doesnt matter if the implementation of the methods are
different.
Anyway, you really dont need to know the nitty gritty details of all that to use classes in Python.
You just need to be aware of the terminology so if you want to dig deeper, you will be able to. You
can find lots of good examples of Python polymorphism that will help you figure out if and how
you might use that concept in your own applications.
Now, when you subclass, you can override as much or as little from the parent class as you want. If
you completely override it, then you would probably be just as well off just creating a new class.

Wrapping Up
Classes are a little complicated, but they are very powerful. They allow you to use variables across
methods which can make code reuse even easier. I would recommend taking a look at Pythons
source for some excellent examples of how classes are defined and used.
We are now at the end of Part I. Congratulations for making it this far! You now have the tools
necessary to build just about anything in Python! In Part II, we will spend time learning about using
some of the wonderful modules that Python includes in its distribution. This will help you to better
understand the power of Python and familiarize yourself with using the Standard Library. Part II
will basically be a set of tutorials to help you on your way to becoming a great Python programmer!

Part II - Learning from the Library


In Part II, you will get an abbreviated tour of some of the Python Standard Library. The reason its
abbreviated is that the Python Standard Library is HUGE! So this section is to get you acquainted
with using the modules that come with Python. I will be going over the modules I use the most in
my day-to-day job and the modules Ive seen used by my co-workers. I think this mini-tour will
prepare you for digging in on your own.
Lets take a look at what well be covering:

Introspection
csv
ConfigParser
logging
os
smtplib / email
subprocess
sys
thread / queues
time / datetime

|
The first chapter in this section will give you a quick tutorial into Pythons introspection abilities;
basically youll learn how to make Python tell you about itself, which sounds kind of weird but is
really quite valuable to know about. Next well learn how to use ConfigParser, a neat little module
that lets you read and write config files. After that well take a look at logging. The os module
can do lots of fun things, but well try to focus on the ones that I think youll find most useful. The
subprocess allows you to open other processes.
You will find the sys module allows you to exit a program, get the Python path, acquire version
information, redirect stdout and a whole lot more. The thread module allows you to create threads
in your program. We wont dive too deep into that subject as it can get confusing pretty quickly. The
time and datetime modules allow you to manipulate dates and time in Python, which has many
applications when it comes to developing programs.
Lets get started!

72

Part II - Learning from the Library

image

Chapter 12 - Introspection
Whether youre new to Python, been using it for a few years or youre an expert, knowing how to use
Pythons introspection capabilities can help your understanding of your code and that new package
you just downloaded with the crappy documentation. Introspection is a fancy word that means to
observe oneself and ponder ones thoughts, senses, and desires. In Python world, introspection is
actually kind of similar. Introspection in this case is to use Python to figure out Python. In this
chapter, you can learn how to use Python to give yourself a clue about the code youre working on
or trying to learn. Some might even call it a form of debugging.
Heres what were going to cover:
type
dir
help

The Python Type


You may not know this, but Python may just be your type. Yes, Python can tell you what type of
variable you have or what type is returned from a function. Its a very handy little tool. Lets look
at a few examples to make it all clear:
1
2
3
4
5
6
7
8
9

>>> x = "test"
>>> y = 7
>>> z = None
>>> type(x)
<class 'str'>
>>> type(y)
<class 'int'>
>>> type(z)
<class 'NoneType'>

As you can see, Python has a keyword called type that can tell you what is what. In my real-life
experiences, Ive used type to help me figure out what is going on when my database data is corrupt
or not what I expect. I just add a couple lines and print out each rows data along with its type. This
has helped me a lot when Ive been dumbfounded by some stupid code I wrote.

Chapter 12 - Introspection

74

The Python Dir


What is dir? Is it something you say when someone says or does something stupid? Not in this
context! No, here on Planet Python, the dir keyword is used to tell the programmer what attributes
and methods there are in the passed in object. If you forget to pass in an object, dir will return a list
of names in the current scope. As usual, this is easier to understand with a few examples.
1
2
3
4
5
6
7
8
9
10
11
12
13
14

>>> dir("test")
['__add__', '__class__', '__contains__', '__delattr__',
'__doc__', '__eq__', '__ge__', '__getattribute__',
'__getitem__', '__getnewargs__', '__getslice__', '__gt__',
'__hash__', '__init__', '__le__', '__len__', '__lt__',
'__mod__', '__mul__', '__ne__', '__new__', '__reduce__',
'__reduce_ex__', '__repr__', '__rmod__', '__rmul__',
'__setattr__', '__str__', 'capitalize', 'center',
'count', 'decode', 'encode', 'endswith', 'expandtabs',
'find', 'index', 'isalnum', 'isalpha', 'isdigit', 'islower',
'isspace', 'istitle', 'isupper', 'join', 'ljust', 'lower',
'lstrip', 'replace', 'rfind', 'rindex', 'rjust', 'rsplit',
'rstrip', 'split', 'splitlines', 'startswith', 'strip',
'swapcase', 'title', 'translate', 'upper', 'zfill']

Since everything in Python is an object, we can pass a string to dir and find out what methods it
has. Pretty neat, huh? Now lets try it with an imported module:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

>>> import sys


>>> dir(sys)
['__displayhook__', '__doc__', '__egginsert', '__excepthook__',
'__name__', '__plen', '__stderr__', '__stdin__', '__stdout__',
'_getframe', 'api_version', 'argv', 'builtin_module_names',
'byteorder', 'call_tracing', 'callstats', 'copyright',
'displayhook', 'dllhandle', 'exc_clear', 'exc_info',
'exc_traceback', 'exc_type', 'exc_value', 'excepthook',
'exec_prefix', 'executable', 'exit', 'exitfunc',
'getcheckinterval', 'getdefaultencoding', 'getfilesystemencoding',
'getrecursionlimit', 'getrefcount', 'getwindowsversion', 'hexversion',
'maxint', 'maxunicode', 'meta_path', 'modules', 'path', 'path_hooks',
'path_importer_cache', 'platform', 'prefix', 'setcheckinterval',
'setprofile', 'setrecursionlimit', 'settrace', 'stderr', 'stdin',
'stdout', 'version', 'version_info', 'warnoptions', 'winver']

75

Chapter 12 - Introspection

Now, thats handy! If you havent figured it out yet, the dir function is extremely handy for
those 3rd party packages that you have downloaded (or will soon download) that have little to
no documentation. How do you find out about what methods are available in these cases? Well,
dir will help you figure it out. Of course, sometimes the documentation is in the code itself, which
brings us to the builtin help utility.

Python Help!
Python comes with a handy help utility. Just type help() (minus the quotes) into a Python shell
and youll see the following directions (Python version may vary)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

>>> help()
Welcome to Python 2.6!

This is the online help utility.

If this is your first time using Python, you should definitely check out
the tutorial on the Internet at http://www.python.org/doc/tut/.
Enter the name of any module, keyword, or topic to get help on writing
Python programs and using Python modules. To quit this help utility and
return to the interpreter, just type "quit".
To get a list of available modules, keywords, or topics, type "modules",
"keywords", or "topics". Each module also comes with a one-line summary
of what it does; to list the modules whose summaries contain a given word
such as "spam", type "modules spam".
help>

Note that you now have a help> prompt instead of the >>>. When you are in help mode, you
can explore the various modules, keywords and topics found in Python. Also note that when typing
the word modules, you will see a delay as Python searches its library folder to acquire a list. If
you have installed a lot of 3rd party modules, this can take quite a while, so be prepared to go fix
yourself a mocha while you wait. Once its done, just follow the directions and play around with it
and I think youll get the gist.

Wrapping Up
Now you know how to take an unknown module and learn a lot about it by just using some of
Pythons built-in functionality. You will find yourself using these commands over and over again
to help you learn Python. As I mentioned earlier, you will find these tools especially helpful for the
3rd Party modules that dont believe in documentation.

Chapter 13 - The csv Module


The csv module gives the Python programmer the ability to parse CSV (Comma Separated Values)
files. A CSV file is a human readable text file where each line has a number of fields, separated by
commas or some other delimiter. You can think of each line as a row and each field as a column.
The CSV format has no standard, but they are similar enough that the csv module will be able to
read the vast majority of CSV files. You can also write CSV files using the csv module.

Reading a CSV File


There are two ways to read a CSV file. You can use the csv modules reader function or you can use
the DictReader class. We will look at both methods. But first, we need to get a CSV file so we have
something to parse. There are many websites that provide interesting information in CSV format.
We will be using the World Health Organizations (WHO) website to download some information
on Tuberculosis. You can go here to get it: http://www.who.int/tb/country/data/download/en/. Once
you have the file, well be ready to start. Ready? Then lets look at some code!
1
2
3
4
5
6
7
8
9
10
11
12
13
14

import csv
def csv_reader(file_obj):
"""
Read a csv file
"""
reader = csv.reader(file_obj)
for row in reader:
print(" ".join(row))
if __name__ == "__main__":
csv_path = "TB_data_dictionary_2014-02-26.csv"
with open(csv_path, "r") as f_obj:
csv_reader(f_obj)

Lets take a moment to break this down a bit. First off, we have to actually import the csv module.
Then we create a very simple function called csv_reader that accepts a file object. Inside the
function, we pass the file object into the csv.reader function, which returns a reader object. The
reader object allows iteration, much like a regular file object does. This lets us iterate over each row
in the reader object and print out the line of data, minus the commas. This works because each row
is a list and we can join each element in the list together, forming one long string.
Now lets create our own CSV file and feed it into the DictReader class. Heres a really simple one:

Chapter 13 - The csv Module

1
2
3
4

77

first_name,last_name,address,city,state,zip_code
Tyrese,Hirthe,1404 Turner Ville,Strackeport,NY,19106-8813
Jules,Dicki,2410 Estella Cape Suite 061,Lake Nickolasville,ME,00621-7435
Dedric,Medhurst,6912 Dayna Shoal,Stiedemannberg,SC,43259-2273

Lets save this in a file named data.csv. Now were ready to parse the file using the DictReader class.
Lets try it out:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

import csv
def csv_dict_reader(file_obj):
"""
Read a CSV file using csv.DictReader
"""
reader = csv.DictReader(file_obj, delimiter=',')
for line in reader:
print(line["first_name"]),
print(line["last_name"])
if __name__ == "__main__":
with open("data.csv") as f_obj:
csv_dict_reader(f_obj)

In the example above, we open a file and pass the file object to our function as we did before. The
function passes the file object to our DictReader class. We tell the DictReader that the delimiter is
a comma. This isnt actually required as the code will still work without that keyword argument.
However, its a good idea to be explicit so you know whats going on here. Next we loop over the
reader object and discover that each line in the reader object is a dictionary. This makes printing out
specific pieces of the line very easy.
Now were ready to learn how to write a csv file to disk.

Writing a CSV File


The csv module also has two methods that you can use to write a CSV file. You can use the writer
function or the DictWriter class. Well look at both of these as well. We will be with the writer
function. Lets look at a simple example:

Chapter 13 - The csv Module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

78

import csv
def csv_writer(data, path):
"""
Write data to a CSV file path
"""
with open(path, "w", newline='') as csv_file:
writer = csv.writer(csv_file, delimiter=',')
for line in data:
writer.writerow(line)
if __name__ == "__main__":
data = ["first_name,last_name,city".split(","),
"Tyrese,Hirthe,Strackeport".split(","),
"Jules,Dicki,Lake Nickolasville".split(","),
"Dedric,Medhurst,Stiedemannberg".split(",")
]
path = "output.csv"
csv_writer(data, path)

In the code above, we create a csv_writer function that accepts two arguments: data and path. The
data is a list of lists that we create at the bottom of the script. We use a shortened version of the
data from the previous example and split the strings on the comma. This returns a list. So we end
up with a nested list that looks like this:
1
2
3
4

[['first_name', 'last_name', 'city'],


['Tyrese', 'Hirthe', 'Strackeport'],
['Jules', 'Dicki', 'Lake Nickolasville'],
['Dedric', 'Medhurst', 'Stiedemannberg']]

The csv_writer function opens the path that we pass in and creates a csv writer object. Then we
loop over the nested list structure and write each line out to disk. Note that we specified what the
delimiter should be when we created the writer object. If you want the delimiter to be something
besides a comma, this is where you would set it.
Now were ready to learn how to write a CSV file using the DictWriter class! Were going to use
the data from the previous version and transform it into a list of dictionaries that we can feed to our
hungry DictWriter. Lets take a look:

Chapter 13 - The csv Module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

79

import csv
def csv_dict_writer(path, fieldnames, data):
"""
Writes a CSV file using DictWriter
"""
with open(path, "w", newline='') as out_file:
writer = csv.DictWriter(out_file, delimiter=',', fieldnames=fieldnames)
writer.writeheader()
for row in data:
writer.writerow(row)
if __name__ == "__main__":
data = ["first_name,last_name,city".split(","),
"Tyrese,Hirthe,Strackeport".split(","),
"Jules,Dicki,Lake Nickolasville".split(","),
"Dedric,Medhurst,Stiedemannberg".split(",")
]
my_list = []
fieldnames = data[0]
for values in data[1:]:
inner_dict = dict(zip(fieldnames, values))
my_list.append(inner_dict)
path = "dict_output.csv"
csv_dict_writer(path, fieldnames, my_list)

We will start in the second section first. As you can see, we start out with the nested list structure
that we had before. Next we create an empty list and a list that contains the field names, which
happens to be the first list inside the nested list. Remember, lists are zero-based, so the first element
in a list starts at zero! Next we loop over the nested list construct, starting with the second element:
1
2
3

for values in data[1:]:


inner_dict = dict(zip(fieldnames, values))
my_list.append(inner_dict)

Inside the for loop, we use Python builtins to create dictionary. The zip method will take two iterators
(lists in this case) and turn them into a list of tuples. Heres an example:

Chapter 13 - The csv Module

1
2

80

zip(fieldnames, values)
[('first_name', 'Dedric'), ('last_name', 'Medhurst'), ('city', 'Stiedemannberg')]

Now when your wrap that call in dict, it turns that list of tuples into a dictionary. Finally we append
the dictionary to the list. When the for finishes, youll end up with a data structure that looks like
this:
[{city: Strackeport, first_name: Tyrese, last_name: Hirthe},
{city: Lake Nickolasville, first_name: Jules, last_name: Dicki}, {city: Stiedemannberg, first_name: Dedric, last_name: Medhurst}]
At the end of the second session, we call our csv_dict_writer function and pass in all
the required arguments. Inside the function, we create a DictWriter instance and pass
it a file object, a delimiter value and our list of field names. Next we write the field
names out to disk and loop over the data one row at a time, writing the data to disk.
The DictWriter class also supports the writerows method, which we could have used
instead of the loop. The csv.writer function also supports this functionality.
You may be interested to know that you can also create Dialects with the csv module. This allows
you to tell the csv module how to read or write a file in a very explicit manner. If you need this sort of
thing because of an oddly formatted file from a client, then youll find this functionality invaluable.

Wrapping Up
Now you know how to use the csv module to read and write CSV files. There are many websites
that put out their data in this format and it is used a lot in the business world. In our next chapter,
we will begin learning about the ConfigParser module.

Chapter 14 - configparser
Configuration files are used by both users and programmers. They are usually used for storing
your applications settings or even your operating systems settings. Pythons core library includes
a module called configparser that you can use for creating and interacting with configuration files.
Well spend a few minutes learning how it works in this chapter.

Creating a Config File


Creating a config file with configparser is extremely simple. Lets create some code to demonstrate:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

import configparser
def createConfig(path):
"""
Create a config file
"""
config = configparser.ConfigParser()
config.add_section("Settings")
config.set("Settings", "font", "Courier")
config.set("Settings", "font_size", "10")
config.set("Settings", "font_style", "Normal")
config.set("Settings", "font_info",
"You are using %(font)s at %(font_size)s pt")
with open(path, "w") as config_file:
config.write(config_file)

if __name__ == "__main__":
path = "settings.ini"
createConfig(path)

The code above will create a config file with one section labelled Settings that will contain four
options: font, font_size, font_style and font_info. Also note that in Python 3 we only need to specify
that were writing the file in write-only mode, i.e. w. Back in Python 2, we had to use wb to
write in binary mode.

Chapter 14 - configparser

82

How to Read, Update and Delete Options


Now were ready to learn how to read the config file, update its options and even how to delete
options. In this case, its easier to learn by actually writing some code! Just add the following function
to the code that you wrote above.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31

import configparser
import os
def crudConfig(path):
"""
Create, read, update, delete config
"""
if not os.path.exists(path):
createConfig(path)
config = configparser.ConfigParser()
config.read(path)
# read some values from the config
font = config.get("Settings", "font")
font_size = config.get("Settings", "font_size")
# change a value in the config
config.set("Settings", "font_size", "12")
# delete a value from the config
config.remove_option("Settings", "font_style")
# write changes back to the config file
with open(path, "w") as config_file:
config.write(config_file)

if __name__ == "__main__":
path = "settings.ini"
crudConfig(path)

This code first checks to see if the path for the config file exists. If it does not, then it uses the
createConfig function we created earlier to create it. Next we create a ConfigParser object and pass
it the config file path to read. To read an option in your config file, we call our ConfigParser objects
get method, passing it the section name and the option name. This will return the options value. If

Chapter 14 - configparser

83

you want to change an options value, then you use the set method, where you pass the section name,
the option name and the new value. Finally, you can use the remove_option method to remove an
option.
In our example code, we change the value of font_size to 12 and we remove the font_style option
completely. Then we write the changes back out to disk.
This isnt really a good example though as you should never have a function that does everything
like this one does. So lets split it up into a series of functions instead:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34

import configparser
import os
def create_config(path):
"""
Create a config file
"""
config = configparser.ConfigParser()
config.add_section("Settings")
config.set("Settings", "font", "Courier")
config.set("Settings", "font_size", "10")
config.set("Settings", "font_style", "Normal")
config.set("Settings", "font_info",
"You are using %(font)s at %(font_size)s pt")
with open(path, "w") as config_file:
config.write(config_file)

def get_config(path):
"""
Returns the config object
"""
if not os.path.exists(path):
create_config(path)
config = configparser.ConfigParser()
config.read(path)
return config

def get_setting(path, section, setting):


"""
Print out a setting

Chapter 14 - configparser

35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70

84

"""
config = get_config(path)
value = config.get(section, setting)
msg = "{section} {setting} is {value}".format(
section=section, setting=setting, value=value)
print(msg)
return value

def update_setting(path, section, setting, value):


"""
Update a setting
"""
config = get_config(path)
config.set(section, setting, value)
with open(path, "w") as config_file:
config.write(config_file)

def delete_setting(path, section, setting):


"""
Delete a setting
"""
config = get_config(path)
config.remove_option(section, setting)
with open(path, "w") as config_file:
config.write(config_file)
if __name__ == "__main__":
path = "settings.ini"
font = get_setting(path, 'Settings', 'font')
font_size = get_setting(path, 'Settings', 'font_size')
update_setting(path, "Settings", "font_size", "12")
delete_setting(path, "Settings", "font_style")

This example is heavily refactored compared to the first one. I even went so far as to name the
functions following PEP8. Each function should be self-explanatory and self-contained. Instead
of putting all the logic into one function, we separate it out into multiple functions and then
demonstrate their functionality within the bottom if statement. Now you can import the module
and use it yourself.

Chapter 14 - configparser

85

Please note that this example has the section hard-coded, so you will want to update this example
further to make it completely generic.

How to Use Interpolation


The configparser module also allows interpolation, which means you can use some options to build
another option. We actually do this with the font_info option in that its value is based on the font
and font_size options. We can change an interpolated value using a Python dictionary. Lets take a
moment to demonstrate both of these facts.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

import configparser
import os
def interpolationDemo(path):
if not os.path.exists(path):
createConfig(path)
config = configparser.ConfigParser()
config.read(path)
print(config.get("Settings", "font_info"))
print(config.get("Settings", "font_info",
vars={"font": "Arial", "font_size": "100"}))

if __name__ == "__main__":
path = "settings.ini"
interpolationDemo(path)

If you run this code, you should see output similar to the following:
1
2

You are using Courier at 12 pt


You are using Arial at 100 pt

Wrapping Up
At this point, you should know enough about the configparsers capabilities that you can use it for
your own projects. Theres another project called ConfigObj that isnt a part of Python that you
might also want to check out. ConfigObj is more flexible and has more features than configparser.
But if youre in a pinch or your organization doesnt allow 3rd party packages, then configparser
will probably fit the bill.

Chapter 15 - Logging
Python provides a very powerful logging library in its standard library. A lot of programmers use
print statements for debugging (myself included), but you can also use logging to do this. Its actually
cleaner to use logging as you wont have to go through all your code to remove the print statements.
In this chapter well cover the following topics:

Creating a simple logger


How to log from multiple modules
Log formatting
Log configuration

By the end of this chapter, you should be able to confidently create your own logs for your
applications. Lets get started!

Creating a Simple Logger


Creating a log with the logging module is easy and straight-forward. Its easiest to just look at a
piece of code and then explain it, so heres some code for you to read:
1
2
3
4
5
6
7
8

import logging
# add filemode="w" to overwrite
logging.basicConfig(filename="sample.log", level=logging.INFO)
logging.debug("This is a debug message")
logging.info("Informational message")
logging.error("An error has happened!")

As you might expect, to access the logging module you have to first import it. The easiest way to
create a log is to use the logging modules basicConfig function and pass it some keyword arguments.
It accepts the following: filename, filemode, format, datefmt, level and stream. In our example, we
pass it a file name and the logging level, which we set to INFO. There are five levels of logging (in
ascending order): DEBUG, INFO, WARNING, ERROR and CRITICAL. By default, if you run this
code multiple times, it will append to the log if it already exists. If you would rather have your
logger overwrite the log, then pass in a filemode=w as mentioned in the comment in the code.
Speaking of running the code, this is what you should get if you ran it once:

Chapter 15 - Logging

1
2

87

INFO:root:Informational message
ERROR:root:An error has happened!

Note that the debugging message isnt in the output. That is because we set the level at INFO, so
our logger will only log if its a INFO, WARNING, ERROR or CRITICAL message. The root part just
means that this logging message is coming from the root logger or the main logger. Well look at
how to change that so its more descriptive in the next section. If you dont use basicConfig, then
the logging module will output to the console / stdout.
The logging module can also log some exceptions to file or wherever you have it configured to log
to. Heres an example:
1
2
3
4
5
6
7
8
9

import logging
logging.basicConfig(filename="sample.log", level=logging.INFO)
log = logging.getLogger("ex")
try:
raise RuntimeError
except RuntimeError:
log.exception("Error!")

Lets break this down a bit. Here we use the logging modules getLogger method to return a logger
object that has been named ex. This is handy when you have multiple loggers in one application
as it allows you to identify which messages came from each logger. This example will force a
RuntimeError to be raised, catch the error and log the entire traceback to file, which can be very
handy when debugging.

How to log From Multiple Modules (and Formatting


too!)
The more you code, the more often you end up creating a set of custom modules that work together.
If you want them all to log to the same place, then youve come to the right place. Well look at the
simple way and then show a more complex method thats also more customizable. Heres one easy
way to do it:

Chapter 15 - Logging

1
2
3
4
5
6
7
8
9
10
11
12
13
14

88

import logging
import otherMod
def main():
"""
The main entry point of the application
"""
logging.basicConfig(filename="mySnake.log", level=logging.INFO)
logging.info("Program started")
result = otherMod.add(7, 8)
logging.info("Done!")
if __name__ == "__main__":
main()

Here we import logging and a module of our own creation (otherMod). Then we create our log
file as before. The other module looks like this:
1
2
3
4
5
6
7

# otherMod.py
import logging
def add(x, y):
""""""
logging.info("added %s and %s to get %s" % (x, y, x+y))
return x+y

If you run the main code, you should end up with a log that has the following contents:
1
2
3

INFO:root:Program started
INFO:root:added 7 and 8 to get 15
INFO:root:Done!

Do you see the problem with doing it this way? You cant really tell very easily where the log
messages are coming from. This will only get more confusing the more modules there are that write
to this log. So we need to fix that. That brings us to the complex way of creating a logger. Lets take
a look at a different implementation:

Chapter 15 - Logging

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

89

import logging
import otherMod2
def main():
"""
The main entry point of the application
"""
logger = logging.getLogger("exampleApp")
logger.setLevel(logging.INFO)
# create the logging file handler
fh = logging.FileHandler("new_snake.log")
formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(me\
ssage)s')
fh.setFormatter(formatter)
# add handler to logger object
logger.addHandler(fh)
logger.info("Program started")
result = otherMod2.add(7, 8)
logger.info("Done!")
if __name__ == "__main__":
main()

Here we create a logger instance named exampleApp. We set its logging level, create a logging file
handler object and a logging Formatter object. The file handler has to set the formatter object as its
formatter and then the file handler has to be added to the logger instance. The rest of the code in
main is mostly the same. Just note that instead of logging.info, its logger.info or whatever you
call your logger variable. Heres the updated otherMod2 code:
1
2
3
4
5
6
7
8
9
10

# otherMod2.py
import logging
module_logger = logging.getLogger("exampleApp.otherMod2")
def add(x, y):
""""""
logger = logging.getLogger("exampleApp.otherMod2.add")
logger.info("added %s and %s to get %s" % (x, y, x+y))
return x+y

Chapter 15 - Logging

90

Note that here we have two loggers defined. We dont do anything with the module_logger in this
case, but we do use the other one. If you run the main script, you should see the following output
in your file:
1
2
3
4

2012-08-02 15:37:40,592 - exampleApp - INFO - Program started


2012-08-02 15:37:40,592 - exampleApp.otherMod2.add - INFO - added 7 and 8 to get\
15
2012-08-02 15:37:40,592 - exampleApp - INFO - Done!

You will notice that all references to root have been removed. Instead it uses our Formatter object
which says that we should get a human readable time, the logger name, the logging level and then the
message. These are actually known as LogRecord attributes. For a full list of LogRecord attributes,
see the documentation as there are too many to list here.

Configuring Logs for Work and Pleasure


The logging module can be configured 3 different ways. You can configure it using methods (loggers,
formatters, handlers) like we did earlier in this article; you can use a configuration file and pass
it to fileConfig(); or you can create a dictionary of configuration information and pass it to the
dictConfig() function. Lets create a configuration file first and then well look at how to execute it
with Python. Heres an example config file:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

[loggers]
keys=root,exampleApp
[handlers]
keys=fileHandler, consoleHandler
[formatters]
keys=myFormatter
[logger_root]
level=CRITICAL
handlers=consoleHandler
[logger_exampleApp]
level=INFO
handlers=fileHandler
qualname=exampleApp
https://docs.python.org/3/library/logging.html#logrecord-attributes

Chapter 15 - Logging

18
19
20
21
22
23
24
25
26
27
28
29
30
31
32

91

[handler_consoleHandler]
class=StreamHandler
level=DEBUG
formatter=myFormatter
args=(sys.stdout,)
[handler_fileHandler]
class=FileHandler
formatter=myFormatter
args=("config.log",)
[formatter_myFormatter]
format=%(asctime)s - %(name)s - %(levelname)s - %(message)s
datefmt=

Youll notice that we have two loggers specified: root and exampleApp. For whatever reason, root
is required. If you dont include it, Python will raise a ValueError from config.pys _install_loggers
function, which is a part of the logging module. If you set the roots handler to fileHandler, then
youll end up doubling the log output, so to keep that from happening, we send it to the console
instead. Study this example closely. Youll need a section for every key in the first three sections.
Now lets see how we load it in the code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

# log_with_config.py
import logging
import logging.config
import otherMod2
def main():
"""
Based on http://docs.python.org/howto/logging.html#configuring-logging
"""
logging.config.fileConfig('logging.conf')
logger = logging.getLogger("exampleApp")
logger.info("Program started")
result = otherMod2.add(7, 8)
logger.info("Done!")
if __name__ == "__main__":
main()

Chapter 15 - Logging

92

As you can see, all you need to do is pass the config file path to logging.config.fileConfig. Youll
also notice that we dont need all that setup code any more as thats all in the config file. Also we
can just import the otherMod2 module with no changes. Anyway, if you run the above, you should
end up with the following in your log file:
1
2
3
4

2012-08-02 18:23:33,338 - exampleApp - INFO - Program started


2012-08-02 18:23:33,338 - exampleApp.otherMod2.add - INFO - added 7 and 8 to get\
15
2012-08-02 18:23:33,338 - exampleApp - INFO - Done!

As you might have guessed, its very similar to the other example. Now well move on to the other
config method. The dictionary configuration method (dictConfig) wasnt added until Python 2.7, so
make sure you have that or a later version, otherwise you wont be able to follow along. Its not
well documented how this works. In fact, the examples in the documentation show YAML for some
reason. Anyway, heres some working code for you to look over:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

# log_with_config.py
import logging
import logging.config
import otherMod2
def main():
"""
Based on http://docs.python.org/howto/logging.html#configuring-logging
"""
dictLogConfig = {
"version":1,
"handlers":{
"fileHandler":{
"class":"logging.FileHandler",
"formatter":"myFormatter",
"filename":"config2.log"
}
},
"loggers":{
"exampleApp":{
"handlers":["fileHandler"],
"level":"INFO",
}
},
"formatters":{

Chapter 15 - Logging

27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

93

"myFormatter":{
"format":"%(asctime)s - %(name)s - %(levelname)s - %(message)s"
}
}
}
logging.config.dictConfig(dictLogConfig)
logger = logging.getLogger("exampleApp")
logger.info("Program started")
result = otherMod2.add(7, 8)
logger.info("Done!")
if __name__ == "__main__":
main()

If you run this code, youll end up with the same output as the previous method. Note that you dont
need the root logger when you use a dictionary configuration.

Wrapping Up
At this point you should know how to get started using loggers and how to configure them in
several different ways. You should also have gained the knowledge of how to modify the output
using the Formatter object. The logging module is very handy for troubleshooting what went wrong
in your application. Be sure to spend some time practicing with this module before writing a large
application.
In the next chapter we will be looking at how to use the os module.

Chapter 16 - The os Module


The os module has many uses. We wont be covering everything that it can do. Instead, we will
get an overview of its uses and well also take a look at one of its sub-modules, known as os.path.
Specifically, we will be covering the following:

os.name
os.environ
os.chdir()
os.getcwd()
os.getenv()
os.putenv()
os.mkdir()
os.makedirs()
os.remove()
os.rename()
os.rmdir()
os.startfile()
os.walk()
os.path

That looks like a lot to cover, but there is at least ten times as many other actions that the os module
can do. This chapter is just going to give you a little taste of whats available. To use any of the
methods mentioned in this section, you will need to import the os module, like this:
1

import os

Lets start learning how to use this module!

os.name
The os module has both callable functions and normal values. In the case of os.name, it is just a
value. When you access os.name, you will get information about what platform you are running on.
You will receive one of the following values: posix, nt, os2, ce, java, riscos. Lets see what we
get when we run it on Windows 7:

Chapter 16 - The os Module

1
2
3

95

>>> import os
>>> os.name
'nt'

This tells us that our Python instance is running on a Windows box. How do we know this? Because
Microsoft started calling its operating system NT many years ago. For example, Windows 7 is also
known as Windows NT 6.1.

os.environ, os.getenv() and os.putenv()


The os.environ value is known as a mapping object that returns a dictionary of the users
environmental variables. You may not know this, but every time you use your computer, some
environment variables are set. These can give you valuable information, such as number of
processors, type of CPU, the computer name, etc. Lets see what we can find out about our machine:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

>>> import os
>>> os.environ
{'ALLUSERSPROFILE': 'C:\\ProgramData',
'APPDATA': 'C:\\Users\\mike\\AppData\\Roaming',
'CLASSPATH': '.;C:\\Program Files\\QuickTime\\QTSystem\\QTJava.zip',
'COMMONPROGRAMFILES': 'C:\\Program Files\\Common Files',
'COMPUTERNAME': 'MIKE-PC',
'COMSPEC': 'C:\\Windows\\system32\\cmd.exe',
'FP_NO_HOST_CHECK': 'NO',
'HOMEDRIVE': 'C:',
'HOMEPATH': '\\Users\\mike',
'LOCALAPPDATA': 'C:\\Users\\mike\\AppData\\Local',
'LOGONSERVER': '\\\\MIKE-PC',
'NUMBER_OF_PROCESSORS': '2',
'OS': 'Windows_NT',
'PATHEXT': '.COM;.EXE;.BAT;.CMD;.VBS;.VBE;.JS;.JSE;.WSF;.WSH;.MSC',
'PROCESSOR_ARCHITECTURE': 'x86',
'PROCESSOR_IDENTIFIER': 'x86 Family 6 Model 15 Stepping 13, GenuineIntel',
'PROCESSOR_LEVEL': '6',
'PROGRAMDATA': 'C:\\ProgramData',
'PROGRAMFILES': 'C:\\Program Files',
'PSMODULEPATH': 'C:\\Windows\\system32\\WindowsPowerShell\\v1.0\\Modules\\',
'PUBLIC': 'C:\\Users\\Public',
'PYTHONIOENCODING': 'cp437',
'QTJAVA': 'C:\\Program Files\\QuickTime\\QTSystem\\QTJava.zip',
'SESSIONNAME': 'Console',

Chapter 16 - The os Module

27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

96

'SYSTEMDRIVE': 'C:',
'SYSTEMROOT': 'C:\\Windows',
'TEMP': 'C:\\Users\\mike\\AppData\\Local\\Temp',
'TMP': 'C:\\Users\\mike\\AppData\\Local\\Temp',
'USERDOMAIN': 'mike-PC',
'USERNAME': 'mike',
'USERPROFILE': 'C:\\Users\\mike',
'VBOX_INSTALL_PATH': 'C:\\Program Files\\Oracle\\VirtualBox\\',
'VS90COMNTOOLS': 'C:\\Program Files\\Microsoft Visual Studio 9.0\\Common7\\Tool\
s\\',
'WINDIR': 'C:\\Windows',
'WINDOWS_TRACING_FLAGS': '3',
'WINDOWS_TRACING_LOGFILE': 'C:\\BVTBin\\Tests\\installpackage\\csilogfile.log',
'WINGDB_ACTIVE': '1',
'WINGDB_PYTHON': 'c:\\python27\\python.exe',
'WINGDB_SPAWNCOOKIE': 'rvlxwsGdD7SHYIJm'}

Your output wont be the same as mine as everyones PC configuration is a little different, but youll
see something similar. As you may have noticed, this returned a dictionary. That means you can
access the environmental variables using your normal dictionary methods. Heres an example:
1
2

>>> print(os.environ["TMP"])
'C:\\Users\\mike\\AppData\\Local\\Temp'

You could also use the os.getenv function to access this environmental variable:
1
2

>>> os.getenv("TMP")
'C:\\Users\\mike\\AppData\\Local\\Temp'

The benefit of using os.getenv() instead of the os.environ dictionary is that if you happen to try to
access an environmental variable that doesnt exist, the getenv function will just return None. If you
did the same thing with os.environ, you would receive an error. Lets give it a try so you can see
what happens:

Chapter 16 - The os Module

1
2
3
4
5
6
7
8
9
10

97

>>> os.environ["TMP2"]
Traceback (most recent call last):
File "<pyshell#1>", line 1, in <module>
os.environ["TMP2"]
File "C:\Python27\lib\os.py", line 423, in __getitem__
return self.data[key.upper()]
KeyError: 'TMP2'
>>> print(os.getenv("TMP2"))
None

os.chdir() and os.getcwd()


The os.chdir function allows us to change the directory that were currently running our Python
session in. If you want to actually know what path you are currently in, then you would call
os.getcwd(). Lets try them both out:
1
2
3
4
5

>>> os.getcwd()
'C:\\Python27'
>>> os.chdir(r"c:\Users\mike\Documents")
>>> os.getcwd()
'c:\\Users\\mike\\Documents'

The code above shows us that we started out in the Python directory by default when we run this
code in IDLE. Then we change folders using os.chdir(). Finally we call os.getcwd() a second time to
make sure that we changed to the folder successfully.

os.mkdir() and os.makedirs()


You might have guessed this already, but the two methods covered in this section are used for
creating directories. The first one is os.mkdir(), which allows us to create a single folder. Lets try
it out:
1
2
3

>>> os.mkdir("test")
>>> path = r'C:\Users\mike\Documents\pytest'
>>> os.mkdir(path)

The first line of code will create a folder named test in the current directory. You can use the methods
in the previous section to figure out where you just ran your code if youve forgotten. The second

Chapter 16 - The os Module

98

example assigns a path to a variable and then we pass the path to os.mkdir(). This allows you to
create a folder anywhere on your system that you have permission to.
The os.makedirs() function will create all the intermediate folders in a path if they dont already
exist. Basically this means that you can created a path that has nested folders in it. I find myself
doing this a lot when I create a log file that is in a dated folder structure, like Year/Month/Day. Lets
look at an example:
1
2

>>> path = r'C:\Users\mike\Documents\pytest\2014\02\19'


>>> os.makedirs(path)

What happened here? This code just created a bunch of folders! If you still had the pytest folder in
your system, then it just added a 2014 folder with another folder inside of it which also contained a
folder. Try it out for yourself using a valid path on your system.

os.remove() and os.rmdir()


The os.remove() and os.rmdir() functions are used for deleting files and directories respectively.
Lets look at an example of os.remove():
1

>>> os.remove("test.txt")

This code snippet will attempt to remove a file named test.txt from your current working directory.
If it cannot find the file, you will likely receive some sort of error. You will also receive an error if
the file is in use (i.e. locked) or you dont have permission to delete the file. You might also want to
check out os.unlink, which does the same thing. The term unlink is the traditional Unix name for
this procedure.
Now lets look at an example of os.rmdir():
1

>>> os.rmdir("pytest")

The code above will attempt to remove a directory named pytest from your current working
directory. If its successful, you will see that the directory no longer exists. An error will be raised if
the directory does not exist, you do not have permission to remove it or if the directory is not empty.
You might also want to take a look at os.removedirs() which can remove nested empty directories
recursively.

os.rename(src, dst)
The os.rename() function will rename a file or folder. Lets take a look at an example where we
rename a file:

Chapter 16 - The os Module

99

>>> os.rename("test.txt", "pytest.txt")

In this example, we tell os.rename to rename a file named test.txt to pytest.txt. This occurs in our
current working directory. You will see an error occur if you try to rename a file that doesnt exist
or that you dont have the proper permission to rename the file.
There is also an os.renames function that recursively renames a directory or file.

os.startfile()
The os.startfile() method allows us to start a file with its associated program. In other words, we
can open a file with its associated program, just like when you double-click a PDF and it opens in
Adobe Reader. Lets give it a try!
1

>>> os.startfile(r'C:\Users\mike\Documents\labels.pdf')

In the example above, I pass a fully qualified path to os.startfile that tells it to open a file called
labels.pdf. On my machine, this will open the PDF in Adobe Reader. You should try opening your
own PDFs, MP3s, and photos using this method to see how it works.

os.walk()
The os.walk() method gives us a way to iterate over a root level path. What this means is that we
can pass a path to this function and get access to all its sub-directories and files. Lets use one of the
Python folders that we have handy to test this function with. Well use: C:\Python27\Tools
1
2
3
4
5
6
7
8
9
10
11

>>> path = r'C:\Python27\Tools'


>>> for root, dirs, files in os.walk(path):
print(root)
C:\Python27\Tools
C:\Python27\Tools\i18n
C:\Python27\Tools\pynche
C:\Python27\Tools\pynche\X
C:\Python27\Tools\Scripts
C:\Python27\Tools\versioncheck
C:\Python27\Tools\webchecker

If you want, you can also loop over dirs and files too. Heres one way to do it:

Chapter 16 - The os Module

1
2
3
4
5
6

100

>>> for root, dirs, files in os.walk(path):


print(root)
for _dir in dirs:
print(_dir)
for _file in files:
print(_file)

This piece of code will print a lot of stuff out, so I wont be showing its output here, but feel free to
give it a try. Now were ready to learn about working with paths!

os.path
The os.path sub-module of the os module has lots of great functionality built into it. Well be looking
at the following functions:

basename
dirname
exists
isdir and isfile
join
split

There are lots of other functions in this sub-module. You are welcome to go read about them in the
Python documentation, section 10.1.

os.path.basename
The basename function will return just the filename of a path. Here is an example:
1
2

>>> os.path.basename(r'C:\Python27\Tools\pynche\ChipViewer.py')
'ChipViewer.py'

I have found this useful whenever I need to use a filename for naming some related file, such as a
log file. This happens a lot when Im processing a data file.

os.path.dirname
The dirname function will return just the directory portion of the path. Its easier to understand if
we take a look at some code:

Chapter 16 - The os Module

1
2

101

>>> os.path.dirname(r'C:\Python27\Tools\pynche\ChipViewer.py')
'C:\\Python27\\Tools\\pynche'

In this example, we just get the directory path back. This is also useful when you want to store other
files next to the file youre processing, like the aforementioned log file.

os.path.exists
The exists function will tell you if a path exists or not. All you have to do is pass it a path. Lets take
a look:
1
2
3
4

>>> os.path.exists(r'C:\Python27\Tools\pynche\ChipViewer.py')
True
>>> os.path.exists(r'C:\Python27\Tools\pynche\fake.py')
False

In the first example, we pass the exists function a real path and it returns True, which means that
the path exists. In the second example, we passed it a bad path and it told us that the path did not
exist by returning False.

os.path.isdir / os.path.isfile
The isdir and isfile methods are closely related to the exists method in that they also test for
existence. However, isdir only checks if the path is a directory and isfile only checks if the path
is a file. If you want to check if a path exists regardless of whether it is a file or a directory, then
youll want to use the exists method. Anyway, lets study some examples:
1
2
3
4
5
6
7
8

>>> os.path.isfile(r'C:\Python27\Tools\pynche\ChipViewer.py')
True
>>> os.path.isdir(r'C:\Python27\Tools\pynche\ChipViewer.py')
False
>>> os.path.isdir(r'C:\Python27\Tools\pynche')
True
>>> os.path.isfile(r'C:\Python27\Tools\pynche')
False

Take a moment to study this set of examples. In the first one we pass a path to a file and check if
the path is really a file. Then the second example checks the same path to see if its a directory. You
can see for yourself how that turned out. Then in the last two examples, we switched things up a bit
by passing a path to a directory to the same two functions. These examples demonstrate how these
two functions work.

Chapter 16 - The os Module

102

os.path.join
The join method give you the ability to join one or more path components together using the
appropriate separator. For example, on Windows, the separator is the backslash, but on Linux, the
separator is the forward slash. Heres how it works:
1
2

>>> os.path.join(r'C:\Python27\Tools\pynche', 'ChipViewer.py')


'C:\\Python27\\Tools\\pynche\\ChipViewer.py'

In this example, we joined a directory path and a file path together to get a fully qualified path. Note
however that the join method does not check if the result actually exists!

os.path.split
The split method will split a path into a tuple that contains the directory and the file. Lets take a
look:
1
2

>>> os.path.split(r'C:\Python27\Tools\pynche\ChipViewer.py')
('C:\\Python27\\Tools\\pynche', 'ChipViewer.py')

This example shows what happens when we path in a path with a file. Lets see what happens if the
path doesnt have a filename on the end:
1

>>> os.path.split(r'C:\Python27\Tools\pynche')

(C:\Python27\Tools, pynche)
As you can see, it took the path and split it in such a way that the last sub-folder became the second
element of the tuple with the rest of the path in the first element.
For our final example, I thought you might like to see a commmon use case of the split:
1
2
3
4
5

>>> dirname, fname = os.path.split(r'C:\Python27\Tools\pynche\ChipViewer.py')


>>> dirname
'C:\\Python27\\Tools\\pynche'
>>> fname
'ChipViewer.py'

This shows how to do multiple assignment. When you split the path, it returns a two-element tuple.
Since we have two variables on the left, the first element of the tuple is assigned to the first variable
and the second element to the second variable.

Chapter 16 - The os Module

103

Wrapping Up
At this point you should be pretty familiar with the os module. In this chapter you learned the
following:

how to work with environment variables


change directories and discover your current working directory
create and remove folders and files
rename files / folders
start a file with its associated application
walk a directory
work with paths

There are a lot of other functions in the os module that are not covered here. Be sure to read the
documentation to see what else you can do. In the next chapter, we will be learning about the email
and smtplib modules.

Chapter 17 - The email / smtplib


Module
Python provides a couple of really nice modules that we can use to craft emails with. They are the
email and smtplib modules. Instead of going over various methods in these two modules, well
spend some time learning how to actually use these modules. Specifically, well be covering the
following:

The basics of emailing


How to send to multiple addresses at once
How to send email using the TO, CC and BCC lines
How to add an attachment and a body using the email module

|
Lets get started!

Email Basics - How to Send an Email with smtplib


The smtplib module is very intuitive to use. Lets write a quick example that shows how to send an
email. Save the following code to a file on your hard drive:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

import smtplib
HOST = "mySMTP.server.com"
SUBJECT = "Test email from Python"
TO = "mike@someAddress.org"
FROM = "python@mydomain.com"
text = "Python 3.4 rules them all!"
BODY = "\r\n".join((
"From: %s" % FROM,
"To: %s" % TO,
"Subject: %s" % SUBJECT ,
"",
text
))

Chapter 17 - The email / smtplib Module

16
17
18
19

105

server = smtplib.SMTP(HOST)
server.sendmail(FROM, [TO], BODY)
server.quit()

We imported two modules, smtplib and the string module. Two thirds of this code is used for setting
up the email. Most of the variables are pretty self-explanatory, so well focus on the odd one only,
which is BODY. Here we use the string module to combine all the previous variables into a single
string where each lines ends with a carriage return (/r) plus new line (/n). If you print BODY
out, it would look like this:
1
2

'From: python@mydomain.com\r\nTo: mike@mydomain.com\r\nSubject: Test email from \


Python\r\n\r\nblah blah blah'

After that, we set up a server connection to our host and then we call the smtplib modules sendmail
method to send the email. Then we disconnect from the server. You will note that this code doesnt
have a username or password in it. If your server requires authentication, then youll need to add
the following code:
1

server.login(username, password)

This should be added right after you create the server object. Normally, you would want to put this
code into a function and call it with some of these parameters. You might even want to put some of
this information into a config file. Lets put this code into a function.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

import smtplib
def send_email(host, subject, to_addr, from_addr, body_text):
"""
Send an email
"""
BODY = "\r\n".join((
"From: %s" % from_addr,
"To: %s" % to_addr,
"Subject: %s" % subject ,
"",
body_text
))
server = smtplib.SMTP(host)
server.sendmail(from_addr, [to_addr], BODY)
server.quit()

Chapter 17 - The email / smtplib Module

17
18
19
20
21
22
23
24

106

if __name__ == "__main__":
host = "mySMTP.server.com"
subject = "Test email from Python"
to_addr = "mike@someAddress.org"
from_addr = "python@mydomain.com"
body_text = "Python rules them all!"
send_email(host, subject, to_addr, from_addr, body_text)

Now you can see how small the actual code is by just looking at the function itself. Thats 13 lines!
And we could make it shorter if we didnt put every item in the BODY on its own line, but it
wouldnt be as readable. Now well add a config file to hold the server information and the from
address. Why? Well in the work I do, we might use different email servers to send email or if the
email server gets upgraded and the name changes, then we only need to change the config file
rather than the code. The same thing could apply to the from address if our company was bought
and merged into another.
Lets take a look at the config file (save it as email.ini):
1
2
3

[smtp]
server = some.server.com
from_addr = python@mydomain.com

That is a very simple config file. In it we have a section labelled smtp in which we have two items:
server and from_addr. Well use configObj to read this file and turn it into a Python dictionary.
Heres the updated version of the code (save it as smtp_config.py):
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

import os
import smtplib
import sys
from configparser import ConfigParser
def send_email(subject, to_addr, body_text):
"""
Send an email
"""
base_path = os.path.dirname(os.path.abspath(__file__))
config_path = os.path.join(base_path, "email.ini")
if os.path.exists(config_path):
cfg = ConfigParser()

Chapter 17 - The email / smtplib Module

16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39

107

cfg.read(config_path)
else:
print("Config not found! Exiting!")
sys.exit(1)
host = cfg.get("smtp", "server")
from_addr = cfg.get("smtp", "from_addr")
BODY = "\r\n".join((
"From: %s" % from_addr,
"To: %s" % to_addr,
"Subject: %s" % subject ,
"",
body_text
))
server = smtplib.SMTP(host)
server.sendmail(from_addr, [to_addr], BODY)
server.quit()
if __name__ == "__main__":
subject = "Test email from Python"
to_addr = "mike@someAddress.org"
body_text = "Python rules them all!"
send_email(subject, to_addr, body_text)

Weve added a little check to this code. We want to first grab the path that the script itself is in,
which is what base_path represents. Next we combine that path with the file name to get a fully
qualified path to the config file. We then check for the existence of that file. If its there, we create
a ConfigParser and if its not, we print a message and exit the script. We should add an exception
handler around the ConfigParser.read() call just to be on the safe side though as the file could exist,
but be corrupt or we might not have permission to open it and that will throw an exception. That
will be a little project that you can attempt on your own. Anyway, lets say that everything goes
well and the ConfigParser object is created successfully. Now we can extract the host and from_addr
information using the usual ConfigParser syntax.
Now were ready to learn how to send multiple emails at the same time!

Sending Multiple Emails at Once


Lets modify our last example a little so we send multiple emails!

Chapter 17 - The email / smtplib Module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39

108

import os
import smtplib
import sys
from configparser import ConfigParser
def send_email(subject, body_text, emails):
"""
Send an email
"""
base_path = os.path.dirname(os.path.abspath(__file__))
config_path = os.path.join(base_path, "email.ini")
if os.path.exists(config_path):
cfg = ConfigParser()
cfg.read(config_path)
else:
print("Config not found! Exiting!")
sys.exit(1)
host = cfg.get("smtp", "server")
from_addr = cfg.get("smtp", "from_addr")
BODY = "\r\n".join((
"From: %s" % from_addr,
"To: %s" % ', '.join(emails),
"Subject: %s" % subject ,
"",
body_text
))
server = smtplib.SMTP(host)
server.sendmail(from_addr, emails, BODY)
server.quit()
if __name__ == "__main__":
emails = ["mike@someAddress.org", "someone@gmail.com"]
subject = "Test email from Python"
body_text = "Python rules them all!"
send_email(subject, body_text, emails)

Youll notice that in this example, we removed the to_addr parameter and added an emails parameter,
which is a list of email addresses. To make this work, we need to create a comma-separated string

Chapter 17 - The email / smtplib Module

109

in the To: portion of the BODY and also pass the email list to the sendmail method. Thus we do the
following to create a simple comma separated string: , .join(emails). Simple, huh?

Send email using the TO, CC and BCC lines


Now we just need to figure out how to send using the CC and BCC fields. Lets create a new version
of this code that supports that functionality!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33

import os
import smtplib
import sys
from configparser import ConfigParser
def send_email(subject, body_text, to_emails, cc_emails, bcc_emails):
"""
Send an email
"""
base_path = os.path.dirname(os.path.abspath(__file__))
config_path = os.path.join(base_path, "email.ini")
if os.path.exists(config_path):
cfg = ConfigParser()
cfg.read(config_path)
else:
print("Config not found! Exiting!")
sys.exit(1)
host = cfg.get("smtp", "server")
from_addr = cfg.get("smtp", "from_addr")
BODY = "\r\n".join((
"From: %s" % from_addr,
"To: %s" % ', '.join(to_emails),
"CC: %s" % ', '.join(cc_emails),
"BCC: %s" % ', '.join(bcc_emails),
"Subject: %s" % subject ,
"",
body_text
))
emails = to_emails + cc_emails + bcc_emails

Chapter 17 - The email / smtplib Module

34
35
36
37
38
39
40
41
42
43
44
45
46

110

server = smtplib.SMTP(host)
server.sendmail(from_addr, emails, BODY)
server.quit()
if __name__ == "__main__":
emails = ["mike@somewhere.org"]
cc_emails = ["someone@gmail.com"]
bcc_emails = ["schmuck@newtel.net"]
subject = "Test email from Python"
body_text = "Python rules them all!"
send_email(subject, body_text, emails, cc_emails, bcc_emails)

In this code, we pass in 3 lists, each with one email address a piece. We create the CC and BCC
fields exactly the same as before, but we also need to combine the 3 lists into one so we can pass the
combined list to the sendmail() method. There is some talk on forums like StackOverflow that some
email clients may handle the BCC field in odd ways that allow the recipient to see the BCC list via
the email headers. I am unable to confirm this behavior, but I do know that Gmail successfully strips
the BCC information from the email header.
Now were ready to move on to using Pythons email module!

Add an attachment / body using the email module


Now well take what we learned from the previous section and mix it together with the Python email
module so that we can send attachments. The email module makes adding attachments extremely
easy. Heres the code:
1
2
3
4
5
6
7
8
9
10
11
12
13

import os
import smtplib
import sys
from
from
from
from
from
from

configparser import ConfigParser


email import encoders
email.mime.text import MIMEText
email.mime.base import MIMEBase
email.mime.multipart import MIMEMultipart
email.utils import formatdate

#---------------------------------------------------------------------def send_email_with_attachment(subject, body_text, to_emails,

Chapter 17 - The email / smtplib Module

14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55

cc_emails, bcc_emails, file_to_attach):


"""
Send an email with an attachment
"""
base_path = os.path.dirname(os.path.abspath(__file__))
config_path = os.path.join(base_path, "email.ini")
header = 'Content-Disposition', 'attachment; filename="%s"' % file_to_attach
# get the config
if os.path.exists(config_path):
cfg = ConfigParser()
cfg.read(config_path)
else:
print("Config not found! Exiting!")
sys.exit(1)
# extract server and from_addr from config
host = cfg.get("smtp", "server")
from_addr = cfg.get("smtp", "from_addr")
# create the message
msg = MIMEMultipart()
msg["From"] = from_addr
msg["Subject"] = subject
msg["Date"] = formatdate(localtime=True)
if body_text:
msg.attach( MIMEText(body_text) )
msg["To"] = ', '.join(to_emails)
msg["cc"] = ', '.join(cc_emails)
attachment = MIMEBase('application', "octet-stream")
try:
with open(file_to_attach, "rb") as fh:
data = fh.read()
attachment.set_payload( data )
encoders.encode_base64(attachment)
attachment.add_header(*header)
msg.attach(attachment)
except IOError:
msg = "Error opening attachment file %s" % file_to_attach
print(msg)

111

Chapter 17 - The email / smtplib Module

56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73

112

sys.exit(1)
emails = to_emails + cc_emails
server = smtplib.SMTP(host)
server.sendmail(from_addr, emails, msg.as_string())
server.quit()
if __name__ == "__main__":
emails = ["mike@someAddress.org", "nedry@jp.net"]
cc_emails = ["someone@gmail.com"]
bcc_emails = ["anonymous@circe.org"]
subject = "Test email with attachment from Python"
body_text = "This email contains an attachment!"
path = "/path/to/some/file"
send_email_with_attachment(subject, body_text, emails,
cc_emails, bcc_emails, path)

Here we have renamed our function and added a new argument, file_to_attach. We also need to
add a header and create a MIMEMultipart object. The header could be created any time before
we add the attachment. We add elements to the MIMEMultipart object (msg) like we would keys
to a dictionary. Youll note that we have to use the email modules formatdate method to insert
the properly formatted date. To add the body of the message, we need to create an instance of
MIMEText. If youre paying attention, youll see that we didnt add the BCC information, but you
could easily do so by following the conventions in the code above. Next we add the attachment. We
wrap it in an exception handler and use the with statement to extract the file and place it in our
MIMEBase object. Finally we add it to the msg variable and we send it out. Notice that we have to
convert the msg to a string in the sendmail method.

Wrapping Up
Now you know how to send out emails with Python. For those of you that like mini projects,
you should go back and add additional error handling around the server.sendmail portion of the
code in case something odd happens during the process, such as an SMTPAuthenticationError or
SMTPConnectError. We could also beef up the error handling during the attachment of the file
to catch other errors. Finally, we may want to take those various lists of emails and create one
normalized list that has removed duplicates. This is especially important if we are reading a list of
email addresses from a file.
Also note that our from address is fake. We can spoof emails using Python and other programming
languages, but that is very bad etiquette and possibly illegal depending on where you live. You have
been warned! Use your knowledge wisely and enjoy Python for fun and profit!

Chapter 18 - The sqlite Module


SQLite is a self-contained, server-less, config-free transactional SQL database engine. Python gained
the sqlite3 module all the way back in version 2.5 which means that you can create SQLite database
with any current Python without downloading any additional dependencies. Mozilla uses SQLite
databases for its popular Firefox browser to store bookmarks and other various pieces of information.
In this chapter you will learn the following:

How to create a SQLite database


How to insert data into a table
How to edit the data
How to delete the data
Basic SQL queries

In other words, rather than covering bits and pieces of the sqlite3 module, well go through how to
actually use it.
If you want to inspect your database visually, you can use the SQLite Manager plugin for Firefox
(just Google for it) or if you like the command line, you can use SQLites command line shell.

How to Create a Database and INSERT Some Data


Creating a database in SQLite is really easy, but the process requires that you know a little SQL to
do it. Heres some code that will create a database to hold music albums:
1
2
3
4
5
6
7
8
9
10
11

import sqlite3
conn = sqlite3.connect("mydatabase.db") # or use :memory: to put it in RAM
cursor = conn.cursor()
# create a table
cursor.execute("""CREATE TABLE albums
(title text, artist text, release_date text,
publisher text, media_type text)
""")

Chapter 18 - The sqlite Module

114

First we have to import the sqlite3 module and create a connection to the database. You can pass it
a file path, file name or just use use the special string :memory: to create the database in memory.
In our case, we created it on disk in a file called mydatabase.db. Next we create a cursor object,
which allows you to interact with the database and add records, among other things. Here we use
SQL syntax to create a table named albums with 5 text fields: title, artist, release_date, publisher
and media_type. SQLite only supports five data types: null, integer, real, text and blob. Lets build
on this code and insert some data into our new table!
Note: If you run the CREATE TABLE command and the database already exists, you will
receive an error message.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

# insert some data


cursor.execute("""INSERT INTO albums
VALUES ('Glow', 'Andy Hunter', '7/24/2012',
'Xplore Records', 'MP3')"""
)
# save data to database
conn.commit()
# insert multiple records using the more secure "?" method
albums = [('Exodus', 'Andy Hunter', '7/9/2002', 'Sparrow Records', 'CD'),
('Until We Have Faces', 'Red', '2/1/2011', 'Essential Records', 'CD'),
('The End is Where We Begin', 'Thousand Foot Krutch', '4/17/2012', 'TF\
Kmusic', 'CD'),
('The Good Life', 'Trip Lee', '4/10/2012', 'Reach Records', 'CD')]
cursor.executemany("INSERT INTO albums VALUES (?,?,?,?,?)", albums)
conn.commit()

Here we use the INSERT INTO SQL command to insert a record into our database. Note that each
item had to have single quotes around it. This can get complicated when you need to insert strings
that include single quotes in them. Anyway, to save the record to the database, we have to commit it.
The next piece of code shows how to add multiple records at once by using the cursors executemany
method. Note that were using question marks (?) instead of string substitution (%s) to insert the
values. Using string substitution is NOT safe and should not be used as it can allow SQL injection
attacks to occur. The question mark method is much better and using SQLAlchemy is even better
because it does all the escaping for you so you wont have to mess with the annoyances of converting
embedded single quotes into something that SQLite will accept.

Updating and Deleting Records


Being able to update your database records is key to keeping your data accurate. If you cant update,
then your data will become out of date and pretty useless very quickly. Sometimes you will need to

Chapter 18 - The sqlite Module

115

delete rows from your data too. Well be covering both of those topics in this section. First, lets do
an update!
1
2
3
4
5
6
7
8
9
10
11
12

import sqlite3
conn = sqlite3.connect("mydatabase.db")
cursor = conn.cursor()
sql = """
UPDATE albums
SET artist = 'John Doe'
WHERE artist = 'Andy Hunter'
"""
cursor.execute(sql)
conn.commit()

Here we use SQLs UPDATE command to update out albums table. You can use SET to change a
field, so in this case we change the artist field to be John Doe in any record WHERE the artist field is
set to Andy Hunter. Wasnt that easy? Note that if you dont commit the changes, then your changes
wont be written out to the database. The DELETE command is almost as easy. Lets check that out!
1
2
3
4
5
6
7
8
9
10
11

import sqlite3
conn = sqlite3.connect("mydatabase.db")
cursor = conn.cursor()
sql = """
DELETE FROM albums
WHERE artist = 'John Doe'
"""
cursor.execute(sql)
conn.commit()

Deleting is even easier than updating. The SQL is only 2 lines! In this case, all we had to do was tell
SQLite which table to delete from (albums) and which records to delete using the WHERE clause.
Thus it looked for any records that had John Doe in its artist field and deleted it.

Basic SQLite Queries


Queries in SQLite are pretty much the same as what youd use for other databases, such as MySQL
or Postgres. You just use normal SQL syntax to run the queries and then have the cursor object
execute the SQL. Here are a few examples:

Chapter 18 - The sqlite Module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

116

import sqlite3
conn = sqlite3.connect("mydatabase.db")
#conn.row_factory = sqlite3.Row
cursor = conn.cursor()
sql = "SELECT * FROM albums WHERE artist=?"
cursor.execute(sql, [("Red")])
print(cursor.fetchall()) # or use fetchone()
print("\nHere's a listing of all the records in the table:\n")
for row in cursor.execute("SELECT rowid, * FROM albums ORDER BY artist"):
print(row)
print("\nResults from a LIKE query:\n")
sql = """
SELECT * FROM albums
WHERE title LIKE 'The%'"""
cursor.execute(sql)
print(cursor.fetchall())

The first query we execute is a SELECT * which means that we want to select all the records that
match the artist name we pass in, which in this case is Red. Next we execute the SQL and use
fetchall() to return all the results. You can also use fetchone() to grab the first result. Youll also
notice that theres a commented out section related to a mysterious row_factory. If you un-comment
that line, the results will be returned as Row objects that are kind of like Python dictionaries and
give you access to the rows fields just like a dictionary. However, you cannot do item assignment
with a Row object.
The second query is much like the first, but it returns every record in the database and orders the
results by the artist name in ascending order. This also demonstrates how we can loop over the
results. The last query shows how to use SQLs LIKE command to search for partial phrases. In
this case, we do a search of the entire table for titles that start with The. The percent sign (%) is a
wildcard operator.

Wrapping Up
Now you know how to use Python to create a SQLite database. You can also create, update and
delete records as well as run queries against your database.

Chapter 19 - The subprocess Module


The subprocess module gives the developer the ability to start processes or programs from Python.
In other words, you can start applications and pass arguments to them using the subprocess module.
The subprocess module was added way back in Python 2.4 to replace the os modules set of os.popen,
os.spawn and os.system calls as well as replace popen2 and the old commands module. We will be
looking at the following aspects of the subprocess module:
the call function
the Popen class
how to communicate with a spawned process
Lets get started!

The call function


The subprocess module provides a function named call. This function allows you to call another
program, wait for the command to complete and then return the return code. It accepts one or
more arguments as well as the following keyword arguments (with their defaults): stdin=None,
stdout=None, stderr=None, shell=False.
Lets look at a simple example:
1
2
3

>>> import subprocess


>>> subprocess.call("notepad.exe")
0

If you run this on a Windows machine, you should see Notepad open up. You will notice that IDLE
waits for you to close Notepad and then it returns a code zero (0). This means that it completed
successfully. If you receive anything except for a zero, then it usually means you have had some
kind of error.
Normally when you call this function, you would want to assign the resulting return code to a
variable so you can check to see if it was the result you expected. Lets do that:

Chapter 19 - The subprocess Module

1
2
3
4
5
6

118

>>> code = subprocess.call("notepad.exe")


>>> if code == 0:
print("Success!")
else:
print("Error!")
Success!

If you run this code, you will see that it prints out Success! to the screen. The call method also
accepts arguments to be passed to the program that youre executing. Lets see how that works:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

>>> code = subprocess.call(["ping", "www.yahoo.com"])


Pinging ds-any-fp3-real.wa1.b.yahoo.com [98.139.180.149] with 32 bytes of data:
Reply from 98.139.180.149: bytes=32 time=66ms TTL=45
Reply from 98.139.180.149: bytes=32 time=81ms TTL=45
Reply from 98.139.180.149: bytes=32 time=81ms TTL=45
Reply from 98.139.180.149: bytes=32 time=69ms TTL=45
Ping statistics for 98.139.180.149:
Packets: Sent = 4, Received = 4, Lost = 0 (0% loss),
Approximate round trip times in milli-seconds:
Minimum = 66ms, Maximum = 81ms, Average = 74ms
>>> code
0

You will notice that in this example we are passing a list of arguments. The first item in the list is
the program we want to call. Anything else in the list are arguments that we want to pass to that
program. So in this example, we are executing a ping against Yahoos website. You will notice that
the return code was zero, so everything completed successfully.
You can also execute the program using the operating systems shell. This does add a level
of abstraction to the process and it raises the possibility of security issues. Here is the Python
documentations official warning on the matter:
Executing shell commands that incorporate unsanitized input from an untrusted source makes
a program vulnerable to shell injection, a serious security flaw which can result in arbitrary
command execution. For this reason, the use of shell=True is strongly discouraged in cases
where the command string is constructed from external input.
The usual recommendation is not to use it if an outside process or person can modify the calls
arguments. If youre hard-coding something yourself, then it doesnt matter as much.

Chapter 19 - The subprocess Module

119

The Popen Class


The Popen class executes a child program in a new process. Unlike the call method, it does not wait
for the called process to end unless you tell it to using by using the wait method. Lets try running
Notepad through Popen and see if we can detect a difference:
1
2
3

>>> program = "notepad.exe"


>>> subprocess.Popen(program)
<subprocess.Popen object at 0x01EE0430>

Here we create a variable called program and assign it the value of notepad.exe. Then we pass it to
the Popen class. When you run this, you will see that it immediately returns the subprocess.Popen
object and the application that was called executes. Lets make Popen wait for the program to finish:
1
2
3
4
5

>>>
>>>
>>>
>>>
0

program = "notepad.exe"
process = subprocess.Popen(program)
code = process.wait()
print(code)

When you do this in IDLE, Notepad will pop up and may be in front of your IDLE session. Just move
Notepad out of the way, but do not close it! You need to tell your process to wait for you wont be
able to get the return code. Once youve typed that line, close Notepad and print the code out. Or
you could just put all this code into a saved Python file and run that.
Note that using the wait method can cause the child process to deadlock when using the stdout/stderr=PIPE commands when the process generates enough output to block the pipe. You can use the
communicate method to alleviate this situation. Well be looking at that method in the next section.
Now lets try running Popen using multiple arguments:
1
2

>>> subprocess.Popen(["ls", "-l"])


<subprocess.Popen object at 0xb7451001>

If you run this code in Linux, youll see it print out the Popen object message and then a listing of
the permissions and contents of whatever folder you ran this in. You can use the shell argument
with Popen too, but the same caveats apply to Popen as they did to the call method.

Learning to Communicate
There are several ways to communicate with the process you have invoked. Were just going to focus
on how to use the subprocess modules communicate method. Lets take a look:

Chapter 19 - The subprocess Module

1
2
3
4
5
6

120

args = ["ping", "www.yahoo.com"]


process = subprocess.Popen(args,
stdout=subprocess.PIPE)
data = process.communicate()
print(data)

In this code example, we create an args variable to hold our list of arguments. Then we redirect
standard out (stdout) to our subprocess so we can communicate with it. The communicate method
itself allows us to communicate with the process we just spawned. We can actually pass input to
the process using this method. But in this example, we just use communicate to read from standard
out. You will notice when you run this code that communicate will wait for the process to finish
and then returns a two-element tuple that contains what was in stdout and stderr. Here is the result
from my run:
1
2
3
4
5
6
7
8
9
10

('Pinging ds-any-fp3-real.wa1.b.yahoo.com [98.139.180.149] with 32 bytes of data:


Reply from 98.139.180.149: bytes=32 time=139ms TTL=45
Reply from 98.139.180.149: bytes=32 time=162ms TTL=45
Reply from 98.139.180.149: bytes=32 time=164ms TTL=45
Reply from 98.139.180.149: bytes=32 time=110ms TTL=45
Ping statistics for 98.139.180.149:
Packets: Sent = 4, Received = 4, Lost = 0 (0% loss),
Approximate round trip times in milli-seconds:
Minimum = 110ms, Maximum = 164ms, Average = 143ms
', None)

Thats kind of ugly. Lets make it print out the result in a more readable format, shall we?
1
2
3
4
5
6
7
8

import subprocess
args = ["ping", "www.yahoo.com"]
process = subprocess.Popen(args, stdout=subprocess.PIPE)
data = process.communicate()
for line in data:
print(line)

If you run this code, you should see something like the following printed to your screen:

Chapter 19 - The subprocess Module

1
2
3
4
5
6
7
8
9
10
11
12

121

Pinging ds-any-fp3-real.wa1.b.yahoo.com [98.139.180.149] with 32 bytes of data:


Reply from 98.139.180.149: bytes=32 time=67ms TTL=45
Reply from 98.139.180.149: bytes=32 time=68ms TTL=45
Reply from 98.139.180.149: bytes=32 time=70ms TTL=45
Reply from 98.139.180.149: bytes=32 time=69ms TTL=45
Ping statistics for 98.139.180.149:
Packets: Sent = 4, Received = 4, Lost = 0 (0% loss),
Approximate round trip times in milli-seconds:
Minimum = 67ms, Maximum = 70ms, Average = 68ms
None

That last line that says None is the result of stderr, which means that there were no errors.

Wrapping Up
At this point, you have the knowledge to use the subprocess module effectively. You can open a
process in two different ways, you know how to wait for a return code, and you know how to
communicate with the child process that you created.
In the next chapter, we will be looking at the sys module.

Chapter 20 - The sys Module


The sys module provides system specific parameters and functions. We will be narrowing our study
down to the following:

sys.argv
sys.executable
sys.exit
sys.modules
sys.path
sys.platform
sys.stdin/stdout/stderr

sys.argv
The value of sys.argv is a Python list of command line arguments that were passed to the Python
script. The first argument, argv[0] is the name of the Python script itself. Depending on the platform
that you are running on, the first argument may contain the full path to the script or just the file
name. You should study the documentation for additional details.
Lets try out a few examples to familiarize ourselves with this little tool:
1
2
3

>>> import sys


>>> sys.argv
['']

If you run this in the interpreter, you will receive a list with an empty string. Lets create a file named
sysargv.py with the following contents:
1
2
3
4

# sysargv.py
import sys
print(sys.argv)

123

Chapter 20 - The sys Module

Now run the code in IDLE. You should see it print out a list with a single element that contains the
path to your script in it. Lets try passing the script some arguments. Open up a terminal / console
screen and change directories (use the cd command) to where your script is. Then run something
like this:

image

You will notice that it outputs the following to the screen:


1

['sysargv.py', '-v', 'somefile.py']

The first argument is the name of the script we wrote. The next two arguments in the list are the
ones we passed to our script on the command line.

sys.executable
The value of sys.executable is the absolute path to the Python interpreter. This is useful when you
are using someone elses machine and need to know where Python is installed. On some systems,
this command will fail and it will return an empty string or None. Heres how to use it:
1
2
3

>>> import sys


>>> sys.executable
'C:\\Python27\\pythonw.exe'

sys.exit
The sys.exit() function allows the developer to exit from Python. The exit function takes an optional
argument, typically an integer, that gives an exit status. Zero is considered a successful termination.
Be sure to check if your operating system has any special meanings for its exit statuses so that you
can follow them in your own application. Note that when you call exit, it will raise the SystemExit
exception, which allows cleanup functions to work in the finally clauses of try / except blocks.
Lets take a look at how to call this:

124

Chapter 20 - The sys Module

1
2
3
4
5
6
7

>>> import sys


>>> sys.exit(0)
Traceback (most recent call last):
File "<pyshell#5>", line 1, in <module>
sys.exit(0)
SystemExit: 0

When you run this code in IDLE, you will see the SystemExit error raised. Lets create a couple of
scripts to test this out. First youll want to create a master script, a program that will call another
Python script. Lets name it call_exit.py. Put the following code into it:
1
2
3
4
5

# call_exit.py
import subprocess
code = subprocess.call(["python.exe", "exit.py"])
print(code)

Now create another Python script called exit.py and save it in the same folder. Put the following
code into it:
1
2
3

import sys
sys.exit(0)

Now lets try running this code:

image

In the screenshot above, you can see that the exit script we wrote returned a zero, so it ran
successfully. You have also learned how to call another Python script from within Python!

Chapter 20 - The sys Module

125

sys.path
The sys modules path value is a list of strings that specifies the search path for modules. Basically
this tells Python what locations to look in when it tries to import a module. According to the Python
documentation, sys.path is initialized from an environment variable called PYTHONPATH, plus an
installation-dependent default. Lets give it a try:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

>>> import sys


>>> print(sys.path)
['',
'C:\\Python27\\Lib\\idlelib',
'C:\\Python27\\lib\\site-packages\\setuptools-0.9.5-py2.7.egg',
'C:\\Python27\\lib\\site-packages\\pip-1.3.1-py2.7.egg',
'C:\\Python27\\lib\\site-packages\\sphinx-1.2b3-py2.7.egg',
'C:\\Python27\\lib\\site-packages\\docutils-0.11-py2.7.egg',
'C:\\Python27\\lib\\site-packages\\pygments-1.6-py2.7.egg',
'C:\\Windows\\system32\\python27.zip', '
C:\\Python27\\DLLs',
'C:\\Python27\\lib',
'C:\\Python27\\lib\\plat-win',
'C:\\Python27\\lib\\lib-tk',
'C:\\Python27',
'C:\\Python27\\lib\\site-packages',
'C:\\Python27\\lib\\site-packages\\PIL',
'C:\\Python27\\lib\\site-packages\\wx-2.9.4-msw']

This can be very useful for debugging why a module isnt getting imported. You can also modify
the path. Because its a list, we can add or delete paths from it. Heres how to add a path:
1

>>> sys.path.append("/path/to/my/module")

Ill leave deleting a path as an exercise for the reader.

sys.platform
The sys.platform value is a platform identifier. You can use this to append platform specific modules
to sys.path, import different modules depending on platform or run different pieces of code. Lets
take a look:

Chapter 20 - The sys Module

1
2
3

126

>>> import sys


>>> sys.platform
'win32'

This tells us that Python is running on a Windows machine. Heres an example of how we might
use this information:
1
2
3
4
5
6
7
8

>>> os = sys.platform
>>> if os == "win32":
# use Window-related code here
import _winreg
elif os.startswith('linux'):
# do something Linux specific
import subprocess
subprocess.Popen(["ls, -l"])

The code above shows how we might check to see if were using a particular operating system. If
were on Windows, well get some information from the Windows Registry using a Python module
called _winreg. If were on Linux, we might execute the ls command to get information about the
directory were in.

sys.stdin / stdout / stderr


The stdin, stdout and stderr map to file objects that correspond to the interpreters standard input,
output and error streams, respectively. stdin is used for all input given to the interpreter except
for scripts whereas stdout is used for the output of print and expression statements. The primary
reason I mention this is that you will sometimes need to redirect stdout or stderr or both to a file,
such as a log or to some kind of display in a custom GUI you have created. You could also redirect
stdin, but I have rarely seen this done.

Wrapping Up
There are many other values and methods in the sys module. Be sure to look it up in the Python
documentation, section 27.1. You have learned a lot in this chapter. You now know how to exit a
Python program, how to get platform information, working with arguments passed on the command
line and much more. In the next chapter, well be learning about Python threads!

Chapter 21 - The threading module


Python has a number of different concurrency constructs such as threading, queues and multiprocessing. The threading module used to be the primary way of accomplishing concurrency. A few
years ago, the multiprocessing module was added to the Python suite of standard libraries. This
chapter will be focused on how to use threads and queues.

Using Threads
We will start with a simple example that just demonstrates how threads work. We will sub-class the
Thread class and make it print out its name to stdout. Lets get coding!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

import random
import time
from threading import Thread
class MyThread(Thread):
"""
A threading example
"""
def __init__(self, name):
"""Initialize the thread"""
Thread.__init__(self)
self.name = name
def run(self):
"""Run the thread"""
amount = random.randint(3, 15)
time.sleep(amount)
msg = "%s is running" % self.name
print(msg)
def create_threads():
"""
Create a group of threads
"""

Chapter 21 - The threading module

27
28
29
30
31
32
33

128

for i in range(5):
name = "Thread #%s" % (i+1)
my_thread = MyThread(name)
my_thread.start()
if __name__ == "__main__":
create_threads()

In the code above, we import Pythons random module, the time module and we import the Thread
class from the threading module. Next we sub-class Thread and make override its __init__ method
to accept an argument we label name. To start a thread, you have to call its start() method. When
you start a thread, it will automatically call the threads run method. We have overridden the
threads run method to make it choose a random amount of time to sleep. The random.randint
example here will cause Python to randomly choose a number from 3-15. Then we make the thread
sleep the number of seconds that we just randomly chose to simulate it actually doing something.
Finally we print out the name of the thread to let the user know that the thread has finished.
The create_threads function will create 5 threads, giving each of them a unique name. If you run
this code, you should see something like this:
1
2
3
4
5

Thread
Thread
Thread
Thread
Thread

#2
#3
#1
#4
#5

is
is
is
is
is

running
running
running
running
running

The order of the output will be different each time. Try running the code a few times to see the order
change. Now lets write something a little more practical!

Writing a Threaded Downloader


The previous example wasnt very useful other than as a tool to explain how threads work. So in this
example, we will create a Thread class that can download files from the internet. The U.S. Internal
Revenue Service has lots of PDF forms that it has its citizens use for taxes. We will use this free
resource for our demo. Heres the code:

Chapter 21 - The threading module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

# Python 2 version
import os
import urllib2
from threading import Thread
class DownloadThread(Thread):
"""
A threading example that can download a file
"""
def __init__(self, url, name):
"""Initialize the thread"""
Thread.__init__(self)
self.name = name
self.url = url
def run(self):
"""Run the thread"""
handle = urllib2.urlopen(self.url)
fname = os.path.basename(self.url)
with open(fname, "wb") as f_handler:
while True:
chunk = handle.read(1024)
if not chunk:
break
f_handler.write(chunk)
msg = "%s has finished downloading %s!" % (self.name,
self.url)
print(msg)
def main(urls):
"""
Run the program
"""
for item, url in enumerate(urls):
name = "Thread %s" % (item+1)
thread = DownloadThread(url, name)
thread.start()
if __name__ == "__main__":

129

Chapter 21 - The threading module

43
44
45
46
47
48

130

urls = ["http://www.irs.gov/pub/irs-pdf/f1040.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040a.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040ez.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040es.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040sb.pdf"]
main(urls)

This is basically a complete rewrite of the first script. In this one we import the os and urllib2 modules
as well as the threading module. We will be using urllib2 to do the actual downloading inside the
thread class. The os module is used to extract the name of the file were downloading so we can use
it to create a file with the same name on our machine. In the DownloadThread class, we set up the
__init__ to accept a url and a name for the thread. In the run method, we open up the url, extract
the filename and then use that filename for naming / creating the file on disk. Then we use a while
loop to download the file a kilobyte at a time and write it to disk. Once the file is finished saving,
we print out the name of the thread and which url has finished downloading.
The Python 3 version of the code is slightly different. You have to import urllib instead of urllib2 and
use urllib.request.urlopen instead of urllib2.urlopen. Heres the code so you can see the difference:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

# Python 3 version
import os
import urllib.request
from threading import Thread
class DownloadThread(Thread):
"""
A threading example that can download a file
"""
def __init__(self, url, name):
"""Initialize the thread"""
Thread.__init__(self)
self.name = name
self.url = url
def run(self):
"""Run the thread"""
handle = urllib.request.urlopen(self.url)
fname = os.path.basename(self.url)
with open(fname, "wb") as f_handler:
while True:

Chapter 21 - The threading module

25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48

131

chunk = handle.read(1024)
if not chunk:
break
f_handler.write(chunk)
msg = "%s has finished downloading %s!" % (self.name,
self.url)
print(msg)
def main(urls):
"""
Run the program
"""
for item, url in enumerate(urls):
name = "Thread %s" % (item+1)
thread = DownloadThread(url, name)
thread.start()
if __name__ == "__main__":
urls = ["http://www.irs.gov/pub/irs-pdf/f1040.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040a.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040ez.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040es.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040sb.pdf"]
main(urls)

Using Queues
A Queue can be used for first-in-first-out (FIFO) or last-in-last-out (LILO) stack-like implementations if you just use them directly. In this section, were going to mix threads in and create a simple
file downloader script to demonstrate how Queues work for cases where we want concurrency.
To help explain how Queues work, we will rewrite the downloading script from the previous section
to use Queues. Lets get started!

Chapter 21 - The threading module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

import os
import threading
import urllib.request
from queue import Queue
class Downloader(threading.Thread):
"""Threaded File Downloader"""
def __init__(self, queue):
"""Initialize the thread"""
threading.Thread.__init__(self)
self.queue = queue
def run(self):
"""Run the thread"""
while True:
# gets the url from the queue
url = self.queue.get()
# download the file
self.download_file(url)
# send a signal to the queue that the job is done
self.queue.task_done()
def download_file(self, url):
"""Download the file"""
handle = urllib.request.urlopen(url)
fname = os.path.basename(url)
with open(fname, "wb") as f:
while True:
chunk = handle.read(1024)
if not chunk: break
f.write(chunk)
def main(urls):
"""
Run the program
"""
queue = Queue()

132

Chapter 21 - The threading module

43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62

133

# create a thread pool and give them a queue


for i in range(5):
t = Downloader(queue)
t.setDaemon(True)
t.start()
# give the queue some data
for url in urls:
queue.put(url)
# wait for the queue to finish
queue.join()
if __name__ == "__main__":
urls = ["http://www.irs.gov/pub/irs-pdf/f1040.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040a.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040ez.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040es.pdf",
"http://www.irs.gov/pub/irs-pdf/f1040sb.pdf"]
main(urls)

Lets break this down a bit. First of all, we need to look at the main function definition to see how this
all flows. Here we see that it accepts a list of urls. The main function then creates a queue instance
that it passes to 5 daemonized threads. The main difference between daemonized and non-daemon
threads is that you have to keep track of non-daemon threads and close them yourself whereas with
a daemon thread you basically just set them and forget them and when your app closes, they close
too. Next we load up the queue (using its put method) with the urls we passed in.
Finally we tell the queue to wait for the threads to do their processing via the join method. In the
download class, we have the line self.queue.get() which blocks until the queue has something to
return. That means the threads just sit idly waiting to pick something up. It also means that for a
thread to get something from the queue, it must call the queues get method. Thus as we add or put
items in the queue, the thread pool will pick up or get items and process them. This is also known as
dequeing. Once all the items in the queue are processed, the script ends and exits. On my machine,
it downloads all 5 documents in under a second.

Wrapping Up
Now you know how to use threads and queues both in theory and in a practical way. Threads are
especially useful when you are creating a user interface and you want to keep your interface usable.
Without threads, the user interface would become unresponsive and would appear to hang while
you did a large file download or a big query against a database. To keep that from happening, you

Chapter 21 - The threading module

134

do the long running processes in threads and then communicate back to your interface when you
are done.

Chapter 22 - Working with Dates and


Time
Python gives the developer several tools for working with dates and time. In this chapter, we will be
looking at the datetime and time modules. We will study how they work and some common uses
for them. Lets start with the datetime module!

The datetime Module


We will be learning about the following classes from the datetime module:
datetime.date
datetime.timedelta
datetime.datetime
These will cover the majority of instances where youll need to use date and datetime object in
Python. There is also a tzinfo class for working with time zones that we wont be covering. Feel free
to take a look at the Python documentation for more information on that class.

datetime.date
Python can represent dates several different ways. Were going to look at the datetime.date format
first as it happens to be one of the simpler date objects.
1
2
3
4
5
6

>>> datetime.date(2012, 13, 14)


Traceback (most recent call last):
File "<string>", line 1, in <fragment>
builtins.ValueError: month must be in 1..12
>>> datetime.date(2012, 12, 14)
datetime.date(2012, 12, 14)

This code shows how to create a simple date object. The date class accepts three arguments: the year,
the month and the day. If you pass it an invalid value, you will see a ValueError, like the one above.
Otherwise you will see a datetime.date object returned. Lets take a look at another example:

Chapter 22 - Working with Dates and Time

1
2
3
4
5
6
7
8

136

>>> import datetime


>>> d = datetime.date(2012, 12, 14)
>>> d.year
2012
>>> d.day
14
>>> d.month
12

Here we assign the date object to the variable d. Now we can access the various date components
by name, such as d.year or d.month. Now lets find out what day it is:
1
2

>>> datetime.date.today()
datetime.date(2014, 3, 5)

This can be helpful whenever you need to record what day it is. Or perhaps you need to do a datebased calculation based on today. Its a handy little convenience method though.

datetime.datetime
A datetime.datetime object contains all the information from a datetime.date plus a datetime.time
object. Lets create a couple of examples so we can better understand the difference between this
object and the datetime.date object.
1
2
3
4
5
6
7
8
9
10
11

>>> datetime.datetime(2014, 3, 5)
datetime.datetime(2014, 3, 5, 0, 0)
>>> datetime.datetime(2014, 3, 5, 12, 30, 10)
datetime.datetime(2014, 3, 5, 12, 30, 10)
>>> d = datetime.datetime(2014, 3, 5, 12, 30, 10)
>>> d.year
2014
>>> d.second
10
>>> d.hour
12

Here we can see that datetime.datetime accepts several additional arguments: year, month, day,
hour, minute and second. It also allows you to specify microsecond and timezone information too.
When you work with databases, you will find yourself using these types of objects a lot. Most of
the time, you will need to convert from the Python date or datetime format to the SQL datetime
or timestamp format. You can find out what today is with datetime.datetime using two different
methods:

Chapter 22 - Working with Dates and Time

1
2
3
4

137

>>> datetime.datetime.today()
datetime.datetime(2014, 3, 5, 17, 56, 10, 737000)
>>> datetime.datetime.now()
datetime.datetime(2014, 3, 5, 17, 56, 15, 418000)

The datetime module has another method that you should be aware of called strftime. This method
allows the developer to create a string that represents the time in a more human readable format.
Theres an entire table of formatting options that you should go read in the Python documentation,
section 8.1.7. Were going to look at a couple of examples to show you the power of this method:
1
2
3
4
5
6
7

>>> datetime.datetime.today().strftime("%Y%m%d")
'20140305'
>>> today = datetime.datetime.today()
>>> today.strftime("%m/%d/%Y")
'03/05/2014'
>>> today.strftime("%Y-%m-%d-%H.%M.%S")
'2014-03-05-17.59.53'

The first example is kind of a hack. It shows how to convert todays datetime object into a string that
follows the YYYYMMDD (year, month, day) format. The second example is better. Here we assign
todays datetime object to a variable called today and then try out two different string formatting
operations. The first one adds forward slashes between the datetime elements and also rearranges
it so that it becomes month, day, year. The last example creates a timestamp of sorts that follows
a fairly typical format: YYYY-MM-DD.HH.MM.SS. If you want to go to a two-digit year, you can
swap out the %Y for %y.

datetime.timedelta
The datetime.timedelta object represents a time duration. In other words, it is the difference
between two dates or times. Lets take a look at a simple example:
1
2
3
4
5
6
7
8
9
10
11

>>> now = datetime.datetime.now()


>>> now
datetime.datetime(2014, 3, 5, 18, 13, 51, 230000)
>>> then = datetime.datetime(2014, 2, 26)
>>> delta = now - then
>>> type(delta)
<type 'datetime.timedelta'>
>>> delta.days
7
>>> delta.seconds
65631

Chapter 22 - Working with Dates and Time

138

We create two datetime objects here. One for today and one for a week ago. Then we take the
difference between them. This returns a timedelta object which we can then use to find out the
number of days or seconds between the two dates. If you need to know the number of hours or
minutes between the two, youll have to use some math to figure it out. Heres one way to do it:
1
2
3
4
5
6
7

>>> seconds = delta.total_seconds()


>>> hours = seconds // 3600
>>> hours
186.0
>>> minutes = (seconds % 3600) // 60
>>> minutes
13.0

What this tells us is that there are 186 hours and 13 minutes in a week. Note that we are using a
double-forward slash as our division operator. This is known as floor division.
Now were ready to move on and learn a bit about the time module!

The time Module


The time module provides the Python developer access to various time-related functions. The time
module is based around what it known as an epoch, the point when time starts. For Unix systems,
the epoch was in 1970. To find out what the epoch is on your system, try running the following:
1
2
3
4

>>> import time


>>> time.gmtime(0)
time.struct_time(tm_year=1970, tm_mon=1, tm_mday=1, tm_hour=0, tm_min=0, tm_sec=\
0, tm_wday=3, tm_yday=1, tm_isdst=0)

I ran this on Windows 7 and it too seems to think that time began in 1970. Anyway, in this section,
we will be studying the following time-related functions:

time.ctime
time.sleep
time.strftime
time.time

Lets get started!

time.ctime
The time.ctime function will convert a time in seconds since the epoch to a string representing
local time. If you dont pass it anything, then the current time is returned. Lets try out a couple of
examples:

Chapter 22 - Working with Dates and Time

1
2
3
4
5

139

>>> import time


>>> time.ctime()
'Thu Mar 06 07:28:48 2014'
>>> time.ctime(1384112639)
'Sun Nov 10 13:43:59 2013'

Here we show the results of calling ctime with nothing at all and with a fairly random number of
seconds since the epoch. I have seen sort of thing used when someone saves the date as seconds
since the epoch and then they want to convert it to something a human can understand. Its a bit
simpler to save a big integer (or long) to a database then to mess with formatting it from a datetime
object to whatever date object the database accepts. Of course, that also has the drawback that you
do need to convert the integer or float value back into a string.

time.sleep
The time.sleep function gives the developer the ability to suspend execution of your script a given
number of seconds. Its like adding a pause to your program. I have found this personally useful
when I need to wait a second for a file to finish closing or a database commit to finish committing.
Lets take a look at an example. Open a new window in IDLE and save the following code:
1
2
3
4
5

import time
for x in range(5):
time.sleep(2)
print("Slept for 2 seconds")

Now run the code in IDLE. You can do that by going to the Run menu and then choose the Run
module menu item. When you do so, you will see it print out the phrase Slept for 2 seconds five
times with a two second pause between each print. Its really that easy to use!

time.strftime
The time module has a strftime function that works in pretty much the same manner as the datetime
version. The difference is mainly in what it accepts for input: a tuple or a struct_time object, like
those that are returned when you call time.gmtime() or time.localtime(). Heres a little example:

Chapter 22 - Working with Dates and Time

1
2
3

140

>>> time.strftime("%Y-%m-%d-%H.%M.%S",
time.localtime())
'2014-03-06-20.35.56'

This code is quite similar to the timestamp code we created in the datetime portion of this chapter.
I think the datetime method is a little more intuitive in that you just create a datetime.datetime
object and then call its strftime method with the format you want. With the time module, you have
to pass the format plus a time tuple. Its really up to you to decide which one makes the most sense
to you.

time.time
The time.time function will return the time in seconds since the epoch as a floating point number.
Lets take a look:
1
2

>>> time.time()
1394199262.318

That was pretty simple. You could use this when you want to save the current time to a database
but you didnt want to bother converting it to the databases datetime method. You might also recall
that the ctime method accepts the time in seconds, so we could use time.time to get the number of
seconds to pass to ctime, like this:
1
2

>>> time.ctime(time.time())
'Fri Mar 07 07:36:38 2014'

If you do some digging in the documentation for the time module or if you just experiment with it
a bit, you will likely find a few other uses for this function.

Wrapping Up
At this point you should know how to work with dates and time using Pythons standard modules.
Python gives you a lot of power when it comes to working with dates. You will find these modules
helpful if you ever need to create an application that keeps track of appointments or that needs to
run on particular days. They are also useful when working with databases.

Chapter 23 - The xml module


Python has built-in XML parsing capabilities that you can access via its xml module. In this article,
we will be focusing on two of the xml modules sub-modules:
minidom
ElementTree
Well start with minidom simply because this used to be the de-facto method of XML parsing. Then
we will look at how to use ElementTree instead.

Working with minidom


To start out, well need some actual XML to parse. Take a look at the following short example of
XML:
1
2
3
4
5
6
7
8
9
10
11
12

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
</zAppointments>

This is fairly typical XML and actually pretty intuitive to read. There is some really nasty XML out
in the wild that you may have to work with. Anyway, save the XML code above with the following
name: appt.xml
Lets spend some time getting acquainted with how to parse this file using Pythons minidom
module. This is a fairly long piece of code, so prepare yourself.

Chapter 23 - The xml module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

import xml.dom.minidom
import urllib.request
class ApptParser(object):
def __init__(self, url, flag='url'):
self.list = []
self.appt_list = []
self.flag = flag
self.rem_value = 0
xml = self.getXml(url)
self.handleXml(xml)
def getXml(self, url):
try:
print(url)
f = urllib.request.urlopen(url)
except:
f = url
doc = xml.dom.minidom.parse(f)
node = doc.documentElement
if node.nodeType == xml.dom.Node.ELEMENT_NODE:
print('Element name: %s' % node.nodeName)
for (name, value) in node.attributes.items():
print('
Attr -- Name: %s Value: %s' % (name, value))
return node
def handleXml(self, xml):
rem = xml.getElementsByTagName('zAppointments')
appointments = xml.getElementsByTagName("appointment")
self.handleAppts(appointments)
def getElement(self, element):
return self.getText(element.childNodes)
def handleAppts(self, appts):
for appt in appts:
self.handleAppt(appt)
self.list = []

142

143

Chapter 23 - The xml module

43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78

def handleAppt(self, appt):


begin
= self.getElement(appt.getElementsByTagName("begin")[0])
duration = self.getElement(appt.getElementsByTagName("duration")[0])
subject
= self.getElement(appt.getElementsByTagName("subject")[0])
location = self.getElement(appt.getElementsByTagName("location")[0])
uid
= self.getElement(appt.getElementsByTagName("uid")[0])
self.list.append(begin)
self.list.append(duration)
self.list.append(subject)
self.list.append(location)
self.list.append(uid)
if self.flag == 'file':
try:
state

= self.getElement(appt.getElementsByTagName("state")[0\

])
self.list.append(state)
alarm
= self.getElement(appt.getElementsByTagName("alarmTime\
")[0])
self.list.append(alarm)
except Exception as e:
print(e)
self.appt_list.append(self.list)
def getText(self, nodelist):
rc = ""
for node in nodelist:
if node.nodeType == node.TEXT_NODE:
rc = rc + node.data
return rc
if __name__ == "__main__":
appt = ApptParser("appt.xml")
print(appt.appt_list)

This code is loosely based on an example from the Python documentation and I have to admit that
I think my mutation of it is a bit ugly. Lets break this code down a bit. The url parameter you see
in the ApptParser class can be either a url or a file. In the getXml method, we use an exception
handler to try and open the url. If it happens to raise an error, than we assume that the url is actually
a file path. Next we use minidoms parse method to parse the XML. Then we pull out a node from

Chapter 23 - The xml module

144

the XML. Well ignore the conditional as it isnt important to this discussion. Finally, we return the
node object.
Technically, the node is XML and we pass it on to the handleXml method. To grab all the
appointment instances in the XML, we do this:
1

xml.getElementsByTagName("appointment").

Then we pass that information to the handleAppts method. Thats a lot of passing information
around. It might be a good idea to refactor this code a bit to make it so that instead of passing
information around, it just set class variables and then called the next method without any
arguments. Ill leave this as an exercise for the reader. Anyway, all the handleAppts method does
is loop over each appointment and call the handleAppt method to pull some additional information
out of it, add the data to a list and add that list to another list. The idea was to end up with a list of
lists that held all the pertinent data regarding my appointments.
You will notice that the handleAppt method calls the getElement method which calls the getText
method. Technically, you could skip the call to getElement and just call getText directly. On the other
hand, you may need to add some additional processing to getElement to convert the text to some
other type before returning it back. For example, you may want to convert numbers to integers,
floats or decimal.Decimal objects.
Lets try one more example with minidom before we move on. We will use an XML example from
Microsofts MSDN website: http://msdn.microsoft.com/en-us/library/ms762271%28VS.85%29.aspx.
Save the following XML as example.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

<?xml version="1.0"?>
<catalog>
<book id="bk101">
<author>Gambardella, Matthew</author>
<title>XML Developer's Guide</title>
<genre>Computer</genre>
<price>44.95</price>
<publish_date>2000-10-01</publish_date>
<description>An in-depth look at creating applications
with XML.</description>
</book>
<book id="bk102">
<author>Ralls, Kim</author>
<title>Midnight Rain</title>
<genre>Fantasy</genre>
<price>5.95</price>
http://msdn.microsoft.com/en-us/library/ms762271%28VS.85%29.aspx

Chapter 23 - The xml module

17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32

145

<publish_date>2000-12-16</publish_date>
<description>A former architect battles corporate zombies,
an evil sorceress, and her own childhood to become queen
of the world.</description>
</book>
<book id="bk103">
<author>Corets, Eva</author>
<title>Maeve Ascendant</title>
<genre>Fantasy</genre>
<price>5.95</price>
<publish_date>2000-11-17</publish_date>
<description>After the collapse of a nanotechnology
society in England, the young survivors lay the
foundation for a new society.</description>
</book>
</catalog>

For this example, well just parse the XML, extract the book titles and print them to stdout. Heres
the code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

import xml.dom.minidom as minidom


def getTitles(xml):
"""
Print out all titles found in xml
"""
doc = minidom.parse(xml)
node = doc.documentElement
books = doc.getElementsByTagName("book")
titles = []
for book in books:
titleObj = book.getElementsByTagName("title")[0]
titles.append(titleObj)
for title in titles:
nodes = title.childNodes
for node in nodes:
if node.nodeType == node.TEXT_NODE:
print(node.data)
if __name__ == "__main__":

Chapter 23 - The xml module

23
24

146

document = 'example.xml'
getTitles(document)

This code is just one short function that accepts one argument, the XML file. We import the minidom
module and give it the same name to make it easier to reference. Then we parse the XML. The
first two lines in the function are pretty much the same as the previous example. We use the
getElementsByTagName method to grab the parts of the XML that we want, then iterate over
the result and extract the book titles from them. This actually extracts title objects, so we need to
iterate over that as well and pull out the plain text, which is why we use a nested for loop.
Now lets spend a little time trying out a different sub-module of the xml module named
ElementTree.

Parsing with ElementTree


In this section, you will learn how to create an XML file, edit XML and parse the XML with
ElementTree. For comparisons sake, well use the same XML we used in the previous section to
illustrate the differences between using minidom and ElementTree. Here is the original XML again:
1
2
3
4
5
6
7
8
9
10
11
12

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
</zAppointments>

Lets begin by learning how to create this piece of XML programmatically using Python!

How to Create XML with ElementTree


Creating XML with ElementTree is very simple. In this section, we will attempt to create the XML
above with Python. Heres the code:

Chapter 23 - The xml module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35

import xml.etree.ElementTree as xml


def createXML(filename):
"""
Create an example XML file
"""
root = xml.Element("zAppointments")
appt = xml.Element("appointment")
root.append(appt)
# add appointment children
begin = xml.SubElement(appt, "begin")
begin.text = "1181251680"
uid = xml.SubElement(appt, "uid")
uid.text = "040000008200E000"
alarmTime = xml.SubElement(appt, "alarmTime")
alarmTime.text = "1181572063"
state = xml.SubElement(appt, "state")
location = xml.SubElement(appt, "location")
duration = xml.SubElement(appt, "duration")
duration.text = "1800"
subject = xml.SubElement(appt, "subject")
tree = xml.ElementTree(root)
with open(filename, "w") as fh:
tree.write(fh)
if __name__ == "__main__":
createXML("appt.xml")

If you run this code, you should get something like the following (probably all on one line):

147

Chapter 23 - The xml module

1
2
3
4
5
6
7
8
9
10
11

148

<zAppointments>
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state />
<location />
<duration>1800</duration>
<subject />
</appointment>
</zAppointments>

This is pretty close to the original and is certainly valid XML. While its not quite the same, its close
enough. Lets take a moment to review the code and make sure we understand it. First we create
the root element by using ElementTrees Element function. Then we create an appointment element
and append it to the root. Next we create SubElements by passing the appointment Element object
(appt) to SubElement along with a name, like begin. Then for each SubElement, we set its text
property to give it a value. At the end of the script, we create an ElementTree and use it to write the
XML out to a file.
Now were ready to learn how to edit the file!

How to Edit XML with ElementTree


Editing XML with ElementTree is also easy. To make things a little more interesting though, well
add another appointment block to the XML:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
<appointment>
<begin>1181253977</begin>
<uid>sdlkjlkadhdakhdfd</uid>
<alarmTime>1181588888</alarmTime>

Chapter 23 - The xml module

16
17
18
19
20
21

149

<state>TX</state>
<location>Dallas</location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
</zAppointments>

Now lets write some code to change each of the begin tags values from seconds since the epoch to
something a little more readable. Well use Pythons time module to facilitate this:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

import time
import xml.etree.cElementTree as ET
def editXML(filename):
"""
Edit an example XML file
"""
tree = ET.ElementTree(file=filename)
root = tree.getroot()
for begin_time in root.iter("begin"):
begin_time.text = time.ctime(int(begin_time.text))
tree = ET.ElementTree(root)
with open("updated.xml", "w") as f:
tree.write(f)
if __name__ == "__main__":
editXML("original_appt.xml")

Here we create an ElementTree object (tree) and we extract the root from it. Then we use
ElementTrees iter() method to find all the tags that are labeled begin. Note that the iter() method
was added in Python 2.7. In our for loop, we set each items text property to a more human readable
time format via time.ctime(). Youll note that we had to convert the string to an integer when passing
it to ctime. The output should look something like the following:

Chapter 23 - The xml module

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

150

<zAppointments reminder="15">
<appointment>
<begin>Thu Jun 07 16:28:00 2007</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state />
<location />
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
<appointment>
<begin>Thu Jun 07 17:06:17 2007</begin>
<uid>sdlkjlkadhdakhdfd</uid>
<alarmTime>1181588888</alarmTime>
<state>TX</state>
<location>Dallas</location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
</zAppointments>

You can also use ElementTrees find() or findall() methods to search for specific tags in your XML.
The find() method will just find the first instance whereas the findall() will find all the tags with the
specified label. These are helpful for editing purposes or for parsing, which is our next topic!

How to Parse XML with ElementTree


Now we get to learn how to do some basic parsing with ElementTree. First well read through the
code and then well go through bit by bit so we can understand it. Note that this code is based around
the original example, but it should work on the second one as well.
1
2
3
4
5
6
7
8
9
10

import xml.etree.cElementTree as ET
def parseXML(xml_file):
"""
Parse XML with ElementTree
"""
tree = ET.ElementTree(file=xml_file)
print(tree.getroot())
root = tree.getroot()
print("tag=%s, attrib=%s" % (root.tag, root.attrib))

Chapter 23 - The xml module

11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37

151

for child in root:


print(child.tag, child.attrib)
if child.tag == "appointment":
for step_child in child:
print(step_child.tag)
# iterate over the entire tree
print("-" * 40)
print("Iterating using a tree iterator")
print("-" * 40)
iter_ = tree.getiterator()
for elem in iter_:
print(elem.tag)
# get the information via the children!
print("-" * 40)
print("Iterating using getchildren()")
print("-" * 40)
appointments = root.getchildren()
for appointment in appointments:
appt_children = appointment.getchildren()
for appt_child in appt_children:
print("%s=%s" % (appt_child.tag, appt_child.text))
if __name__ == "__main__":
parseXML("appt.xml")

You may have already noticed this, but in this example and the last one, weve been importing
cElementTree instead of the normal ElementTree. The main difference between the two is that
cElementTree is C-based instead of Python-based, so its much faster. Anyway, once again we create
an ElementTree object and extract the root from it. Youll note that we print out the root and the
roots tag and attributes. Next we show several ways of iterating over the tags. The first loop just
iterates over the XML child by child. This would only print out the top level child (appointment)
though, so we added an if statement to check for that child and iterate over its children too.
Next we grab an iterator from the tree object itself and iterate over it that way. You get the same
information, but without the extra steps in the first example. The third method uses the roots
getchildren() function. Here again we need an inner loop to grab all the children inside each
appointment tag. The last example uses the roots iter() method to just loop over any tags that
match the string begin.
As mentioned in the last section, you could also use find() or findall() to help you find specific tags
or sets of tags respectively. Also note that each Element object has a tag and a text property that

Chapter 23 - The xml module

152

you can use to acquire that exact information.

Wrapping Up
Now you know how to use minidom to parse XML. You have also learned how to use ElementTree
to create, edit and parse XML. There are other libraries outside of Python that provide additional
methods for working with XML too. Be sure you do some research to make sure youre using a tool
that you understand as this topic can get pretty confusing if the tool youre using is obtuse.

Part III - Intermediate Odds and Ends


In Part III, you will learn about some Python internals that many would consider intermediate-level
Python. You have graduated from the milk and youre ready for some meat! In this section, we will
take a look at the following topics:

Debugging
Decorators
The lambda statement
Profiling your code
Testing

The first chapter in this section will introduce you to Pythons debugging module, pdb and how
to use it to debug your code. The next chapter is all about decorators. You will learn about how to
create them and about a few of the decorators that are built into Python. For the third chapter, we
will be looking at the lambda statement, which basically creates a one-line anonymous function. Its
a little weird, but fun! The fourth chapter will cover how you profile your code. This discipline gives
you the ability to find possible bottlenecks in your code so that you know where to focus to optimize
your code. The final chapter in this section is about testing your code. In it you will discover how
to test your code using a couple of Pythons own built-in modules.
I think you will find this section very helpful in continuing your Python education. Lets jump right
in!

154

Part III - Intermediate Odds and Ends

image

Chapter 24 - The Python Debugger


Python comes with its own debugger module that is named pdb. This module provides an interactive
source code debugger for your Python programs. You can set breakpoints, step through your code,
inspect stack frames and more. We will look at the following aspects of the module:
How to start the debugger
Stepping through your code
Setting breakpoints
Lets start by creating a quick piece of code to attempt debugging with. Heres a silly example:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

# debug_test.py
def doubler(a):
""""""
result = a*2
print(result)
return result
def main():
""""""
for i in range(1,10):
doubler(i)
if __name__ == "__main__":
main()

Now lets learn how to run the debugger against this piece of code.

How to Start the Debugger


You can start the debugger three different ways. The first is to just import it and insert pdb.set_trace() into your code to start the debugger. You can import the debugger in IDLE and have it run
your module. Or you can call the debugger on the command line. Well focus on the last two methods
in this section. We will start with using it in the interpreter (IDLE). Open up a terminal (command
line window) and navigate to where you save the above code example. Then start Python. Now do
the following:

Chapter 24 - The Python Debugger

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

156

>>> import debug_test


>>> import pdb
>>> pdb.run('debug_test.main()')
> <string>(1)<module>()
(Pdb) continue
2
4
6
8
10
12
14
16
18
>>>

Here we import our module and pdb. Then we execute pdbs run method and tell it to call our
modules main method. This brings up the debuggers prompt. Here we typed continue to tell it
to go ahead and run the script. You can also type the letter c as a shortcut for continue. When you
continue, the debugger will continue execution until it reaches a breakpoint or the script ends.
The other way to start the debugger is to execute the following command via your terminal session:
1

python -m pdb debug_test.py

If you run it this way, you will see a slightly different result:
1
2
3
4
5
6
7
8
9
10
11
12

-> def doubler(a):


(Pdb) c
2
4
6
8
10
12
14
16
18
The program finished and will be restarted

You will note that in this example we used c instead of continue. You will also note that the debugger
restarts at the end. This preserves the debuggers state (such as breakpoints) and can be more useful

Chapter 24 - The Python Debugger

157

than having the debugger stop. Sometimes youll need to go through the code several times to
understand whats wrong with it.
Lets dig a little deeper and learn how to step through the code.

Stepping Through the Code


If you want to step through your code one line at a time, then you can use the step (or simply s)
command. Heres a session for your viewing pleasure:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

C:\Users\mike>cd c:\py101
c:\py101>python -m pdb debug_test.py
> c:\py101\debug_test.py(4)<module>()
-> def doubler(a):
(Pdb) step
> c:\py101\debug_test.py(11)<module>()
-> def main():
(Pdb) s
> c:\py101\debug_test.py(16)<module>()
-> if __name__ == "__main__":
(Pdb) s
> c:\py101\debug_test.py(17)<module>()
-> main()
(Pdb) s
--Call-> c:\py101\debug_test.py(11)main()
-> def main():
(Pdb) next
> c:\py101\debug_test.py(13)main()
-> for i in range(1,10):
(Pdb) s
> c:\py101\debug_test.py(14)main()
-> doubler(i)
(Pdb)

Here we start up the debugger and tell it to step into the code. It starts at the top and goes through
the first two function definitions. Then it reaches the conditional and finds that its supposed to
execute the main function. We step into the main function and then use the next command. The
next command will execute a called function if it encounters it without stepping into it. If you want
to step into the called function, then youll only want to just use the step command.

Chapter 24 - The Python Debugger

158

When you see a line like > c:py101debug_test.py(13)main(), you will want to pay attention to the
number thats in the parentheses. This number is the current line number in the code.
You can use the args (or a) to print the current argument list to the screen. Another handy command
is jump (or j) followed by a space and the line number that you want to jump to. This gives you
the ability to skip a bunch of monotonous stepping to get to the line that you want to get to. This
leads us to learning about breakpoints!

Setting breakpoints
A breakpoint is a line in the code where you want to pause execution. You can set a breakpoint by
calling the break (or b) command followed by a space and the line number that you want to break on.
You can also prefix the line number with a filename and colon to specify a breakpoint in a different
file. The break command also allows you to set a breakpoint with the function argument. There
is also a tbreak command which will set a temporary breakpoint which is automatically removed
when it gets hit.
Heres an example:
1
2
3
4
5
6
7
8

c:\py101>python -m pdb debug_test.py


> c:\py101\debug_test.py(4)<module>()
-> def doubler(a):
(Pdb) break 6
Breakpoint 1 at c:\py101\debug_test.py:6
(Pdb) c
> c:\py101\debug_test.py(6)doubler()
-> result = a*2

We start the debugger and then tell it to set a breakpoint on line 6. Then we continue and it stops
on line 6 just like its supposed to. Now would be a good time to check the argument list to see if
its what you expect. Give it a try by typing args now. Then do another continue and another args
to see how it changed.

Wrapping Up
There are a lot of other commands that you can use in the debugger. I recommend reading the
documentation to learn about the others. However, at this point you should be able to use the
debugger effectively to debug your own code.

Chapter 25 - Decorators
Python decorators are really cool, but they can be a little hard to understand at first. A decorator
in Python is a function that accepts another function as an argument. The decorator will usually
modify or enhance the function it accepted and return the modified function. This means that when
you call a decorated function, you will get a function that may be a little different that may have
additional features compared with the base definition. But lets back up a bit. We should probably
review the basic building block of a decorator, namely, the function.

A Simple Function
A function is a block of code that begins with the Python keyword def followed by the actual name
of the function. A function can accept zero or more arguments, keyword arguments or a mixture of
the two. A function always returns something. If you do not specify what a function should return,
it will return None. Here is a very simple function that just returns a string:
1
2
3
4
5
6
7

def a_function():
"""A pretty useless function"""
return "1+1"
if __name__ == "__main__":
value = a_function()
print(value)

All we do in the code above is call the function and print the return value. Lets create another
function:
1
2
3
4
5
6
7
8
9
10

def another_function(func):
"""
A function that accepts another function
"""
def other_func():
val = "The result of %s is %s" % (func(),
eval(func())
)
return val
return other_func

Chapter 25 - Decorators

160

This function accepts one argument and that argument has to be a function or callable. In fact, it
really should only be called using the previously defined function. You will note that this function
has a nested function inside of it that we are calling other_func. It will take the result of the function
passed to it, evaluate it and create a string that tells us about what it did, which it then returns. Lets
look at the full version of the code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

def another_function(func):
"""
A function that accepts another function
"""
def other_func():
val = "The result of %s is %s" % (func(),
eval(func())
)
return val
return other_func
def a_function():
"""A pretty useless function"""
return "1+1"
if __name__ == "__main__":
value = a_function()
print(value)
decorator = another_function(a_function)
print(decorator())

This is how a decorator works. We create one function and then pass it into a second function. The
second function is the decorator function. The decorator will modify or enhance the function that
was passed to it and return the modification. If you run this code, you should see the following as
output to stdout:
1
2

1+1
The result of 1+1 is 2

Lets change the code slightly to turn another_function into a decorator:

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

161

def another_function(func):
"""
A function that accepts another function
"""
def other_func():
val = "The result of %s is %s" % (func(),
eval(func())
)
return val
return other_func
@another_function
def a_function():
"""A pretty useless function"""
return "1+1"
if __name__ == "__main__":
value = a_function()
print(value)

You will note that in Python, a decorator starts with the <@*>* symbol followed by the name of
the function that we will be using to decorate our regular with. To apply the decorator, you
just put it on the line before the function definition. Now when we call **a_function, it will
get decorated and well get the following result:
1

The result of 1+1 is 2

Lets create a decorator that actually does something useful.

Creating a Logging Decorator


Sometimes you will want to create a log of what a function is doing. Most of the time, you will
probably be doing your logging within the function itself. Occasionally you might want to do it at
the function level to get an idea of the flow of the program or perhaps to fulfill some business rules,
like auditing. Heres a little decorator that we can use to record any functions name and what it
returns:

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33

162

import logging
def log(func):
"""
Log what function is called
"""
def wrap_log(*args, **kwargs):
name = func.__name__
logger = logging.getLogger(name)
logger.setLevel(logging.INFO)
# add file handler
fh = logging.FileHandler("%s.log" % name)
fmt = '%(asctime)s - %(name)s - %(levelname)s - %(message)s'
formatter = logging.Formatter(fmt)
fh.setFormatter(formatter)
logger.addHandler(fh)
logger.info("Running function: %s" % name)
result = func(*args, **kwargs)
logger.info("Result: %s" % result)
return func
return wrap_log
@log
def double_function(a):
"""
Double the input parameter
"""
return a*2
if __name__ == "__main__":
value = double_function(2)

This little script has a log function that accepts a function as its sole argument. It will create a logger
object and a log file name based on the name of the function. Then the log function will log what
function was called and what the function returned, if anything.

Built-in Decorators
Python comes with several built-in decorators. The big three are:

Chapter 25 - Decorators

163

@classmethod
@staticmethod
@property
There are also decorators in various parts of Pythons standard library. One example would be
functools.wraps. We will be limiting our scope to the three above though.

@classmethod and @staticmethod


I have never actually used these myself, so I did a fair bit of research. The <*@classmethod>*
decorator can be called with with an instance of a class or directly by the class itself as its first
argument. According to the Python documentation: It can be called either on the class (such as C.f())
or on an instance (such as C().f()). The instance is ignored except for its class. If a class method is
called for a derived class, the derived class object is passed as the implied first argument. The primary
use case of a @classmethod decorator that I have found in my research is as an alternate constructor
or helper method for initialization.
The <*@staticmethod>* decorator is just a function inside of a class. You can call it both with and
without instantiating the class. A typical use case is when you have a function where you believe it
has a connection with a class. Its a stylistic choice for the most part.
It might help to see a code example of how these two decorators work:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

class DecoratorTest(object):
"""
Test regular method vs @classmethod vs @staticmethod
"""
def __init__(self):
"""Constructor"""
pass
def doubler(self, x):
""""""
print("running doubler")
return x*2
@classmethod
def class_tripler(klass, x):
""""""
print("running tripler: %s" % klass)
return x*3

Chapter 25 - Decorators

21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37

164

@staticmethod
def static_quad(x):
""""""
print("running quad")
return x*4
if __name__ == "__main__":
decor = DecoratorTest()
print(decor.doubler(5))
print(decor.class_tripler(3))
print(DecoratorTest.class_tripler(3))
print(DecoratorTest.static_quad(2))
print(decor.static_quad(3))
print(decor.doubler)
print(decor.class_tripler)
print(decor.static_quad)

This example demonstrates that you can call a regular method and both decorated methods in the
same way. You will notice that you can call both the @classmethod and the @staticmethod decorated
functions directly from the class or from an instance of the class. If you try to call a regular function
with the class (i.e. DecoratorTest.doubler(2)) you will receive a TypeError. You will also note that
the last print statement shows that decor.static_quad returns a regular function instead of a bound
method.

Python Properties
Python has a neat little concept called a property that can do several useful things. We will be looking
into how to do the following:
Convert class methods into read-only attributes
Reimplement setters and getters into an attribute
One of the simplest ways to use a property is to use it as a decorator of a method. This allows
you to turn a class method into a class attribute. I find this useful when I need to do some kind of
combination of values. Others have found it useful for writing conversion methods that they want
to have access to as methods. Lets take a look at a simple example:

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14

165

class Person(object):
""""""
def __init__(self, first_name, last_name):
"""Constructor"""
self.first_name = first_name
self.last_name = last_name
@property
def full_name(self):
"""
Return the full name
"""
return "%s %s" % (self.first_name, self.last_name)

In the code above, we create two class attributes or properties: self.first_name and self.last_name.
Next we create a full_name method that has a <*@property>* decorator attached to it. This allows
us to do the following in an interpreter session:
1
2
3
4
5
6
7
8
9

>>> person = Person("Mike", "Driscoll")


>>> person.full_name
'Mike Driscoll'
>>> person.first_name
'Mike'
>>> person.full_name = "Jackalope"
Traceback (most recent call last):
File "<string>", line 1, in <fragment>
AttributeError: can't set attribute

As you can see, because we turned the method into a property, we can access it using normal
dot notation. However, if we try to set the property to something different, we will cause an
AttributeError to be raised. The only way to change the full_name property is to do so indirectly:
1
2
3

>>> person.first_name = "Dan"


>>> person.full_name
'Dan Driscoll'

This is kind of limiting, so lets look at another example where we can make a property that does
allow us to set it.

Replacing Setters and Getters with a Python property


Lets pretend that we have some legacy code that someone wrote who didnt understand Python
very well. If youre like me, youve already seen this kind of code before:

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

166

from decimal import Decimal


class Fees(object):
""""""
def __init__(self):
"""Constructor"""
self._fee = None
def get_fee(self):
"""
Return the current fee
"""
return self._fee
def set_fee(self, value):
"""
Set the fee
"""
if isinstance(value, str):
self._fee = Decimal(value)
elif isinstance(value, Decimal):
self._fee = value

To use this class, we have to use the setters and getters that are defined:
1
2
3
4

>>> f = Fees()
>>> f.set_fee("1")
>>> f.get_fee()
Decimal('1')

If you want to add the normal dot notation access of attributes to this code without breaking all the
applications that depend on this piece of code, you can change it very simply by adding a property:

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25

167

from decimal import Decimal


class Fees(object):
""""""
def __init__(self):
"""Constructor"""
self._fee = None
def get_fee(self):
"""
Return the current fee
"""
return self._fee
def set_fee(self, value):
"""
Set the fee
"""
if isinstance(value, str):
self._fee = Decimal(value)
elif isinstance(value, Decimal):
self._fee = value
fee = property(get_fee, set_fee)

We added one line to the end of this code. Now we can do stuff like this:
1
2
3
4
5
6
7

>>> f = Fees()
>>> f.set_fee("1")
>>> f.fee
Decimal('1')
>>> f.fee = "2"
>>> f.get_fee()
Decimal('2')

As you can see, when we use property in this manner, it allows the fee property to set and get the
value itself without breaking the legacy code. Lets rewrite this code using the property decorator
and see if we can get it to allow setting.

Chapter 25 - Decorators

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

168

from decimal import Decimal


class Fees(object):
""""""
def __init__(self):
"""Constructor"""
self._fee = None
@property
def fee(self):
"""
The fee property - the getter
"""
return self._fee
@fee.setter
def fee(self, value):
"""
The setter of the fee property
"""
if isinstance(value, str):
self._fee = Decimal(value)
elif isinstance(value, Decimal):
self._fee = value
if __name__ == "__main__":
f = Fees()

The code above demonstrates how to create a setter for the fee property. You can do this by
decorating a second method that is also called fee with a decorator called <@fee.setter>. The setter
is invoked when you do something like this:
1
2

>>> f = Fees()
>>> f.fee = "1"

If you look at the signature for property, it has fget, fset, fdel and doc as arguments. You can
create another decorated method using the same name to correspond to a delete function using
<@fee.deleter*>* if you want to catch the **del command against the attribute.

Chapter 25 - Decorators

169

Wrapping Up
At this point you should know how to create your own decorators and how to use a few of Pythons
built-in decorators. We looked at @classmethod, @property and @staticmethod. I would be curious
to know how my readers use the built-in decorators and how they use their own custom decorators.

Chapter 26 - The lambda


The Python lambda statement is an anonymous or unbound function and a pretty limited function
at that. Lets take a look at a few typical examples and see if we can find a use case for it. The typical
examples that one normally sees for teaching the lambda are some sort of boring doubling function.
Just to be contrary, our simple example will show how to find the square root. First well show a
normal function and then the lambda equivalent:
1
2
3
4
5
6
7
8
9

import math
def sqroot(x):
"""
Finds the square root of the number passed in
"""
return math.sqrt(x)
square_rt = lambda x: math.sqrt(x)

If you try each of these functions, youll end up with a float. Here are a couple examples:
1
2
3
4

>>> sqroot(49)
7.0
>>> square_rt(64)
8.0

Pretty slick, right? But where would we actually use a lambda in real life? Maybe a calculator
program? Well, that would work, but its a pretty limited application for a builtin of Python! One
of the major pieces of Python that lambda examples are applied to regularly are Tkinter callbacks.
Tkinter is a toolkit for building GUIs that is included with Python.

Tkinter + lambda
Well start with Tkinter since its included with the standard Python package. Heres a really simple
script with three buttons, two of which are bound to their event handler using a lambda:

Chapter 26 - The lambda

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

171

import Tkinter as tk
class App:
""""""
def __init__(self, parent):
"""Constructor"""
frame = tk.Frame(parent)
frame.pack()
btn22 = tk.Button(frame, text="22", command=lambda: self.printNum(22))
btn22.pack(side=tk.LEFT)
btn44 = tk.Button(frame, text="44", command=lambda: self.printNum(44))
btn44.pack(side=tk.LEFT)
quitBtn = tk.Button(frame, text="QUIT", fg="red", command=frame.quit)
quitBtn.pack(side=tk.LEFT)

def printNum(self, num):


""""""
print("You pressed the %s button" % num)
if __name__ == "__main__":
root = tk.Tk()
app = App(root)
root.mainloop()

Notice the btn22 and btn44 variables. This is where the action is. We create a tk.Button instance
here and bind to our printNum method in one fell swoop. The lambda is assigned to the buttons
command parameter. What this means is that were creating a one-off function for the command,
much like in the quit button where we call the frames quit method. The difference here is that
this particular lambda is a method that calls another method and passes the latter an integer. In the
printNum method, we print to stdout which button was pressed by using the information that was
passed to it from the lambda function. Did you follow all that? If so, we can continue; if not, re-read
this paragraph as many times as necessary until the information sinks in or you go crazy, whichever
comes first.

Wrapping Up
The lambda statement is used in all kinds of other projects as well. If you Google a Python project
name and lambda, you can find lots of live code out there. For example, if you search for django

Chapter 26 - The lambda

172

lambda, youll find out that django has a modelformset factory that utilizes lambdas. The Elixir
plugin for SqlAlchemy also uses lambdas. Keep your eyes open and youll be surprised how many
times youll stumble across this handy little function maker.

Chapter 27 - Code Profiling


Code profiling is an attempt to find bottlenecks in your code. Profiling is supposed to find what
parts of your code take the longest. Once you know that, then you can look at those pieces of your
code and try to find ways to optimize it. Python comes with three profilers built in: cProfile, profile
and hotshot. According to the Python documentation, hotshot is no longer maintained and may
be dropped in a future version of Python. The profile module is a pure Python module, but adds a
lot of overhead to profiled programs. Thus we will be focusing on cProfile, which has an interface
that mimics the profile module.

Profiling Your Code with cProfile


Profiling code with cProfile is really quite easy. All you need to do is import the module and call its
run function. Lets look at a simple example:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

>>> import hashlib


>>> import cProfile
>>> cProfile.run("hashlib.md5(b'abcdefghijkl').digest()")
4 function calls in 0.000 CPU seconds
Ordered by: standard name
ncalls tottime
1
0.000
1
0.000
1
0.000
' objects}
1
0.000
iler' objects}

percall
0.000
0.000
0.000

cumtime
0.000
0.000
0.000

0.000

0.000

percall
0.000
0.000
0.000

filename:lineno(function)
<string>:1(<module>)
{_hashlib.openssl_md5}
{method 'digest' of '_hashlib.HASH\

0.000 {method 'disable' of '_lsprof.Prof\

Here we import the hashlib module and use cProfile to profile the creation of an MD5 hash. The
first line shows that there were 4 function calls. The next line tells us how the results are ordered.
According to the documentation, standard name refers to the far right column. There are a number
of columns here.
ncalls is the number of calls made.
tottime is a total of the time spent in the given function.

Chapter 27 - Code Profiling

174

percall refers to the quotient of tottime divided by ncalls


cumtime is the cumulative time spent in this and all subfunctions. Its even
accurate for recursive functions!
The second percall column is the quotient of cumtime divided by primitive calls
filename:lineno(function) provides the respective data of each function
|
A primitive call is one that was not induced via recursion.
This isnt a very interesting example as there are no obvious bottlenecks. Lets create a piece of code
with some built in bottlenecks and see if the profiler detects them.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

import time
def fast():
""""""
print("I run fast!")
def slow():
""""""
time.sleep(3)
print("I run slow!")
def medium():
""""""
time.sleep(0.5)
print("I run a little slowly...")
def main():
""""""
fast()
slow()
medium()
if __name__ == '__main__':
main()

In this example, we create four functions. The first three run at different rates. The fast function
will run at normal speed; the medium function will take approximately half a second to run and the
slow function will take around 3 seconds to run. The main function calls the other three. Now lets
run cProfile against this silly little program:

175

Chapter 27 - Code Profiling

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

>>> import cProfile


>>> import ptest
>>> cProfile.run('ptest.main()')
I run fast!
I run slow!
I run a little slowly...
8 function calls in 3.500 seconds
Ordered by: standard name
ncalls tottime
1
0.000
1
0.000
1
0.000
1
0.000
1
0.000
1
0.000
iler' objects}
2
3.499

percall
0.000
0.000
0.000
0.000
0.000
0.000

cumtime
3.500
0.500
3.500
0.000
3.000
0.000

1.750

3.499

percall
3.500
0.500
3.500
0.000
3.000
0.000

filename:lineno(function)
<string>:1(<module>)
ptest.py:15(medium)
ptest.py:21(main)
ptest.py:4(fast)
ptest.py:9(slow)
{method 'disable' of '_lsprof.Prof\

1.750 {time.sleep}

This time around we see the program took 3.5 seconds to run. If you examine the results, you will see
that cProfile has identified the slow function as taking 3 seconds to run. Thats the biggest bottleneck
after the main function. Normally when you find a bottleneck like this, you would try to find a faster
way to execute your code or perhaps decide that the runtime was acceptable. In this example, we
know that the best way to speed up the function is to remove the time.sleep call or at least reduce
the sleep length.
You can also call cProfile on the command line rather than using it in the interpreter. Heres one
way to do it:
1

python -m cProfile ptest.py

This will run cProfile against your script in much the same way as we did before. But what if you
want to save the profilers output? Well, thats easy with cProfile! All you need to do is pass it the
-o command followed by the name (or path) of the output file. Heres an example:
1

python -m cProfile -o output.txt ptest.py

Unfortunately, the file it outputs isnt exactly human-readable. If you want to read the file, then
youll need to use Pythons pstats module. You can use pstats to format the output in various ways.
Heres some code that shows how to get some output thats similar to what weve seen so far:

176

Chapter 27 - Code Profiling

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

>>>
>>>
>>>
Thu

import pstats
p = pstats.Stats("output.txt")
p.strip_dirs().sort_stats(-1).print_stats()
Mar 20 18:32:16 2014
output.txt
8 function calls in 3.501 seconds

Ordered by: standard name


ncalls tottime
1
0.000
1
0.001
1
0.000
1
0.001
1
0.001
1
0.000
iler' objects}
2
3.499

percall
0.000
0.001
0.000
0.001
0.001
0.000

cumtime
3.501
0.500
3.501
0.001
3.000
0.000

1.750

3.499

percall
3.501
0.500
3.501
0.001
3.000
0.000

filename:lineno(function)
ptest.py:1(<module>)
ptest.py:15(medium)
ptest.py:21(main)
ptest.py:4(fast)
ptest.py:9(slow)
{method 'disable' of '_lsprof.Prof\

1.750 {time.sleep}

<pstats.Stats instance at 0x017C9030>

The strip_dirs call will strip out all the paths to the modules from the output while the sort_stats
call does the sorting that were used to seeing. There are a bunch of really interesting examples in
the cProfile documentation showing different ways to extract information using the pstats module.

Wrapping Up
At this point you should be able to use the cProfile module to help you diagnose why your code is
so slow. You might also want to take a look at Pythons timeit module. It allows you to time small
pieces of your code if you dont want to deal with the complexities involved with profiling. There
are also several other 3rd party modules that are good for profiling such as the line_profiler and the
memory_profiler projects.

Chapter 28 - An Intro to Testing


Python includes a couple of built-in modules for testing your code. They two methods are called
doctest and unittest. We will look at how to use doctest first and in the second section we will
introduce unit tests using Test Driven Development techniques.

Testing with doctest


The doctest module will search for pieces of text in your code that resemble interactive Python
sessions. It will then execute those sessions to verify that they work exactly as written. This means
that if you wrote an example in a docstring that showed the output with a trailing space or tab,
then the actual output of the function has to have that trailing whitespace too. Most of the time, the
docstring is where you will want to put your tests. The following aspects of doctest will be covered:
How to run doctest from the terminal
How to use doctest inside a module
How to run a doctest from a separate file
Lets get started!

Running doctest via the Terminal


We will start by creating a really simple function that will double whatever is given to it. We will
include a couple of tests inside the functions docstring. Heres the code (be sure and save it as
dtest1.py):
1
2
3
4
5
6
7
8
9
10

# dtest1.py
def double(a):
"""
>>> double(4)
8
>>> double(9)
18
"""
return a*2

178

Chapter 28 - An Intro to Testing

Now we just need to run this code in doctest. Open up a terminal (or command line) and change
directories to the folder that contains your script. Heres a screenshot of what I did:

image

You will notice that in the first example, I executed the following:
1

python -m doctest dtest1.py

That ran the test and nothing printed out to the screen. When you dont see anything printed, that
means that all the tests passed successfully. The second example shows the following command:

179

Chapter 28 - An Intro to Testing

python -m doctest -v dtest1.py

The -v means that we want verbose output, which is exactly what we received. Open up the
code again and add a space after the 18 in the docstring. Then re-run the test. Heres the output I
received:

image

The error message says it expected 18 and it got 18. Whats going on here? Well we added a
space after 18 to our docstring, so doctest actually expected the number 18 followed by a space.
Also beware of putting dictionaries as output in your docstring examples. Dictionaries can be in any
order, so the likelihood of it matching your actual output isnt very good.

Running doctest Inside a Module


Lets modify the example slightly so that we import the doctest module and use its testmod function.

Chapter 28 - An Intro to Testing

1
2
3
4
5
6
7
8
9
10
11
12

180

def double(a):
"""
>>> double(4)
8
>>> double(9)
18
"""
return a*2
if __name__ == "__main__":
import doctest
doctest.testmod(verbose=True)

Here we import doctest and call doctest.testmod. We pass it the keyword argument of verbose=True so that we can see some output. Otherwise, this script will run with no output
whatsoever, signifying that the tests ran successfully.
If you do not want to hard-code the verbose option, you can also do it on the command line:
1

python dtest2.py -v

Now were ready to learn how to put the tests into a separate file.

Running doctest From a Separate File


The doctest module also supports putting the testing into a separate file. This allows us to separate
the tests from the code. Lets strip the tests from the previous example and put them into a text file
named tests.txt:
1
2
3
4
5
6
7

The following are tests for dtest2.py


>>> from dtest2 import double
>>> double(4)
8
>>> double(9)
18

Lets run this test file on the command line. Heres how:

181

Chapter 28 - An Intro to Testing

image

You will notice that the syntax for calling doctest with a text file is the same as calling it with a
Python file. The results are the same as well. In this case, there are three tests instead of two because
were also importing a module. You can also run the tests that are in a text file inside the Python
interpreter. Heres one example:

182

Chapter 28 - An Intro to Testing

image

Here we just import doctest and call its testfile method. Note that you need to also pass the filename
or path to the testfile function. It will return a TestResults object that contains how many tests were
attempted and how many failed.

Test Driven Development with unittest


In this section, you will learn about Test Driven Development (TDD) with Python using Pythons
built-in unittest module. I do want to thank Matt and Aaron for their assistance in showing me
how TDD works in the real world. To demonstrate the concepts of TDD, well be covering how to
score bowling in Python. You may want to use Google to look up the rules for bowling if you dont
already know them. Once you know the rules, its time to write some tests. In case you didnt know,
the idea behind Test Driven Development is that you write the tests BEFORE you write the actual
code. In this chapter, we will write a test, then some code to pass the test. We will iterate back and
forth between writing tests and code until were done. For this chapter, well write just three tests.
Lets get started!

Chapter 28 - An Intro to Testing

183

The First Test


Our first test will be to test our game object and see if it can calculate the correct total if we roll
eleven times and only knock over one pin each time. This should give us a total of eleven.
1
2
3
4
5
6
7
8
9
10

import unittest
class TestBowling(unittest.TestCase):
""""""
def test_all_ones(self):
"""Constructor"""
game = Game()
game.roll(11, 1)
self.assertEqual(game.score, 11)

This is a pretty simple test. We create a game object and then call its roll method eleven times with
a score of one each time. Then we use the assertEqual method from the unittest module to test if
the game objects score is correct (i.e. eleven). The next step is to write the simplest code you can
think of to make the test pass. Heres one example:
1
2
3
4
5
6
7
8
9
10
11

class Game:
""""""
def __init__(self):
"""Constructor"""
self.score = 0
def roll(self, numOfRolls, pins):
""""""
for roll in numOfRolls:
self.score += pins

For simplicitys sake, you can just copy and paste that into the same file with your test. Well break
them into two files for our next test. Anyway, as you can see, our Game class is super simple. All
that was needed to pass the test was a score property and a roll method that can update it.
Lets run the test and see if it passes! The easiest way to run the tests is to add the following two
lines of code to the bottom of the file:

Chapter 28 - An Intro to Testing

1
2

184

if __name__ == '__main__':
unittest.main()

Then just run the Python file via the command line, If you do, you should get something like the
following:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

E
---------------------------------------------------------------------ERROR: test_all_ones (__main__.TestBowling)
Constructor
---------------------------------------------------------------------Traceback (most recent call last):
File "C:\Users\Mike\Documents\Scripts\Testing\bowling\test_one.py",
line 27, in test_all_ones
game.roll(11, 1)
File "C:\Users\Mike\Documents\Scripts\Testing\bowling\test_one.py",
line 15, in roll
for roll in numOfRolls:
TypeError: 'int' object is not iterable
---------------------------------------------------------------------Ran 1 test in 0.001s
FAILED (errors=1)

Oops! Weve got a mistake in there somewhere. It looks like were passing an Integer and then
trying to iterate over it. That doesnt work! We need to change our Game objects roll method to the
following to make it work:
1
2
3
4

def roll(self, numOfRolls, pins):


""""""
for roll in range(numOfRolls):
self.score += pins

If you run the test now, you should get the following:

Chapter 28 - An Intro to Testing

1
2
3
4
5

185

.
---------------------------------------------------------------------Ran 1 test in 0.000s
OK

Note the . because its important. That little dot means that one test has run and that it passed.
The OK at the end clues you into that fact as well. If you study the original output, youll notice
it leads off with an E for error and theres no dot! Lets move on to test #2.

The Second Test


For the second test, well test what happens when we get a strike. Well need to change the first test
to use a list for the number of pins knocked down in each frame though, so well look at both tests
here. Youll probably find this to be a fairly common process where you may need to edit a couple
of tests due to fundamental changes in what youre testing for. Normally this will only happen at
the beginning of your coding and you will get better later on such that you shouldnt need to do
this. Since this is my first time doing this, I wasnt thinking far enough ahead. Anyway, lets take a
look at the code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

from game import Game


import unittest
class TestBowling(unittest.TestCase):
""""""
def test_all_ones(self):
"""Constructor"""
game = Game()
pins = [1 for i in range(11)]
game.roll(11, pins)
self.assertEqual(game.score, 11)
def test_strike(self):
"""
A strike is 10 + the value of the next two rolls. So in this case
the first frame will be 10+5+4 or 19 and the second will be
5+4. The total score would be 19+9 or 28.
"""
game = Game()
game.roll(11, [10, 5, 4])

Chapter 28 - An Intro to Testing

22
23
24
25

186

self.assertEqual(game.score, 28)
if __name__ == '__main__':
unittest.main()

Lets take a look at our first test and how it changed. Yes, were breaking the rules here a bit when
it comes to TDD. Feel free to NOT change the first test and see what breaks. In the test_all_ones
method, we set the pins variable to equal a list comprehension which created a list of eleven ones.
Then we passed that to our game objects roll method along with the number of rolls.
In the second test, we roll a strike in our first roll, a five in our second and a four in our third. Note
that we went a head and told it that we were passing in eleven rolls and yet we only pass in three.
This means that we need to set the other eight rolls to zeros. Next, we use our trusty assertEqual
method to check if we get the right total. Finally, note that were now importing the Game class
rather than keeping it with the tests. Now we need to implement the code necessary to pass these
two tests. Lets take a look at one possible solution:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

class Game:
""""""
def __init__(self):
"""Constructor"""
self.score = 0
self.pins = [0 for i in range(11)]
def roll(self, numOfRolls, pins):
""""""
x = 0
for pin in pins:
self.pins[x] = pin
x += 1
x = 0
for roll in range(numOfRolls):
if self.pins[x] == 10:
self.score = self.pins[x] + self.pins[x+1] + self.pins[x+2]
else:
self.score += self.pins[x]
x += 1
print(self.score)

Right off the bat, you will notice that we have a new class attribute called self.pins that holds the
default pin list, which is eleven zeroes. Then in our roll method, we add the correct scores to the
correct position in the self.pins list in the first loop. Then in the second loop, we check to see if the

Chapter 28 - An Intro to Testing

187

pins knocked down equals ten. If it does, we add it and the next two scores to score. Otherwise, we
do what we did before. At the end of the method, we print out the score to check if its what we
expect. At this point, were ready to code up our final test.

The Third (and Final) Test


Our last test will test for the correct score that would result should someone roll a spare. The test is
easy, the solution is slightly more difficult. While were at it, were going to refactor the test code a
bit. As usual, we will look at the test first.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31

from game_v2 import Game


import unittest
class TestBowling(unittest.TestCase):
""""""
def setUp(self):
""""""
self.game = Game()
def test_all_ones(self):
"""
If you don't get a strike or a spare, then you just add up the
face value of the frame. In this case, each frame is worth
one point, so the total is eleven.
"""
pins = [1 for i in range(11)]
self.game.roll(11, pins)
self.assertEqual(self.game.score, 11)
def test_spare(self):
"""
A spare is worth 10, plus the value of your next roll. So in this
case, the first frame will be 5+5+5 or 15 and the second will be
5+4 or 9. The total is 15+9, which equals 24,
"""
self.game.roll(11, [5, 5, 5, 4])
self.assertEqual(self.game.score, 24)
def test_strike(self):
"""

Chapter 28 - An Intro to Testing

32
33
34
35
36
37
38
39
40

188

A strike is 10 + the value of the next two rolls. So in this case


the first frame will be 10+5+4 or 19 and the second will be
5+4. The total score would be 19+9 or 28.
"""
self.game.roll(11, [10, 5, 4])
self.assertEqual(self.game.score, 28)
if __name__ == '__main__':
unittest.main()

First off, we added a setUp method that will create a self.game object for us for each test. If we were
accessing a database or something like that, we would probably have a tear down method too for
closing connections or files or that sort of thing. These are run at the beginning and end of each test
respectively should they exist. The test_all_ones and test_strike tests are basically the same except
that they are using self.game now. The only the new test is test_spare. The docstring explains
how spares work and the code is just two lines. Yes, you can figure this out. Lets look at the code
well need to pass these tests:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

# game_v2.py
class Game:
""""""
def __init__(self):
"""Constructor"""
self.score = 0
self.pins = [0 for i in range(11)]
def roll(self, numOfRolls, pins):
""""""
x = 0
for pin in pins:
self.pins[x] = pin
x += 1
x = 0
spare_begin = 0
spare_end = 2
for roll in range(numOfRolls):
spare = sum(self.pins[spare_begin:spare_end])
if self.pins[x] == 10:
self.score = self.pins[x] + self.pins[x+1] + self.pins[x+2]
elif spare == 10:

Chapter 28 - An Intro to Testing

25
26
27
28
29
30
31
32
33
34

189

self.score = spare + self.pins[x+2]


x += 1
else:
self.score += self.pins[x]
x += 1
if x == 11:
break
spare_begin += 2
spare_end += 2
print(self.score)

For this part of the puzzle, we add to our conditional statement in our loop. To calculate the spares
value, we use the spare_begin and spare_end list positions to get the right values from our list and
then we sum them up. Thats what the spare variable is for. That may be better placed in the elif,
but Ill leave that for the reader to experiment with. Technically, thats just the first half of the spare
score. The second half are the next two rolls, which is what youll find in the calculation in the elif
portion of the current code. The rest of the code is the same.

Other Notes
As you may have guessed, there is a whole lot more to the unittest module than what is covered
here. There are lots of other asserts you can use to test results with. You can skip tests, run the tests
from the command line, use a TestLoader to create a test suite and much, much more. Be sure to
read the full documentation when you get the chance as this tutorial barely scratches the surface
of this library.

Wrapping Up
At this point, you should be able to understand how to use the doctest and unittest modules
effectively in your own code. You should go and read the Python documentation about these two
modules as there is additional information about other options and functionality that you might
find useful. You also know a little about how to use the concepts of Test Driven Development when
writing your own scripts.
http://docs.python.org/2/library/unittest.html

Part IV - Tips, Tricks and Tutorials


In Part IV, you will learn how to install 3rd party packages from the Python Package Index (PyPI).
You will learn a bit about easy_install, pip and setup.py and how to use these tools to install the
packages. This is just the first chapter though. Heres a listing of the packages you will learn about:

configobj - working with Config files in a more Pythonic way.


lxml - a package for working with XML
pylint / pyflakes - Python code analyzers
requests - a Python-friendly version of urllib
SQLAlchemy - an Object Relational Mapper for Python
virtualenv - learn about virtual environments in Python

The reason we will be looking at configobj is because I think it works better than ConfigParser, the
module that comes with Python. The configobj package has an interface that is just more intuitive
and powerful than ConfigParser. In the next chapter, well look at the lxml module and learn a
couple of new ways to read, parse and create XML. In the fourth chapter, we will look at pylint and
pyflakes, which are great for code analysis. They can look at your module and check for errors.
pylint can also be used to help you get your code to conform to PEP8, the Python style guide.
The requests package is great replacement for the urllib module. Its interface is simpler and the
documentation is quite good. SQLAlchemy is the premier Object Relational Mapper for Python. It
allows you to write SQL queries, tables, etc in Python code. One of its best features is that if you need
to switch database backends, you wont have to change your code very much to continue working
with said database. For the last chapter, well look at virtualenv, a neat module that allows you to
create mini-virtual environments that you can write your code in. These virtual environments are
especially handy for testing out new modules or just new releases of modules before you apply them
to your Python core installation.

191

Part IV - Tips, Tricks and Tutorials

image

Chapter 29 - Installing Modules


When youre first starting out as a Python programmer, you dont think about how you might need
to install an external package or module. But when that need appears, youll want to know how
to in a hurry! Python packages can be found all over the internet. Most of the popular ones can be
found on the Python Package Index (PyPI). You will also find a lot of Python packages on github,
bitbucket, and Google code. In this article, we will be covering the following methods of installing
Python packages:

Install from source


easy_install
pip
Other ways to install packages

Installing from Source


Installing from source is a great skill to have. There are easier ways, which well be getting to later on
in the article. However, there are some packages that you have to install from source. For example, to
use easy_install, you will need to first install setuptools. To do that, you will want to download the
tar or zip file from the Python Package Index and extract it somewhere on your system. Then look
for the setup.py file. Open up a terminal session and change directories to the folder that contains
the setup file. Then run the following command:
1

python setup.py install

If Python isnt on your system path, you will receive an error message stating that the python
command wasnt found or is an unknown application. You can call this command by using the
full path to Python instead. Heres how you might do it if you were on Windows:
1

c:\python34\python.exe setup.py install

This method is especially handy if you have multiple versions of Python installed and you need to
install the package to different ones. All you need to do is type the full path to the right Python
version and install the package against it.
Some packages contain C code, such as C header files that will need to be compiled for the package to
install correctly. On Linux, you will normally already have a C/C++ compiler installed and you can
get the package installed with minimal headaches. On Windows, you will need to have the correct
version of Visual Studio installed to compile the package correctly. Some people say you can use
MingW too, but I have yet to find a way to make that work. If the package has a Windows installer
already pre-made, use it. Then you dont have to mess with compiling at all.

Chapter 29 - Installing Modules

193

Using easy_install
Once you have setuptools installed, you can use easy_install. You can find it installed in your
Python installations Scripts folder. Be sure to add the Scripts folder to your system path so you can
call easy_install on the command line without specifying its full path. Try running the following
command to learn about all of easy_installs options:
1

easy_install -h

When you want to install a package with easy_install, all you have to do is this:
1

easy_install package_name

easy_install will attempt to download the package from PyPI, compile it (if necessary) and install
it. If you go into your Pythons site-packages directory, you will find a file named easy-install.pth
that will contain an entry for all packages installed with easy_install. This file is used by Python to
help in importing the module or package.
You can also tell easy_install to install from a URL or from a path on your computer. It can also
install packages directly from a tar file. You can use easy_install to upgrade a package by using upgrade (or -U). Finally, you can use easy_install to install Python eggs. You can find egg files on
PyPI and other locations. An egg is basically a special zip file. In fact, if you change the extension
to .zip, you can unzip the egg file.
Here are some examples:
1
2
3

easy_install -U SQLAlchemy
easy_install http://example.com/path/to/MyPackage-1.2.3.tgz
easy_install /path/to/downloaded/package

There are some issues with easy_install. It will try to install a package before its finished
downloading. There is no way to uninstall a package using easy_install. You will have to delete the
package yourself and update the easy-install.pth file by removing the entry to the package. For these
reasons and others, there was movement in the Python community to create something different,
which caused pip to be born.

Using pip
The pip program actually comes with Python 3.4. If you have an older version of Python, then you
will need to install pip manually. Installing pip is a little bit different than what we have previously
discussed. You still go to PyPI, but instead of downloading the package and running its setup.py
script, you will be asked to download a single script called get-pip.py. Then you will need to execute
it by doing the following:

Chapter 29 - Installing Modules

194

python get-pip.py

This will install setuptools or an alternative to setuptools called distribute if one of them is not
already installed. It will also install pip. pip works with CPython versions 2.6, 2.7, 3.1, 3.2, 3.3, 3.4
and also pypy. You can use pip to install anything that easy_install can install, but the invocation is
a bit different. To install a package, do the following:
1

pip install package_name

To upgrade a package, you would do this:


1

pip install -U PackageName

You may want to call pip -h to get a full listing of everything pip can do. One thing that pip can install
that easy_install cannot is the Python wheel format. A wheel is a ZIP-format archive with a specially
formatted filename and the .whl extension. You can also install wheels via its own command line
utility. On the other hand, pip cannot install an egg. If you need to install an egg, you will want to
use easy_install.

A Note on Dependencies
One of the many benefits of using easy_install and pip is that if the package has dependencies
specified in their setup.py script, both easy_install and pip will attempt to download and install
them too. This can alleviate a lot of frustration when youre trying new packages and you didnt
realize that package A depended on Packages B, C and D. With easy_install or pip, you dont need
to worry about that any more.

Wrapping Up
At this point, you should be able to install just about any package that you need, assuming the
package supports your version of Python. There are a lot of tools available to the Python programmer.
While packaging in Python is a bit confusing right now, once you know how to use the proper tools,
you can usually get what you want installed or packaged up. We will be looking more at creating
our own packages, eggs and wheels in Part V.

Chapter 30 - ConfigObj
Python comes with a handy module called ConfigParser. Its good for creating and reading
configuration files (aka INI files). However, Michael Foord (author of IronPython in Action) and
Nicola Larosa decided to write their own configuration module called ConfigObj. In many ways, it
is an improvement over the standard librarys module. For example, it will return a dictionary-like
object when it reads a config file. ConfigObj can also understand some Python types. Another neat
feature is that you can create a configuration spec that ConfigObj will use to validate the config file.

Getting Started
First of all, you need to go and get ConfigObj. This is a good time to use your knowledge from the
last chapter on installing packages. Here is how you would get ConfigObj with pip:
1

pip install configobj

Once you have it installed, we can move on. To start off, open a text editor and create a file with
some contents like this:
1
2
3
4

product = Sony PS3


accessories = controller, eye, memory stick
# This is a comment that will be ignored
retail_price = $400

Save it where ever you like. Im going to call mine config.ini. Now lets see how ConfigObj can be
used to extract that information:
1
2
3
4
5
6
7
8

>>> from configobj import ConfigObj


>>> config = ConfigObj(r"path to config.ini")
>>> config["product"]
'Sony PS3'
>>> config["accessories"]
['controller', 'eye', 'memory stick']
>>> type(config["accessories"])
<type 'list'>

Chapter 30 - ConfigObj

196

As you can see, ConfigObj uses Pythons dict API to access the information it has extracted. All
you had to do to get ConfigObj to parse the file was to pass the files path to ConfigObj. Now, if
the information had been under a section (i.e. [Sony]), then you would have had to do pre-pend
everything with [Sony], like this: config[Sony][product]. Also take note that the accessories
section was returned as a list of strings. ConfigObj will take any valid line with a comma-separated
list and return it as a Python list. You can also create multi-line strings in the config file as long as
you enclose them with triple single or double quotes.
If you need to create a sub-section in the file, then use extra square brackets. For example, [Sony] is
the top section, [[Playstation]] is the sub-section and [[[PS3]]] is the sub-section of the sub-section.
You can create sub-sections up to any depth. For more information on the formatting of the file, I
recommend reading ConfigObjs documentation.
Now well do the reverse and create the config file programmatically.
1
2
3
4
5
6
7
8
9
10
11
12
13

import configobj
def createConfig(path):
config = configobj.ConfigObj()
config.filename = path
config["Sony"] = {}
config["Sony"]["product"] = "Sony PS3"
config["Sony"]["accessories"] = ['controller', 'eye', 'memory stick']
config["Sony"]["retail price"] = "$400"
config.write()
if __name__ == "__main__":
createConfig("config.ini")

As you can see, all it took was 13 lines of code. In the code above, we create a function and pass
it the path for our config file. Then we create a ConfigObj object and set its filename property. To
create the section, we create an empty dict with the name Sony. Then we pre-pend each line of
the sections contents in the same way. Finally, we call our config objects write method to write the
data to the file.

Using a configspec
ConfigObj also provides a way to validate your configuration files using a configspec. When I
mentioned that I was going to write on this topic, Steven Sproat (creator of Whyteboard) volunteered
his configspec code as an example. I took his specification and used it to create a default config file.
In this example, we use Foords validate module to do the validation. I dont think its included in
your ConfigObj download, so you may need to download it as well. Now, lets take a look at the
code:

Chapter 30 - ConfigObj

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

import configobj, validate


cfg = """
bmp_select_transparent = boolean(default=False)
canvas_border = integer(min=10, max=35, default=15)
colour1 = list(min=3, max=3, default=list('280', '0', '0'))
colour2 = list(min=3, max=3, default=list('255', '255', '0'))
colour3 = list(min=3, max=3, default=list('0', '255', '0'))
colour4 = list(min=3, max=3, default=list('255', '0', '0'))
colour5 = list(min=3, max=3, default=list('0', '0', '255'))
colour6 = list(min=3, max=3, default=list('160', '32', '240'))
colour7 = list(min=3, max=3, default=list('0', '255', '255'))
colour8 = list(min=3, max=3, default=list('255', '165', '0'))
colour9 = list(min=3, max=3, default=list('211', '211', '211'))
convert_quality = option('highest', 'high', 'normal', default='normal')
default_font = string
default_width = integer(min=1, max=12000, default=640)
default_height = integer(min=1, max=12000, default=480)
imagemagick_path = string
handle_size = integer(min=3, max=15, default=6)
language = option('English', 'English (United Kingdom)', 'Russian',
'Hindi', default='English')
print_title = boolean(default=True)
statusbar = boolean(default=True)
toolbar = boolean(default=True)
toolbox = option('icon', 'text', default='icon')
undo_sheets = integer(min=5, max=50, default=10)
"""
def createConfig(path):
"""
Create a config file using a configspec
and validate it against a Validator object
"""
spec = cfg.split("\n")
config = configobj.ConfigObj(path, configspec=spec)
validator = validate.Validator()
config.validate(validator, copy=True)
config.filename = path
config.write()
if __name__ == "__main__":

197

Chapter 30 - ConfigObj

43

198

createConfig("config.ini")

The configspec allows the programmer the ability to specify what types are returned for each line
in the configuration file. It also can be used to set a default value and a min and max values (among
other things). If you run the code above, you will see a config.ini file generated in the current working
directory that has just the default values. If the programmer didnt specify a default, then that line
isnt even added to the configuration.
Lets take a closer look at whats going on just to make sure you understand. In the createConfig
function, we create a ConfigObj instance by passing in the file path and setting the configspec. Note
that the configspec can also be a normal text file or a python file rather than the string that is in this
example. Next, we create a Validator object. Normal usage is to just call config.validate(validator),
but in this code I set the copy argument to True so that I could create a file. Otherwise, all it would
do is validate that the file I passed in fit the configspecs rules. Finally I set the configs filename and
write the data out.

Wrapping Up
Now you know just enough to get you started on the ins and outs of ConfigObj. I hope youll find it
as helpful as I have. Be sure to go to the modules documentation and read more about what it and
validate can do.

Chapter 31 - Parsing XML with lxml


In Part I, we looked at some of Pythons built-in XML parsers. In this chapter, we will look at the
fun third-party package, lxml from codespeak. It uses the ElementTree API, among other things.
The lxml package has XPath and XSLT support, includes an API for SAX and a C-level API for
compatibility with C/Pyrex modules. Here is what we will cover:

How to Parse XML with lxml


A Refactoring example
How to Parse XML with lxml.objectify
How to Create XML with lxml.objectify

For this chapter, we will use the examples from the minidom parsing example and see how to parse
those with lxml. Heres an XML example from a program that was written for keeping track of
appointments:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
<appointment>
<begin>1234360800</begin>
<duration>1800</duration>
<subject>Check MS Office website for updates</subject>
<location></location>
<uid>604f4792-eb89-478b-a14f-dd34d3cc6c21-1234360800</uid>
<state>dismissed</state>
</appointment>
</zAppointments>

Lets learn how to parse this with lxml!

Chapter 31 - Parsing XML with lxml

200

Parsing XML with lxml


The XML above shows two appointments. The beginning time is in seconds since the epoch; the
uid is generated based on a hash of the beginning time and a key; the alarm time is the number of
seconds since the epoch, but should be less than the beginning time; and the state is whether or not
the appointment has been snoozed, dismissed or not. The rest of the XML is pretty self-explanatory.
Now lets see how to parse it.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

from lxml import etree


def parseXML(xmlFile):
"""
Parse the xml
"""
with open(xmlFile) as fobj:
xml = fobj.read()
root = etree.fromstring(xml)
for appt in root.getchildren():
for elem in appt.getchildren():
if not elem.text:
text = "None"
else:
text = elem.text
print(elem.tag + " => " + text)
if __name__ == "__main__":
parseXML("example.xml")

First off, we import the needed modules, namely the etree module from the lxml package and
the StringIO function from the built-in StringIO module. Our parseXML function accepts one
argument: the path to the XML file in question. We open the file, read it and close it. Now comes
the fun part! We use etrees parse function to parse the XML code that is returned from the StringIO
module. For reasons I dont completely understand, the parse function requires a file-like object.
Anyway, next we iterate over the context (i.e. the lxml.etree.iterparse object) and extract the tag
elements. We add the conditional if statement to replace the empty fields with the word None to
make the output a little clearer. And thats it.

Chapter 31 - Parsing XML with lxml

201

Parsing the Book Example


Well, the result of that example was kind of boring. Most of the time, you want to save the data you
extract and do something with it, not just print it out to stdout. So for our next example, well create
a data structure to contain the results. Our data structure for this example will be a list of dicts.
Well use the MSDN book example here from the earlier chapter again. Save the following XML as
example.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32

<?xml version="1.0"?>
<catalog>
<book id="bk101">
<author>Gambardella, Matthew</author>
<title>XML Developer's Guide</title>
<genre>Computer</genre>
<price>44.95</price>
<publish_date>2000-10-01</publish_date>
<description>An in-depth look at creating applications
with XML.</description>
</book>
<book id="bk102">
<author>Ralls, Kim</author>
<title>Midnight Rain</title>
<genre>Fantasy</genre>
<price>5.95</price>
<publish_date>2000-12-16</publish_date>
<description>A former architect battles corporate zombies,
an evil sorceress, and her own childhood to become queen
of the world.</description>
</book>
<book id="bk103">
<author>Corets, Eva</author>
<title>Maeve Ascendant</title>
<genre>Fantasy</genre>
<price>5.95</price>
<publish_date>2000-11-17</publish_date>
<description>After the collapse of a nanotechnology
society in England, the young survivors lay the
foundation for a new society.</description>
</book>
</catalog>

Now lets parse this XML and put it in our data structure!

Chapter 31 - Parsing XML with lxml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26

202

from lxml import etree


def parseBookXML(xmlFile):
with open(xmlFile) as fobj:
xml = fobj.read()
root = etree.fromstring(xml)
book_dict = {}
books = []
for book in root.getchildren():
for elem in book.getchildren():
if not elem.text:
text = "None"
else:
text = elem.text
print(elem.tag + " => " + text)
book_dict[elem.tag] = text
if book.tag == "book":
books.append(book_dict)
book_dict = {}
return books
if __name__ == "__main__":
parseBookXML("books.xml")

This example is pretty similar to our last one, so well just focus on the differences present here.
Right before we start iterating over the context, we create an empty dictionary object and an empty
list. Then inside the loop, we create our dictionary like this:
1

book_dict[elem.tag] = text

The text is either elem.text or None. Finally, if the tag happens to be book, then were at the end of
a book section and need to add the dict to our list as well as reset the dict for the next book. As you
can see, that is exactly what we have done. A more realistic example would be to put the extracted
data into a Book class. I have done the latter with json feeds before.
Now were ready to learn how to parse XML with lxml.objectify!

Parsing XML with lxml.objectify


The lxml module has a module called objectify that can turn XML documents into Python objects.
I find objectified XML documents very easy to work with and I hope you will too. You may need

Chapter 31 - Parsing XML with lxml

203

to jump through a hoop or two to install it as pip doesnt work with lxml on Windows. Be sure to
go to the Python Package index and look for a version thats been made for your version of Python.
Also note that the latest pre-built installer for lxml only supports Python 3.2 (at the time of writing),
so if you have a newer version of Python, you may have some difficulty getting lxml installed for
your version.
Anyway, once you have it installed, we can start going over this wonderful piece of XML again:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
<appointment>
<begin>1234360800</begin>
<duration>1800</duration>
<subject>Check MS Office website for updates</subject>
<location></location>
<uid>604f4792-eb89-478b-a14f-dd34d3cc6c21-1234360800</uid>
<state>dismissed</state>
</appointment>
</zAppointments>

Now we need to write some code that can parse and modify the XML. Lets take a look at this little
demo that shows a bunch of the neat abilities that objectify provides.
1
2
3
4
5
6
7
8
9
10
11

from lxml import etree, objectify


def parseXML(xmlFile):
"""Parse the XML file"""
with open(xmlFile) as f:
xml = f.read()
root = objectify.fromstring(xml)
# returns attributes in element node as dict
attrib = root.attrib

Chapter 31 - Parsing XML with lxml

12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

204

# how to extract element data


begin = root.appointment.begin
uid = root.appointment.uid
# loop over elements and print their tags and text
for appt in root.getchildren():
for e in appt.getchildren():
print("%s => %s" % (e.tag, e.text))
print()
# how to change an element's text
root.appointment.begin = "something else"
print(root.appointment.begin)
# how to add a new element
root.appointment.new_element = "new data"
# remove the py:pytype stuff
objectify.deannotate(root)
etree.cleanup_namespaces(root)
obj_xml = etree.tostring(root, pretty_print=True)
print(obj_xml)
# save your xml
with open("new.xml", "w") as f:
f.write(obj_xml)
if __name__ == "__main__":
f = r'path\to\sample.xml'
parseXML(f)

The code is pretty well commented, but well spend a little time going over it anyway. First we pass
it our sample XML file and objectify it. If you want to get access to a tags attributes, use the attrib
property. It will return a dictionary of the attributes of the tag. To get to sub-tag elements, you just
use dot notation. As you can see, to get to the begin tags value, we can just do something like this:
1

begin = root.appointment.begin

One thing to be aware of is if the value happens to have leading zeroes, the returned value may have
them truncated. If that is important to you, then you should use the following syntax instead:

Chapter 31 - Parsing XML with lxml

205

begin = root.appointment.begin.text

If you need to iterate over the children elements, you can use the iterchildren method. You may
have to use a nested for loop structure to get everything. Changing an elements value is as simple
as just assigning it a new value.
1

root.appointment.new_element = "new data"

Now were ready to learn how to create XML using lxml.objectify.

Creating XML with lxml.objectify


The lxml.objectify sub-package is extremely handy for parsing and creating XML. In this section, we
will show how to create XML using the lxml.objectify module. Well start with some simple XML
and then try to replicate it. Lets get started!
We will continue using the following XML for our example:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

<?xml version="1.0" ?>


<zAppointments reminder="15">
<appointment>
<begin>1181251680</begin>
<uid>040000008200E000</uid>
<alarmTime>1181572063</alarmTime>
<state></state>
<location></location>
<duration>1800</duration>
<subject>Bring pizza home</subject>
</appointment>
<appointment>
<begin>1234360800</begin>
<duration>1800</duration>
<subject>Check MS Office website for updates</subject>
<location></location>
<uid>604f4792-eb89-478b-a14f-dd34d3cc6c21-1234360800</uid>
<state>dismissed</state>
</appointment>
</zAppointments>

Lets see how we can use lxml.objectify to recreate this XML:

Chapter 31 - Parsing XML with lxml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

from lxml import etree, objectify


def create_appt(data):
"""
Create an appointment XML element
"""
appt = objectify.Element("appointment")
appt.begin = data["begin"]
appt.uid = data["uid"]
appt.alarmTime = data["alarmTime"]
appt.state = data["state"]
appt.location = data["location"]
appt.duration = data["duration"]
appt.subject = data["subject"]
return appt
def create_xml():
"""
Create an XML file
"""
xml = '''<?xml version="1.0" encoding="UTF-8"?>
<zAppointments>
</zAppointments>
'''
root = objectify.fromstring(xml)
root.set("reminder", "15")
appt = create_appt({"begin":1181251680,
"uid":"040000008200E000",
"alarmTime":1181572063,
"state":"",
"location":"",
"duration":1800,
"subject":"Bring pizza home"}
)
root.append(appt)
uid = "604f4792-eb89-478b-a14f-dd34d3cc6c21-1234360800"
appt = create_appt({"begin":1234360800,
"uid":uid,

206

Chapter 31 - Parsing XML with lxml

43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67

207

"alarmTime":1181572063,
"state":"dismissed",
"location":"",
"duration":1800,
"subject":"Check MS Office website for updates"}
)
root.append(appt)
# remove lxml annotation
objectify.deannotate(root)
etree.cleanup_namespaces(root)
# create the xml string
obj_xml = etree.tostring(root,
pretty_print=True,
xml_declaration=True)
try:
with open("example.xml", "wb") as xml_writer:
xml_writer.write(obj_xml)
except IOError:
pass
if __name__ == "__main__":
create_xml()

Lets break this down a bit. We will start with the create_xml function. In it we create an XML root
object using the objectify modules fromstring function. The root object will contain zAppointment
as its tag. We set the roots reminder attribute and then we call our create_appt function using
a dictionary for its argument. In the create_appt function, we create an instance of an Element
(technically, its an ObjectifiedElement) that we assign to our appt variable. Here we use dotnotation to create the tags for this element. Finally we return the appt element back and append it
to our root object. We repeat the process for the second appointment instance.
The next section of the create_xml function will remove the lxml annotation. If you do not do this,
your XML will end up looking like the following:

Chapter 31 - Parsing XML with lxml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

208

<?xml version="1.0" ?>


<zAppointments py:pytype="TREE" reminder="15">
<appointment py:pytype="TREE">
<begin py:pytype="int">1181251680</begin>
<uid py:pytype="str">040000008200E000</uid>
<alarmTime py:pytype="int">1181572063</alarmTime>
<state py:pytype="str"/>
<location py:pytype="str"/>
<duration py:pytype="int">1800</duration>
<subject py:pytype="str">Bring pizza home</subject>
</appointment><appointment py:pytype="TREE">
<begin py:pytype="int">1234360800</begin>
<uid py:pytype="str">604f4792-eb89-478b-a14f-dd34d3cc6c21-1234360800</ui\
d>
<alarmTime py:pytype="int">1181572063</alarmTime>
<state py:pytype="str">dismissed</state>
<location py:pytype="str"/>
<duration py:pytype="int">1800</duration>
<subject py:pytype="str">Check MS Office website for updates</subject>
</appointment>
</zAppointments>

To remove all that unwanted annotation, we call the following two functions:
1
2

objectify.deannotate(root)
etree.cleanup_namespaces(root)

The last piece of the puzzle is to get lxml to generate the XML itself. Here we use lxmls etree module
to do the hard work:
1
2
3

obj_xml = etree.tostring(root,
pretty_print=True,
xml_declaration=True)

The tostring function will return a nice string of the XML and if you set pretty_print to True, it
will usually return the XML in a nice format too. The xml_declaration keyword argument tells the
etree module whether or not to include the first declaration line (i.e. <?xml version=1.0 ?>.

Wrapping Up
Now you know how to use lxmls etree and objectify modules to parse XML. You also know how
to use objectify to create XML. Knowing how to use more than one module to accomplish the same
task can be valuable in seeing how to approach the same problem from different angles. It will also
help you choose the tool that youre most comfortable with.

Chapter 32 - Python Code Analysis


Python code analysis can be a heavy subject, but it can be very helpful in making your programs
better. There are several Python code analyzers that you can use to check your code and see if they
conform to standards. pylint is probably the most popular. Its very configurable, customizable and
pluggable too. It also checks your code to see if it conforms to PEP8, the official style guide of Python
Core and it looks for programming errors too.
Note that pylint checks your code against most, but not all of PEP8s standards. We will spend a
little time learning about another code analysis package that is called pyflakes.

Getting Started with pylint


The pylint package isnt included with Python, so you will need to go to the Python Package Index
(PyPI) or the packages website to download it. You can use the following command to do all the
work for you:
1

pip install pylint

If everything goes as planned, you should now have pylint installed and well be ready to continue.

Analyzing Your Code


Once pylint is installed, you can run it on the command line without any arguments to see what
options it accepts. If that doesnt work, you can type out the full path like this:
1

c:\Python34\Scripts\pylint

Now we need some code to analyze. Heres a piece of code that has four errors in it. Save this to a
file named crummy_code.py:

Chapter 32 - Python Code Analysis

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

210

import sys
class CarClass:
""""""
def __init__(self, color, make, model, year):
"""Constructor"""
self.color = color
self.make = make
self.model = model
self.year = year
if "Windows" in platform.platform():
print("You're using Windows!")
self.weight = self.getWeight(1, 2, 3)
def getWeight(this):
""""""
return "2000 lbs"

Can you spot the errors without running the code? Lets see if pylint can find the issues!
1

pylint crummy_code.py

When you run this command, you will see a lot of output sent to your screen. Heres a partial
example:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

c:\py101>c:\Python34\Scripts\pylint crummy_code.py
No config file found, using default configuration
************* Module crummy_code
C: 2, 0: Trailing whitespace (trailing-whitespace)
C: 5, 0: Trailing whitespace (trailing-whitespace)
C: 12, 0: Trailing whitespace (trailing-whitespace)
C: 15, 0: Trailing whitespace (trailing-whitespace)
C: 17, 0: Trailing whitespace (trailing-whitespace)
C: 1, 0: Missing module docstring (missing-docstring)
C: 3, 0: Empty class docstring (empty-docstring)
C: 3, 0: Old-style class defined. (old-style-class)
E: 13,24: Undefined variable 'platform' (undefined-variable)
E: 16,36: Too many positional arguments for function call (too-many-function-arg\
s)

Chapter 32 - Python Code Analysis

15
16
17
18
19
20

C:
C:
E:
R:
R:
W:

18,
18,
18,
18,
3,
1,

4:
4:
4:
4:
0:
0:

211

Invalid method name "getWeight" (invalid-name)


Empty method docstring (empty-docstring)
Method should have "self" as first argument (no-self-argument)
Method could be a function (no-self-use)
Too few public methods (1/2) (too-few-public-methods)
Unused import sys (unused-import)

Lets take a moment to break this down a bit. First we need to figure out what the letters designate:
C is for convention, R is refactor, W is warning and E is error. pylint found 3 errors, 4 convention
issues, 2 lines that might be worth refactoring and 1 warning. The 3 errors plus the warning were
what I was looking for. We should try to make this crummy code better and reduce the number of
issues. Well fix the imports and change the getWeight function to get_weight since camelCase isnt
allowed for method names. We also need to fix the call to get_weight so it passes the right number
of arguments and fix it so it has self as the first argument. Heres the new code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21

# crummy_code_fixed.py
import platform
class CarClass:
""""""
def __init__(self, color, make, model, year):
"""Constructor"""
self.color = color
self.make = make
self.model = model
self.year = year
if "Windows" in platform.platform():
print("You're using Windows!")
self.weight = self.get_weight(3)
def get_weight(self, this):
""""""
return "2000 lbs"

Lets run this new code against pylint and see how much weve improved the results. For brevity,
well just show the first section again:

Chapter 32 - Python Code Analysis

1
2
3
4
5
6
7
8
9

212

c:\py101>c:\Python34\Scripts\pylint crummy_code_fixed.py
No config file found, using default configuration
************* Module crummy_code_fixed
C: 1,0: Missing docstring
C: 4,0:CarClass: Empty docstring
C: 21,4:CarClass.get_weight: Empty docstring
W: 21,25:CarClass.get_weight: Unused argument 'this'
R: 21,4:CarClass.get_weight: Method could be a function
R: 4,0:CarClass: Too few public methods (1/2)

That helped a lot! If we added docstrings, we could halve the number of issues. Now were ready to
take a look at pyflakes!

Getting Started with pyflakes


The pyflakes project is a part of something known as the Divmod Project. Pyflakes doesnt actually
execute the code it checks much like pylint doesnt execute the code it analyzes. You can install
pyflakes using pip, easy_install or from source.
We will start by running pyflakes against the original version of the same piece of code that we used
with pylint. Here it is again:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

import sys
class CarClass:
""""""
def __init__(self, color, make, model, year):
"""Constructor"""
self.color = color
self.make = make
self.model = model
self.year = year
if "Windows" in platform.platform():
print("You're using Windows!")
self.weight = self.getWeight(1, 2, 3)
def getWeight(this):
""""""
return "2000 lbs"

Chapter 32 - Python Code Analysis

213

As was noted in the previous section, this silly piece of code has 4 issues, 3 of which would stop the
program from running. Lets see what pyflakes can find! Try running the following command and
youll see the following output:
1
2
3

c:\py101>c:\Python34\Scripts\pyflakes.exe crummy_code.py
crummy_code.py:1: 'sys' imported but unused
crummy_code.py:13: undefined name 'platform'

While pyflakes was super fast at returning this output, it didnt find all the errors. The getWeight
method call is passing too many arguments and the getWeight method itself is defined incorrectly
as it doesnt have a self argument. Well, you can actually call the first argument anything you want,
but by convention its usually called self. If you fixed your code according to what pyflakes told you,
your code still wouldnt work.

Wrapping Up
The next step would be to try running pylint and pyflakes against some of your own code or against a
Python package like SQLAlchemy and seeing what you get for output. You can learn a lot about your
own code using these tools. pylint is integrated with many popular Python IDEs, such as Wingware,
Editra, and PyDev. You may find some of the warnings from pylint to be annoying or not even really
applicable. There are ways to suppress such things as deprecation warnings through command line
options. Or you can use the -generate-rcfile to create an example config file that will help you
control pylint. Note that pylint and pyflakes does not import your code, so you dont have to worry
about undesirable side effects.

Chapter 33 - The requests package


The requests package is a more Pythonic replacement for Pythons own urllib. You will find that
requests packages API is quite a bit simpler to work with. You can install the requests library by
using pip or easy_install or from source.

Using requests
Lets take a look at a few examples of how to use the requests package. We will use a series of small
code snippets to help explain how to use this library.
1

>>> r = requests.get("http://www.google.com")

This example returns a Response object. You can use the Response objects methods to learn a lot
about how you can use requests. Lets use Pythons dir function to find out what methods we have
available:
1
2
3
4
5
6
7
8
9
10
11

>>> dir(r)
['__attrs__', '__bool__', '__class__', '__delattr__', '__dict__',
'__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__',
'__getstate__', '__gt__', '__hash__', '__init__', '__iter__', '__le__', '__lt__',
'__module__', '__ne__', '__new__', '__nonzero__', '__reduce__', '__reduce_ex__',
'__repr__', '__setattr__', '__setstate__', '__sizeof__', '__str__', '__subclassh\
ook__',
'__weakref__', '_content', '_content_consumed', 'apparent_encoding', 'close',
'connection', 'content', 'cookies', 'elapsed', 'encoding', 'headers', 'history',
'iter_content', 'iter_lines', 'json', 'links', 'ok', 'raise_for_status', 'raw',
'reason', 'request', 'status_code', 'text', 'url']

If you run the following method, you can see the web pages source code:
1

>>> r.content()

The output from this command is way too long to include in the book, so be sure to try it out yourself.
If youd like to take a look at the web pages headers, you can run the following:

Chapter 33 - The requests package

215

>>> r.headers

Note that the headers attribute returns a dict-like object and isnt a function call. Were not showing
the output as web page headers tend to be too wide to show correctly in a book. There are a bunch
of other great functions and attributes in the Response object. For example, you can get the cookies,
the links in the page, and the status_code that the page returned.
The requests package supports the following HTTP request types: POST, GET, PUT, DELETE, HEAD
and OPTIONS. If the page returns json, you can access it by calling the Response objects json
method. Lets take a look at a practical example.

How to Submit a Web Form


In this section, we will compare how to submit a web form with requests versus urllib. Lets start by
learning how to submit a web form. We will be doing a web search with duckduckgo.com searching
on the term python and saving the result as an HTML file. Well start with an example that uses
urllib:
1
2
3
4
5
6
7
8
9
10
11
12

import urllib.request
import urllib.parse
import webbrowser
data = urllib.parse.urlencode({'q': 'Python'})
url = 'http://duckduckgo.com/html/'
full_url = url + '?' + data
response = urllib.request.urlopen(full_url)
with open("results.html", "wb") as f:
f.write(response.read())
webbrowser.open("results.html")

The first thing you have to do when you want to submit a web form is figure out what the form
is called and what the url is that you will be posting to. If you go to duckduckgos website and
view the source, youll notice that its action is pointing to a relative link, /html. So our url is
http://duckduckgo.com/html. The input field is named q, so to pass duckduckgo a search term,
we have to concatenate the url to the q field. The results are read and written to disk. Now lets
find out how this process differs when using the requests package.
The requests package does form submissions a little bit more elegantly. Lets take a look:

Chapter 33 - The requests package

1
2
3
4
5
6
7

216

import requests
url = 'https://duckduckgo.com/html/'
payload = {'q':'python'}
r = requests.get(url, params=payload)
with open("requests_results.html", "wb") as f:
f.write(r.content)

With requests, you just need to create a dictionary with the field name as the key and the search
term as the value. Then you use requests.get to do the search. Finally you use the resulting requests
object, r, and access its content property which you save to disk.

Wrapping Up
Now you know the basics of the requests package. I would recommend reading the packages online
documentation as it has many additional examples that you may find useful. I personally think this
module is more intuitive to use than the standard librarys equivalent.

Chapter 34 - SQLAlchemy
SQLAlchemy is usually referred to as an Object Relational Mapper (ORM), although it is much
more full featured than any of the other Python ORMs that Ive used, such as SqlObject or the one
thats built into Django. SQLAlchemy was created by a fellow named Michael Bayer. Since Im a
music nut, well be creating a simple database to store album information. A database isnt a database
without some relationships, so well create two tables and connect them. Here are a few other things
well be learning:

Adding data to each table


Modifying data
Deleting data
Basic queries

But first we need to actually make the database, so thats where well begin our journey.
You will need to install SQLAlchemy to follow along in this tutorial. Well use pip to
get this job done:

pip install sqlalchemy

Now were ready to get started!

How to Create a Database


Creating a database with SQLAlchemy is really easy. SQLAlchemy uses a Declarative method for
creating databases. We will write some code to generate the database and then explain how the code
works. If you want a way to view your SQLite database, I would recommend the SQLite Manager
plugin for Firefox. Heres some code to create our database tables:

Chapter 34 - SQLAlchemy

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31

# table_def.py
from sqlalchemy import create_engine, ForeignKey
from sqlalchemy import Column, Date, Integer, String
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy.orm import relationship, backref
engine = create_engine('sqlite:///mymusic.db', echo=True)
Base = declarative_base()
class Artist(Base):
""""""
__tablename__ = "artists"
id = Column(Integer, primary_key=True)
name = Column(String)
class Album(Base):
""""""
__tablename__ = "albums"
id = Column(Integer, primary_key=True)
title = Column(String)
release_date = Column(Date)
publisher = Column(String)
media_type = Column(String)
artist_id = Column(Integer, ForeignKey("artists.id"))
artist = relationship("Artist", backref=backref("albums", order_by=id))
# create tables
Base.metadata.create_all(engine)

If you run this code, then you should see something similar to the following output:

218

219

Chapter 34 - SQLAlchemy

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37

2014-04-03 09:43:57,541 INFO sqlalchemy.engine.base.Engine


in returns' AS VARCHAR(60)) AS anon_1
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
code returns' AS VARCHAR(60)) AS anon_1
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
tists")
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
bums")
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine
CREATE TABLE artists (
id INTEGER NOT NULL,
name VARCHAR,
PRIMARY KEY (id)
)

SELECT CAST('test pla\


()
SELECT CAST('test uni\
()
PRAGMA table_info("ar\
()
PRAGMA table_info("al\
()

2014-04-03 09:43:57,551 INFO sqlalchemy.engine.base.Engine ()


2014-04-03 09:43:57,661 INFO sqlalchemy.engine.base.Engine COMMIT
2014-04-03 09:43:57,661 INFO sqlalchemy.engine.base.Engine
CREATE TABLE albums (
id INTEGER NOT NULL,
title VARCHAR,
release_date DATE,
publisher VARCHAR,
media_type VARCHAR,
artist_id INTEGER,
PRIMARY KEY (id),
FOREIGN KEY(artist_id) REFERENCES artists (id)
)

2014-04-03 09:43:57,661 INFO sqlalchemy.engine.base.Engine ()


2014-04-03 09:43:57,741 INFO sqlalchemy.engine.base.Engine COMMIT

Why did this happen? Because when we created the engine object, we set its echo parameter to
True. The engine is where the database connection information is and it has all the DBAPI stuff in
it that makes communication with your database possible. Youll note that were creating a SQLite
database. Ever since Python 2.5, SQLite has been supported by the language. If you want to connect

Chapter 34 - SQLAlchemy

220

to some other database, then youll need to edit the connection string. Just in case youre confused
about what were talking about, here is the code in question:
1

engine = create_engine('sqlite:///mymusic.db', echo=True)

The string, sqlite:///mymusic.db, is our connection string. Next we create an instance of the
declarative base, which is what well be basing our table classes on. Next we have two classes, Artist
and Album that define what our database tables will look like. Youll notice that we have Columns,
but no column names. SQLAlchemy actually used the variable names as the column names unless
you specifically specify one in the Column definition. Youll note that we are using an id Integer
field as our primary key in both classes. This field will auto-increment. The other columns are pretty
self-explanatory until you get to the ForeignKey. Here youll see that were tying the artist_id to the
id in the Artist table. The relationship directive tells SQLAlchemy to tie the Album class/table to the
Artist table. Due to the way we set up the ForeignKey, the relationship directive tells SQLAlchemy
that this is a many-to-one relationship, which is what we want. Many albums to one artist. You
can read more about table relationships here.
The last line of the script will create the tables in the database. If you run this script multiple times,
it wont do anything new after the first time as the tables are already created. You could add another
table though and then it would create the new one.

How to Insert / Add Data to Your Tables


A database isnt very useful unless it has some data in it. In this section well show you how to
connect to your database and add some data to the two tables. Its much easier to take a look at
some code and then explain it, so lets do that!
1
2
3
4
5
6
7
8
9
10
11
12
13

# add_data.py
import datetime
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker
from table_def import Album, Artist
engine = create_engine('sqlite:///mymusic.db', echo=True)
# create a Session
Session = sessionmaker(bind=engine)
session = Session()
# Create an artist
http://docs.sqlalchemy.org/en/rel_0_7/orm/relationships.html#relationship-patterns

Chapter 34 - SQLAlchemy

14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43

221

new_artist = Artist(name="Newsboys")
new_artist.albums = [Album(title="Read All About It",
release_date=datetime.date(1988,12,1),
publisher="Refuge", media_type="CD")]
# add more albums
more_albums = [Album(title="Hell Is for Wimps",
release_date=datetime.date(1990,7,31),
publisher="Star Song", media_type="CD"),
Album(title="Love Liberty Disco",
release_date=datetime.date(1999,11,16),
publisher="Sparrow", media_type="CD"),
Album(title="Thrive",
release_date=datetime.date(2002,3,26),
publisher="Sparrow", media_type="CD")]
new_artist.albums.extend(more_albums)
# Add the record to the session object
session.add(new_artist)
# commit the record the database
session.commit()
# Add several artists
session.add_all([
Artist(name="MXPX"),
Artist(name="Kutless"),
Artist(name="Thousand Foot Krutch")
])
session.commit()

First we need to import our table definitions from the previous script. Then we connect to the
database with our engine and create something new, the Session object. The session is our handle
to the database and lets us interact with it. We use it to create, modify, and delete records and we
also use sessions to query the database. Next we create an Artist object and add an album. Youll
note that to add an album, you just create a list of Album objects and set the artist objects albums
property to that list or you can extend it, as you see in the second part of the example. At the end of
the script, we add three additional Artists using the add_all. As you have probably noticed by now,
you need to use the session objects commit method to write the data to the database. Now its time
to turn our attention to modifying the data.

Chapter 34 - SQLAlchemy

222

How to Modify Records with SQLAlchemy


What happens if you saved some bad data. For example, you typed your favorite albums title
incorrectly or you got the release date wrong for that fan edition you own? Well you need to learn
how to modify that record! This will actually be our jumping off point into learning SQLAlchemy
queries as you need to find the record that you need to change and that means you need to write a
query for it. Heres some code that shows us the way:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

# modify_data.py
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker
from table_def import Album, Artist
engine = create_engine('sqlite:///mymusic.db', echo=True)
# create a Session
Session = sessionmaker(bind=engine)
session = Session()
# querying for a record in the Artist table
res = session.query(Artist).filter(Artist.name=="Kutless").first()
print(res.name)
# changing the name
res.name = "Beach Boys"
session.commit()
# editing Album data
artist, album = session.query(Artist, Album).filter(
Artist.id==Album.artist_id).filter(Album.title=="Thrive").first()
album.title = "Step Up to the Microphone"
session.commit()

Our first query goes out and looks up an Artist by name using the filter method. The .first() tells
SQLAlchemy that we only want the first result. We could have used .all() if we thought there would
be multiple results and we wanted all of them. Anyway, this query returns an Artist object that
we can manipulate. As you can see, we changed the name from Kutless to Beach Boys and then
committed out changes.
Querying a joined table is a little bit more complicated. This time we wrote a query that queries both
our tables. It filters using the Artist id AND the Album title. It returns two objects: an artist and an
album. Once we have those, we can easily change the title for the album. Wasnt that easy? At this

Chapter 34 - SQLAlchemy

223

point, we should probably note that if we add stuff to the session erroneously, we can rollback our
changes/adds/deletes by using session.rollback(). Speaking of deleting, lets tackle that subject!

How to Delete Records in SQLAlchemy


Sometimes you just have to delete a record. Whether its because youre involved in a cover-up or
because you dont want people to know about your love of Britney Spears music, you just have
to get rid of the evidence. In this section, well show you how to do just that! Fortunately for us,
SQLAlchemy makes deleting records really easy. Just take a look at the following code!
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

# deleting_data.py
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker
from table_def import Album, Artist
engine = create_engine('sqlite:///mymusic.db', echo=True)
# create a Session
Session = sessionmaker(bind=engine)
session = Session()
res = session.query(Artist).filter(Artist.name=="MXPX").first()
session.delete(res)
session.commit()

As you can see, all you had to do was create another SQL query to find the record you want to
delete and then call session.delete(res). In this case, we deleted our MXPX record. Some people
think punk will never die, but they must not know any DBAs! Weve already seen queries in action,
but lets take a closer look and see if we can learn anything new.

The Basic SQL Queries of SQLAlchemy


SQLAlchemy provides all the queries youll probably ever need. Well be spending a little time just
looking at a few of the basic ones though, such as a couple simple SELECTs, a JOINed SELECT and
using the LIKE query. Youll also learn where to go for information on other types of queries. For
now, lets look at some code:

Chapter 34 - SQLAlchemy

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33

224

# queries.py
from sqlalchemy import create_engine
from sqlalchemy.orm import sessionmaker
from table_def import Album, Artist
engine = create_engine('sqlite:///mymusic.db', echo=True)
# create a Session
Session = sessionmaker(bind=engine)
session = Session()
# how to do a SELECT * (i.e. all)
res = session.query(Artist).all()
for artist in res:
print(artist.name)
# how to SELECT the first result
res = session.query(Artist).filter(Artist.name=="Newsboys").first()
# how to sort the results (ORDER_BY)
res = session.query(Album).order_by(Album.title).all()
for album in res:
print(album.title)
# how to do a JOINed query
qry = session.query(Artist, Album)
qry = qry.filter(Artist.id==Album.artist_id)
artist, album = qry.filter(Album.title=="Step Up to the Microphone").first()
# how to use LIKE in a query
res = session.query(Album).filter(Album.publisher.like("S%a%")).all()
for item in res:
print(item.publisher)

The first query we run will grab all the artists in the database (a SELECT ) and print out each of their
name fields. Next youll see how to just do a query for a specific artist and return just the first result.
The third query shows how do a SELECT on the Album table and order the results by album title. The
fourth query is the same query (a query on a JOIN) we used in our editing section except that weve
broken it down to better fit PEP8 standards regarding line length. Another reason to break down
long queries is that they become more readable and easier to fix later on if you messed something
up. The last query uses LIKE, which allows us to pattern match or look for something thats like a
specified string. In this case, we wanted to find any records that had a Publisher that started with a

Chapter 34 - SQLAlchemy

225

capital S, some character, an a and then anything else. So this will match the publishers Sparrow
and Star, for example.
SQLAlchemy also supports IN, IS NULL, NOT, AND, OR and all the other filtering keywords that
most DBAs use. SQLAlchemy also supports literal SQL, scalars, etc, etc.

Wrapping Up
At this point you should know SQLAlchemy well enough to get started using it confidently. The
project also has excellent documentation that you should be able to use to answer just about anything
you need to know. If you get stuck, the SQLAlchemy users group / mailing list is very responsive to
new users and even the main developers are there to help you figure things out.

Chapter 35 - virtualenv
Virtual environments can be really handy for testing software. Thats true in programming circles
too. Ian Bicking created the virtualenv project, which is a tool for creating isolated Python
environments. You can use these environments to test out new versions of your software, new
versions of packages you depend on or just as a sandbox for trying out some new package in general.
You can also use virtualenv as a workspace when you cant copy files into site-packages because its
on a shared host. When you create a virtual environment with virtualenv, it creates a folder and
copies Python into it along with a site-packages folder and a couple others. It also installs pip. Once
your virtual environment is active, its just like using your normal Python. And when youre done,
you can just delete the folder to cleanup. No muss, no fuss. Alternatively, you can keep on using it
for development.
In this chapter, well spend some time getting to know virtualenv and how to use it to make our own
magic.

Installation
First of all, you probably need to install virtualenv. You can use pip or easy_install to install it or you
can download the virtualenv.py file from their website and install it from source using its setup.py
script.
If you have Python 3.4, you will find that you actually have the venv module, which follows an
API that is very similar to the virtualenv package. This chapter will focus on just the virtualenv
package, however.

Creating a Virtual Environment


Creating a virtual sandbox with the virtualenv package is quite easy. All you need to do is the
following:
1

python virtualenv.py FOLDER_NAME

Where FOLDER_NAME is the name of the folder that you want your sandbox to go. On my
Windows 7 machine, I have C:\Python34\Scripts added to my path so I can just call virtualenv.py
FOLDER_NAME without the python part. If you dont pass it anything, then youll get a list of
options printed out on your screen. Lets say we create a project called sandbox. How do we use it?
Well, we need to activate. it. Heres how:

Chapter 35 - virtualenv

227

On Posix you would do source bin/activate while on Windows, you would do .\path\to\env\Scripts\activate.
on the command line. Lets actually go through these steps. Well create the sandbox folder on our
desktop so you can see an example. Heres what it looks like on my machine:
1
2
3
4
5
6
7

C:\Users\mdriscoll\Desktop>virtualenv sandbox
New python executable in sandbox\Scripts\python.exe
Installing setuptools................done.
Installing pip...................done.
C:\Users\mdriscoll\Desktop>sandbox\Scripts\activate
(sandbox) C:\Users\mdriscoll\Desktop>

Youll note that once your virtual environment is activated, youll see your prompt change to include
a prefix of the folder name that you created, which is sandbox in this case. This lets you know
that youre using your sandbox. Now you can use pip to install other packages to your virtual
environment. When youre done, you just call the deactivate script to exit the environment.
There are a couple of flags you can pass to virtualenv when creating your virtual playground that
you should be aware of. For example, you can use -system-site-packages to inherit the packages
from your default Pythons site packages. If you want to use distribute rather than setuptools, you
can pass virtualenv the -distribute flag.
virtualenv also provides a way for you to just install libraries but use the system Python itself to run
them. According to the documentation, you just create a special script to do it.
Theres also a neat (and experimental) flag called -relocatable that can be used to make the folder
relocatable. However, this does NOT work on Windows at the time of this writing, so I wasnt able
to test it out.
Finally, theres an -extra-search-dir flag that you can use to keep your virtual environment offline.
Basically it allows you to add a directory to the search path for distributions that pip or easy_install
can install from. In this way, you dont need to have access to the internet to install packages.

Wrapping Up
At this point, you should be able to use virtualenv yourself. There are a couple of other projects
worth mentioning at this point. Theres Doug Hellmans virtualenvwrapper library that makes it
even easier to create, delete and manage virtual environments and then theres zc.buildout which
is probably the closest thing to virtualenv that could be called a competitor. I recommend checking
them both out as they might help you in your programming adventures.

Part V - Packaging and Distribution


In Part V you will learn about Python packaging and the various methods to distribute your code.
You will learn about the following:

How to create a module and package


Publishing your Packages to the Python Packaging Index (PyPI)
Python eggs
Python wheels
py2exe
bb_freeze
cx_Freeze
PyInstaller
GUI2Exe
How to Create an Installer with InnoSetup

The first chapter of this section describes how to create a module or package. Then in the following
chapter, we will go over publishing our package to PyPI. Next up, we will learn how to create and
install a Python egg and the Python wheel.
The next four chapters will cover how to create binaries using the following 3rd party packages:
py2exe, bb_freeze, cx_Freeze and PyInstaller. The only package in this list that is actually compatible
with Python 3 is cx_Freeze. Because of this fact, we will be showing Python 2 examples in these four
chapters so that we can easily compare all 4 packages and their capabilities.
The next to last chapter will show you how to use GUI2Exe, a neat little user interface that was
created to go on top of py2exe, bb_freeze, etc. GUI2Exe makes creating binaries even easier!
The last chapter of this section will show how to create an installer using InnoSetup. Lets get started!

229

Part V - Packaging and Distribution

image

Chapter 36 - Creating Modules and


Packages
Creating Python modules is something that most Python programmers do every day without even
thinking about it. Any time you save a new Python script, you have created a new module. You can
import your module into other modules. A package is a collection of related modules. The things
you import into your scripts from the standard library are modules or packages. In this chapter, we
will learn how to create modules and packages. Well spend more time on packages since theyre
more complicated than modules.

How to Create a Python Module


We will begin by creating a super simple module. This module will provide us with basic arithmetic
and no error handling. Heres our first example:
1
2
3
4
5
6
7
8
9
10
11

def add(x, y):


return x + y
def division(x, y):
return x / y
def multiply(x, y):
return x * y
def subtract(x, y):
return x - y

This code has issues, of course. If you pass in two Integers to the division method, then youll end
up getting an Integer back if you are using Python 2.x. This may not be what youre expecting.
Theres also no error checking for division by zero or mixing of strings and numbers. But thats not
the point. The point is that if you save this code, you have a fully qualified module. Lets call it
arithmetic.py. Now what can you do with a module anyway? You can import it and use any of the
defined functions or classes that are inside it. And we could make it executable with a little spit and
polish. Lets do both!
First well write a little script that imports our module and runs the functions in it. Save the following
as math_test.py:

Chapter 36 - Creating Modules and Packages

1
2
3
4
5
6

231

import arithmetic
print(arithmetic.add(5, 8))
print(arithmetic.subtract(10, 5))
print(arithmetic.division(2, 7))
print(arithmetic.multiply(12, 6))

Now lets modify the original script so that we can run it from the command line. Heres one of the
simplest ways to do it:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

def add(x, y):


return x + y
def division(x, y):
return x / y
def multiply(x, y):
return x * y
def subtract(x, y):
return x - y
if __name__ == "__main__":
import sys
print(sys.argv)
v = sys.argv[1].lower()
valOne = int(sys.argv[2])
valTwo = int(sys.argv[3])
if v == "a":
print(add(valOne, valTwo))
elif v == "d":
print(division(valOne, valTwo))
elif v == "m":
print(multiply(valOne, valTwo))
elif v == "s":
print(subtract(valOne, valTwo))
else:
pass

The proper way to do this script would be to use Pythons optparse (pre-2.7) or argparse (2.7+)
module. You should spend some time to figure out one of these modules as a learning exercise. In
the meantime, we will move on to packages!

Chapter 36 - Creating Modules and Packages

232

How to Create a Python Package


The main difference between a module and a package is that a package is a collection of modules
AND it has an __init__.py file. Depending on the complexity of the package, it may have more than
one __init__.py. Lets take a look at a simple folder structure to make this more obvious, then well
create some simple code to follow the structure we define.
1
2
3
4
5
6
7
8
9

mymath/
__init__.py
adv/
__init__.py
sqrt.py
add.py
subtract.py
multiply.py
divide.py

Now we just need to replicate this structure in our own package. Lets give that a whirl! Create each
of these files in a folder tree like the above example. For the add, subtract, multiply and divide files,
you can use the functions we created in the earlier example. For the sqrt.py module, well use the
following code.
1
2
3
4
5

# sqrt.py
import math
def squareroot(n):
return math.sqrt(n)

You can leave both __init__.py files blank, but then youll have to write code like mymath.add.add(x,y)
which is pretty ugly, so well add the following code to the outer __init__.py to make using our
package easier to understand and use.
1
2
3
4
5
6

# outer __init__.py
from . add import add
from . divide import division
from . multiply import multiply
from . subtract import subtract
from .adv.sqrt import squareroot

Now we should be able to use our module once we have it on our Python path. You can copy the
folder into your Pythons site-packages folder to do this. On Windows its in the following general
location: C:\Python34\Lib\site-packages. Alternatively, you can edit the path on the fly in your
test code. Lets see how thats done:

Chapter 36 - Creating Modules and Packages

1
2
3
4
5
6
7
8
9
10
11

233

import sys
# modify this path to match your environment
sys.path.append('C:\Users\mdriscoll\Documents')
import mymath
print(mymath.add(4,5))
print(mymath.division(4, 2))
print(mymath.multiply(10, 5))
print(mymath.squareroot(48))

Note that my path does NOT include the mymath folder. You want to append the parent folder that
holds your new module, NOT the module folder itself. If you do this, then the code above should
work.
You can also create a setup.py script and install your package in develop mode. Heres an example
setup.py script:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

#!/usr/bin/env python
from setuptools import setup

# This setup is suitable for "python setup.py develop".


setup(name='mymath',
version='0.1',
description='A silly math package',
author='Mike Driscoll',
author_email='mike@mymath.org',
url='http://www.mymath.org/',
packages=['mymath', 'mymath.adv'],
)

You would save this script one level above the mymath folder. To install the package in develop
mode, you would do the following:
1

python setup.py develop

This will install a link file in the site-packages folder that points to where ever your package resides.
This is great for testing without actually installing your package.
Congratulations! Youve just created a Python package!

Chapter 36 - Creating Modules and Packages

234

Wrapping Up
Youve just learned how to create your very own, custom-made modules and packages. You will
find that the more you code, the more often youll create programs that have parts that you want
to re-use. You can put those reusable pieces of code into modules. Eventually you will have enough
related modules that you may want to turn them into a package. Now you have the tools to actually
do that!

Chapter 37 - How to Add Your Code to


PyPI
We created a package called mymath in the previous chapter. In this chapter, we will learn how to
share it on the Python Packaging Index (PyPI). To do that, we will first need to learn how to create
a setup.py file. Just to review, here is our current folder hierarchy:
1
2
3
4
5
6
7
8
9

mymath/
__init__.py
adv/
__init__.py
sqrt.py
add.py
subtract.py
multiply.py
divide.py

This means that you have a mymath folder with the following files in it: __init__.py, add.py,
subtract.py, multiply.py and divide.py. You will also have an adv folder inside the mymath folder.
In the adv folder, you will have two files: __init__.py and sqrt.py.

Creating a setup.py File


We will start out by creating a super simple setup.py script. Heres a bare-bones one:
1
2
3
4
5
6

from distutils.core import setup


setup(name='mymath',
version='0.1',
packages=['mymath', 'mymath.adv'],
)

This is something you might write for a package internally. To upload to PyPI, you will need to
include a little more information:

Chapter 37 - How to Add Your Code to PyPI

1
2
3
4
5
6
7
8
9
10

236

from distutils.core import setup


setup(name='mymath',
version='0.1',
description='A silly math package',
author='Mike Driscoll',
author_email='mike@mymath.org',
url='http://www.mymath.org/',
packages=['mymath', 'mymath.adv'],
)

Now that were done with that, we should test our script. You can create a virtual environment using
the directions from chapter 35 or you can just install your code to your Python installation by calling
the following command:
1

python setup.py install

Alternatively, you can use the method at the end of the last chapter in which you created a special
setup.py that you installed in develop mode. You will note that in the last chapter, we used
setuptools whereas in this chapter we used distutils. The only reason we did this is that setuptools
has the develop command and distutils does not.
Now we need to register our package with PyPI!

Registering Packages
Registering you package is very easy. Since this is your first package, you will want to register with
the Test PyPI server instead of the real one. You may need to create a .pypirc file and enter the Test
PyPI server address. See the next section for more information. Once you have done that, you just
need to run the following command:
1

python setup.py register

You will receive a list of options that ask you to login, register, have the server send you a password or
quit. If you have your username and password saved on your machine, you wont see that message.
If youre already registered, you can login and your packages metadata will be uploaded.

Chapter 37 - How to Add Your Code to PyPI

237

Uploading Packages to PyPI


You will probably want to start out by testing with PyPIs test server, which is at https://testpypi.
python.org/pypi. You will have to register with that site too as it uses a different database than
the main site. Once youve done that, you may want to create a .pypirc file somewhere on your
operating systems path. On Linux, you can use $HOME to find it and on Windows, you can use
the HOME environ variable. This path is where you would save that file. Following is a sample of
what could go in your pypirc file from https://wiki.python.org/moin/TestPyPI:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

[distutils]
index-servers=
pypi
test
[test]
repository = https://testpypi.python.org/pypi
username = richard
password = <your password goes here>
[pypi]
repository = http://pypi.python.org/pypi
username = richard
password = <your password goes here>

I would highly recommend that you read the documentation in depth to understand all the options
you can add to this configuration file.
To upload some files to PyPI, you will need to create some distributions.
1

python setup.py sdist bdist_wininst upload

When you run the command above, it will create a dist folder. The sdist command will create an
archive file (a zip on Windows, a tarball on Linux). The bdist_wininst will create a simple Windows
installer executable. The upload command will then upload these two files to PyPI.
In your setup.py file, you can add a long_description field that will be used by PyPI to generate a
home page for your package on PyPI. You can use reStructuredText to format your description. Or
you can skip adding the description and accept PyPIs default formatting.
If you would like a full listing of the commands you can use with setup.py, try running the following
command:

Chapter 37 - How to Add Your Code to PyPI

238

python setup.py --help-commands

You should also add a README.txt file that explains how to install and use your package. It can
also contain a Thank You section if you have a lot of contributors.

Wrapping Up
Now you know the basics for adding your package to the Python Packaging Index. If you want to
add a Python egg to PyPI, you will need to use easy_install instead of distutils. When you release
your next version, you may want to add a CHANGES.txt file that lists the changes to your code.
There is a great website called The Hitchhikers Guide to Packaging which would be a great place
for you to check out for additional information on this exciting topic. Alternatively, you may also
want to check out this tutorial by Scott Torborg to get a different take on the process.
http://www.scotttorborg.com/python-packaging/index.html

Chapter 38 - The Python egg


Python eggs are an older distribution format for Python. The new format is called a Python wheel,
which we will look at in the next chapter. An egg file is basically a zip file with a different extension.
Python can import directly from an egg. You will need the SetupTools package to work with eggs.
SetupTools is the original mainstream method of downloading and installing Python packages from
PyPI and other sources via the command line, kind of like apt-get for Python. There was a fork of
SetupTools called distribute that was eventually merged back into SetupTools. I only mention it
because you may see references to that fork if you do much reading about Python eggs outside of
this book.
While the egg format is being migrated away from, you do still need to know about it as there are
many packages that are distributed using this technology. It will probably be years before everyone
has stopped using eggs. Lets learn how to make our own!

Creating an egg
You can think of an egg as just an alternative to a source distribution or Windows executable, but it
should be noted that for pure Python eggs, the egg file is completely cross-platform. We will take a
look at how to create our own egg using the package we created in a previous modules and packages
chapter. To get started creating an egg, you will need to create a new folder and put the mymath
folder inside it. Then create a setup.py file in the parent directory to mymath with the following
contents:
1
2
3
4
5
6
7

from setuptools import setup, find_packages


setup(
name = "mymath",
version = "0.1",
packages = find_packages()
)

Python has its own package for creating distributions that is called distutils. However instead of
using Pythons distutils setup function, were using setuptools setup. Were also using setuptools
find_packages function which will automatically look for any packages in the current directory and
add them to the egg. To create said egg, youll need to run the following from the command line:

Chapter 38 - The Python egg

240

c:\Python34\python.exe setup.py bdist_egg

This will generate a lot of output, but when its done youll see that you have three new folders: build,
dist, and mymath.egg-info. The only one we care about is the dist folder in which you will find
your egg file, mymath-0.1-py3.4.egg. Note that on my machine, I forced it to run against Python
3.4 so that it would create the egg against that version of Python. The egg file itself is basically a
zip file. If you change the extension to zip, you can look inside it and see that it has two folders:
mymath and EGG-INFO. At this point, you should be able to point easy_install at your egg on
your file system and have it install your package.

Wrapping Up
Now its your turn. Go onto the Python Package Index and find some pure Python modules to
download. Then try creating eggs using the techniques you learned in this chapter. If you want to
install an egg, you can use easy_install. Uninstalling an egg is a bit tougher. You will have to go
to its install location and delete the folder and / or egg file it installed as well as remove the entry
for the package from the easy-install.pth file. All of these items can be found in your Pythons
site-packages folder.

Chapter 39 - Python wheels


Pythons first mainstream packaging format was the .egg file. Now theres a new format in town
called the wheel (.whl). According to the Python Packaging Indexs description, a wheelis designed to
contain all the files for a PEP 376 compatible install in a way that is very close to the on-disk format*.
In this chapter, we will learn how to create a wheel and then install our wheel in a virtualenv.

Getting Started
Using pip is the recommended way to work with wheels. Make sure you have installed the latest
copy of pip as older versions did not support the wheel format. If youre not sure if you have the
latest pip, you can run the following command:
1

pip install --upgrade pip

If you didnt have the latest, then this command will upgrade pip. Now were ready to create a wheel!

Creating a wheel
First of all, you will need to install the wheel package:
1

pip install wheel

That was easy! Next, well be using the unidecode package for creating our first wheel as it doesnt
already have one made at the time of writing and Ive used this package myself in several projects.
The unidecode package will take a string of text and attempt to replace any unicode with its
ASCII equivalent. This is extremely handy for when you have to scrub user provided data of weird
anomalies. Heres the command you should run to create a wheel for this package:
1

pip wheel --wheel-dir=my_wheels Unidecode

Heres a screenshot of the output I received when I ran this:

242

Chapter 39 - Python wheels

image

Now you should have a wheel named Unidecode-0.04.14-py26-none-any.whl in a folder named


my_wheels. Lets learn how to install our new wheel!

Installing a Python wheel


Lets create a virtualenv to test with. We will use the following command to create our virtual testing
environment:
1

virtualenv test

This assumes that virtualenv is on your system path. If you get an unrecognized command error,
then youll probably have to specify the full path (i.e. c:\Python34\Scripts\virtualenv). Running
this command will create a virtual sandbox for us to play in that includes pip. Be sure to run the
activate script from the test folders Scripts folder to enable the virtuanenv before continuing. Your
virtualenv does not include wheel, so youll have to install wheel again:
1

pip install wheel

Once that is installed, we can install our wheel with the following command:

Chapter 39 - Python wheels

243

pip install --use-wheel --no-index --find-links=path/to/my_wheels Unidecode

To test that this worked, run Python from the Scripts folder in your virtualenv and try importing
unidecode. If it imports, then you successfully installed your wheel!
The .whl file is similar to an.egg in that its basically a .zip file in disguise. If you rename the extension
from.whl to *.zip, you can open it up with your zip application of choice and examine the files and
folders inside at your leisure.

Wrapping Up
Now you should be ready to create your own wheels. They are a nice way to create a local repository
of dependencies for your project(s) that you can install quickly. You could create several different
wheel repositories to easily switch between different version sets for testing purposes. When
combined with virtualenv, you have a really easy way to see how newer versions of dependencies
could affect your project without needing to download them multiple times.

Chapter 40 - py2exe
The py2exe project used to be the primary way to create Windows executables from your Python
applications. The regular version of py2exe just supports Python 2.3-2.7. There is a new version listed
on PyPI that will work with Python 3.4 as well. We will focus on the Python 2.x version, although
this chapter should work with the Python 3 version too.
You have several choices for your application. You can create a program that only runs in a terminal,
you can create a desktop graphical user interface (GUI) or you can create a web application. We will
create a very simple desktop interface that doesnt do anything except display a form that the user
can fill out. We will use the wxPython GUI toolkit to help demonstrate how py2exe can pick up
packages without us telling it to.

Creating a Simple GUI


You will want to go to wxPythons website (www.wxpython.org) and download a copy that matches
your Python version. If you have a 32-bit Python, make sure you download a 32-bit wxPython. You
cannot use easy_install or pip to install wxPython unless you get the bleeding edge Phoenix version
of wxPython, so youll have to grab a copy that is pre-built for your system either from the wxPython
website or from your systems package manager. I recommend using at least wxPython 2.9 or higher.
Lets write some code!
1
2
3
4
5
6
7
8
9
10
11
12
13
14

import wx
class DemoPanel(wx.Panel):
""""""
def __init__(self, parent):
"""Constructor"""
wx.Panel.__init__(self, parent)
labels = ["Name", "Address", "City", "State", "Zip",
"Phone", "Email", "Notes"]
mainSizer = wx.BoxSizer(wx.VERTICAL)
lbl = wx.StaticText(self, label="Please enter your information here:")
https://pypi.python.org/pypi/py2exe/0.9.2.0/

Chapter 40 - py2exe

15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56

lbl.SetFont(wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD))


mainSizer.Add(lbl, 0, wx.ALL, 5)
for lbl in labels:
sizer = self.buildControls(lbl)
mainSizer.Add(sizer, 1, wx.EXPAND)
self.SetSizer(mainSizer)
mainSizer.Layout()
def buildControls(self, label):
"""
Put the widgets together
"""
sizer = wx.BoxSizer(wx.HORIZONTAL)
size = (80,40)
font = wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD)
lbl = wx.StaticText(self, label=label, size=size)
lbl.SetFont(font)
sizer.Add(lbl, 0, wx.ALL|wx.CENTER, 5)
if label != "Notes":
txt = wx.TextCtrl(self, name=label)
else:
txt = wx.TextCtrl(self, style=wx.TE_MULTILINE, name=label)
sizer.Add(txt, 1, wx.ALL, 5)
return sizer
class DemoFrame(wx.Frame):
"""
Frame that holds all other widgets
"""
def __init__(self):
"""Constructor"""
wx.Frame.__init__(self, None, wx.ID_ANY,
"Py2Exe Tutorial",
size=(600,400)
)
panel = DemoPanel(self)
self.Show()
if __name__ == "__main__":
app = wx.App(False)

245

246

Chapter 40 - py2exe

57
58

frame = DemoFrame()
app.MainLoop()

If you run the code above, you should see something like the following:

image

Lets break this down a bit. We create two classes, DemoPanel and DemoFrame. In wxPython, the
wx.Frame object is what you use to create the actual window that you see in most cases. You add
a wx.Panel to give your application the proper look and feel and to add tabbing between fields. The
panel objects parent is the frame. The frame, being the top level widget, has no parent. The panel
contains all the other widgets in this example. We use sizers to help layout the widgets. Sizers allow
the developer to create widgets that will resize appropriately when the window itself is resized. You
can also place the widgets on the panel using absolute positioning, which is not recommended. We
call the MainLoop method of the wx.App object at the end to start the event loop, which allows
wxPython to respond to mouse and keyboard events (like clicking, typing, etc).
Now were ready to learn how to package this application up into an executable!
Note: I tested on Windows 7 using Python 2.7.3, wxPython 2.9.4.0 (classic) and py2exe 0.6.9.

The py2exe setup.py file


The key to any py2exe script is the setup.py file. This file controls what gets included or excluded,
how much we compress and bundle, and much more! Here is the simplest setup that we can use
with the wx script above:

247

Chapter 40 - py2exe

1
2
3
4

from distutils.core import setup


import py2exe
setup(windows=['sampleApp.py'])

As you can see, we import the setup method from distutils.core and then we import py2exe. Next
we call setup with a windows keyword parameter and pass it the name of the main file inside a
python list object. If you were creating a non-GUI project, than you would use the console key
instead of windows. To run this snippet, save it into the same folder as your wxPython script, open
up a command prompt and navigate to the location that you saved the two files. Then type python
setup.py py2exe to run it. If all goes well, you will see a lot of output ending with something like
this:

image

If you happen to use Python 2.6, you might get an error about MSVCP90.dll not being found. Should
you see that error, you will probably need to go find the Microsoft Visual C++ 2008 Redistributable
Package and install it to make the DLL available on your system. Occasionally you will create the
executable and then when you run it, it just wont load correctly. A log file is normally created when

Chapter 40 - py2exe

248

this happens that you can use to try to figure out what happened. I have also found a tool called
Dependency Walker that you can run against your executable and it can tell you about non-Python
items that are missing (like DLLs, etc).
I would like to point out that the setup.py file doesnt explicitly include wxPython. That means that
py2exe was smart enough to include the wxPython package automatically. Lets spend some time
learning a bit more about including and excluding packages.

Creating an Advanced setup.py File


Lets see what other options py2exe gives us for creating binaries by creating a more complex
setup.py file.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

from distutils.core import setup


import py2exe
includes = []
excludes = ['_gtkagg', '_tkagg', 'bsddb', 'curses', 'email', 'pywin.debugger',
'pywin.debugger.dbgcon', 'pywin.dialogs', 'tcl',
'Tkconstants', 'Tkinter']
packages = []
dll_excludes = ['libgdk-win32-2.0-0.dll', 'libgobject-2.0-0.dll', 'tcl84.dll',
'tk84.dll']
setup(
options = {"py2exe": {"compressed": 2,
"optimize": 2,
"includes": includes,
"excludes": excludes,
"packages": packages,
"dll_excludes": dll_excludes,
"bundle_files": 3,
"dist_dir": "dist",
"xref": False,
"skip_archive": False,
"ascii": False,
"custom_boot_script": '',
}
},
windows=['sampleApp.py']
)

Chapter 40 - py2exe

249

This is pretty self-explanatory, but lets unpack it anyway. First we set up a few lists that we pass to
the options parameter of the setup function.
The includes list is for special modules that you need to specifically include.
Sometimes py2exe cant find certain modules, so you get to manually specify them
here.
The excludes list is a list of which modules to exclude from your program. In this
case, we dont need Tkinter since were using wxPython. This list of excludes is
what GUI2Exe will exclude by default.
The packages list is a list of specific packages to include. Again, sometimes py2exe
just cant find something. Ive had to include email, PyCrypto, or lxml here before.
Note that if the excludes list contains something youre trying to include in the
packages or includes lists, py2exe may continue to exclude them.
dll_excludes - excludes dlls that we dont need in our project.
In the options dictionary, we have a few other options to look at. The compressed key tells py2exe
whether or not to compress the zipfile, if its set. The optimize key sets the optimization level. Zero
is no optimization and 2 is the highest. By setting optimize to 2, we can reduce the size of folder
by about one megabyte. The bundle_files key bundles dlls in the zipfile or the exe. Valid values for
bundle_files are:
1 = bundle everything, including the Python interpreter.
2 = bundle everything but the Python interpreter
3 = dont bundle (default)
A few years ago, when I was first learning py2exe, I asked on their mailing list what the best option
was because I was having issues with bundle option 1. I was told that 3 was probably the most stable.
I went with that and stopped having random problems, so thats what I currently recommend. If you
dont like distributing more than one file, zip them up or create an installer. The only other option
I use in this list is the dist_dir one. I use it to experiment with different build options or to create
custom builds when I dont want to overwrite my main good build. You can read about all the other
options on the py2exe website.
The py2exe package does not support including Python eggs in its binaries, so if you have installed a
package that your application depends on as an egg, when you go to create the executable, it wont
work. You will have to make sure your dependencies are installed normally.

Wrapping Up
At this point, you should know enough to get started using py2exe yourself. You can get busy and
start distributing your latest creation now. It should be noted that there are several alternatives to
py2exe, such as bbfreeze, cx_freeze and PyInstaller. You should try at least a couple of the others
to see how they compare. Creating executables can be frustrating, but have patience and persevere
through it. The Python packaging community is quite willing to help. All you need to do is ask.

Chapter 41 - bbfreeze
The bbfreeze package also allows us to create binaries, but only on Linux and Windows. When
you create a binary on Linux, the result will only run on machines that have the same hardware
architecture and version of libc, which limits its usefulness on Linux. It should also be noted that
bbfreeze only works with Python versions 2.4 - 2.7. You can use easy_install or pip to install the
bbfreeze package onto your system. The bbfreeze package includes egg support, so it can include
egg dependencies in your binary, unlike py2exe. You can also freeze multiple scripts at once, include
the Python interpreter and more.

Getting Started with bbfreeze


You can use easy_install to download and install bbfreeze or you can just download its source or
the egg file directly from the Python Package Index (PyPI). In this article, well try using it on a
simple configuration file generator script and well also try it against the wxPython program from
the py2exe chapter.
Note: I tested on Windows 7 using Python 2.7.3, wxPython 2.9.4.0 (classic) and bbfreeze 1.1.3
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

# config_1.py
import configobj
def createConfig(configFile):
"""
Create the configuration file
"""
config = configobj.ConfigObj()
inifile = configFile
config.filename = inifile
config['server'] = "http://www.google.com"
config['username'] = "mike"
config['password'] = "dingbat"
config['update interval'] = 2
config.write()
def getConfig(configFile):
"""
Open the config file and return a configobj

Chapter 41 - bbfreeze

20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36

251

"""
return configobj.ConfigObj(configFile)
def createConfig2(path):
"""
Create a config file
"""
config = configobj.ConfigObj()
config.filename = path
config["Sony"] = {}
config["Sony"]["product"] = "Sony PS3"
config["Sony"]["accessories"] = ['controller', 'eye', 'memory stick']
config["Sony"]["retail price"] = "$400"
config.write()
if __name__ == "__main__":
createConfig2("sampleConfig2.ini")

This script has a couple of functions that are pretty pointless, but well leave them in for illustrative
purposes. According to the bbfreeze documentation, we should be able to create a binary with the
following string typed into the command line:
1

bb-freeze config_1.py

This assumes that you have C:\Python27\Scripts on your path. If you dont, youll need to type the
complete path out (i.e. C:\Python27\Scripts\bb-freeze config_1.py). If you run this, you should see
a folder named dist get created. Heres what mine looked like after I ran config_1.exe:

252

Chapter 41 - bbfreeze

image

You will note that when you run the executable, it creates a config file named sampleconfig2.ini.
You may see a warning about the pywin32 package not being installed. You can ignore the warning
or download and install pywin32.
Now were ready to move on and try to create an executable from code that use wxPython!

Using bbfreezes Advanced Configuration


The PyPI page for bbfreeze (which is also its home page) has very little documentation. However,
the page does say that the preferred way to use bbfreeze is with little scripts. Were going to try
creating a binary with the wxPython example, mentioned earlier. Heres the wx code:
1
2
3
4
5
6
7
8
9
10
11
12

import wx
class DemoPanel(wx.Panel):
""""""
def __init__(self, parent):
"""Constructor"""
wx.Panel.__init__(self, parent)
labels = ["Name", "Address", "City", "State", "Zip",
"Phone", "Email", "Notes"]

Chapter 41 - bbfreeze

13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54

mainSizer = wx.BoxSizer(wx.VERTICAL)
lbl = wx.StaticText(self, label="Please enter your information here:")
lbl.SetFont(wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD))
mainSizer.Add(lbl, 0, wx.ALL, 5)
for lbl in labels:
sizer = self.buildControls(lbl)
mainSizer.Add(sizer, 1, wx.EXPAND)
self.SetSizer(mainSizer)
mainSizer.Layout()
def buildControls(self, label):
"""
Put the widgets together
"""
sizer = wx.BoxSizer(wx.HORIZONTAL)
size = (80,40)
font = wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD)
lbl = wx.StaticText(self, label=label, size=size)
lbl.SetFont(font)
sizer.Add(lbl, 0, wx.ALL|wx.CENTER, 5)
if label != "Notes":
txt = wx.TextCtrl(self, name=label)
else:
txt = wx.TextCtrl(self, style=wx.TE_MULTILINE, name=label)
sizer.Add(txt, 1, wx.ALL, 5)
return sizer
class DemoFrame(wx.Frame):
"""
Frame that holds all other widgets
"""
def __init__(self):
"""Constructor"""
wx.Frame.__init__(self, None, wx.ID_ANY,
"Py2Exe Tutorial",
size=(600,400)
)
panel = DemoPanel(self)
self.Show()

253

Chapter 41 - bbfreeze

55
56
57
58

254

if __name__ == "__main__":
app = wx.App(False)
frame = DemoFrame()
app.MainLoop()

Now lets create a simple freezing script!


1
2
3
4
5
6

# bb_setup.py
from bbfreeze import Freezer
f = Freezer(distdir="bb-binary")
f.addScript("sampleApp.py")
f()

First off, we import the Freezer class from the bbfreeze package. Freezer accepts three arguments: a
destination folder, an includes iterable and an excludes iterable (i.e. a tuple or list). Just to see how
well bbfreeze works with only its defaults, we leave out the includes and excludes tuples/lists. Once
you have a Freezer object, you can add your script(s) by calling the Freezer object names addScript
method. Then you just need to call the object (i.e. f() ).
Note: You may see a warning about bb_freeze not being able to find MSVCP90.dll or similar.
If you see that message, you may need to include it explicitly or add it as a dependency when
you create an installer. We will be learning about how to create an installer in a later chapter.
To run this script, you just have to do something like this:
1

python bb_setup.py

When I ran this script, it created a folder named bb-binary that contained 19 files that weighed in
at 17.2 MB. When I ran the sampleApp.exe file, it ran just fine and was properly themed, however
it also had a console screen. Well have to edit our script a bit to fix that:
1
2
3
4
5
6
7
8
9
10

# bb_setup2.py
from bbfreeze import Freezer
includes = []
excludes = ['_gtkagg', '_tkagg', 'bsddb', 'curses', 'email', 'pywin.debugger',
'pywin.debugger.dbgcon', 'pywin.dialogs', 'tcl',
'Tkconstants', 'Tkinter']
bbFreeze_Class = Freezer('dist', includes=includes, excludes=excludes)

Chapter 41 - bbfreeze

11
12
13
14
15

255

bbFreeze_Class.addScript("sampleApp.py", gui_only=True)
bbFreeze_Class.use_compression = 0
bbFreeze_Class.include_py = True
bbFreeze_Class()

If you run this, you should end up with a dist folder with about 19 files, but a slightly different
size of 19.6 MB. Notice that we added a second argument to the addScript method: gui_only=True.
This makes that annoying console go away. We also set compression to zero (no compression) and
include the Python interpreter. Turning on compression only reduced the result back down to 17.2
MB though.
The bbfreeze package also handles recipes and includes several examples, however they are not
documented well either. Feel free to study them yourself as an exercise.

Wrapping Up
Now you should know the basics of using bbfreeze to create binaries from your programs. I noticed
that when I ran bbfreeze on my machine, it was considerably slower in producing the wxPython
executable compared with py2exe. This is one of those things that you will have to experiment with
when you are determining which tool to use to create your binaries.

Chapter 42 - cx_Freeze
In this chapter, we will be learning about cx_Freeze, a cross-platform set of scripts designed to
freeze Python scripts into executables in a manner similar to py2exe, PyInstaller, etc. We will freeze
one console script and one window (i.e GUI) script, using the examples from the previous chapter.
The cx_Freeze tool is the only binary creation tool that can work with both Python 2.x and 3.x on
multiple operating systems at this time. We will be using it with Python 2.7 in this chapter only
because we want to compare it directly to the other binary creation tools.
You can install cx_Freeze using one of their Windows installers, via their provided Linux RPMs, via
a source RPM or directly from source. You can also use pip to install cx_Freeze.
Note: I tested on Windows 7 using Python 2.7.3, wxPython 2.9.4.0 (classic) and cx_Freeze 4.3.2.

Getting Started with cx_Freeze


As mentioned on the cx_Freeze website, there are three ways to use this script. The first is to just
use the included cxfreeze script; the second is to create a distutils setup script (think py2exe) which
you can save for future use; and the third is to work with the internals of cxfreeze. We will focus on
the first two ways of using cx_Freeze. Well begin with the console script:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18

# config_1.py
import configobj
def createConfig(configFile):
"""
Create the configuration file
"""
config = configobj.ConfigObj()
inifile = configFile
config.filename = inifile
config['server'] = "http://www.google.com"
config['username'] = "mike"
config['password'] = "dingbat"
config['update interval'] = 2
config.write()
def getConfig(configFile):
"""

257

Chapter 42 - cx_Freeze

19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36

Open the config file and return a configobj


"""
return configobj.ConfigObj(configFile)
def createConfig2(path):
"""
Create a config file
"""
config = configobj.ConfigObj()
config.filename = path
config["Sony"] = {}
config["Sony"]["product"] = "Sony PS3"
config["Sony"]["accessories"] = ['controller', 'eye', 'memory stick']
config["Sony"]["retail price"] = "$400"
config.write()
if __name__ == "__main__":
createConfig2("sampleConfig2.ini")

All this script does is create a really simple configuration file using Michael Foords configobj
module. You can set it up to read the config too, but for this example, well skip that. Lets find
out how to build a binary with cx_Freeze! According to the documentation, all it should take is the
following string on the command line (assuming you are in the correct directory):
1

cxfreeze config_1.py --target-dir dirName

This assumes that you have C:\PythonXX\Scripts on your path. If not, youll either have to fix that
or type out the fully qualified path. Anyway, if the cxfreeze script runs correctly, you should have
a folder with the following contents:

image

Chapter 42 - cx_Freeze

258

As you can see, the total file size should around 5 megabytes. That was pretty easy. It even picked
up the configobj module without our having to tell it to. There are 18 command line arguments you
can pass to cx_Freeze to control how it does things. These range from what modules to include or
exclude, optimization, compression, include a zip file, path manipulation and more.
Now lets try something a little more advanced.

Advanced cx_Freeze - Using a setup.py File


First off we need a script to use. We will use the wxPython form example from the previous chapters.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31

import wx
class DemoPanel(wx.Panel):
""""""
def __init__(self, parent):
"""Constructor"""
wx.Panel.__init__(self, parent)
labels = ["Name", "Address", "City", "State", "Zip",
"Phone", "Email", "Notes"]
mainSizer = wx.BoxSizer(wx.VERTICAL)
lbl = wx.StaticText(self, label="Please enter your information here:")
lbl.SetFont(wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD))
mainSizer.Add(lbl, 0, wx.ALL, 5)
for lbl in labels:
sizer = self.buildControls(lbl)
mainSizer.Add(sizer, 1, wx.EXPAND)
self.SetSizer(mainSizer)
mainSizer.Layout()
def buildControls(self, label):
"""
Put the widgets together
"""
sizer = wx.BoxSizer(wx.HORIZONTAL)
size = (80,40)
font = wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD)
lbl = wx.StaticText(self, label=label, size=size)

Chapter 42 - cx_Freeze

32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58

259

lbl.SetFont(font)
sizer.Add(lbl, 0, wx.ALL|wx.CENTER, 5)
if label != "Notes":
txt = wx.TextCtrl(self, name=label)
else:
txt = wx.TextCtrl(self, style=wx.TE_MULTILINE, name=label)
sizer.Add(txt, 1, wx.ALL, 5)
return sizer
class DemoFrame(wx.Frame):
"""
Frame that holds all other widgets
"""
def __init__(self):
"""Constructor"""
wx.Frame.__init__(self, None, wx.ID_ANY,
"Py2Exe Tutorial",
size=(600,400)
)
panel = DemoPanel(self)
self.Show()
if __name__ == "__main__":
app = wx.App(False)
frame = DemoFrame()
app.MainLoop()

Now lets create a setup.py file in the cx_Freeze style:


1
2
3
4
5
6
7
8
9

# setup.py
from cx_Freeze import setup, Executable
setup(
name = "wxSampleApp",
version = "0.1",
description = "An example wxPython script",
executables = [Executable("sampleApp.py")]
)

As you can see, this is a pretty simple one. We import a couple classes from cx_Freeze and pass
some parameters into them. In this case, we give the setup class a name, version, description and

Chapter 42 - cx_Freeze

260

Executable class. The Executable class also gets one parameter, the script name that it will use to
create the binary from.
Alternatively, you can create a simple setup.py using cx_Freezes quickstart command (assuming
its on your systems path) in the same folder as your code:
cxfreeze-quickstart
To get the setup.py to build the binary, you need to do the following on the command
line:
1

python setup.py build

After running this, you should end up with the following folders: buildexe.win32-2.7. Inside that
last folder I ended up with 15 files that total 16.6 MB. When you run the sampleApp.exe file, you
will notice that weve screwed something up. Theres a console window loading in addition to our
GUI! To rectify this, well need to change our setup file slightly. Take a look at our new one:
1
2
3
4
5
6
7
8
9
10
11
12
13

from cx_Freeze import setup, Executable


exe = Executable(
script="sampleApp.py",
base="Win32GUI",
)
setup(
name = "wxSampleApp",
version = "0.1",
description = "An example wxPython script",
executables = [exe]
)

First off, we separated the Executable class from the setup class and assigned the Executable class
to a variable. We also added a second parameter to the Executable class that is key. That parameter
is called base. By setting base=Win32GUI, we are able to suppress the console window. The
documentation on the cx_Freeze website shows the many other options that the Executable class
takes.

Wrapping Up
Now you should know how to create binaries with cx_Freeze. Its pretty easy to do and it ran a lot
faster than bbfreeze did in my testing. If you have the need to create binaries for both Python 2.x
and 3.x on all major platforms, then this is the tool for you!

Chapter 43 - PyInstaller
PyInstaller is the last tool we will be looking at for creating binaries. It supports Python 2.4 - 2.7.
We will continue to use our simple console and wxPython GUI scripts for our testing. PyInstaller
is supposed to work on Windows, Linux, Mac, Solaris and AIX. The support for Solaris and AIX is
experimental. PyInstaller supports code-signing (Windows), eggs, hidden imports, single executable,
single directory, and lots more!
Note: I tested on Windows 7 using Python 2.7.3, wxPython 2.9.4.0 (classic) and PyInstaller 2.1.

Getting Started with PyInstaller


To install PyInstaller, you can download the source code in a tarball or zip archive, decompress it
and run its setup.py file:
1

python setup.py install

You can also install PyInstaller using pip. We will start with our little piece of config creation code:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

# config_1.py
import configobj
def createConfig(configFile):
"""
Create the configuration file
"""
config = configobj.ConfigObj()
inifile = configFile
config.filename = inifile
config['server'] = "http://www.google.com"
config['username'] = "mike"
config['password'] = "dingbat"
config['update interval'] = 2
config.write()
def getConfig(configFile):
"""
Open the config file and return a configobj

Chapter 43 - PyInstaller

20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36

262

"""
return configobj.ConfigObj(configFile)
def createConfig2(path):
"""
Create a config file
"""
config = configobj.ConfigObj()
config.filename = path
config["Sony"] = {}
config["Sony"]["product"] = "Sony PS3"
config["Sony"]["accessories"] = ['controller', 'eye', 'memory stick']
config["Sony"]["retail price"] = "$400"
config.write()
if __name__ == "__main__":
createConfig2("sampleConfig2.ini")

Now lets try to create an executable! You should be able to just do this to get PyInstaller to work:
1

pyinstaller config_1.py

When I ran this, I got the following error:


1
2

Error: PyInstaller for Python 2.6+ on Windows needs pywin32.


Please install from http://sourceforge.net/projects/pywin32/

To use PyInstaller on Windows, you will need to install PyWin32 first! Once you have PyWin32
installed, try re-running that command. You should see a lot of output sent to the screen and you
should also see these two folders appear next to your script: build and dist. If you go into the dist
folder, and then into its config_1 folder, you should see something like this:

263

Chapter 43 - PyInstaller

image

When I ran the executable, it created the config file just as it should. You will notice that PyInstaller
was able to grab configobj without you needing to tell it to.

PyInstaller and wxPython


Now lets try creating a binary from a simple wxPython script. Heres the wxPython code that weve
been using in previous chapters:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

import wx
class DemoPanel(wx.Panel):
""""""
def __init__(self, parent):
"""Constructor"""
wx.Panel.__init__(self, parent)
labels = ["Name", "Address", "City", "State", "Zip",
"Phone", "Email", "Notes"]
mainSizer = wx.BoxSizer(wx.VERTICAL)
lbl = wx.StaticText(self, label="Please enter your information here:")

Chapter 43 - PyInstaller

15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56

lbl.SetFont(wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD))


mainSizer.Add(lbl, 0, wx.ALL, 5)
for lbl in labels:
sizer = self.buildControls(lbl)
mainSizer.Add(sizer, 1, wx.EXPAND)
self.SetSizer(mainSizer)
mainSizer.Layout()
def buildControls(self, label):
"""
Put the widgets together
"""
sizer = wx.BoxSizer(wx.HORIZONTAL)
size = (80,40)
font = wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD)
lbl = wx.StaticText(self, label=label, size=size)
lbl.SetFont(font)
sizer.Add(lbl, 0, wx.ALL|wx.CENTER, 5)
if label != "Notes":
txt = wx.TextCtrl(self, name=label)
else:
txt = wx.TextCtrl(self, style=wx.TE_MULTILINE, name=label)
sizer.Add(txt, 1, wx.ALL, 5)
return sizer
class DemoFrame(wx.Frame):
"""
Frame that holds all other widgets
"""
def __init__(self):
"""Constructor"""
wx.Frame.__init__(self, None, wx.ID_ANY,
"Py2Exe Tutorial",
size=(600,400)
)
panel = DemoPanel(self)
self.Show()
if __name__ == "__main__":
app = wx.App(False)

264

Chapter 43 - PyInstaller

57
58

265

frame = DemoFrame()
app.MainLoop()

If you execute the pyinstaller command against this script, you will see ever more output sent to
the screen. It will create 23 files that total 19.4 MB. You will also notice that when you run the
sampleApp.exe, it shows a console window in addition to your GUI, which is not what we want.
The simplest way to fix that is to call PyInstaller with the -w command which tells PyInstaller to
suppress the console window:
1

pyinstaller -w sampleApp.py

The PyInstaller package has many command line options that you can use to change the way
PyInstaller processes your program. Whenever you run PyInstaller, it will create a spec file that
it uses to process your program. If youd like to save a copy of the spec file to help you better
understand what PyInstaller is doing, you can do so using the following command:
1

pyi-makespec sampleApp.py

You can pass the same commands to pyi-makespec as you do to PyInstaller, which will change the
spec appropriately. Heres the contents of the spec that was created with the previous command:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

# -*- mode: python -*a = Analysis(['sampleApp.py'],


pathex=['c:\\py101\\wxpy'],
hiddenimports=[],
hookspath=None,
runtime_hooks=None)
pyz = PYZ(a.pure)
exe = EXE(pyz,
a.scripts,
exclude_binaries=True,
name='sampleApp.exe',
debug=False,
strip=None,
upx=True,
console=False )
coll = COLLECT(exe,
a.binaries,
a.zipfiles,
a.datas,
strip=None,
upx=True,
name='sampleApp')

Chapter 43 - PyInstaller

266

In earlier versions of PyInstaller, you would actually create the spec file and edit it directly. Now
unless you need something really special, you can generate the right spec by just using flags. Be sure
to read the documentation for full details as there are many flags and describing them all is outside
the scope of this chapter.

Wrapping Up
This ends our quick tour of PyInstaller. I hope you found this helpful in your Python binary-making
endeavors. The PyInstaller project is pretty well documented and worth your time to check out.

Chapter 44 - Creating an Installer


In this chapter, we will walk you through the process of creating an executable and then packaging
it up into an installer. We will be using a really neat little user interface called GUI2Exe that was
written by Andrea Gavana to create the executable. It is based on wxPython, so you will need to
have that installed to use it. GUI2Exe supports py2exe, bbfreeze, cx_Freeze, PyInstaller and py2app.
Then once we have the dist folder created, we will use Inno Setup to create our installer.
We will be using the following code once more:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

# sampleApp.py
import wx
class DemoPanel(wx.Panel):
""""""
def __init__(self, parent):
"""Constructor"""
wx.Panel.__init__(self, parent)
labels = ["Name", "Address", "City", "State", "Zip",
"Phone", "Email", "Notes"]
mainSizer = wx.BoxSizer(wx.VERTICAL)
lbl = wx.StaticText(self, label="Please enter your information here:")
lbl.SetFont(wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD))
mainSizer.Add(lbl, 0, wx.ALL, 5)
for lbl in labels:
sizer = self.buildControls(lbl)
mainSizer.Add(sizer, 1, wx.EXPAND)
self.SetSizer(mainSizer)
mainSizer.Layout()
def buildControls(self, label):
"""
Put the widgets together
"""
sizer = wx.BoxSizer(wx.HORIZONTAL)

Chapter 44 - Creating an Installer

30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60

268

size = (80,40)
font = wx.Font(12, wx.SWISS, wx.NORMAL, wx.BOLD)
lbl = wx.StaticText(self, label=label, size=size)
lbl.SetFont(font)
sizer.Add(lbl, 0, wx.ALL|wx.CENTER, 5)
if label != "Notes":
txt = wx.TextCtrl(self, name=label)
else:
txt = wx.TextCtrl(self, style=wx.TE_MULTILINE, name=label)
sizer.Add(txt, 1, wx.ALL, 5)
return sizer
class DemoFrame(wx.Frame):
"""
Frame that holds all other widgets
"""
def __init__(self):
"""Constructor"""
wx.Frame.__init__(self, None, wx.ID_ANY,
"Py2Exe Tutorial",
size=(600,400)
)
panel = DemoPanel(self)
self.Show()
if __name__ == "__main__":
app = wx.App(False)
frame = DemoFrame()
app.MainLoop()

Lets get started!

Getting Started with GUI2Exe


To use GUI2Exe, you just have to go to its website (http://code.google.com/p/gui2exe/) and download
a release. Then you unzip it and run the script thats called GUI2Exe.py. The GUI2Exe project is
based on wxPython, so make sure you have that installed as well. I ran mine successfully with
wxPython 2.9. Heres how you would call it:

269

Chapter 44 - Creating an Installer

python GUI2Exe.py

If that executed successfully, you should see a screen similar to this one:

image

Now go to File -> New Project and give your project a name. In this case, I called the project wxForm.
If you want to, you can add a fake Company Name, Copyright and give it a Program Name. Be sure
to also browse for your main Python script (i.e. sampleApp.py). According to Andreas website,
you should set Optimize to 2, Compressed to 2 and Bundled Files to 1. This seems to work most
of the time, but Ive had some screwy errors that seem to stem from setting the last one to 1. In fact,
according to one of my contacts on the py2exe mailing list, the bundle option should be set to 3 to
minimize errors. The nice thing about setting bundle to 1 is that you end up with just one file, but
since Im going to roll it up with Inno Im going to go with option 3 to make sure my program works
well.
Once you have everything the way you want it, click the Compile button in the lower righthand corner. This will create all the files you want to distribute in the dist folder unless you have
changed the name by checking the dist checkbox and editing the subsequent textbox. When its
done compiling, GUI2Exe will ask you if you want to test your executable. Go ahead and hit Yes. If

270

Chapter 44 - Creating an Installer

you receive any errors about missing modules, you can add them in the Python Modules or Python
Packages section as appropriate. For this example, you shouldnt have that issue though.
Now were ready to learn about creating the installer!

Lets Make an Installer!


Now that we have an executable and a bunch of dependencies, how do we make an installer? For
this chapter well be using Inno Setup, but you could also use NSIS or a Microsoft branded installer.
You will need to go to their website (http://www.jrsoftware.org/isdl.php), download the program
and install it. Then run the program. You should see the main program along with the following
dialog on top of it:

image

Choose the Create a new script using the Script Wizard option and then press the OK button.
Click Next and you should see something like this:

271

Chapter 44 - Creating an Installer

image

Fill this out however you like and click Next (I called mine wxForm). This next screen allows you
to choose where you want the application to be installed by default. It defaults to Program Files
which is fine. Click Next. Now you should see the following screen:

272

Chapter 44 - Creating an Installer

image

Browse to the executable you created to add it. Then click the Add file(s) button to add the rest.
You can actually select all of the files except for the exe and hit OK. This is how mine turned out:

273

Chapter 44 - Creating an Installer

image

Now youre ready to click Next. Make sure the Start Menu folder has the right name (in this case,
wxForm) and continue. You can ignore the next two screens or experiment with them if you like.
Im not using a license or putting information files in to display to the user though. The last screen
before finishing allows you to choose a directory to put the output into. I just left that empty since
it defaults to where the executable is located, which is fine with this example. Click Next, Next
and Finish. This will generate a full-fledged .iss file, which is what Inno Setup uses to turn your
application into an installer. It will ask you if youd like to go ahead and compile the script now.
Go ahead and do that. Then it will ask if youd like to save your .iss script. Thats a good idea, so
go ahead and do that too. Hopefully you didnt receive any errors and you can try out your new
installer.
If youre interested in learning about Innos scripting language, feel free to read Innos Documentation. You can do quite a bit with it. If you happen to make changes to your build script, you can
rebuild your installer by going to the build menu and choosing the compile menu item.

Wrapping Up
At this point, you now know how to create a real, live installer that you can use to install your
application and any files it needs to run. This is especially handy when you have a lot of custom
icons for your toolbars or a default database, config file, etc that you need to distribute with your

Chapter 44 - Creating an Installer

274

application. Go back and try creating the installer again, but choose different options to see what
else you can do. Experimenting is a great way to learn. Just make sure you always have a backup in
case something goes wrong!

Appendix A: Putting It All Together


Now that youve finished reading the book, what do you do? Most beginner programming books
Ive read would leave you hanging at this point, but not this one. In this chapter, we will take what
weve learned and apply it to create a real program! First we need some background and a set
of requirements. In the business world, we might call this a specification. The more specific the
specification is, the more likely your code will match the customers expectations. Unfortunately,
you will find that in the real world, good specifications are hard to come by and you have to spend
a lot of time nailing down exactly what the customer wants. But I digress. Lets get started on our
program!

The Background
This book was written using a markup language called RestructuredText. Each chapter of the book
was saved in its own file (Ex. chapter1.rst) as I have found that my proofers would rather proof
individual chapters than get overwhelmed by large chunks of the book. Here is a really short example
that demonstrates how to create a chapter heading:

image

You may want to take a look at this Quick Reference to learn more about RestructuredText.

The Specification
What I need to be able to do is find a way to produce a PDF out of these individual chapters, add
the customers name and email address to each page and email the PDF to each customer. This was
http://docutils.sourceforge.net/docs/user/rst/quickref.html

Appendix A: Putting It All Together

276

my original idea for distributing the book to my Kickstarter backers. Note that Kickstarter provides
the customer information in a CSV file.

Breaking the Specification Down


Most programs that you will write can be broken down into smaller programs. In fact, you will want
to do this to make your code more modular. This means that if you break the code down into small
enough pieces that the pieces can be reused by other programs. We wont go that far in this example,
but you should have an idea of how to do that by the end of this chapter.
Lets break the specification down a bit. Here are the tasks that occurred to me:

Write a program to put all the chapters into one book file
Figure out how to add a footer
Find some way to turn the book file into a PDF
Write a program that can email PDFs

Lets start with the first two items since theyre related.

Turning Chapters Into a Book


I went the easy route and just created a Python list of the chapters in the order that I wanted them
to appear in the book. I could have used Pythons glob module to grab all the files in a folder that
had the .rst extension, but then they might not have been in the right order.
Lets take a look at a simple example of what I did:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

import os
import subprocess
chapters = ['chapter_1.rst',
'chapter_2.rst',
'chapter_3.rst']
def read_chapter(chapter):
"""
Reads a chapter and returns the stream
"""
path = os.path.join("data", chapter)
try:
with open(path) as chp_handler:

Appendix A: Putting It All Together

15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56

data = chp_handler.read()
except (IOError, OSError):
raise Exception("Unable to open chapter: %s" % chapter)
return data
def make_book(name="Mike", email_address="test@py.com"):
"""
Creates Python 101 book
"""
book_path = "output/python101.rst"
pdf_path = "output/python101.pdf"
page_break = """
.. raw:: pdf
PageBreak
"""
footer = """
.. footer::
Copyright |copy| 2014 by Michael Driscoll, all rights reserved.
Licensed to %s <%s>
.. |copy| unicode:: 0xA9 .. copyright sign
""" % (name, email_address)
try:
with open(book_path, "w") as book:
book.write(footer + "\n")
for chapter in chapters:
data = read_chapter(chapter)
book.write(data)
book.write("\n")
book.write(page_break + "\n")
except:
print("Error writing book!")
raise
cmd = [r"C:\Python27\Scripts\rst2pdf.exe",
book_path, "-o", pdf_path]
subprocess.call(cmd)

277

Appendix A: Putting It All Together

57
58

278

if __name__ == "__main__":
make_book()

Lets break this down a bit. At the beginning of the script, we import two modules: os and
subprocess. Then we create a really simple function that will open a chapter file thats located
in the data folder, read the chapter and return the files contents. The meat of the program is in the
make_book function. Here we set up the names of the RestructuredText file and the PDF. Then we
create a page_break variable and a footer variable. These two variables contain RestructuredText
that we insert into the document. Right now, the program does not insert a customers name or
email, but will just use a couple of defaults instead.
Next we open the RestructuredText book document in write mode and write the footer text to it.
Then we loop over the chapters, writing them to disk with a page break between each of them. Then
at the very end of the script, we call a program called rst2pdf that will turn our new book file into
a PDF. At this point, we have about a third of the program written. Now we just need to figure out
how to read CSV file and email the PDF.

Reading the Customer CSV File


You may recall that we already covered a module that can help us in this quest back in Chapter 13.
Its the csv module, of course! I wont reveal exactly what information Kickstarter has in their files,
but we can create our own pretend version. Lets say its something like this:
1
2
3

Mike Driscoll,test@py.com
Andy Hunter,andwa@hunter.org
Toby Mac,macdaddy@gotee.com

Lets create a quick function that can extract the name and email address from this file.
1
2
3
4
5
6
7
8
9
10
11
12

import csv
def read_csv(path):
""""""
try:
with open(path) as csv_file:
reader = csv.reader(csv_file)
for line in reader:
name, email = line
print(name)
print(email)
except IOError:

Appendix A: Putting It All Together

13
14
15
16
17
18

279

print("Error reading file: %s" % path)


raise
if __name__ == "__main__":
path = "backers.csv"
read_csv(path)

All this script does is read the CSV file and print out the name and email address. Lets rename the
function to main and have it call our make_book function:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

def main(path):
""""""
try:
with open(path) as csv_file:
reader = csv.reader(csv_file)
for line in reader:
name, email = line
make_book(name, email)
except IOError:
print("Error reading file: %s" % path)
raise
if __name__ == "__main__":
main("backers.csv")

This will call the make_book function 3 times, so we will create the PDF 3 times and overwrite
each time. You can check the file at the end of the run to see which name and email address is in
the footer. This gets us two-thirds of the way there. All thats left is to figure out how to send the
PDF out in an email. Fortunately, we covered that back in Chapter 17 when you learned about the
email and smtplib modules!

Emailing the PDF


Lets spend a few minutes putting together a simple program that can email PDFs. Heres a fairly
simple example:

Appendix A: Putting It All Together

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42

import os
import smtplib
from
from
from
from
from

email import encoders


email.mime.text import MIMEText
email.mime.base import MIMEBase
email.mime.multipart import MIMEMultipart
email.utils import formatdate

def send_email(email, pdf):


"""
Send an email out
"""
header0 = 'Content-Disposition'
header1 ='attachment; filename="%s"' % os.path.basename(pdf)
header = header0, header1
host = "mail.server.com"
server = smtplib.SMTP(host)
subject = "Test email from Python"
to = email
from_addr = "test@pylib.com"
body_text = "Here is the Alpha copy of Python 101, Part I"
# create the message
msg = MIMEMultipart()
msg["From"] = from_addr
msg["Subject"] = subject
msg["Date"] = formatdate(localtime=True)
msg["To"] = email
msg.attach( MIMEText(body_text) )
attachment = MIMEBase('application', "octet-stream")
try:
with open(pdf, "rb") as fh:
data = fh.read()
attachment.set_payload( data )
encoders.encode_base64(attachment)
attachment.add_header(*header)
msg.attach(attachment)
except IOError:

280

Appendix A: Putting It All Together

43
44
45
46
47
48
49

281

msg = "Error opening attachment file %s" % file_to_attach


print(msg)
server.sendmail(from_addr, to, msg.as_string())
if __name__ == "__main__":
send_email("mike@example.org", "output/python101.pdf")

This is basically a modified example of the email attachment code from earlier in the book. This
function is made to accept two arguments: an email address and a path to a PDF file. Technically,
you could pass it any type of file and it would work. You will need to modify the host variable
to point to your own SMTP server. I tested this on my machine with the proper host name and it
worked great. Now we just need to integrate this code into the main code.

Putting it all Together


Now we get to put all the code together and see it as a whole. Here it is:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23

import
import
import
import
from
from
from
from
from

csv
os
smtplib
subprocess

email import encoders


email.mime.text import MIMEText
email.mime.base import MIMEBase
email.mime.multipart import MIMEMultipart
email.utils import formatdate

chapters = ['chapter_1.rst',
'chapter_2.rst',
'chapter_3.rst']
def make_book(name="Mike", email_address="test@py.com"):
"""
Creates Python 101 book
"""
book_path = "output/python101.rst"
pdf_path = "output/python101.pdf"
page_break = """
.. raw:: pdf

Appendix A: Putting It All Together

24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65

PageBreak
"""
footer = """
.. footer::
Copyright |copy| 2014 by Michael Driscoll, all rights reserved.
Licensed to %s <%s>
.. |copy| unicode:: 0xA9 .. copyright sign
""" % (name, email_address)
try:
with open(book_path, "w") as book:
book.write(footer + "\n")
for chapter in chapters:
data = read_chapter(chapter)
book.write(data)
book.write("\n")
book.write(page_break + "\n")
except:
print("Error writing book!")
raise
cmd = [r"C:\Python27\Scripts\rst2pdf.exe",
book_path, "-o", pdf_path]
subprocess.call(cmd)
return pdf_path
def read_chapter(chapter):
"""
Reads a chapter and returns the stream
"""
path = os.path.join("data", chapter)
try:
with open(path) as chp_handler:
data = chp_handler.read()
except (IOError, OSError):
raise Exception("Unable to open chapter: %s" % chapter)
return data

282

Appendix A: Putting It All Together

66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107

def send_email(email, pdf):


"""
Send an email out
"""
header0 = 'Content-Disposition'
header1 ='attachment; filename="%s"' % os.path.basename(pdf)
header = header0, header1
host = "mail.server.com"
server = smtplib.SMTP(host)
subject = "Test email from Python"
to = email
from_addr = "test@pylib.com"
body_text = "Here is the Alpha copy of Python 101, Part I"
# create the message
msg = MIMEMultipart()
msg["From"] = from_addr
msg["Subject"] = subject
msg["Date"] = formatdate(localtime=True)
msg["To"] = email
msg.attach( MIMEText(body_text) )
attachment = MIMEBase('application', "octet-stream")
try:
with open(pdf, "rb") as fh:
data = fh.read()
attachment.set_payload( data )
encoders.encode_base64(attachment)
attachment.add_header(*header)
msg.attach(attachment)
except IOError:
msg = "Error opening attachment file %s" % file_to_attach
print(msg)
server.sendmail(from_addr, to, msg.as_string())
def main(path):
""""""
try:
with open(path) as csv_file:

283

Appendix A: Putting It All Together

108
109
110
111
112
113
114
115
116
117
118

284

reader = csv.reader(csv_file)
for line in reader:
name, email = line
pdf = make_book(name, email)
send_email(email, pdf)
except IOError:
print("Error reading file: %s" % path)
raise
if __name__ == "__main__":
main("backers.csv")

Lets go over the minor changes. First off, we had to change the make_book function so that it
returned the PDF path. Then we changed the main function so that it took that path and passed it
to the send_email function along with the email address. Two changes and were done!

Wrapping Up
Now you have the knowledge to take apart a new program and turn it into a set of smaller tasks.
Then you can join the smaller programs into one large program. There are lots of different ways we
could have done this. For example, we could have left each piece in its own file or module. Then we
could import each piece into a different script and run them that way. That would be a good exercise
for to try on your own.
When I did this, I ran into issues with my email host. Most email hosts will not let you send out
lots of emails per hour or even make many connections at once. You can find a few services on the
internet that you can use instead of your own internet provider that may work better for you.
I hope you found this chapter helpful in your own endeavours. Thanks for reading!

You might also like