# frozen_string_literal: true class ActivityTracker include Redisable EXPIRE_AFTER = 6.months.seconds def initialize(prefix, type) @prefix = prefix @type = type end def add(value = 1, at_time = Time.now.utc) key = key_at(at_time) case @type when :basic redis.incrby(key, value) when :unique redis.pfadd(key, value) end redis.expire(key, EXPIRE_AFTER) end def get(start_at, end_at = Time.now.utc) (start_at.to_date...end_at.to_date).map do |date| key = key_at(date.to_time(:utc)) value = begin case @type when :basic redis.get(key).to_i when :unique redis.pfcount(key) end end [date, value] end end def sum(start_at, end_at = Time.now.utc) keys = (start_at.to_date...end_at.to_date).flat_map { |date| [key_at(date.to_time(:utc)), legacy_key_at(date)] }.uniq case @type when :basic redis.mget(*keys).map(&:to_i).sum when :unique redis.pfcount(*keys) end end class << self def increment(prefix) new(prefix, :basic).add end def record(prefix, value) new(prefix, :unique).add(value) end end private def key_at(at_time) "#{@prefix}:#{at_time.beginning_of_day.to_i}" end def legacy_key_at(at_time) "#{@prefix}:#{at_time.to_date.cweek}" end end