summaryrefslogtreecommitdiff
path: root/app/models/concerns/queryable.rb
blob: 1d9468abefe85d1479dbc4947b49446fe95b8059 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
module Queryable
  extend ActiveSupport::Concern

  included { send :include, ClassMethods }

  module ClassMethods
    def filter_by(filters)
      return all if filters.empty?
      filters.reduce(self) do |current_scope, filter|
        filter.call(current_scope)
      end
    end

    def query_builder_for(params)
      Queryable::Builder.new(params).tap do |builder|
        yield builder if block_given?
      end
    end
  end

  class Builder
    include Enumerable
    attr_reader :params, :filters
    delegate :empty?, to: :filters

    def initialize(params)
      @params = params
      @filters = []
    end

    def always(&filter)
      add do |relation|
        filter.call(relation)
      end
    end

    def if_present(key, &filter)
      return if params[key].nil? || params[key] == ''
      add do |relation|
        filter.call(relation, normalize(params[key]))
      end
    end

    def if_missing(key, &filter)
      return if params[key].present?
      add do |relation|
        filter.call(relation, normalize(params[key]))
      end
    end

    def each(&block)
      @filters.each(&block)
    end

    private

    def add(&filter)
      @filters.push(filter)
    end

    def boolean?(value)
      true?(value) || false?(value)
    end

    def true?(value)
      "true" == value.to_s.downcase
    end

    def false?(value)
      "false" == value.to_s.downcase
    end

    def normalize(value)
      boolean?(value) ? true?(value) : value
    end
  end
end