New data scientists typically import entire tables from a database into R, and then merge, filter, and groom the data.frames. A more efficient approach is to submit sql that executes on the database and returns a more specialized dataset.
This provides several advantages:
- A database will be much more efficient when filtering and joining tables than any programing language, such as R or Python. A well-designed database will have indexed columns and other optimizations that surpass R and Python capabilities.
- A database handles datasets that are thousands of times larger than what R and Python can accommodate in RAM. For large datasets, database engines persist the data on a hard drive (instead of just RAM) and are optimized to read the necessary information into RAM the moment before it is needed, and then return the processed back to disk before progressing to the next block of data.
- Frequently, only a portion of the table's rows and columns are ultimately needed by the analysis. Reducing the size of the dataset leaving the database has two benefits: less information travels across the network and R's and Python's limited memory space is conserved.
In some scenarios, it is desirable to use the INSERT
SQL command to transfer data within the database; and never travel across the network and never touch R or your local machine. For our large and complicated projects, the majority of data movement uses INSERT
commands within SQL files. Among these scenarios, the analysis-focused projects use R to call the sequence of SQL files (see flow.R
), while the database-focused project uss SSIS.
In both cases, we try to write the SQL files to conform to similar standards and conventions. As stated in Consistency across Files (and in the previous chapter), using a consistent file structure can (a) improve the quality of the code because the structure has been proven over time to facilitate good practices and (b) allow your intentions to be more clear to teammates because they are familiar with the order and intentions of the chunks.
The major relational database engines use roughly the same syntax, but they all have slight deviations and enhancements beyond the SQL standards. Most of our databases are hosted by SQL Server, since that is what OUHSC's campus seems most comfortable supporting. Consequently, this chapter uses SQL Server 2017+ syntax.
But like most data science teams, we still need to consume other databases, such as Oracle and MySQL. Outside OUHSC projects, we tend to use PostgreSQL and Redshift.
This basic sql file moves data within a database to create a table named dx
, which is contained in the ley_covid_1
schema of the cdw_staging
database.
--use cdw_staging
declare @start_date date = '2020-02-01'; -- sync with config.yml
declare @stop_date date = dateadd(day, -1, cast(getdate() as date)); -- sync with config.yml
DROP TABLE if exists ley_covid_1.dx;
CREATE TABLE ley_covid_1.dx(
dx_id int identity primary key,
patient_id int not null,
covid_confirmed bit not null,
problem_date date,
icd10_code varchar(20) not null
);
-- TRUNCATE TABLE ley_covid_1.dx;
INSERT INTO ley_covid_1.dx
SELECT
pr.patient_id
,ss.covid_confirmed
,pr.invoice_date as problem_date
,pr.code as icd10_code
-- into ley_covid_1.dx
FROM cdw.star_1.fact_problem as pr
inner join beasley_covid_1.ss_dx as ss on pr.code = ss.icd10_code
WHERE
pr.problem_date_start between @start_date and @stop_date
and
pr.patient_id is not null
ORDER BY pr.patient_id, pr.problem_date_start desc
CREATE INDEX ley_covid_1_dx_patient_id on ley_covid_1.dx (patient_id);
CREATE INDEX ley_covid_1_dx_icd10_code on ley_covid_1.dx (icd10_code);
We prefer not to specify the database of each table, and instead control it through the connection (such as the DSN's "default database" value). Nevertheless, it's helpful to include the default database behind a comment for two reasons. First, it communicates to the default database to the human reader. Second, during debugging, the code can be highlighted in ADS/SSMS and executed with "F5"; this will mimic what happens when the file is run via automation with a DSN.
--use cdw_staging
Similar to the Declare Globals chunk in a prototypical R file, values set at the top of the file are easy to read and modify.
declare @start_date date = '2020-02-01'; -- sync with config.yml
declare @stop_date date = dateadd(day, -1, cast(getdate() as date)); -- sync with config.yml
When batch-loading data, it is typically easiest drop and recreate a database table. In the snippet below, any table with the specific name is dropped/deleted from the database and replaced with a (possibly new) definition. We like to dedicate a line to each table column, with at least three elements per line: the name, the data type, and if nulls are allowed.
Many other features and keywords are available when designing tables. The ones we occasionally use are:
primary key
helps database optimization when later querying the table, and enforces uniqueness, such as a patient table should not have any two rows with the samepatient_id
value. Primary keys must be nonmissing, so thenot null
keyword is redundant.unique
is helpful when a table has additional columns that need to be unique (such aspatient_ssn
andpatient_id
). A more advanced scenario using a clustered columnar table, which is incompatible with theprimary key
designation.identity(1, 1)
creates a 1, 2, 3, ... sequence, which relieves the client of creating the sequence with something likerow_number()
. Note that when identity column exists, the number columns in theSELECT
clause will be one fewer than the columns defined inCREATE TABLE
.
DROP TABLE if exists ley_covid_1.dx;
CREATE TABLE ley_covid_1.dx(
dx_id int identity(1, 1) primary key,
patient_id int not null,
covid_confirmed bit not null,
problem_date date null,
icd10_code varchar(20) not null
);
To jump-start the creation of the table definition, we frequently use the INTO
clause. This operation creates a new table, informed the column properties of the source tables. Within ADS and SSMS, refresh the list of tables and select the new table; there will be an option to copy the CREATE TABLE
statement (similar to the snippet above) and paste it into the sql file. The definition can then be modified, such as tightening from null
to not null
.
-- into ley_covid_1.dx
In scenarios where the table definition is stable and the data is refreshed frequently (say, daily), consider TRUNCATE-ing the table. When taking this approach, we prefer to keep the DROP
and CREATE
code in the file, but commented out. This saves development time in the future if the table definition needs to be modified.
-- TRUNCATE TABLE ley_covid_1.dx;
The INSERT INTO
(when followed by a SELECT
clause), simply moves data from the query into the specified table.
The INSERT INTO
clause transfers the columns in the exact order of the query. It does not try to match to the names of the destination table. An error will be thrown if the column types are mismatched (e.g., attempting to insert a character string into an integer value).
Even worse, no error will be thrown if the mismatched columns have compatible types. This will occur if the table's columns are patient_id
, weight_kg
, and height_cm
, but the query's columns are patient_id
, height_cm
, and weight_in
. Not only will the weight and height be written to the incorrect columns, but the execution will not catch that the source is weight_kg
, but the destination is weight_in
.
INSERT INTO ley_covid_1.dx
The SELECT
clause specifies the desired columns. It can also rename columns and perform manipulations.
We prefer to specify the aliased table of each column. If two source tables have the same column name, an error will be thrown regarding the ambiguity. Even if that's not a concern, we believe that explicitly specifying the source improves readability and reduces errors.
SELECT
pr.patient_id
,ss.covid_confirmed
,cast(pr.invoice_datetime as date) as problem_date
,pr.code as icd10_code
FROM cdw.star_1.fact_problem as pr
inner join beasley_covid_1.ss_dx as ss on pr.code = ss.icd10_code
The WHERE
clause reduces the number of returned rows (as opposed to reducing the number of columns in the SELECT
clause). Use the indention level to communicate to reader how the subclauses are combined. This is especially important if it both AND
and OR
operators are used, since their order of operations can be confused easily.
WHERE
pr.problem_date_start between @start_date and @stop_date
and
pr.patient_id is not null
The ORDER BY
clause simply specifies the order of the rows. Be default, a column's values will be in ascending order, but can be descending if desired.
ORDER BY pr.patient_id, pr.problem_date_start desc
If the table is large or queried in a variety of ways, indexing the table can speed up performance dramatically.
CREATE INDEX ley_covid_1_dx_patient_id on ley_covid_1.dx (patient_id);
CREATE INDEX ley_covid_1_dx_icd10_code on ley_covid_1.dx (icd10_code);