forked from swcarpentry/sql-novice-survey
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* add r lesson * add bobby tables and key points * add R helper functions * add polish * move text * minor text change * fix exercises time * increase teaching time to reflect added content * fix spacing
- Loading branch information
Showing
2 changed files
with
273 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,267 @@ | ||
--- | ||
title: Programming with Databases - R | ||
teaching: 30 | ||
exercises: 15 | ||
questions: | ||
- "How can I access databases from programs written in R?" | ||
objectives: | ||
- "Write short programs that execute SQL queries." | ||
- "Trace the execution of a program that contains an SQL query." | ||
- "Explain why most database applications are written in a general-purpose language rather than in SQL." | ||
keypoints: | ||
- "Data analysis languages have libraries for accessing databases." | ||
- "To connect to a database, a program must use a library specific to that database manager." | ||
- "R's libraries can be used to directly query or read from a database." | ||
- "Programs can read query results in batches or all at once." | ||
- "Queries should be written using parameter substitution, not string formatting." | ||
- "R has multiple helper functions to make working with databases easier." | ||
--- | ||
|
||
To close, | ||
let's have a look at how to access a database from | ||
a data analysis language like R. | ||
Other languages use almost exactly the same model: | ||
library and function names may differ, | ||
but the concepts are the same. | ||
|
||
Here's a short R program that selects latitudes and longitudes | ||
from an SQLite database stored in a file called `survey.db`: | ||
|
||
~~~ | ||
library(RSQLite) | ||
connection <- dbConnect(SQLite(), "survey.db") | ||
results <- dbGetQuery(connection, "SELECT Site.lat, Site.long FROM Site;") | ||
print(results) | ||
dbDisconnect(connection) | ||
~~~ | ||
{: .r} | ||
~~~ | ||
lat long | ||
1 -49.85 -128.57 | ||
2 -47.15 -126.72 | ||
3 -48.87 -123.40 | ||
~~~ | ||
{: .output} | ||
|
||
The program starts by importing the `RSQLite` library. | ||
If we were connecting to MySQL, DB2, or some other database, | ||
we would import a different library, | ||
but all of them provide the same functions, | ||
so that the rest of our program does not have to change | ||
(at least, not much) | ||
if we switch from one database to another. | ||
|
||
Line 2 establishes a connection to the database. | ||
Since we're using SQLite, | ||
all we need to specify is the name of the database file. | ||
Other systems may require us to provide a username and password as well. | ||
|
||
On line 3, we retrieve the results from an SQL query. | ||
It's our job to make sure that SQL is properly formatted; | ||
if it isn't, | ||
or if something goes wrong when it is being executed, | ||
the database will report an error. | ||
This result is a dataframe with one row for each entry and one column for each column in the database. | ||
|
||
Finally, the last line closes our connection, | ||
since the database can only keep a limited number of these open at one time. | ||
Since establishing a connection takes time, | ||
though, | ||
we shouldn't open a connection, | ||
do one operation, | ||
then close the connection, | ||
only to reopen it a few microseconds later to do another operation. | ||
Instead, | ||
it's normal to create one connection that stays open for the lifetime of the program. | ||
|
||
Queries in real applications will often depend on values provided by users. | ||
For example, | ||
this function takes a user's ID as a parameter and returns their name: | ||
|
||
~~~ | ||
library(RSQLite) | ||
connection <- dbConnect(SQLite(), "survey.db") | ||
getName <- function(personID) { | ||
query <- paste0("SELECT personal || ' ' || family FROM Person WHERE ident =='", | ||
personID, "';") | ||
return(dbGetQuery(connection, query)) | ||
} | ||
print(paste("full name for dyer:", getName('dyer'))) | ||
dbDisconnect(connection) | ||
~~~ | ||
{: .r} | ||
~~~ | ||
full name for dyer: William Dyer | ||
~~~ | ||
{: .output} | ||
|
||
We use string concatenation on the first line of this function | ||
to construct a query containing the user ID we have been given. | ||
This seems simple enough, | ||
but what happens if someone gives us this string as input? | ||
|
||
~~~ | ||
dyer'; DROP TABLE Survey; SELECT ' | ||
~~~ | ||
{: .sql} | ||
|
||
It looks like there's garbage after the user's ID, | ||
but it is very carefully chosen garbage. | ||
If we insert this string into our query, | ||
the result is: | ||
|
||
~~~ | ||
SELECT personal || ' ' || family FROM Person WHERE id='dyer'; DROP TABLE Survey; SELECT ''; | ||
~~~ | ||
{: .sql} | ||
|
||
If we execute this, | ||
it will erase one of the tables in our database. | ||
|
||
This is called an [SQL injection attack](reference.html#sql-injection-attack), | ||
and it has been used to attack thousands of programs over the years. | ||
In particular, | ||
many web sites that take data from users insert values directly into queries | ||
without checking them carefully first. | ||
A very [relevant XKCD](https://xkcd.com/327/) that explains the | ||
dangers of using raw input in queries a little more succinctly: | ||
|
||
![relevant XKCD](https://imgs.xkcd.com/comics/exploits_of_a_mom.png) | ||
|
||
Since an unscrupulous parent might try to smuggle commands into our queries in many different ways, | ||
the safest way to deal with this threat is | ||
to replace characters like quotes with their escaped equivalents, | ||
so that we can safely put whatever the user gives us inside a string. | ||
We can do this by using a [prepared statement](reference.html#prepared-statement) | ||
instead of formatting our statements as strings. | ||
Here's what our example program looks like if we do this: | ||
|
||
~~~ | ||
library(RSQLite) | ||
connection <- dbConnect(SQLite(), "survey.db") | ||
getName <- function(personID) { | ||
query <- "SELECT personal || ' ' || family FROM Person WHERE ident == ?" | ||
return(dbGetPreparedQuery(connection, query, data.frame(personID))) | ||
} | ||
print(paste("full name for dyer:", getName('dyer'))) | ||
dbDisconnect(connection) | ||
~~~ | ||
{: .r} | ||
~~~ | ||
full name for dyer: William Dyer | ||
~~~ | ||
{: .output} | ||
|
||
The key changes are in the query string and the `dbGetQuery` call (we use dbGetPreparedQuery instead). | ||
Instead of formatting the query ourselves, | ||
we put question marks in the query template where we want to insert values. | ||
When we call `dbGetPreparedQuery`, | ||
we provide a dataframe | ||
that contains as many values as there are question marks in the query. | ||
The library matches values to question marks in order, | ||
and translates any special characters in the values | ||
into their escaped equivalents | ||
so that they are safe to use. | ||
|
||
> ## Filling a Table vs. Printing Values | ||
> | ||
> Write an R program that creates a new database in a file called | ||
> `original.db` containing a single table called `Pressure`, with a | ||
> single field called `reading`, and inserts 100,000 random numbers | ||
> between 10.0 and 25.0. How long does it take this program to run? | ||
> How long does it take to run a program that simply writes those | ||
> random numbers to a file? | ||
{: .challenge} | ||
|
||
> ## Filtering in SQL vs. Filtering in R | ||
> | ||
> Write an R program that creates a new database called | ||
> `backup.db` with the same structure as `original.db` and copies all | ||
> the values greater than 20.0 from `original.db` to `backup.db`. | ||
> Which is faster: filtering values in the query, or reading | ||
> everything into memory and filtering in R? | ||
{: .challenge} | ||
|
||
## Database helper functions in R | ||
|
||
R's database interface packages (like `RSQLite`) all share | ||
a common set of helper functions useful for exploring databases and | ||
reading/writing entire tables at once. | ||
|
||
To view all tables in a database, we can use `dbListTables()`: | ||
|
||
~~~ | ||
connection <- dbConnect(SQLite(), "survey.db") | ||
dbListTables(connection) | ||
~~~ | ||
{: .r} | ||
~~~ | ||
"Person" "Site" "Survey" "Visited" | ||
~~~ | ||
{: .output} | ||
|
||
|
||
To view all column names of a table, use `dbListFields()`: | ||
|
||
~~~ | ||
dbListFields(connection, "Survey") | ||
~~~ | ||
{: .r} | ||
~~~ | ||
"taken" "person" "quant" "reading" | ||
~~~ | ||
{: .output} | ||
|
||
|
||
To read an entire table as a dataframe, use `dbReadTable()`: | ||
|
||
~~~ | ||
dbReadTable(connection, "Person") | ||
~~~ | ||
{: .r} | ||
~~~ | ||
ident personal family | ||
1 dyer William Dyer | ||
2 pb Frank Pabodie | ||
3 lake Anderson Lake | ||
4 roe Valentina Roerich | ||
5 danforth Frank Danforth | ||
~~~ | ||
{: .output} | ||
|
||
|
||
Finally to write an entire table to a database, you can use `dbWriteTable()`. | ||
Note that we will always want to use the `row.names = FALSE` argument or R | ||
will write the row names as a separate column. | ||
In this example we will write R's built-in `iris` dataset as a table in `survey.db`. | ||
|
||
~~~ | ||
dbWriteTable(connection, "iris", iris, row.names = FALSE) | ||
head(dbReadTable(connection, "iris")) | ||
~~~ | ||
{: .r} | ||
~~~ | ||
Sepal.Length Sepal.Width Petal.Length Petal.Width Species | ||
1 5.1 3.5 1.4 0.2 setosa | ||
2 4.9 3.0 1.4 0.2 setosa | ||
3 4.7 3.2 1.3 0.2 setosa | ||
4 4.6 3.1 1.5 0.2 setosa | ||
5 5.0 3.6 1.4 0.2 setosa | ||
6 5.4 3.9 1.7 0.4 setosa | ||
~~~ | ||
{: .output} | ||
|
||
And as always, remember to close the database connection when done! | ||
|
||
~~~ | ||
dbDisconnect(connection) | ||
~~~ | ||
{: .r} | ||
|