Extracting Data

Overview

Teaching: 20 min
Exercises: 10 min
Questions
  • What do we do with data we cannot quickly bulk load into a database?

  • How does python read csvs?

Objectives
  • Explain Why do we call csv.reader a wrapper?

  • Write a short Python program to extract data from a CSV file.

  • Explain the pitfalls of parsing data formats like CSV using string splitting.

  • Explain why string splitting is nevertheless an acceptable approach for extracting authors’ names from this data.

The extract, transform, and load process

When our data complexity is higher than what can be accomplished by .import, we need to perform what is called an “Extract, Transform, and Load.”

We’d like to know:

Unfortunately, we can’t use .import to load this bibliography into our database, because one of the fields is a dreaded and evil “multi-valued field.” (Hint: this is a bad thing for any sort of data analysis.)

If we only cared about one author, we could search for her name in the spreadsheet to answer the first question, then select those rows and manually tally her co-authors to answer the second. But doing that for all of the authors one by one would take days, we’d almost certainly make mistakes, and then someone would almost certainly hand us another, larger spreadsheet and we’d have to start over.

This might seem like a lot of work to answer two questions, but for anything more than a half-dozen rows, it will save us a lot of time:

Exploring the data

Our starting point is a spreadsheet called bibliography.csv with 2937 rows like this:

key type year authors title journal
8SW85SQM journalArticle 2013 McClelland, James L Incorporating Rapid Neocortical Learning of New Schema-Consistent Information Into Complementary Learning Systems Theory. J Exp Psychol Gen
85QV9X5F journalArticle 1995 McClelland, J. L.; McNaughton, B. L.; O’Reilly, R. C. Why There are Complementary Learning Systems in the Hippocampus and Neocortex: Insights from the Successes and Failures of Connectionist Models of Learning and Memory Psychological Review
Z4X6DT6N journalArticle 1990 Ratcliff, R. Connectionist models of recognition memory: constraints imposed by learning and forgetting functions. Psychological review

The first step is to turn the rows of the spreadsheet into (key, author) pairs. Let’s start by making sure that Python can read the spreadsheet properly:

# count-lines.py
# Count how many lines there are in the spreadsheet
import sys

filename = sys.argv[1]
count = 0

with open(filename, 'r') as reader:
    for line in reader:
        count += 1

print(count)

This should look familiar by now: the filename is given as the first command-line argument (sys.argv[1]), so we open that file and use a for loop to read it line by line. Each time the loop executes, it adds 1 to a variable called count; when the loop finishes, we close the file and print the count.

We can run this program like this:

$ python count-lines.py bibliography_data/bibliography.csv

Sure enough, its output is:

2937

so we know that Python is reading all of the rows.

The next step is to break each line into fields so that we can get each entry’s key and authors. The fields are separated by commas, so we could try using str.split. This won’t work, though, because authors’ names also contain commas (since they are formatted as “last, first”).

What we can do instead is ask our favorite search engine for help. Sure enough, a search for “python csv” turns up a library called csv, which is part of the standard Python distribution. Its documentation includes a few examples, and after a couple of experiments, we come up with this:

# read-fields.py
# Make sure we can read the fields from a CSV file.

import sys
import csv

with open(sys.argv[1], 'r') as raw:
    reader = csv.reader(raw);
    for line in reader:
        print(line)

This program starts by opening the bibliography file (again, we’ll pass its name as the first command-line argument). It then calls csv.reader to create a wrapper around the file. When the basic file object created by open reads a line at a time, the wrapper created by csv.reader breaks that line into fields at the right places. It knows how to handle commas embedded in fields, special characters, and a bunch of other things that we don’t want to have to worry about.

To check that it’s working correctly, we just print out each line after it has been processed by the CSV reader. Its first few lines of output are:

$ python read-fields.py bibliography_data/bibliography.csv | head -5
['8SW85SQM', 'journalArticle', '2013', 'McClelland, James L', 'Incorporating Rapid Neocortical Learning of New Schema-Consistent Information Into Complementary Learning Systems Theory.', 'J Exp Psychol Gen', '', '1939-2222', '', 'http://www.biomedsearch.com/nih/Incorporating-Rapid-Neocortical-Learning-New/23978185.html', '', '', '', '', '', '', '', '', '']
['85QV9X5F', 'journalArticle', '1995', "McClelland, J. L.; McNaughton, B. L.; O'Reilly, R. C.", 'Why There are Complementary Learning Systems in the Hippocampus and Neocortex: Insights from the Successes and Failures of Connectionist Models of Learning and Memory', 'Psychological Review', '', '', '', '', '', '', '', '', '', '', '', '', '']
['Z4X6DT6N', 'journalArticle', '1990', 'Ratcliff, R.', 'Connectionist models of recognition memory: constraints imposed by learning and forgetting functions.', 'Psychological review', '', '0033-295X', '', 'http://view.ncbi.nlm.nih.gov/pubmed/2186426', '', '', '', '', '', '', '', '', '']
['F5DGU3Q4', 'bookSection', '1989', 'McCloskey, M.; Cohen, N. J.', 'Catastrophic Interference in Connectionist Networks: The Sequential Learning Problem', 'The Psychology of Learning and Motivation, Vol. 24', '', '', '', '', '', '', '', '', '', '', '', '', '']
['PNGQMCP5', 'conferencePaper', '2006', 'Buciluǎ, Cristian; Caruana, Rich; Niculescu-Mizil, Alexandru', 'Model compression', 'Proceedings of the 12th ACM SIGKDD international conference on Knowledge discovery and data mining', '', '', '', '', '', '', '', '', '', '', '', '', '']

(Notice that we run the program’s output through head to only display the first few lines rather than scrolling back through its output.) This is exactly what we need: the key is in the first element of each list, and the authors are all together in the fourth. Let’s modify the program to print out just those two fields:

# display-fields.py
# Print the key and all the authors

import sys
import csv

with open(sys.argv[1], 'r') as raw:
    reader = csv.reader(raw);
    for line in reader:
        print(line[0], line[3])

Its output is:

8SW85SQM McClelland, James L
85QV9X5F McClelland, J. L.; McNaughton, B. L.; O'Reilly, R. C.
Z4X6DT6N Ratcliff, R.
F5DGU3Q4 McCloskey, M.; Cohen, N. J.
PNGQMCP5 Buciluǎ, Cristian; Caruana, Rich; Niculescu-Mizil, Alexandru

The last step is to turn lines with multiple authors into multiple lines, each with a single author. This is the right time to use str.split: the authors’ names are separated by semicolons, so we can break each list of authors on those and use another loop to print the results one by one.

# display-authors-1.py
# Print (key, author) pairs.

import sys
import csv

with open(sys.argv[1], 'r') as raw:
    reader = csv.reader(raw);
    for line in reader:
        key, authors = line[0], line[3]
        for auth in authors.split(';'):
            print(key, auth)

$ python display-authors-1.py bibliography_data/bibliography.csv | head -10
8SW85SQM McClelland, James L
85QV9X5F McClelland, J. L.
85QV9X5F  McNaughton, B. L.
85QV9X5F  O'Reilly, R. C.
Z4X6DT6N Ratcliff, R.
F5DGU3Q4 McCloskey, M.
F5DGU3Q4  Cohen, N. J.
PNGQMCP5 Buciluǎ, Cristian
PNGQMCP5  Caruana, Rich
PNGQMCP5  Niculescu-Mizil, Alexandru

That’s close to what we want, but not quite right: since authors’ names are actually separated by a semicolon and a space, and we’re only splitting on semicolons, the second and subsequent name on each line comes out with an unwanted space at the front. What happens if we try to split on a semicolon plus a space?

# display-authors-1.py
# Print (key, author) pairs.

import sys
import csv

with open(sys.argv[1], 'r') as raw:
    reader = csv.reader(raw);
    for line in reader:
        key, authors = line[0], line[3]
        for auth in authors.split('; '): # semicolon plus space instead of semicolon
            print(key, auth)

8SW85SQM McClelland, James L
85QV9X5F McClelland, J. L.
85QV9X5F McNaughton, B. L.
85QV9X5F O'Reilly, R. C.
Z4X6DT6N Ratcliff, R.
F5DGU3Q4 McCloskey, M.
F5DGU3Q4 Cohen, N. J.
PNGQMCP5 Buciluǎ, Cristian
PNGQMCP5 Caruana, Rich
PNGQMCP5 Niculescu-Mizil, Alexandru

And that’s that: the first step of our data extraction is done. Since we’ve achieved something useful, we save it for posterity:

$ git init .
Initialized empty Git repository in /Users/aturing/lessons/capstone-novice-spreadsheet-biblio/.git
$ git add -A
$ git status
On branch master

Initial commit

Changes to be committed:
  (use "git rm --cached <file>..." to unstage)

  new file:   count-lines.py
  new file:   display-authors-1.py
  new file:   display-authors-2.py
  new file:   display-fields.py
  new file:   read-fields.py
  new file:   bibliography_data/bibliography.csv
$ git commit -m "Extracting (key, author) pairs from bibliography"
[master (root-commit) 9db78ed] Extracting (key, author) pairsfrom bibliography
 6 files changed, 2996 insertions(+)
 create mode 100644 count-lines.py
 create mode 100644 display-authors-1.py
 create mode 100644 display-authors-2.py
 create mode 100644 display-fields.py
 create mode 100644 read-fields.py
 create mode 100644 bibliography_data/bibliography.csv

Checking Assumptions

You suspect that a handful of authors’ names are separated only by a semicolon rather than by a semicolon and space. What Unix shell command or commands could you use to check for this?

Safer Splitting

Suppose you find that some authors’ names are separated only by a semicolon rather than by a semicolon and a space. Modify the program so that it splits the author field on semicolons, then strips unwanted spaces from individual authors’ names while printing.

Breaking Even

If it takes 10 minutes to write a program to do a task that only takes 5 minutes to do by hand, the program is only worth writing if the task has to be done more than twice. Similarly, if we only wanted to find out who had co-authored papers with one specific author, and we were sure we would never have any other questions or need to re-do the analysis, manually searching the spreadsheet would probably be faster than transcribing the data into a database.

Choose some task that you currently do by hand. Estimate how long it takes you to do it each time, how often you do it, and how long it would take you to write a program to do the task instead. How much time would programming actually save you? How sure are you?

Key Points

  • Spreadsheets are inflexible and can present challenges when needing to answer novel questions.

  • Any programming language can be used to transform data into a different format which is better for analysis.