-> scales -> measure. Precalculate averages for performance.pull/1/head
parent
3f2279e2e8
commit
105f30f220
@ -0,0 +1,6 @@
|
||||
class RaceScore < ApplicationRecord
|
||||
belongs_to :measure
|
||||
belongs_to :school
|
||||
belongs_to :academic_year
|
||||
belongs_to :race
|
||||
end
|
||||
@ -0,0 +1,17 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
class RaceScoreCalculator
|
||||
include Analyze::Graph::Column::RacialScore
|
||||
attr_reader :measure, :school, :academic_year, :race
|
||||
|
||||
def initialize(measure:, school:, academic_year:, race:)
|
||||
@measure = measure
|
||||
@school = school
|
||||
@academic_year = academic_year
|
||||
@race = race
|
||||
end
|
||||
|
||||
def score
|
||||
race_score(measure:, school:, academic_year:, race:)
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,12 @@
|
||||
class Sample
|
||||
attr_reader :school, :academic_year, :category, :measure, :race
|
||||
|
||||
def initialize
|
||||
@school = School.find_by_slug 'milford-high-school'
|
||||
@academic_year = AcademicYear.last
|
||||
@category = Category.find_by_category_id '1'
|
||||
@subcategory = Subcategory.find_by_subcategory_id '1A'
|
||||
@measure = Measure.find_by_measure_id '1A-ii'
|
||||
@race = Race.find_by_qualtrics_code 1
|
||||
end
|
||||
end
|
||||
@ -1,7 +1,7 @@
|
||||
class Student < ApplicationRecord
|
||||
has_many :survey_item_responses
|
||||
has_many :student_races
|
||||
has_many :races, through: :student_races
|
||||
has_and_belongs_to_many :races, join_table: :student_races
|
||||
|
||||
encrypts :lasid, deterministic: true
|
||||
end
|
||||
|
||||
@ -1,20 +0,0 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module Analyze
|
||||
module Graph
|
||||
module Column
|
||||
module RaceScore
|
||||
def race_score(measure:, school:, academic_year:, race:)
|
||||
survey_items = measure.student_survey_items
|
||||
students = StudentRace.where(race:).pluck(:student_id)
|
||||
average = SurveyItemResponse.where(school:,
|
||||
academic_year:,
|
||||
survey_item: survey_items,
|
||||
student: students)
|
||||
.average(:likert_score)
|
||||
Score.new(average, true, true, true)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,72 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module Analyze
|
||||
module Graph
|
||||
module Column
|
||||
module RacialScore
|
||||
def race_score(measure:, school:, academic_year:, race:)
|
||||
rate = response_rate(school:, academic_year:, measure:)
|
||||
return Score.new(0, false, false, false) unless rate.meets_student_threshold
|
||||
|
||||
survey_items = measure.student_survey_items
|
||||
|
||||
students = StudentRace.where(race:).pluck(:student_id).uniq
|
||||
averages = grouped_responses(school:, academic_year:, survey_items:, students:)
|
||||
number_of_responses = total_responses(school:, academic_year:, students:, survey_items:)
|
||||
scorify(responses: averages, number_of_responses:)
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def grouped_responses(school:, academic_year:, survey_items:, students:)
|
||||
SurveyItemResponse.where(school:,
|
||||
academic_year:,
|
||||
student: students,
|
||||
survey_item: survey_items)
|
||||
.group(:survey_item_id)
|
||||
.average(:likert_score)
|
||||
end
|
||||
|
||||
def total_responses(school:, academic_year:, students:, survey_items:)
|
||||
@total_responses ||= SurveyItemResponse.where(school:,
|
||||
academic_year:,
|
||||
student: students,
|
||||
survey_item: survey_items).count
|
||||
end
|
||||
|
||||
def response_rate(school:, academic_year:, measure:)
|
||||
@response_rate ||= Hash.new do |memo, (school, academic_year)|
|
||||
memo[[school, academic_year]] =
|
||||
ResponseRate.find_by(subcategory: measure.subcategory, school:, academic_year:)
|
||||
end
|
||||
|
||||
@response_rate[[school, academic_year]]
|
||||
end
|
||||
|
||||
def scorify(responses:, number_of_responses:)
|
||||
averages = bubble_up_averages(responses:)
|
||||
average = averages.average
|
||||
|
||||
meets_student_threshold = sufficient_responses(averages:, number_of_responses:)
|
||||
average = 0 unless meets_student_threshold
|
||||
|
||||
Score.new(average, false, meets_student_threshold, false)
|
||||
end
|
||||
|
||||
def sufficient_responses(averages:, number_of_responses:)
|
||||
total_questions = averages.count
|
||||
average_num_of_responses = number_of_responses.to_f / total_questions
|
||||
meets_student_threshold = average_num_of_responses >= 10
|
||||
end
|
||||
|
||||
def bubble_up_averages(responses:)
|
||||
measure.student_scales.map do |scale|
|
||||
scale.survey_items.map do |survey_item|
|
||||
responses[survey_item.id]
|
||||
end.remove_blanks.average
|
||||
end.remove_blanks
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,16 @@
|
||||
module Analyze
|
||||
module Graph
|
||||
module Column
|
||||
module ScoreForRace
|
||||
def score(year_index)
|
||||
s = ::RaceScore.find_by(measure:, school:, academic_year: academic_years[year_index], race:)
|
||||
average = s.average unless s.nil?
|
||||
average ||= 0
|
||||
meets_student_threshold = s.meets_student_threshold? unless s.nil?
|
||||
meets_student_threshold ||= false
|
||||
Score.new(average, false, meets_student_threshold, false)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,29 @@
|
||||
class RaceScoreLoader
|
||||
def self.reset(schools: School.all, academic_years: AcademicYear.all, measures: Measure.all, races: Race.all)
|
||||
RaceScore.where(school: schools, academic_year: academic_years, measure: measures, race: races).delete_all
|
||||
scores = []
|
||||
measures.each do |measure|
|
||||
schools.each do |school|
|
||||
academic_years.each do |academic_year|
|
||||
races.each do |race|
|
||||
scores << process_score(measure:, school:, academic_year:, race:)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
RaceScore.import scores, batch_size: 1000
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def self.process_score(measure:, school:, academic_year:, race:)
|
||||
score = RaceScoreCalculator.new(measure:, school:, academic_year:, race:).score
|
||||
rs = RaceScore.find_by(measure:, school:, academic_year:, race:)
|
||||
rs ||= RaceScore.new(measure:, school:, academic_year:, race:)
|
||||
rs.average = score.average
|
||||
rs.meets_student_threshold = score.meets_student_threshold?
|
||||
rs
|
||||
end
|
||||
|
||||
private_class_method :process_score
|
||||
end
|
||||
@ -0,0 +1,14 @@
|
||||
class CreateRaceScores < ActiveRecord::Migration[7.0]
|
||||
def change
|
||||
create_table :race_scores do |t|
|
||||
t.references :measure, null: false, foreign_key: true
|
||||
t.references :school, null: false, foreign_key: true
|
||||
t.references :academic_year, null: false, foreign_key: true
|
||||
t.references :race, null: false, foreign_key: true
|
||||
t.float :average
|
||||
t.boolean :meets_student_threshold
|
||||
|
||||
t.timestamps
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,41 @@
|
||||
require 'rails_helper'
|
||||
|
||||
describe RaceScoreCalculator do
|
||||
let(:measure) { create(:measure, :with_student_survey_items) }
|
||||
let(:school) { create(:school) }
|
||||
let(:academic_year) { create(:academic_year) }
|
||||
let(:race) { create(:race) }
|
||||
let(:student) do
|
||||
s = create(:student)
|
||||
s.races << race
|
||||
s.save
|
||||
s
|
||||
end
|
||||
let(:survey_item_1) { measure.survey_items[0] }
|
||||
let(:survey_item_2) { measure.survey_items[1] }
|
||||
let(:survey_item_3) { measure.survey_items[2] }
|
||||
let(:response_rate) do
|
||||
create(:response_rate, school:, academic_year:, subcategory: measure.subcategory, meets_student_threshold: true)
|
||||
end
|
||||
|
||||
context 'when survey item responses exist' do
|
||||
before :each do
|
||||
response_rate
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 1, survey_item: survey_item_1, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 2, survey_item: survey_item_1,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 3, survey_item: survey_item_1, student:)
|
||||
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 2, survey_item: survey_item_2, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 3, survey_item: survey_item_2,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 4, survey_item: survey_item_2, student:)
|
||||
end
|
||||
|
||||
it 'returns a list of averages' do
|
||||
expect(measure.student_survey_items.count).to eq 2
|
||||
american_indian_score = RaceScoreCalculator.new(measure:, school:, academic_year:, race:).score
|
||||
expect(american_indian_score).to eq Score.new(2.5, false, true, false)
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,46 @@
|
||||
require 'rails_helper'
|
||||
|
||||
include Analyze::Graph::Column
|
||||
|
||||
# RacialScore is a module used in the RaceScoreCalculator class
|
||||
describe RacialScore do
|
||||
let(:measure) { create(:measure, :with_student_survey_items) }
|
||||
let(:school) { create(:school) }
|
||||
let(:academic_year) { create(:academic_year) }
|
||||
let(:race) { create(:race) }
|
||||
let(:student) do
|
||||
s = create(:student)
|
||||
s.races << race
|
||||
s.save
|
||||
s
|
||||
end
|
||||
let(:survey_item_1) { measure.survey_items[0] }
|
||||
let(:survey_item_2) { measure.survey_items[1] }
|
||||
let(:survey_item_3) { measure.survey_items[2] }
|
||||
let(:response_rate) do
|
||||
create(:response_rate, school:, academic_year:, subcategory: measure.subcategory, meets_student_threshold: true)
|
||||
end
|
||||
|
||||
context 'when survey item responses exist' do
|
||||
before :each do
|
||||
response_rate
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 1, survey_item: survey_item_1, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 2, survey_item: survey_item_1,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 3, survey_item: survey_item_1, student:)
|
||||
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 2, survey_item: survey_item_2, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 3, survey_item: survey_item_2,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 4, survey_item: survey_item_2, student:)
|
||||
end
|
||||
|
||||
it 'returns a list of averages' do
|
||||
expect(measure.student_survey_items.count).to eq 2
|
||||
students = StudentRace.where(race:).pluck(:student_id)
|
||||
|
||||
american_indian_score = RaceScoreCalculator.new(measure:, school:, academic_year:, race:).score
|
||||
expect(american_indian_score).to eq Score.new(2.5, false, true, false)
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,50 @@
|
||||
require 'rails_helper'
|
||||
|
||||
describe RaceScoreLoader do
|
||||
let(:measure) { create(:measure, :with_student_survey_items) }
|
||||
let(:school) { create(:school) }
|
||||
let(:academic_year) { create(:academic_year) }
|
||||
let(:race) { create(:race) }
|
||||
let(:student) do
|
||||
s = create(:student)
|
||||
s.races << race
|
||||
s.save
|
||||
s
|
||||
end
|
||||
let(:survey_item_1) { measure.survey_items[0] }
|
||||
let(:survey_item_2) { measure.survey_items[1] }
|
||||
let(:survey_item_3) { measure.survey_items[2] }
|
||||
let(:response_rate) do
|
||||
create(:response_rate, school:, academic_year:, subcategory: measure.subcategory, meets_student_threshold: true)
|
||||
end
|
||||
|
||||
context 'when survey item responses exist' do
|
||||
before :each do
|
||||
response_rate
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 1, survey_item: survey_item_1, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 2, survey_item: survey_item_1,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 3, survey_item: survey_item_1, student:)
|
||||
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 2, survey_item: survey_item_2, student:)
|
||||
create_list(:survey_item_response, 8, school:, academic_year:, likert_score: 3, survey_item: survey_item_2,
|
||||
student:)
|
||||
create(:survey_item_response, school:, academic_year:, likert_score: 4, survey_item: survey_item_2, student:)
|
||||
RaceScoreLoader.reset
|
||||
end
|
||||
|
||||
it 'returns a list of averages' do
|
||||
expect(measure.student_survey_items.count).to eq 2
|
||||
american_indian_score = RaceScore.find_by(measure:, school:, academic_year:, race:)
|
||||
expect(american_indian_score.average).to eq 2.5
|
||||
expect(american_indian_score.meets_student_threshold).to eq true
|
||||
end
|
||||
|
||||
it 'is idempotent' do
|
||||
original_count = RaceScore.count
|
||||
RaceScoreLoader.reset
|
||||
new_count = RaceScore.count
|
||||
expect(original_count).to eq new_count
|
||||
end
|
||||
end
|
||||
end
|
||||
Loading…
Reference in new issue