class Conductor MAX_WEIGHTING_FACTOR = 1.25 EQUALIZATION_PERIOD_DEFAULT = 7 MINIMUM_CONVERSIONS_PER_GROUP_DEFAULT = 10 DBG = false cattr_writer :cache def self.cache @@cache || Rails.cache end class << self # Specifies a unique identity for the current visitor. If no identity is specified # then a random value is selected. Conductor makes sure that the same visitor # will always see the same alternative selections to reduce confusion. def identity=(value) @conductor_identity = value end def identity return (@conductor_identity || ActiveSupport::SecureRandom.hex(16)) end # The number of days to include when calculating weights # The inclusion period MUST be higher than then equalization period # The default is 14 days def inclusion_period=(value) raise "Conductor.inclusion_period must be a positive number > 0" unless value.is_a?(Numeric) && value > 0 raise "Conductor.inclusion_period must be greater than the equalization period" if value < equalization_period @inclusion_period = value end def inclusion_period return (@inclusion_period || 14) end # The minimum number of conversions that a group needs to have in TOTAL before # weighting is allowed. # # TODO: trigger a notification if a post equalized group hits below this number def minimum_conversions_per_group=(value) raise "Conductor.minimum_conversions_per_group must be a positive number > 0" unless value.is_a?(Numeric) && value > 0 @minimum_conversions_per_group = value end def minimum_conversions_per_group return (@minimum_conversions_per_group || MINIMUM_CONVERSIONS_PER_GROUP_DEFAULT) end # The equalization period is the initial amount of time, in days, that conductor # should apply the max_weighting_factor towards a new alternative to ensure # that it receives a far shot of performing. # # If an equalization period was not used then any new alternative would # immediately be weighed very low since it has no conversions and would # never have a chance of performing def equalization_period=(value) raise "Conductor.equalization_period must be a positive number > 0" unless value.is_a?(Numeric) && value > 0 @equalization_period = value end def equalization_period return (@equalization_period || EQUALIZATION_PERIOD_DEFAULT) end # The attribute for weighting specifies if the conversion_value OR number of conversions # should be used to calculate the weight. The default is conversions. # # TODO: Change this to only use conversion rate when normalization is figured out def attribute_for_weighting=(value) raise "Conductor.attribute_for_weighting must be either :views, :conversions or :conversion_value (default)" unless [:views, :conversions, :conversion_value].include?(value) @attribute_for_weighting = value end def attribute_for_weighting return (@attribute_for_weighting || :conversions) end def log(msg) puts msg if DBG end def sanitize(str) str.gsub(/\s/,'_').downcase end end end class Array def sum_it(attribute) self.map {|x| x.send(attribute) }.compact.sum end def weighted_mean_of_attribute(attribute) self.map {|x| x.send(attribute) }.compact.weighted_mean end def weighted_mean w_sum = sum(self) return 0.00 if w_sum == 0.00 w_prod = 0 self.each_index {|i| w_prod += (i+1) * self[i].to_f} w_prod.to_f / w_sum.to_f end end