code-mancers/rapidfire

View on GitHub
app/services/rapidfire/survey_results.rb

Summary

Maintainability
A
0 mins
Test Coverage
module Rapidfire
  class SurveyResults < Rapidfire::BaseService
    attr_accessor :survey

    # extracts question along with results
    # each entry will have the following:
    # 1. question type and question id
    # 2. question text
    # 3. if aggregatable, return each option with value
    # 4. else return an array of all the answers given
    def extract(filter_params = {})
      @survey.questions.collect do |question|
        results =
          case question
          when Rapidfire::Questions::Select, Rapidfire::Questions::Radio,
            Rapidfire::Questions::Checkbox
            answers = question.answers.where(*filter(filter_params)).map(&:answer_text).map do |text|
              text.to_s.split(Rapidfire.answers_delimiter)
            end.flatten

            count_hash = Hash.new(0)
            question.options.map{|o| count_hash[o] = 0 }

            answers.inject(count_hash) { |total, e| total[e] += 1; total }
          when Rapidfire::Questions::Short, Rapidfire::Questions::Date,
            Rapidfire::Questions::Long, Rapidfire::Questions::Numeric
            question.answers.where(*filter(filter_params)).pluck(:answer_text)
          end

        QuestionResult.new(question: question, results: results)
      end
    end

    def filter(filter_params, column = 'attempt_id')
      @filter_result ||= begin
        return ["0=0"] unless Array(filter_params[:question_ids]).compact.count == Array(filter_params[:options]).compact.count
        return ["0=0"] unless %w(id attempt_id).include?(column)
        return ["0=0"] if Array(filter_params[:question_ids]).compact.count == 0

        collected_filters = {}
        Array(filter_params[:question_ids]).each_with_index do |question_id, i|
          collected_filters[question_id] ||= []
          collected_filters[question_id] << filter_params[:options][i]
        end

        attempt_ids = nil

        collected_filters.each do |question_id, options|
          these_matches = Answer.where("question_id = ? and answer_text in (?)", question_id, options).pluck(:attempt_id)
          if attempt_ids.nil?
            attempt_ids = these_matches
          else
            attempt_ids = attempt_ids & these_matches
          end
        end

        ["#{column} in (?)", attempt_ids]
      end
    end

    def self.filter(filter_params, column)
      self.new.filter(filter_params, column)
    end
  end
end