require File.expand_path(File.dirname(__FILE__) + '/test_helper') class TestConductor < Test::Unit::TestCase # Wipes cache, D/B prior to doing a test run. def setup Conductor.cache.clear wipe end context "conductor" do should "assign an identity if none is specified" do assert Conductor.identity != nil end should "select one of the specified options randomly" do selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # => value must be unique assert ["a", "b", "c"].include? selected end should "use the cache if working" do Conductor.cache.write('testing','value') x = Conductor.cache.read('testing') assert_equal x, 'value' end should "allow for the equalization_period to be configurable" do Conductor.equalization_period = 3 assert_equal(3, Conductor.equalization_period) end should "raise an error if a non-numeric value, negative or 0 value is specified for the equalization_period" do assert_raise(RuntimeError, LoadError) { Conductor.equalization_period = 'junk'} assert_raise(RuntimeError, LoadError) { Conductor.equalization_period = -1.0} assert_raise(RuntimeError, LoadError) { Conductor.equalization_period = 0} assert_nothing_raised(RuntimeError, LoadError) { Conductor.equalization_period = 3} end should "raise an error if an improper attribute is specified for @attribute_for_weighting" do assert_raise(RuntimeError, LoadError) { Conductor.attribute_for_weighting = :random} end should "almost equally select each option if no weights exist" do a = 0 b = 0 c = 0 (1..1000).each do |x| Conductor.identity = ActiveSupport::SecureRandom.hex(16) selected_lander = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # => value must be unique case selected_lander when 'a' then a += 1 when 'b' then b += 1 when 'c' then c += 1 end end nums = [] << a << b << c nums.sort! range = nums.last - nums.first assert (nums.first * 0.20) >= range end end context "a single site visitor" do setup do Conductor.identity = ActiveSupport::SecureRandom.hex(16) end should "always select the same alternative when using the cache" do selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # => value must be unique different = false (1..100).each do |x| different = true if selected != Conductor::Experiment.pick('a_group', ["a", "b", "c"]) end assert !different end should "select a lander and then successfully record a conversion" do selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # => value must be unique Conductor::Experiment.track! experiments = Conductor::Experiment::Raw.find_all_by_identity_id(Conductor.identity) assert_equal 1, experiments.count assert_equal 1, experiments.first.conversion_value end should "select a lander and then successfully record custom conversion value" do selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # => value must be unique Conductor::Experiment.track!({:value => 12.34}) experiments = Conductor::Experiment::Raw.find_all_by_identity_id(Conductor.identity) assert_equal 1, experiments.count assert_equal 12.34, experiments.first.conversion_value end should "record three different experiments with two goals but a single conversion for all goals for the same identity" do first = Conductor::Experiment.pick('a_group', ["a", "b", "c"], {:goal => 'goal_1'}) # => value must be unique second = Conductor::Experiment.pick('b_group', ["1", "2", "3"], {:goal => 'goal_2'}) # => value must be unique third = Conductor::Experiment.pick('c_group', ["zz", "xx", "yy"], {:goal => 'goal_1'}) # => value must be unique Conductor::Experiment.track! experiments = Conductor::Experiment::Raw.find_all_by_identity_id(Conductor.identity) assert_equal 3, experiments.count assert_equal 2, experiments.count {|x| x.goal == 'goal_1'} assert_equal 3, experiments.sum_it(:conversion_value) end should "record three different experiments with two goals but only track a conversion for goal_1" do first = Conductor::Experiment.pick('a_group', ["a", "b", "c"], {:goal => 'goal_1'}) # => value must be unique second = Conductor::Experiment.pick('b_group', ["1", "2", "3"], {:goal => 'goal_2'}) # => value must be unique third = Conductor::Experiment.pick('c_group', ["zz", "xx", "yy"], {:goal => 'goal_1'}) # => value must be unique Conductor::Experiment.track!({:goal => 'goal_1'}) experiments = Conductor::Experiment::Raw.find_all_by_identity_id(Conductor.identity) assert_equal 3, experiments.count assert_equal 2, experiments.count {|x| x.goal == 'goal_1'} assert_equal 2, experiments.sum_it(:conversion_value) end end context "conductor" do setup do seed_raw_data(100) Conductor::RollUp.process Conductor.identity = ActiveSupport::SecureRandom.hex(16) end should "correctly RollUp daily data" do assert Conductor::Experiment::Daily.count > 2 assert Conductor::Experiment::Daily.all.detect {|x| x.conversions > 0} assert Conductor::Experiment::Daily.all.detect {|x| x.views > 0} assert Conductor::Experiment::Daily.all.detect {|x| x.conversion_value > 0} end should "correctly populate weighting table when selecting a value" do selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) assert_equal 3, Conductor::Experiment::Weight.count end should "pull weights from the cache" do Conductor::Experiment.pick('a_group', ["a", "b", "c"]) (1..100).each do |x| Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor::Experiment.pick('a_group', ["a", "b", "c"]) end # => if this works the history table should have only been updated one time not 101 so there should # => be three records (one for a, b and c) assert_equal 3, Conductor::Experiment::History.count end should "pull weights from the cache and then recreate weights when the alternative list changes" do Conductor::Experiment.pick('a_group', ["a", "b", "c"]) (1..100).each do |x| Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor::Experiment.pick('a_group', ["a", "b", "c"]) end Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor::Experiment.pick('a_group', ["a", "c"]) # => if this works the history table should have only been updated one time not 101 so there should # => be FIVE records (one for a, b and c and then one for a and c) assert_equal 5, Conductor::Experiment::History.count end end context "conductor" do setup do wipe seed_raw_data(100, 7) Conductor::RollUp.process end should "populate the weighting table with equal weights if all new options are launched" do # hit after rollup to populare weight table Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor.equalization_period = 7 weights = Conductor::Experiment.weights('a_group', ["a", "b", "c"]) # each weight will be equal to 0.18 assert_equal 7, Conductor.equalization_period assert_equal weights['a'], weights['b'] assert_equal weights['c'], weights['b'] end end context "conductor" do setup do seed_raw_data(100, 14); # rollup Conductor::RollUp.process # hit after rollup to populare weight table Conductor.identity = ActiveSupport::SecureRandom.hex(16) selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) end should "populate the weighting table with different weights" do # if this DOES NOT work then each weight will be equal to 0.18 assert_not_equal 0.54, Conductor::Experiment::Weight.all.sum_it(:weight).to_f end should "record the new weights in the weight history table in database" do assert Conductor::Experiment::History.count > 1 end should "return a weight 1.25 times higher than the highest weight for a newly launched and non-recorded alernative" do # get the highest weight max_weight = Conductor::Experiment::Weight.maximum(:weight) # pick something weights = Conductor::Experiment.weights('a_group', ["a", "b", "c", "f"]) # => value must be unique assert_equal weights['f'], (max_weight * 1.25) end end context "conductor" do should "correctly record the launch window in the weight histories table" do seed_raw_data(10, 6) # rollup Conductor::RollUp.process # hit after rollup to populare weight table Conductor.identity = ActiveSupport::SecureRandom.hex(16) selected = Conductor::Experiment.pick('a_group', ["a", "b", "c"]) # make sure that launch_window values can be detected assert_not_nil Conductor::Experiment::History.find(:all, :conditions => 'launch_window > 0') end end context "conductor" do setup do seed_raw_data(500, 30) # rollup Conductor::RollUp.process end should "correctly calculate weights even if there are no conversions" do Conductor::Experiment::Daily.update_all('conversion_value = 0.00, conversions = 0') Conductor.identity = ActiveSupport::SecureRandom.hex(16) assert_nil Conductor::Experiment::Daily.all.detect {|x| x.conversions > 0 || x.conversion_value > 0} assert_equal 3, Conductor::Experiment.weights('a_group', ["a", "b", "c"]).values.sum end should "correctly calculate weights even if an alternative has no conversions" do Conductor::Experiment::Daily.update_all('conversion_value = 0.00, conversions = 0', "alternative = 'a'") Conductor.identity = ActiveSupport::SecureRandom.hex(16) assert_nil Conductor::Experiment::Daily.find_all_by_alternative('a').detect {|x| x.conversions > 0 || x.conversion_value > 0} assert_equal 0, Conductor::Experiment.weights('a_group', ["a", "b", "c"])['a'] end should "allow for the number of conversions to be used for weighting instead of conversion_value" do Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor::Experiment.pick('a_group', ["a", "b", "c"]) weights_cv = Conductor::Experiment::Weight.all.map(&:weight).sort Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor.attribute_for_weighting = :conversions Conductor::Experiment.pick('a_group', ["a", "b", "c"]) weights_c = Conductor::Experiment::Weight.all.map(&:weight).sort # since one is using conversion_value and the other is using conversions, they two weight arrays should be different assert_equal :conversions, Conductor.attribute_for_weighting assert_not_equal weights_cv, weights_c end end context "conductor" do setup do seed_raw_data(500, 30) # rollup Conductor::RollUp.process end should "weight everything equally if the minimum number of conversions per group has not been hit" do Conductor::Experiment::Daily.update_all('conversion_value = 0.00, conversions = 0') Conductor::Experiment::Daily.update_all('conversion_value = 1.00, conversions = 1', '', :limit => 10) Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor.attribute_for_weighting = :conversions Conductor.minimum_conversions_per_group = 15 weights = Conductor::Experiment.weights('a_group', ["a", "b", "c"]) assert_equal weights['a'], weights['b'] assert_equal weights['c'], weights['b'] end should "weight everything correctly if the minimum number of conversions per group HAS been hit" do Conductor::Experiment::Daily.update_all('conversion_value = 0.00, conversions = 0') Conductor::Experiment::Daily.all.each_with_index {|x,ndx| if ndx < 10 x.update_attributes(:conversion_value => rand(100), :conversions => rand(20)) end } Conductor.identity = ActiveSupport::SecureRandom.hex(16) Conductor.attribute_for_weighting = :conversions weights = Conductor::Experiment.weights('a_group', ["a", "b", "c"]) assert_not_equal weights['a'], weights['b'] assert_not_equal weights['c'], weights['b'] end end private def wipe Conductor::Experiment::Daily.delete_all Conductor::Experiment::Raw.delete_all Conductor::Experiment::Weight.delete_all Conductor::Experiment::History.delete_all end def seed_raw_data(num, days_ago=14) # seed the raw data (1..num).each do |x| Conductor.identity = ActiveSupport::SecureRandom.hex(16) options = {:created_at => rand(days_ago).days.ago} options.merge!({:conversion_value => rand(100)}) if rand() < 0.20 # => convert 20% of traffic selected_lander = Conductor::Experiment.pick('a_group', ["a", "b", "c"], options) # => value must be unique end end end