require 'spec_helper' # TODO change some of these tests to use Rack::Test describe Split::Helper do include Split::Helper let(:experiment) { Split::Experiment.find_or_create('link_color', 'blue', 'red') } describe "ab_test" do it "should not raise an error when passed strings for alternatives" do lambda { ab_test('xyz', '1', '2', '3') }.should_not raise_error end it "should raise the appropriate error when passed integers for alternatives" do lambda { ab_test('xyz', 1, 2, 3) }.should raise_error(ArgumentError) end it "should raise the appropriate error when passed symbols for alternatives" do lambda { ab_test('xyz', :a, :b, :c) }.should raise_error(ArgumentError) end it "should not raise error when passed an array for goals" do lambda { ab_test({'link_color' => ["purchase", "refund"]}, 'blue', 'red') }.should_not raise_error end it "should not raise error when passed just one goal" do lambda { ab_test({'link_color' => "purchase"}, 'blue', 'red') }.should_not raise_error end it "should assign a random alternative to a new user when there are an equal number of alternatives assigned" do ab_test('link_color', 'blue', 'red') ['red', 'blue'].should include(ab_user['link_color']) end it "should increment the participation counter after assignment to a new user" do previous_red_count = Split::Alternative.new('red', 'link_color').participant_count previous_blue_count = Split::Alternative.new('blue', 'link_color').participant_count ab_test('link_color', 'blue', 'red') new_red_count = Split::Alternative.new('red', 'link_color').participant_count new_blue_count = Split::Alternative.new('blue', 'link_color').participant_count (new_red_count + new_blue_count).should eql(previous_red_count + previous_blue_count + 1) end it 'should not increment the counter for an experiment that the user is not participating in' do ab_test('link_color', 'blue', 'red') e = Split::Experiment.find_or_create('button_size', 'small', 'big') lambda { # User shouldn't participate in this second experiment ab_test('button_size', 'small', 'big') }.should_not change { e.participant_count } end it 'should not increment the counter for an ended experiment' do e = Split::Experiment.find_or_create('button_size', 'small', 'big') e.winner = 'small' lambda { a = ab_test('button_size', 'small', 'big') a.should eq('small') }.should_not change { e.participant_count } end it "should return the given alternative for an existing user" do alternative = ab_test('link_color', 'blue', 'red') repeat_alternative = ab_test('link_color', 'blue', 'red') alternative.should eql repeat_alternative end it 'should always return the winner if one is present' do experiment.winner = "orange" ab_test('link_color', 'blue', 'red').should == 'orange' end it "should allow the alternative to be force by passing it in the params" do @params = {'link_color' => 'blue'} alternative = ab_test('link_color', 'blue', 'red') alternative.should eql('blue') alternative = ab_test('link_color', {'blue' => 1}, 'red' => 5) alternative.should eql('blue') @params = {'link_color' => 'red'} alternative = ab_test('link_color', 'blue', 'red') alternative.should eql('red') alternative = ab_test('link_color', {'blue' => 5}, 'red' => 1) alternative.should eql('red') end it "should allow passing a block" do alt = ab_test('link_color', 'blue', 'red') ret = ab_test('link_color', 'blue', 'red') { |alternative| "shared/#{alternative}" } ret.should eql("shared/#{alt}") end it "should allow the share of visitors see an alternative to be specificed" do ab_test('link_color', {'blue' => 0.8}, {'red' => 20}) ['red', 'blue'].should include(ab_user['link_color']) end it "should allow alternative weighting interface as a single hash" do ab_test('link_color', {'blue' => 0.01}, 'red' => 0.2) experiment = Split::Experiment.find('link_color') experiment.alternatives.map(&:name).should eql(['blue', 'red']) # TODO: persist alternative weights # experiment.alternatives.collect{|a| a.weight}.should == [0.01, 0.2] end it "should only let a user participate in one experiment at a time" do link_color = ab_test('link_color', 'blue', 'red') ab_test('button_size', 'small', 'big') ab_user.should eql({'link_color' => link_color}) big = Split::Alternative.new('big', 'button_size') big.participant_count.should eql(0) small = Split::Alternative.new('small', 'button_size') small.participant_count.should eql(0) end it "should let a user participate in many experiment with allow_multiple_experiments option" do Split.configure do |config| config.allow_multiple_experiments = true end link_color = ab_test('link_color', 'blue', 'red') button_size = ab_test('button_size', 'small', 'big') ab_user.should eql({'link_color' => link_color, 'button_size' => button_size}) button_size_alt = Split::Alternative.new(button_size, 'button_size') button_size_alt.participant_count.should eql(1) end it "should not over-write a finished key when an experiment is on a later version" do experiment.increment_version ab_user = { experiment.key => 'blue', experiment.finished_key => true } finshed_session = ab_user.dup ab_test('link_color', 'blue', 'red') ab_user.should eql(finshed_session) end end describe 'finished' do before(:each) do @experiment_name = 'link_color' @alternatives = ['blue', 'red'] @experiment = Split::Experiment.find_or_create(@experiment_name, *@alternatives) @alternative_name = ab_test(@experiment_name, *@alternatives) @previous_completion_count = Split::Alternative.new(@alternative_name, @experiment_name).completed_count end it 'should increment the counter for the completed alternative' do finished(@experiment_name) new_completion_count = Split::Alternative.new(@alternative_name, @experiment_name).completed_count new_completion_count.should eql(@previous_completion_count + 1) end it "should set experiment's finished key if reset is false" do finished(@experiment_name, {:reset => false}) ab_user.should eql(@experiment.key => @alternative_name, @experiment.finished_key => true) end it 'should not increment the counter if reset is false and the experiment has been already finished' do 2.times { finished(@experiment_name, {:reset => false}) } new_completion_count = Split::Alternative.new(@alternative_name, @experiment_name).completed_count new_completion_count.should eql(@previous_completion_count + 1) end it 'should not increment the counter for an experiment that the user is not participating in' do ab_test('button_size', 'small', 'big') # So, user should be participating in the link_color experiment and # receive the control for button_size. As the user is not participating in # the button size experiment, finishing it should not increase the # completion count for that alternative. lambda { finished('button_size') }.should_not change { Split::Alternative.new('small', 'button_size').completed_count } end it 'should not increment the counter for an ended experiment' do e = Split::Experiment.find_or_create('button_size', 'small', 'big') e.winner = 'small' a = ab_test('button_size', 'small', 'big') a.should eq('small') lambda { finished('button_size') }.should_not change { Split::Alternative.new(a, 'button_size').completed_count } end it "should clear out the user's participation from their session" do ab_user.should eql(@experiment.key => @alternative_name) finished(@experiment_name) ab_user.should == {} end it "should not clear out the users session if reset is false" do ab_user.should eql(@experiment.key => @alternative_name) finished(@experiment_name, {:reset => false}) ab_user.should eql(@experiment.key => @alternative_name, @experiment.finished_key => true) end it "should reset the users session when experiment is not versioned" do ab_user.should eql(@experiment.key => @alternative_name) finished(@experiment_name) ab_user.should eql({}) end it "should reset the users session when experiment is versioned" do @experiment.increment_version @alternative_name = ab_test(@experiment_name, *@alternatives) ab_user.should eql(@experiment.key => @alternative_name) finished(@experiment_name) ab_user.should eql({}) end it "should do nothing where the experiment was not started by this user" do ab_user = nil lambda { finished('some_experiment_not_started_by_the_user') }.should_not raise_exception end it 'should not be doing other tests when it has completed one that has :reset => false' do ab_user[@experiment.key] = @alternative_name ab_user[@experiment.finished_key] = true doing_other_tests?(@experiment.key).should be false end end context "finished with config" do it "passes reset option" do Split.configuration.experiments = { :my_experiment => { :alternatives => ["one", "two"], :resettable => false, } } alternative = ab_test(:my_experiment) experiment = Split::Experiment.find :my_experiment finished :my_experiment ab_user.should eql(experiment.key => alternative, experiment.finished_key => true) end end context "finished with metric name" do before { Split.configuration.experiments = {} } before { Split::Alternative.stub(:new).and_call_original } def should_finish_experiment(experiment_name, should_finish=true) alts = Split.configuration.experiments[experiment_name][:alternatives] experiment = Split::Experiment.find_or_create(experiment_name, *alts) alt_name = ab_user[experiment.key] = alts.first alt = mock('alternative') alt.stub(:name).and_return(alt_name) Split::Alternative.stub(:new).with(alt_name, experiment_name.to_s).and_return(alt) if should_finish alt.should_receive(:increment_completion) else alt.should_not_receive(:increment_completion) end end it "completes the test" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "other_opt" ], :metric => :my_metric } should_finish_experiment :my_experiment finished :my_metric end it "completes all relevant tests" do Split.configuration.experiments = { :exp_1 => { :alternatives => [ "1-1", "1-2" ], :metric => :my_metric }, :exp_2 => { :alternatives => [ "2-1", "2-2" ], :metric => :another_metric }, :exp_3 => { :alternatives => [ "3-1", "3-2" ], :metric => :my_metric }, } should_finish_experiment :exp_1 should_finish_experiment :exp_2, false should_finish_experiment :exp_3 finished :my_metric end it "passes reset option" do Split.configuration.experiments = { :my_exp => { :alternatives => ["one", "two"], :metric => :my_metric, :resettable => false, } } alternative_name = ab_test(:my_exp) exp = Split::Experiment.find :my_exp finished :my_metric ab_user[exp.key].should == alternative_name ab_user[exp.finished_key].should == true end it "passes through options" do Split.configuration.experiments = { :my_exp => { :alternatives => ["one", "two"], :metric => :my_metric, } } alternative_name = ab_test(:my_exp) exp = Split::Experiment.find :my_exp finished :my_metric, :reset => false ab_user[exp.key].should == alternative_name ab_user[exp.finished_key].should == true end end describe 'conversions' do it 'should return a conversion rate for an alternative' do alternative_name = ab_test('link_color', 'blue', 'red') previous_convertion_rate = Split::Alternative.new(alternative_name, 'link_color').conversion_rate previous_convertion_rate.should eql(0.0) finished('link_color') new_convertion_rate = Split::Alternative.new(alternative_name, 'link_color').conversion_rate new_convertion_rate.should eql(1.0) end end describe 'when user is a robot' do before(:each) do @request = OpenStruct.new(:user_agent => 'Googlebot/2.1 (+http://www.google.com/bot.html)') end describe 'ab_test' do it 'should return the control' do alternative = ab_test('link_color', 'blue', 'red') alternative.should eql experiment.control.name end it "should not increment the participation count" do previous_red_count = Split::Alternative.new('red', 'link_color').participant_count previous_blue_count = Split::Alternative.new('blue', 'link_color').participant_count ab_test('link_color', 'blue', 'red') new_red_count = Split::Alternative.new('red', 'link_color').participant_count new_blue_count = Split::Alternative.new('blue', 'link_color').participant_count (new_red_count + new_blue_count).should eql(previous_red_count + previous_blue_count) end end describe 'finished' do it "should not increment the completed count" do alternative_name = ab_test('link_color', 'blue', 'red') previous_completion_count = Split::Alternative.new(alternative_name, 'link_color').completed_count finished('link_color') new_completion_count = Split::Alternative.new(alternative_name, 'link_color').completed_count new_completion_count.should eql(previous_completion_count) end end end describe 'when providing custom ignore logic' do context "using a proc to configure custom logic" do before(:each) do Split.configure do |c| c.ignore_filter = proc{|request| !!"i_am_going_to_be_disabled" } end end it "ignores the ab_test" do ab_test('link_color', 'blue', 'red') red_count = Split::Alternative.new('red', 'link_color').participant_count blue_count = Split::Alternative.new('blue', 'link_color').participant_count (red_count + blue_count).should be(0) end end end shared_examples_for "a disabled test" do describe 'ab_test' do it 'should return the control' do alternative = ab_test('link_color', 'blue', 'red') alternative.should eql experiment.control.name end it "should not increment the participation count" do previous_red_count = Split::Alternative.new('red', 'link_color').participant_count previous_blue_count = Split::Alternative.new('blue', 'link_color').participant_count ab_test('link_color', 'blue', 'red') new_red_count = Split::Alternative.new('red', 'link_color').participant_count new_blue_count = Split::Alternative.new('blue', 'link_color').participant_count (new_red_count + new_blue_count).should eql(previous_red_count + previous_blue_count) end end describe 'finished' do it "should not increment the completed count" do alternative_name = ab_test('link_color', 'blue', 'red') previous_completion_count = Split::Alternative.new(alternative_name, 'link_color').completed_count finished('link_color') new_completion_count = Split::Alternative.new(alternative_name, 'link_color').completed_count new_completion_count.should eql(previous_completion_count) end end end describe 'when ip address is ignored' do context "individually" do before(:each) do @request = OpenStruct.new(:ip => '81.19.48.130') Split.configure do |c| c.ignore_ip_addresses << '81.19.48.130' end end it_behaves_like "a disabled test" end context "for a range" do before(:each) do @request = OpenStruct.new(:ip => '81.19.48.129') Split.configure do |c| c.ignore_ip_addresses << /81\.19\.48\.[0-9]+/ end end it_behaves_like "a disabled test" end context "using both a range and a specific value" do before(:each) do @request = OpenStruct.new(:ip => '81.19.48.128') Split.configure do |c| c.ignore_ip_addresses << '81.19.48.130' c.ignore_ip_addresses << /81\.19\.48\.[0-9]+/ end end it_behaves_like "a disabled test" end end describe 'versioned experiments' do it "should use version zero if no version is present" do alternative_name = ab_test('link_color', 'blue', 'red') experiment.version.should eql(0) ab_user.should eql({'link_color' => alternative_name}) end it "should save the version of the experiment to the session" do experiment.reset experiment.version.should eql(1) alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color:1' => alternative_name}) end it "should load the experiment even if the version is not 0" do experiment.reset experiment.version.should eql(1) alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color:1' => alternative_name}) return_alternative_name = ab_test('link_color', 'blue', 'red') return_alternative_name.should eql(alternative_name) end it "should reset the session of a user on an older version of the experiment" do alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color' => alternative_name}) alternative = Split::Alternative.new(alternative_name, 'link_color') alternative.participant_count.should eql(1) experiment.reset experiment.version.should eql(1) alternative = Split::Alternative.new(alternative_name, 'link_color') alternative.participant_count.should eql(0) new_alternative_name = ab_test('link_color', 'blue', 'red') ab_user['link_color:1'].should eql(new_alternative_name) new_alternative = Split::Alternative.new(new_alternative_name, 'link_color') new_alternative.participant_count.should eql(1) end it "should cleanup old versions of experiments from the session" do alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color' => alternative_name}) alternative = Split::Alternative.new(alternative_name, 'link_color') alternative.participant_count.should eql(1) experiment.reset experiment.version.should eql(1) alternative = Split::Alternative.new(alternative_name, 'link_color') alternative.participant_count.should eql(0) new_alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color:1' => new_alternative_name}) end it "should only count completion of users on the current version" do alternative_name = ab_test('link_color', 'blue', 'red') ab_user.should eql({'link_color' => alternative_name}) alternative = Split::Alternative.new(alternative_name, 'link_color') experiment.reset experiment.version.should eql(1) finished('link_color') alternative = Split::Alternative.new(alternative_name, 'link_color') alternative.completed_count.should eql(0) end end context 'when redis is not available' do before(:each) do Split.stub(:redis).and_raise(Errno::ECONNREFUSED.new) end context 'and db_failover config option is turned off' do before(:each) do Split.configure do |config| config.db_failover = false end end describe 'ab_test' do it 'should raise an exception' do lambda { ab_test('link_color', 'blue', 'red') }.should raise_error(Errno::ECONNREFUSED) end end describe 'finished' do it 'should raise an exception' do lambda { finished('link_color') }.should raise_error(Errno::ECONNREFUSED) end end describe "disable split testing" do before(:each) do Split.configure do |config| config.enabled = false end end after(:each) do Split.configure do |config| config.enabled = true end end it "should not attempt to connect to redis" do lambda { ab_test('link_color', 'blue', 'red') }.should_not raise_error(Errno::ECONNREFUSED) end it "should return control variable" do ab_test('link_color', 'blue', 'red').should eq('blue') lambda { finished('link_color') }.should_not raise_error(Errno::ECONNREFUSED) end end end context 'and db_failover config option is turned on' do before(:each) do Split.configure do |config| config.db_failover = true end end describe 'ab_test' do it 'should not raise an exception' do lambda { ab_test('link_color', 'blue', 'red') }.should_not raise_error(Errno::ECONNREFUSED) end it 'should call db_failover_on_db_error proc with error as parameter' do Split.configure do |config| config.db_failover_on_db_error = proc do |error| error.should be_a(Errno::ECONNREFUSED) end end Split.configuration.db_failover_on_db_error.should_receive(:call) ab_test('link_color', 'blue', 'red') end it 'should always use first alternative' do ab_test('link_color', 'blue', 'red').should eq('blue') ab_test('link_color', {'blue' => 0.01}, 'red' => 0.2).should eq('blue') ab_test('link_color', {'blue' => 0.8}, {'red' => 20}).should eq('blue') ab_test('link_color', 'blue', 'red') do |alternative| "shared/#{alternative}" end.should eq('shared/blue') end context 'and db_failover_allow_parameter_override config option is turned on' do before(:each) do Split.configure do |config| config.db_failover_allow_parameter_override = true end end context 'and given an override parameter' do it 'should use given override instead of the first alternative' do @params = {'link_color' => 'red'} ab_test('link_color', 'blue', 'red').should eq('red') ab_test('link_color', 'blue', 'red', 'green').should eq('red') ab_test('link_color', {'blue' => 0.01}, 'red' => 0.2).should eq('red') ab_test('link_color', {'blue' => 0.8}, {'red' => 20}).should eq('red') ab_test('link_color', 'blue', 'red') do |alternative| "shared/#{alternative}" end.should eq('shared/red') end end end context 'and preloaded config given' do before do Split.configuration.experiments[:link_color] = { :alternatives => [ "blue", "red" ], } end it "uses first alternative" do ab_test(:link_color).should eq("blue") end end end describe 'finished' do it 'should not raise an exception' do lambda { finished('link_color') }.should_not raise_error(Errno::ECONNREFUSED) end it 'should call db_failover_on_db_error proc with error as parameter' do Split.configure do |config| config.db_failover_on_db_error = proc do |error| error.should be_a(Errno::ECONNREFUSED) end end Split.configuration.db_failover_on_db_error.should_receive(:call) finished('link_color') end end end end context "with preloaded config" do before { Split.configuration.experiments = {}} it "pulls options from config file" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "other_opt" ], :goals => ["goal1", "goal2"] } ab_test :my_experiment Split::Experiment.new(:my_experiment).alternatives.map(&:name).should == [ "control_opt", "other_opt" ] Split::Experiment.new(:my_experiment).goals.should == [ "goal1", "goal2" ] end it "can be called multiple times" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "other_opt" ], :goals => ["goal1", "goal2"] } 5.times { ab_test :my_experiment } experiment = Split::Experiment.new(:my_experiment) experiment.alternatives.map(&:name).should == [ "control_opt", "other_opt" ] experiment.goals.should == [ "goal1", "goal2" ] experiment.participant_count.should == 1 end it "accepts multiple goals" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "other_opt" ], :goals => [ "goal1", "goal2", "goal3" ] } ab_test :my_experiment experiment = Split::Experiment.new(:my_experiment) experiment.goals.should == [ "goal1", "goal2", "goal3" ] end it "allow specifying goals to be optional" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "other_opt" ] } experiment = Split::Experiment.new(:my_experiment) experiment.goals.should == [] end it "accepts multiple alternatives" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ "control_opt", "second_opt", "third_opt" ], } ab_test :my_experiment experiment = Split::Experiment.new(:my_experiment) experiment.alternatives.map(&:name).should == [ "control_opt", "second_opt", "third_opt" ] end it "accepts probability on alternatives" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ { :name => "control_opt", :percent => 67 }, { :name => "second_opt", :percent => 10 }, { :name => "third_opt", :percent => 23 }, ], } ab_test :my_experiment experiment = Split::Experiment.new(:my_experiment) experiment.alternatives.collect{|a| [a.name, a.weight]}.should == [['control_opt', 0.67], ['second_opt', 0.1], ['third_opt', 0.23]] end it "accepts probability on some alternatives" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ { :name => "control_opt", :percent => 34 }, "second_opt", { :name => "third_opt", :percent => 23 }, "fourth_opt", ], } ab_test :my_experiment experiment = Split::Experiment.new(:my_experiment) names_and_weights = experiment.alternatives.collect{|a| [a.name, a.weight]} names_and_weights.should == [['control_opt', 0.34], ['second_opt', 0.215], ['third_opt', 0.23], ['fourth_opt', 0.215]] names_and_weights.inject(0){|sum, nw| sum + nw[1]}.should == 1.0 end it "allows name param without probability" do Split.configuration.experiments[:my_experiment] = { :alternatives => [ { :name => "control_opt" }, "second_opt", { :name => "third_opt", :percent => 64 }, ], } ab_test :my_experiment experiment = Split::Experiment.new(:my_experiment) names_and_weights = experiment.alternatives.collect{|a| [a.name, a.weight]} names_and_weights.should == [['control_opt', 0.18], ['second_opt', 0.18], ['third_opt', 0.64]] names_and_weights.inject(0){|sum, nw| sum + nw[1]}.should == 1.0 end it "fails gracefully if config is missing experiment" do Split.configuration.experiments = { :other_experiment => { :foo => "Bar" } } lambda { ab_test :my_experiment }.should raise_error(/not found/i) end it "fails gracefully if config is missing" do Split.configuration.experiments = nil lambda { ab_test :my_experiment }.should raise_error(/not found/i) end it "fails gracefully if config is missing alternatives" do Split.configuration.experiments[:my_experiment] = { :foo => "Bar" } lambda { ab_test :my_experiment }.should raise_error(/alternatives/i) end end it 'should handle multiple experiments correctly' do experiment2 = Split::Experiment.find_or_create('link_color2', 'blue', 'red') alternative_name = ab_test('link_color', 'blue', 'red') alternative_name2 = ab_test('link_color2', 'blue', 'red') finished('link_color2') experiment2.alternatives.each do |alt| alt.unfinished_count.should eq(0) end end context "with goals" do before do @experiment = {'link_color' => ["purchase", "refund"]} @alternatives = ['blue', 'red'] @experiment_name, @goals = normalize_experiment(@experiment) @goal1 = @goals[0] @goal2 = @goals[1] end it "should normalize experiment" do @experiment_name.should eql("link_color") @goals.should eql(["purchase", "refund"]) end describe "ab_test" do it "should allow experiment goals interface as a single hash" do ab_test(@experiment, *@alternatives) experiment = Split::Experiment.find('link_color') experiment.goals.should eql(['purchase', "refund"]) end end describe "finished" do before do @alternative_name = ab_test(@experiment, *@alternatives) end it "should increment the counter for the specified-goal completed alternative" do @previous_completion_count_for_goal1 = Split::Alternative.new(@alternative_name, @experiment_name).completed_count(@goal1) @previous_completion_count_for_goal2 = Split::Alternative.new(@alternative_name, @experiment_name).completed_count(@goal2) finished({"link_color" => "purchase"}) new_completion_count_for_goal1 = Split::Alternative.new(@alternative_name, @experiment_name).completed_count(@goal1) new_completion_count_for_goal1.should eql(@previous_completion_count_for_goal1 + 1) new_completion_count_for_goal2 = Split::Alternative.new(@alternative_name, @experiment_name).completed_count(@goal2) new_completion_count_for_goal2.should eql(@previous_completion_count_for_goal2) end end end end