OpenStreetMap Project Data Wrangling with MongoDB
Wang Xiaochu
Map Area
Birmingham, Westmidlands, England
This city is where I studied in the university for a master's degree of urban and reginal planning last year. So study the city from a different perspective using new skills I've learned means a lot to me.
Problems Encountered in the Map
After initially downloading a small sample size of the Birmingham area and running it against a provisional data.py file, I noticed three main problems with the data, which I will discuss in the following order:
Overabbreviated & Misspelled street names ("Roxburgh G4rove", "Hidcote Aveune")
"Incorrect" postal codes ( Birmingham postcodes all begin with "B" however a large portion of all documented postcodes were outside this region.)
Overabbreviated & Misspelled Street Names
Before the data was imported into MongoDB, it was audited in audit.py using the following function:
street_type_re = re.compile(r'\b\S+\.?$', re.IGNORECASE)
expected = ["Street", "Avenue", "Boulevard", "Drive", "Court", "Place", "Square", "Lane", "Road",
"Trail", "Parkway", "Commons", "Way", "Walk"]
def audit_street_type(street_types, street_name):
m = street_type_re.search(street_name)
if m:
street_type = m.group()
if street_type not in expected:
street_types[street_type].add(street_name)
def is_street_name(elem):
return (elem.attrib['k'] == "addr:street")
def audit(osmfile):
osm_file = open(osmfile, "r")
street_types = defaultdict(set)
for event, elem in ET.iterparse(osm_file, events=("start",)):
if elem.tag == "node" or elem.tag == "way":
for tag in elem.iter("tag"):
if is_street_name(tag):
audit_street_type(street_types, tag.attrib['v'])
osm_file.close()
return street_types
This basic querying revealed street name abbreviations and misspellings. I updated all substrings in problematic address strings, such as:
"Aveune" becomes "Avenue", "G4rove" becomes "Grove".
Postal Codes
Postcodes turn out to be a non-technical problem. It revealed some of the data out of the range of Birmingham city, which force me to filter out all of nodes with postcodes start with "C" or "D" in the process of import data into JSON file with adding following codes:
if m == 'addr:postcode' and elem.attrib['v'][0] != 'B':
return None
Data Overview
This section contains basic statistics about the dataset and the MongoDB queries used to gather them.
File Size
birmingham_england.osm ........... 1,535 MB
birmingham_england.osm.json .... 1,780 MB
-
Number of documents
> db.birm.find().count() 7017662
-
Number of nodes
> db.birm.find({"type":"node"}).count() 6906846
-
Number of ways
> db.birm.find({"type":"way"}).count() 110816
-
Number of unique users
> db.birm.distinct("created.user").length 2249
-
Top 10 contributing user
> db.birm.aggregate([{"$group":{"_id":"$created.user", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "brianboru", "count" : 3549542 } { "_id" : "blackadder", "count" : 447958 } { "_id" : "Miked29", "count" : 370951 } { "_id" : "James Derrick", "count" : 311132 } { "_id" : "mrpacmanmap", "count" : 164581 } { "_id" : "Curran1980", "count" : 136896 } { "_id" : "PeterP", "count" : 115926 } { "_id" : "srbrook", "count" : 105316 } { "_id" : "The Maarssen Mapper", "count" : 104912 } { "_id" : "richardwest", "count" : 104404 }
-
Number of users appearing only once (having 1 post)
> db.birm.aggregate([{"$group":{"_id":"$created.user", "count":{"$sum":1}}}, {"$group":{"_id":"$count", "num_users":{"$sum":1}}}, {"$sort":{"_id":1}}, {"$limit":1}]) { "_id" : 1, "num_users" : 365 }
Additional Ideas
Region statistics and suggestion
Although some of nodes out of Birmingham region have been filtered by their postcodes, there are still indiscriminable nodes do not have postcode attribute left in the file. One way to solve this problem could be comparing the longitude and latitude of Birmingham city and the attribute of each node. Another way is to ensure the property “addr.city” is properly informed. Using the following query, I explore the statistics further:
> db.birm.aggregate([{"$match":{"addr.city":{"$exists":1}}},
{"$group":{"_id":"$addr.city","count":{"$sum":1}}},
{"$sort":{"count":-1}}])
{ "_id" : "Birmingham", "count" : 3884 }
{ "_id" : "Solihull", "count" : 910 }
{ "_id" : "Bromsgrove", "count" : 238 }
{ "_id" : "Alcester", "count" : 158 }
{ "_id" : "Sutton Coldfield", "count" : 106 }
{ "_id" : "Tipton", "count" : 63 }
{ "_id" : "Madeley", "count" : 49 }
.....
{ "_id" : "Ironbridge", "count" : 26 }
{ "_id" : "Wolverhampton", "count" : 22 }
{ "_id" : "bm", "count" : 16 }
{ "_id" : "West Bromwich", "count" : 16 }
{ "_id" : "Redditch", "count" : 13 }
It turns out I didn't wrangle this data properly. Of course better data wrangling should be done ideally, while in the circumstance, the citys are all in the big region of Birmingham, which is just not very influential if I take it as Birmingham metropolitan area.
Additional data exploration using MongoDB queries
-
Top 10 Contributers
> db.birm.aggregate([{"$group":{"_id":"$created.user", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "brianboru", "count" : 3549542 } { "_id" : "blackadder", "count" : 447958 } { "_id" : "Miked29", "count" : 370951 } { "_id" : "James Derrick", "count" : 311132 } { "_id" : "mrpacmanmap", "count" : 164581 } { "_id" : "Curran1980", "count" : 136896 } { "_id" : "PeterP", "count" : 115926 } { "_id" : "srbrook", "count" : 105316 } { "_id" : "The Maarssen Mapper", "count" : 104912 } { "_id" : "richardwest", "count" : 104404 }
-
Most common building types
> db.birm.aggregate([{'$match': {'building': {'$exists': 1, "$ne":"yes"}}}, {'$group': { '_id': '$building','count': {'$sum': 1}}}, {'$sort': {'count': -1}}, {'$limit': 10}]) { "_id" : "residential", "count" : 2147 } { "_id" : "industrial", "count" : 563 } { "_id" : "retail", "count" : 391 } { "_id" : "entrance", "count" : 135 } { "_id" : "commercial", "count" : 109 } { "_id" : "school", "count" : 107 } { "_id" : "garages", "count" : 84 } { "_id" : "university", "count" : 60 } { "_id" : "church", "count" : 52 } { "_id" : "storage_tank", "count" : 48 }
-
Top 10 appearing amenities
> db.birm.aggregate([{"$match":{"amenity":{"$exists":1}}}, {"$group":{"_id":"$amenity","count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "post_box", "count" : 3533 } { "_id" : "bench", "count" : 1789 } { "_id" : "parking", "count" : 1458 } { "_id" : "fast_food", "count" : 1414 } { "_id" : "telephone", "count" : 1248 } { "_id" : "pub", "count" : 1003 } { "_id" : "bicycle_parking", "count" : 922 } { "_id" : "grit_bin", "count" : 753 } { "_id" : "restaurant", "count" : 720 } { "_id" : "cafe", "count" : 637 }
-
Top 5 bank branches
> db.birm.aggregate([{"$match":{"name":{"$exists":1}, "amenity": "bank"}}, {"$group":{"_id":"$name","count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":5}]) { "_id" : "Barclays", "count" : 35 } { "_id" : "HSBC", "count" : 24 } { "_id" : "Santander", "count" : 21 } { "_id" : "Halifax", "count" : 18 } { "_id" : "NatWest", "count" : 15 }
-
Top 10 restaurant trends
> db.birm.aggregate([{"$match":{"amenity":{"$exists":1},"cuisine":{"$exists":1}, "amenity":"restaurant"}}, {"$group":{"_id":"$cuisine", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "indian", "count" : 86 } { "_id" : "chinese", "count" : 33 } { "_id" : "italian", "count" : 26 } { "_id" : "french", "count" : 8 } { "_id" : "thai", "count" : 7 } { "_id" : "american", "count" : 7 } { "_id" : "regional", "count" : 6 } { "_id" : "pizza", "count" : 5 } { "_id" : "mexican", "count" : 5 } { "_id" : "fish_and_chips", "count" : 5 }
-
Top 10 fast-food trends
> db.birm.aggregate([{"$match":{"amenity":{"$exists":1},"cuisine":{"$exists":1}, "amenity":"fast_food"}}, {"$group":{"_id":"$cuisine", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "fish_and_chips", "count" : 177 } { "_id" : "chinese", "count" : 137 } { "_id" : "pizza", "count" : 57 } { "_id" : "sandwich", "count" : 43 } { "_id" : "indian", "count" : 42 } { "_id" : "burger", "count" : 29 } { "_id" : "chicken", "count" : 23 } { "_id" : "italian", "count" : 7 } { "_id" : "caribbean", "count" : 6 } { "_id" : "kebab", "count" : 6 }
-
Top 10 leisure trends
> db.birm.aggregate([{"$match":{"leisure":{"$exists":1}}}, {"$group":{"_id":"$leisure", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "park", "count" : 536 } { "_id" : "playground", "count" : 273 } { "_id" : "pitch", "count" : 181 } { "_id" : "sports_centre", "count" : 116 } { "_id" : "picnic_table", "count" : 91 } { "_id" : "nature_reserve", "count" : 54 } { "_id" : "golf_course", "count" : 49 } { "_id" : "fitness_centre", "count" : 38 } { "_id" : "garden", "count" : 37 } { "_id" : "common", "count" : 33 }
-
Top 10 fast-food brands
> db.birm.aggregate([{"$match":{"name":{"$exists":1},"amenity":"fast_food"}}, {"$group":{"_id":"$name","count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":10}]) { "_id" : "Subway", "count" : 67 } { "_id" : "McDonald's", "count" : 27 } { "_id" : "Greggs", "count" : 25 } { "_id" : "KFC", "count" : 24 } { "_id" : "Burger King", "count" : 17 } { "_id" : "Domino's Pizza", "count" : 11 } { "_id" : "Dixy Chicken", "count" : 9 } { "_id" : "Pizza Hut", "count" : 8 } { "_id" : "Fish & Chips", "count" : 8 } { "_id" : "Papa John's", "count" : 7 }
-
Top 5 shop trends
> db.birm.aggregate([{"$match":{"shop":{"$exists":1,"$ne":"yes"}}}, {"$group":{"_id":"$shop", "count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":5}]) { "_id" : "hairdresser", "count" : 812 } { "_id" : "convenience", "count" : 742 } { "_id" : "clothes", "count" : 731 } { "_id" : "supermarket", "count" : 282 } { "_id" : "car_repair", "count" : 239 }
-
Top 5 supermarket brands
> db.birm.aggregate([{"$match":{"name":{"$exists":1},"shop": "supermarket"}}, {"$group":{"_id":"$name","count":{"$sum":1}}}, {"$sort":{"count":-1}}, {"$limit":5}]) { "_id" : "The Co-operative Food", "count" : 32 } { "_id" : "Tesco Express", "count" : 14 } { "_id" : "Spar", "count" : 13 } { "_id" : "Costcutter", "count" : 13 } { "_id" : "Aldi", "count" : 12 }
-
Accessibility for disabled
> db.birm.aggregate([{"$match":{"wheelchair":{"$exists":1},"type": "way"}}, {"$group":{"_id":"$wheelchair","count":{"$sum":1}}}, {"$sort":{"count":-1}}]) { "_id" : "yes", "count" : 112 } { "_id" : "no", "count" : 7 } { "_id" : "limited", "count" : 4 } { "_id" : "designated", "count" : 4 }
-
Religion of worship
> db.birm.aggregate([{"$match":{"religion":{"$exists":1},"type": "node"}}, {"$group":{"_id":"$religion","count":{"$sum":1}}}, {"$sort":{"count":-1}}]) { "_id" : "christian", "count" : 223 } { "_id" : "muslim", "count" : 18 } { "_id" : "hindu", "count" : 3 } { "_id" : "sikh", "count" : 1 } { "_id" : "jewish", "count" : 1 } { "_id" : "multifaith", "count" : 1 } { "_id" : "spiritualist", "count" : 1 }
Conclusion
OpenStreetMap data presents an ideal opportunity for me to practice data wrangling and a special way to explore citys. Although this review of data is cursory, I think is has been well cleaned for the purposes of the exercise.