Take Home Midterm



Midterm Assignments

Setup

This work should all be housed in a GitHub repo. Establish the repo and add your collaborators.

Loading the data takes a minute, so I would suggest you do it once and cache it. This just means including knitr::opts_chunk$set(cache = TRUE) in one of your chunk options.

The problem with caching is that sometimes results of a later chunk depend upon earlier ones and things can get out of sync. If you make a change and it doesn’t have the result you expect, try clearing the cache and knitting again.

If this is confusing, don’t worry about it. Either check in with me or ignore this part and don’t use caching. It will just take a little longer to render your file each time you click “knit”.

I would also recommend not tracking the cache files. This means adding the cache folder to your .gitignore file. If you initialized your repo with the R .gitignore file this is actually already taken care of for you. If not, add /*_cache/ to your .gitignore.

Part A: Data

20 points

The following function downloads data from the Oregon Department of education website on the number of students who scored in each performance category on the statewide assessment by race/ethnicity for every school in the state. It takes one argument, year, which must be a two digit integer from 15 to 18 (representing the 2014-15 to 2017-18 school years). It actually won’t work on later years because the file structure changed and I wanted to keep the function simple.

NOTE: This function uses the glue function from the package of the same name. If you do not already have this package installed, please first install it with install.packages("glue"). It also uses {rio} for the import, which you should already have installed, but if not, install that first too.

download_file <- function(year) {
  link <- glue::glue("https://www.oregon.gov/ode/educator-resources/assessment/TestResults20{year}/pagr_schools_ela_raceethnicity_{year-1}{year}.xlsx")
  rio::import(link, setclass = "tibble", na = c("-", "--", "*"))
}
  1. (10 points) Use the function above to download all of the data from the 2014-15 to 2017-18 school years and bind it into a single data frame, using a single function (i.e., one line of code). Note, this may take a minute or two to run, depending on your Internet speed.
## # A tibble: 110,943 × 21
##    `Academic Year` `District ID` District   `School ID` School                
##    <chr>                   <dbl> <chr>            <dbl> <chr>                 
##  1 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  2 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  3 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  4 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  5 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  6 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  7 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  8 2014-2015                2063 Adel SD 21         498 Adel Elementary School
##  9 2014-2015                2063 Adel SD 21         498 Adel Elementary School
## 10 2014-2015                2063 Adel SD 21         498 Adel Elementary School
## # … with 110,933 more rows, and 16 more variables: Subject <chr>,
## #   `Student Group` <chr>, `Grade Level` <chr>,
## #   `Number Proficient (Level 3 or 4)` <chr>,
## #   `Percent Proficient (Level 3 or 4)` <chr>, `Number Level 4` <dbl>,
## #   `Percent Level 4` <dbl>, `Number Level 3` <dbl>, `Percent Level 3` <dbl>,
## #   `Number Level 2` <dbl>, `Percent Level 2` <dbl>, `Number Level 1` <dbl>,
## #   `Percent Level 1` <dbl>, `Number of Participants` <dbl>, …

Conduct some basic data cleaning to make your data file look like the following.

## # A tibble: 90,576 × 7
##    academic_year district     school                   student_group  
##    <chr>         <chr>        <chr>                    <chr>          
##  1 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  2 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  3 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  4 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  5 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  6 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  7 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  8 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##  9 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
## 10 2014-2015     Adrian SD 61 Adrian Elementary School Hispanic/Latino
##    grade_level level     n
##    <chr>       <chr> <int>
##  1 Grade 3     1         2
##  2 Grade 3     2         1
##  3 Grade 3     3         2
##  4 Grade 3     4         1
##  5 Grade 6     1         2
##  6 Grade 6     2         3
##  7 Grade 6     3         2
##  8 Grade 6     4         0
##  9 Grade 7     1         2
## 10 Grade 7     2         2
## # … with 90,566 more rows
  1. (10 points) Sum the n across grades and school for each district within each academic year, student group, and level. Calculate the number of schools (unique schools listed) for each district. Also move your data to a wider format where the student groups become new columns, filled by the district n. Remove districts that did not report numbers for one or both student groups (i.e., drop missing data from the new columns). Your data frame should look like the below
## # A tibble: 1,948 × 6
##    academic_year district     n_schools level hispanic_latino white
##    <chr>         <chr>            <int> <chr>           <int> <int>
##  1 2014-2015     Adrian SD 61         2 1                   6     8
##  2 2014-2015     Adrian SD 61         2 2                   6    17
##  3 2014-2015     Adrian SD 61         2 3                   6    23
##  4 2014-2015     Adrian SD 61         2 4                   1    22
##  5 2015-2016     Adrian SD 61         2 1                   5     4
##  6 2015-2016     Adrian SD 61         2 2                   3    15
##  7 2015-2016     Adrian SD 61         2 3                   5    39
##  8 2015-2016     Adrian SD 61         2 4                   0    36
##  9 2016-2017     Adrian SD 61         2 1                   2     4
## 10 2016-2017     Adrian SD 61         2 2                   0    19
## # … with 1,938 more rows

Part B: Achievement disparities

40 points

If you have not already done so, please install the {gapr} package using the following code

remotes::install_github("datalorax/gapr")

The {gapr} package includes, at the moment, one function, estimate_v(), which estimates the average difference between two distributions in terms of an effect size when the only data available are counts within bins (for more information, see Ho & Reardon, 2012). The nice thing about this approach is that we’re able to obtain the effect size of average differences in achievement between two groups of students as if we had the full, student level data even though we just have counts within bins (performance levels).

The estimate_v function takes 3 arguments in the following order: (1) the data frame including the counts for each group, (2) a character string stating the name of the column with the counts for the reference group, and (3) a character string stating the name of the column with the counts for the focal group.

To estimate the average achievement difference between students coded Hispanic/Latino versus White for the first school district in the first year in our data, we would run the following

library(gapr)
d[1:4, ] %>% # data source - just first four rows for first district
  estimate_v("white", "hispanic_latino") # columns for reference/focal groups
##         auc          v       v_se
## 1 0.2824054 -0.8141778 0.07283486

We can see that students coded Hispanic/Latino scored, on average, about 0.81 standard deviations below students coded White within this school district for this year.

  1. (10 points) Estimate the average achievement differences in terms of an effect size for every school district in the state that reported data on both student groups (i.e., using the data we created above), for each academic year. Conduct this analysis twice, once using group_by() %>% nest() map() and once using nest_by() %>% summarize().

Note, on my computer these took about 40 seconds per analysis. It will likely take a bit longer for you, but shouldn’t take too long. You may get a warning. Just go ahead and ignore that for now.

  1. (5 points)

Reproduce the following plot to confirm that the estimates are equivalent across methods.

  1. (15 points) The plot below shows the achievement difference estimates for one school district across years. Produce a similar plot to the below for each of the first 100 school districts. Once you’ve created all the plots, reproduce them again using an alternative method. Make sure you don’t miss the caption noting the number of schools in the district.

Hint: You need to select unique districts. You may want to try something like the following

dists <- unique(v$district)[1:100]

model_list_object %>% 
  filter(district %in% dists)

Extra Credit: For up to one point extra credit, create a third set that “staples together” each of the preceding pairs of plots (i.e., to confirm that the plots were produced identically by both methods).

  1. (10 points) Save the plots into a “plots” directory. Make sure the file names are meaningful.

HINTS