Skip to content

Commit 960d4fa

Browse files
committed
Added Work directory
1 parent ea22c09 commit 960d4fa

17 files changed

+2659
-9
lines changed

Notes/00_Setup.md

+28-9
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,13 @@
11
# Course Setup and Overview
22

3-
Welcome to Practical Python Programming!
3+
Welcome to Practical Python Programming! This page has some important information
4+
about course setup and logistics.
5+
6+
## Course Duration and Time Requirements
7+
8+
This course was originally given as an instructor-led in-person
9+
training that spanned 3 to 4 days. To complete the course in its
10+
entirety, you should plan on committing 25-35 hours of work.
411

512
## Setup and Python Installation
613

@@ -54,18 +61,30 @@ bash %
5461
With this option, you just won't be able to commit code changes except
5562
to the local copy on your machine.
5663

57-
## File Layout
64+
## Coursework Layout
65+
66+
Do all of your coding work in the `Work/` directory. Within that
67+
directory, there is a `Data/` directory. The `Data/` directory
68+
contains a variety of datafiles and other scripts used during the
69+
course. You will frequently have to access files located in `Data/`.
70+
Course exercises are written with the assumption that you are creating
71+
programs in the `Work/` directory.
72+
73+
## Course Order
5874

59-
Do all of your coding work in the `Work/` directory. Within that directory,
60-
there is a `Data/` directory. The `Data/` directory contains a variety of
61-
datafiles and other scripts used during the course. You will frequently have
62-
to access files in `Data/`. Course exercises are written with the assumption
63-
that you are creating programs in the `Work/` directory.
75+
Course material should be completed in section order, starting with
76+
section 1. Course exercises in later sections build upon code written in
77+
earlier sections.
6478

65-
## Solutions
79+
## Solution Code
6680

6781
The `Solutions/` directory contains full solution code to selected exercises.
68-
Feel free to look at this if you need a hint.
82+
Feel free to look at this if you need a hint. To get the most out of the
83+
course, you should try to create your own solutions first.
84+
85+
[Contents](Contents)
86+
87+
6988

7089

7190

Work/Data/dowstocks.csv

+2,341
Large diffs are not rendered by default.

Work/Data/missing.csv

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
name,shares,price
2+
"AA",100,32.20
3+
"IBM",50,91.10
4+
"CAT",150,83.44
5+
"MSFT",,51.23
6+
"GE",95,40.37
7+
"MSFT",50,65.10
8+
"IBM",,70.44

Work/Data/portfolio.csv

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
name,shares,price
2+
"AA",100,32.20
3+
"IBM",50,91.10
4+
"CAT",150,83.44
5+
"MSFT",200,51.23
6+
"GE",95,40.37
7+
"MSFT",50,65.10
8+
"IBM",100,70.44

Work/Data/portfolio.csv.gz

136 Bytes
Binary file not shown.

Work/Data/portfolio.dat

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
name shares price
2+
"AA" 100 32.20
3+
"IBM" 50 91.10
4+
"CAT" 150 83.44
5+
"MSFT" 200 51.23
6+
"GE" 95 40.37
7+
"MSFT" 50 65.10
8+
"IBM" 100 70.44

Work/Data/portfolio2.csv

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
name,shares,price
2+
"AA",50,27.10
3+
"HPQ",250,43.15
4+
"MSFT",25,50.15
5+
"GE",125,52.10

Work/Data/portfolioblank.csv

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
name,shares,price
2+
3+
"AA",100,32.20
4+
5+
"IBM",50,91.10
6+
7+
"CAT",150,83.44
8+
9+
"MSFT",200,51.23
10+
11+
"GE",95,40.37
12+
13+
"MSFT",50,65.10
14+
15+
"IBM",100,70.44
16+

Work/Data/portfoliodate.csv

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
name,date,time,shares,price
2+
"AA","6/11/2007","9:50am",100,32.20
3+
"IBM","5/13/2007","4:20pm",50,91.10
4+
"CAT","9/23/2006","1:30pm",150,83.44
5+
"MSFT","5/17/2007","10:30am",200,51.23
6+
"GE","2/1/2006","10:45am",95,40.37
7+
"MSFT","10/31/2006","12:05pm",50,65.10
8+
"IBM","7/9/2006","3:15pm",100,70.44

Work/Data/prices.csv

+31
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
"AA",9.22
2+
"AXP",24.85
3+
"BA",44.85
4+
"BAC",11.27
5+
"C",3.72
6+
"CAT",35.46
7+
"CVX",66.67
8+
"DD",28.47
9+
"DIS",24.22
10+
"GE",13.48
11+
"GM",0.75
12+
"HD",23.16
13+
"HPQ",34.35
14+
"IBM",106.28
15+
"INTC",15.72
16+
"JNJ",55.16
17+
"JPM",36.90
18+
"KFT",26.11
19+
"KO",49.16
20+
"MCD",58.99
21+
"MMM",57.10
22+
"MRK",27.58
23+
"MSFT",20.89
24+
"PFE",15.19
25+
"PG",51.94
26+
"T",24.79
27+
"UTX",52.61
28+
"VZ",29.26
29+
"WMT",49.74
30+
"XOM",69.35
31+

Work/Data/stocksim.py

+183
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,183 @@
1+
#!/usr/bin/env python
2+
# stocksim.py
3+
#
4+
# Stock market simulator. This simulator creates stock market
5+
# data and provides it in several different ways:
6+
#
7+
# 1. Makes periodic updates to a log file stocklog.dat
8+
# 2. Provides stock data through an embedded HTTP server.
9+
#
10+
# The purpose of this module is to provide data to the user
11+
# in different ways in order to write interesting Python examples
12+
13+
import math
14+
import time
15+
16+
history_file = "dowstocks.csv"
17+
18+
# Convert a time string such as "4:00pm" to minutes past midnight
19+
def minutes(tm):
20+
am_pm = tm[-2:]
21+
fields = tm[:-2].split(":")
22+
hour = int(fields[0])
23+
minute = int(fields[1])
24+
if hour == 12:
25+
hour = 0
26+
if am_pm == 'pm':
27+
hour += 12
28+
return hour*60 + minute
29+
30+
# Convert time in minutes to a format string
31+
def minutes_to_str(m):
32+
frac,m = math.modf(m)
33+
hours = m//60
34+
minutes = m % 60
35+
seconds = frac * 60
36+
return "%02d:%02d.%02.f" % (hours,minutes,seconds)
37+
38+
# Read the stock history file as a list of lists
39+
def read_history(filename):
40+
result = []
41+
f = open(filename)
42+
next(f)
43+
for line in f:
44+
str_fields = line.strip().split(",")
45+
fields = [eval(x) for x in str_fields]
46+
fields[3] = minutes(fields[3])
47+
result.append(fields)
48+
return result
49+
50+
# Format CSV record
51+
def csv_record(fields):
52+
s = '"%s",%0.2f,"%s","%s",%0.2f,%0.2f,%0.2f,%0.2f,%d' % tuple(fields)
53+
return s
54+
55+
class StockTrack(object):
56+
def __init__(self,name):
57+
self.name = name
58+
self.history = []
59+
self.price = 0
60+
self.time = 0
61+
self.index = 0
62+
self.open = 0
63+
self.low = 0
64+
self.high = 0
65+
self.volume = 0
66+
self.initial = 0
67+
self.change = 0
68+
self.date = ""
69+
def add_data(self,record):
70+
self.history.append(record)
71+
def reset(self,time):
72+
self.time = time
73+
# Sort the history by time
74+
self.history.sort(key=lambda t:t[3])
75+
# Find the first entry who's time is behind the given time
76+
self.index = 0
77+
while self.index < len(self.history):
78+
if self.history[self.index][3] > time:
79+
break
80+
self.index += 1
81+
self.open = self.history[0][5]
82+
self.initial = self.history[0][1] - self.history[0][4]
83+
self.date = self.history[0][2]
84+
self.update()
85+
self.low = self.price
86+
self.high = self.price
87+
88+
# Calculate interpolated value of a given field based on
89+
# current time
90+
def interpolate(self,field):
91+
first = self.history[self.index][field]
92+
next = self.history[self.index+1][field]
93+
first_t = self.history[self.index][3]
94+
next_t = self.history[self.index+1][3]
95+
try:
96+
slope = (next - first)/(next_t-first_t)
97+
return first + slope*(self.time - first_t)
98+
except ZeroDivisionError:
99+
return first
100+
101+
# Update all computed values
102+
def update(self):
103+
self.price = round(self.interpolate(1),2)
104+
self.volume = int(self.interpolate(-1))
105+
if self.price < self.low:
106+
self.low = self.price
107+
if self.price >= self.high:
108+
self.high = self.price
109+
self.change = self.price - self.initial
110+
111+
# Increment the time by a delta
112+
def incr(self,dt):
113+
self.time += dt
114+
if self.index < (len(self.history) - 2):
115+
while self.index < (len(self.history) - 2) and self.time >= self.history[self.index+1][3]:
116+
self.index += 1
117+
self.update()
118+
119+
def make_record(self):
120+
return [self.name,round(self.price,2),self.date,minutes_to_str(self.time),round(self.change,2),self.open,round(self.high,2),
121+
round(self.low,2),self.volume]
122+
123+
class MarketSimulator(object):
124+
def __init__(self):
125+
self.stocks = { }
126+
self.prices = { }
127+
self.time = 0
128+
self.observers = []
129+
def register(self,observer):
130+
self.observers.append(observer)
131+
132+
def publish(self,record):
133+
for obj in self.observers:
134+
obj.update(record)
135+
def add_history(self,filename):
136+
hist = read_history(filename)
137+
for record in hist:
138+
if record[0] not in self.stocks:
139+
self.stocks[record[0]] = StockTrack(record[0])
140+
self.stocks[record[0]].add_data(record)
141+
142+
def reset(self,time):
143+
self.time = time
144+
for s in self.stocks.values():
145+
s.reset(time)
146+
147+
# Run forever. Dt is in seconds
148+
def run(self,dt):
149+
for s in self.stocks:
150+
self.prices[s] = self.stocks[s].price
151+
self.publish(self.stocks[s].make_record())
152+
while self.time < 1000:
153+
for s in self.stocks:
154+
self.stocks[s].incr(dt/60.0) # Increment is in minutes
155+
if self.stocks[s].price != self.prices[s]:
156+
self.prices[s] = self.stocks[s].price
157+
self.publish(self.stocks[s].make_record())
158+
time.sleep(dt)
159+
self.time += (dt/60.0)
160+
161+
162+
class BasicPrinter(object):
163+
def update(self,record):
164+
print(csv_record(record))
165+
166+
class LogPrinter(object):
167+
def __init__(self,filename):
168+
self.f = open(filename,"w")
169+
def update(self,record):
170+
self.f.write(csv_record(record)+"\n")
171+
self.f.flush()
172+
173+
m = MarketSimulator()
174+
m.add_history(history_file)
175+
m.reset(minutes("9:30am"))
176+
177+
m.register(BasicPrinter())
178+
m.register(LogPrinter("stocklog.csv"))
179+
180+
m.run(1)
181+
182+
183+

Work/README.md

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# Work Area
2+
3+
Do all of your coding work here, in the `Work/` directory. A number of starting
4+
files have been given (`bounce.py`, `mortgage.py`, `pcost.py`, etc.) along with
5+
their corresponding exercise number.
6+
7+
Many of the programs you write reference files found in the `Data/` directory.
8+
That is also located here.

Work/bounce.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# bounce.py
2+
#
3+
# Exercise 1.5

Work/fileparse.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# fileparse.py
2+
#
3+
# Exercise 3.3

Work/mortgage.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# mortgage.py
2+
#
3+
# Exercise 1.7

Work/pcost.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# pcost.py
2+
#
3+
# Exercise 1.27

Work/report.py

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
# report.py
2+
#
3+
# Exercise 2.4

0 commit comments

Comments
 (0)