forked from greencoder/us-zipcode-to-geojson
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprocess.py
127 lines (103 loc) · 3.95 KB
/
process.py
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
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
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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import json
import os
import shapefile
import sys
class GeonamesEntry(object):
def __init__(self, delimited_string):
parts = delimited_string.split('\t')
self.country_code = parts[0]
self.postal_code = parts[1]
self.name = parts[2]
self.state = parts[4]
self.county_code = parts[6]
self.lat = float(parts[9])
self.lng = float(parts[10])
# Special case - if these are records for US territories,
# we need to change the state. The source data has the
# admin1 code as something other than what we expect
if self.country_code in ('PR', 'VI', 'AS', 'GU'):
self.state = self.country_code
def __repr__(self):
return self.postal_code
class ZipCodeRecord(object):
def __init__(self):
self.postal_code = None
self.county_code = None
self.state = None
self.city = None
self.shape = None
self.latitude = None
self.longitude = None
def __repr__(self):
return self.postal_code
def to_geojson(self):
return {
'type': 'FeatureCollection',
'features': [
{
'type': 'Feature',
'geometry': {
'type': 'Point',
'coordinates': [
self.longitude,
self.latitude
]
},
'properties': {}
},
{
'type': 'Feature',
'geometry': self.shape.__geo_interface__,
'properties': {
'postal-code': self.postal_code,
'county-code': self.county_code,
'state': self.state,
'city': self.city,
}
}
]
}
if __name__ == '__main__':
# Get all the entries from Geonames
with open('source_data/US.txt', 'r') as f:
entries = [line.strip() for line in f.readlines()]
# Index the entries in a dictionary by key
entries_by_zipcode = {}
for line in entries:
entry = GeonamesEntry(line)
entries_by_zipcode[entry.postal_code] = entry
# Get all the records from the shapefile
sf = shapefile.Reader("source_data/cb_2014_us_zcta510_500k.shp")
# Get references to the shapes and records inside the shapefile
shapes = sf.shapes()
records = sf.records()
# Store a list of all the entries
parsed_zcrecords = []
# Iterate over all the data in the shapefiles
for index in range(0, len(shapes)):
current_record = records[index]
current_shape = shapes[index]
zcrecord = ZipCodeRecord()
zcrecord.postal_code = current_record[2]
zcrecord.shape = current_shape
# Find the corresponding entry in the indexed zip code data
try:
entry = entries_by_zipcode[zcrecord.postal_code]
except KeyError:
print "Could not find geonames entry for zip code %s. Skipping." % zcrecord.postal_code
continue
zcrecord.county_code = entry.county_code
zcrecord.state = entry.state
zcrecord.city = entry.name
zcrecord.latitude = entry.lat
zcrecord.longitude = entry.lng
parsed_zcrecords.append(zcrecord)
# Write the entries out to disk
for index, parsed_zcrecord in enumerate(parsed_zcrecords):
# See if the output data directory has a subdirectory for this state
if not os.path.exists('data/%s' % parsed_zcrecord.state):
os.mkdir('data/%s' % parsed_zcrecord.state)
filename = 'data/%s/%s.geojson' % (parsed_zcrecord.state, parsed_zcrecord.postal_code)
print 'Writing file %d of %d: %s' % (index, len(parsed_zcrecords), filename)
with open(filename, 'w') as f:
f.write(json.dumps(parsed_zcrecord.to_geojson(), indent=4))