122 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| class Trends::Query
 | |
|   include Redisable
 | |
|   include Enumerable
 | |
| 
 | |
|   attr_reader :prefix, :klass, :loaded
 | |
| 
 | |
|   alias loaded? loaded
 | |
| 
 | |
|   def initialize(prefix, klass)
 | |
|     @prefix  = prefix
 | |
|     @klass   = klass
 | |
|     @records = []
 | |
|     @loaded  = false
 | |
|     @allowed = false
 | |
|     @limit   = nil
 | |
|     @offset  = nil
 | |
|   end
 | |
| 
 | |
|   def allowed!
 | |
|     @allowed = true
 | |
|     self
 | |
|   end
 | |
| 
 | |
|   def allowed
 | |
|     clone.allowed!
 | |
|   end
 | |
| 
 | |
|   def in_locale!(value)
 | |
|     @locale = value
 | |
|     self
 | |
|   end
 | |
| 
 | |
|   def in_locale(value)
 | |
|     clone.in_locale!(value)
 | |
|   end
 | |
| 
 | |
|   def offset!(value)
 | |
|     @offset = value.to_i
 | |
|     self
 | |
|   end
 | |
| 
 | |
|   def offset(value)
 | |
|     clone.offset!(value)
 | |
|   end
 | |
| 
 | |
|   def limit!(value)
 | |
|     @limit = value.to_i
 | |
|     self
 | |
|   end
 | |
| 
 | |
|   def limit(value)
 | |
|     clone.limit!(value)
 | |
|   end
 | |
| 
 | |
|   def records
 | |
|     load
 | |
|     @records
 | |
|   end
 | |
| 
 | |
|   delegate :each, :empty?, :first, :last, :size, to: :records
 | |
| 
 | |
|   def to_ary
 | |
|     records.dup
 | |
|   end
 | |
| 
 | |
|   alias to_a to_ary
 | |
| 
 | |
|   def to_arel
 | |
|     if ids_for_key.empty?
 | |
|       klass.none
 | |
|     else
 | |
|       scope = klass.joins(sanitized_join_sql).reorder('x.ordering')
 | |
|       scope = scope.offset(@offset) if @offset.present?
 | |
|       scope = scope.limit(@limit) if @limit.present?
 | |
|       scope
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def key
 | |
|     [@prefix, @allowed ? 'allowed' : 'all', @locale].compact.join(':')
 | |
|   end
 | |
| 
 | |
|   def load
 | |
|     unless loaded?
 | |
|       @records = perform_queries
 | |
|       @loaded  = true
 | |
|     end
 | |
| 
 | |
|     self
 | |
|   end
 | |
| 
 | |
|   def ids_for_key
 | |
|     @ids_for_key ||= redis.zrevrange(key, 0, -1).map(&:to_i)
 | |
|   end
 | |
| 
 | |
|   def sanitized_join_sql
 | |
|     ActiveRecord::Base.sanitize_sql_array(join_sql_array)
 | |
|   end
 | |
| 
 | |
|   def join_sql_array
 | |
|     [join_sql_query, ids_for_key]
 | |
|   end
 | |
| 
 | |
|   def join_sql_query
 | |
|     <<~SQL.squish
 | |
|       JOIN unnest(array[?]) WITH ordinality AS x (id, ordering) ON #{klass.table_name}.id = x.id
 | |
|     SQL
 | |
|   end
 | |
| 
 | |
|   def perform_queries
 | |
|     apply_scopes(to_arel).to_a
 | |
|   end
 | |
| 
 | |
|   def apply_scopes(scope)
 | |
|     scope
 | |
|   end
 | |
| end
 |