abingo 1.1.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,25 @@
1
+ require "abingo/conversion_rate"
2
+ class Abingo::Alternative < ActiveRecord::Base
3
+ include Abingo::ConversionRate
4
+
5
+ belongs_to :experiment, :class_name => "Abingo::Experiment"
6
+ attr_accessible :content, :weight, :lookup
7
+ serialize :content
8
+
9
+ def self.calculate_lookup(test_name, alternative_name)
10
+ Digest::MD5.hexdigest(Abingo.salt + test_name + alternative_name.to_s)
11
+ end
12
+
13
+ def self.score_conversion(test_name)
14
+ viewed_alternative = Abingo.find_alternative_for_user(test_name,
15
+ Abingo::Experiment.alternatives_for_test(test_name))
16
+ self.update_all("conversions = conversions + 1", :lookup => self.calculate_lookup(test_name, viewed_alternative))
17
+ end
18
+
19
+ def self.score_participation(test_name)
20
+ viewed_alternative = Abingo.find_alternative_for_user(test_name,
21
+ Abingo::Experiment.alternatives_for_test(test_name))
22
+ self.update_all("participants = participants + 1", :lookup => self.calculate_lookup(test_name, viewed_alternative))
23
+ end
24
+
25
+ end
@@ -0,0 +1,29 @@
1
+ class Abingo
2
+ module Controller
3
+ module Dashboard
4
+
5
+ if Rails::VERSION::MAJOR <= 2
6
+ ActionController::Base.view_paths.unshift File.join(File.dirname(__FILE__), "../views")
7
+ else
8
+ ActionController::Base.prepend_view_path File.join(File.dirname(__FILE__), "../views")
9
+ end
10
+
11
+ def index
12
+ @experiments = Abingo::Experiment.all
13
+ render :template => 'dashboard/index'
14
+ end
15
+
16
+ def end_experiment
17
+ @alternative = Abingo::Alternative.find(params[:id])
18
+ @experiment = Abingo::Experiment.find(@alternative.experiment_id)
19
+ if (@experiment.status != "Completed")
20
+ @experiment.end_experiment!(@alternative.content)
21
+ flash[:notice] = "Experiment marked as ended. All users will now see the chosen alternative."
22
+ else
23
+ flash[:notice] = "This experiment is already ended."
24
+ end
25
+ redirect_to :action => "index"
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,9 @@
1
+ module Abingo::ConversionRate
2
+ def conversion_rate
3
+ 1.0 * conversions / participants
4
+ end
5
+
6
+ def pretty_conversion_rate
7
+ sprintf("%4.2f%%", conversion_rate * 100)
8
+ end
9
+ end
@@ -0,0 +1,107 @@
1
+ require "abingo/statistics"
2
+ require "abingo/conversion_rate"
3
+ class Abingo::Experiment < ActiveRecord::Base
4
+ include Abingo::Statistics
5
+ include Abingo::ConversionRate
6
+
7
+ has_many :alternatives, :dependent => :destroy, :class_name => "Abingo::Alternative"
8
+ validates_uniqueness_of :test_name
9
+ attr_accessible :test_name
10
+ before_destroy :cleanup_cache
11
+
12
+ def cache_keys
13
+ ["Abingo::Experiment::exists(#{test_name})".gsub(" ", "_"),
14
+ "Abingo::Experiment::#{test_name}::alternatives".gsub(" ","_"),
15
+ "Abingo::Experiment::short_circuit(#{test_name})".gsub(" ", "_")
16
+ ]
17
+ end
18
+
19
+ def cleanup_cache
20
+ cache_keys.each do |key|
21
+ Abingo.cache.delete key
22
+ end
23
+ true
24
+ end
25
+
26
+ def participants
27
+ alternatives.sum("participants")
28
+ end
29
+
30
+ def conversions
31
+ alternatives.sum("conversions")
32
+ end
33
+
34
+ def best_alternative
35
+ alternatives.max do |a,b|
36
+ a.conversion_rate <=> b.conversion_rate
37
+ end
38
+ end
39
+
40
+ def self.exists?(test_name)
41
+ cache_key = "Abingo::Experiment::exists(#{test_name})".gsub(" ", "_")
42
+ ret = Abingo.cache.fetch(cache_key) do
43
+ count = Abingo::Experiment.count(:conditions => {:test_name => test_name})
44
+ count > 0 ? count : nil
45
+ end
46
+ (!ret.nil?)
47
+ end
48
+
49
+ def self.alternatives_for_test(test_name)
50
+ cache_key = "Abingo::#{test_name}::alternatives".gsub(" ","_")
51
+ Abingo.cache.fetch(cache_key) do
52
+ experiment = Abingo::Experiment.find_by_test_name(test_name)
53
+ alternatives_array = Abingo.cache.fetch(cache_key) do
54
+ tmp_array = experiment.alternatives.map do |alt|
55
+ [alt.content, alt.weight]
56
+ end
57
+ tmp_hash = tmp_array.inject({}) {|hash, couplet| hash[couplet[0]] = couplet[1]; hash}
58
+ Abingo.parse_alternatives(tmp_hash)
59
+ end
60
+ alternatives_array
61
+ end
62
+ end
63
+
64
+ def self.start_experiment!(test_name, alternatives_array, conversion_name = nil)
65
+ conversion_name ||= test_name
66
+ conversion_name.gsub!(" ", "_")
67
+ cloned_alternatives_array = alternatives_array.clone
68
+ ActiveRecord::Base.transaction do
69
+ experiment = Abingo::Experiment.find_or_create_by_test_name(test_name)
70
+ experiment.alternatives.destroy_all #Blows away alternatives for pre-existing experiments.
71
+ while (cloned_alternatives_array.size > 0)
72
+ alt = cloned_alternatives_array[0]
73
+ weight = cloned_alternatives_array.size - (cloned_alternatives_array - [alt]).size
74
+ experiment.alternatives.build(:content => alt, :weight => weight,
75
+ :lookup => Abingo::Alternative.calculate_lookup(test_name, alt))
76
+ cloned_alternatives_array -= [alt]
77
+ end
78
+ experiment.status = "Live"
79
+ if Rails::VERSION::MAJOR == 2
80
+ experiment.save(false) #Calling the validation here causes problems b/c of transaction.
81
+ else
82
+ experiment.save(:validate => false)
83
+ end
84
+ Abingo.cache.write("Abingo::Experiment::exists(#{test_name})".gsub(" ", "_"), 1)
85
+
86
+ #This might have issues in very, very high concurrency environments...
87
+
88
+ tests_listening_to_conversion = Abingo.cache.read("Abingo::tests_listening_to_conversion#{conversion_name}") || []
89
+ tests_listening_to_conversion += [test_name] unless tests_listening_to_conversion.include? test_name
90
+ Abingo.cache.write("Abingo::tests_listening_to_conversion#{conversion_name}", tests_listening_to_conversion)
91
+ experiment
92
+ end
93
+ end
94
+
95
+ def end_experiment!(final_alternative, conversion_name = nil)
96
+ conversion_name ||= test_name
97
+ ActiveRecord::Base.transaction do
98
+ alternatives.each do |alternative|
99
+ alternative.lookup = "Experiment completed. #{alternative.id}"
100
+ alternative.save!
101
+ end
102
+ update_attribute(:status, "Finished")
103
+ Abingo.cache.write("Abingo::Experiment::short_circuit(#{test_name})".gsub(" ", "_"), final_alternative)
104
+ end
105
+ end
106
+
107
+ end
@@ -0,0 +1,13 @@
1
+ class Abingo
2
+ module Rails
3
+ module Controller
4
+ module Dashboard
5
+
6
+ def index
7
+ @experiments = Abingo::Experiment.all
8
+ end
9
+
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,90 @@
1
+ #Designed to be included into Abingo::Experiment, but you can feel free to adapt this
2
+ #to anything you want.
3
+
4
+ module Abingo::Statistics
5
+
6
+ HANDY_Z_SCORE_CHEATSHEET = [[0.10, 1.29], [0.05, 1.65], [0.01, 2.33], [0.001, 3.08]]
7
+
8
+ PERCENTAGES = {0.10 => '90%', 0.05 => '95%', 0.01 => '99%', 0.001 => '99.9%'}
9
+
10
+ DESCRIPTION_IN_WORDS = {0.10 => 'fairly confident', 0.05 => 'confident',
11
+ 0.01 => 'very confident', 0.001 => 'extremely confident'}
12
+ def zscore
13
+ if alternatives.size != 2
14
+ raise "Sorry, can't currently automatically calculate statistics for A/B tests with > 2 alternatives."
15
+ end
16
+
17
+ if (alternatives[0].participants == 0) || (alternatives[1].participants == 0)
18
+ raise "Can't calculate the z score if either of the alternatives lacks participants."
19
+ end
20
+
21
+ cr1 = alternatives[0].conversion_rate
22
+ cr2 = alternatives[1].conversion_rate
23
+
24
+ n1 = alternatives[0].participants
25
+ n2 = alternatives[1].participants
26
+
27
+ numerator = cr1 - cr2
28
+ frac1 = cr1 * (1 - cr1) / n1
29
+ frac2 = cr2 * (1 - cr2) / n2
30
+
31
+ numerator / ((frac1 + frac2) ** 0.5)
32
+ end
33
+
34
+ def p_value
35
+ index = 0
36
+ z = zscore
37
+ z = z.abs
38
+ found_p = nil
39
+ while index < HANDY_Z_SCORE_CHEATSHEET.size do
40
+ if (z > HANDY_Z_SCORE_CHEATSHEET[index][1])
41
+ found_p = HANDY_Z_SCORE_CHEATSHEET[index][0]
42
+ end
43
+ index += 1
44
+ end
45
+ found_p
46
+ end
47
+
48
+ def is_statistically_significant?(p = 0.05)
49
+ p_value <= p
50
+ end
51
+
52
+ def pretty_conversion_rate
53
+ sprintf("%4.2f%%", conversion_rate * 100)
54
+ end
55
+
56
+ def describe_result_in_words
57
+ begin
58
+ z = zscore
59
+ rescue
60
+ return "Could not execute the significance test because one or more of the alternatives has not been seen yet."
61
+ end
62
+ p = p_value
63
+
64
+ words = ""
65
+ if (alternatives[0].participants < 10) || (alternatives[1].participants < 10)
66
+ words += "Take these results with a grain of salt since your samples are so small: "
67
+ end
68
+
69
+ alts = alternatives - [best_alternative]
70
+ worst_alternative = alts.first
71
+
72
+ words += "The best alternative you have is: [#{best_alternative.content}], which had "
73
+ words += "#{best_alternative.conversions} conversions from #{best_alternative.participants} participants "
74
+ words += "(#{best_alternative.pretty_conversion_rate}). The other alternative was [#{worst_alternative.content}], "
75
+ words += "which had #{worst_alternative.conversions} conversions from #{worst_alternative.participants} participants "
76
+ words += "(#{worst_alternative.pretty_conversion_rate}). "
77
+
78
+ if (p.nil?)
79
+ words += "However, this difference is not statistically significant."
80
+ else
81
+ words += "This difference is #{PERCENTAGES[p]} likely to be statistically significant, which means you can be "
82
+ words += "#{DESCRIPTION_IN_WORDS[p]} that it is the result of your alternatives actually mattering, rather than "
83
+ words += "being due to random chance. However, this statistical test can't measure how likely the currently "
84
+ words += "observed magnitude of the difference is to be accurate or not. It only says \"better\", not \"better "
85
+ words += "by so much\"."
86
+ end
87
+ words
88
+ end
89
+
90
+ end
@@ -0,0 +1,3 @@
1
+ class Abingo
2
+ VERSION = "1.1.0"
3
+ end
@@ -0,0 +1,43 @@
1
+ <h3><%= experiment.test_name.titleize %> <%= %Q|<i>(Test completed)</i>| if experiment.status != "Live" %> </h3>
2
+ <% short_circuit = Abingo.cache.read("Abingo::Experiment::short_circuit(#{experiment.test_name})".gsub(" ", "")) %>
3
+ <table class="experiment" style="border: 1px black;">
4
+ <tr class="header_row">
5
+ <th>Name</th>
6
+ <th>Participants</th>
7
+ <th>Conversions</th>
8
+ <th>Notes</th>
9
+ </tr>
10
+ <tr class="experiment_row">
11
+ <td>Experiment Total: </td>
12
+ <td><%= experiment.participants %> </td>
13
+ <td><%= experiment.conversions %> (<%= experiment.pretty_conversion_rate %>)</td>
14
+ <td></td>
15
+ </tr>
16
+ <% experiment.alternatives.each do |alternative| %>
17
+ <tr class="alternative_row">
18
+ <td>
19
+ <%= h alternative.content %>
20
+ </td>
21
+ <td><%= alternative.participants %></td>
22
+ <td><%= alternative.conversions %> (<%= alternative.pretty_conversion_rate %>)</td>
23
+ <td>
24
+ <% unless experiment.status != "Live" %>
25
+ <%= link_to("End experiment, picking this.", url_for(:id => alternative.id,
26
+ :action => "end_experiment"),
27
+ :method => :post,
28
+ :confirm => "Are you sure you want to terminate this experiment? This is not reversible."
29
+ ) %>
30
+ <% else %>
31
+ <% if alternative.content == short_circuit %>
32
+ <b>(All users seeing this.)</b>
33
+ <% end %>
34
+ <% end %>
35
+ </td>
36
+ </tr>
37
+ <% end %>
38
+ <tr>
39
+ <td colspan="4">
40
+ <b>Significance test results: </b><%= experiment.describe_result_in_words %>
41
+ </td>
42
+ </tr>
43
+ </table>
@@ -0,0 +1,20 @@
1
+ <div id="abingo_dashboard">
2
+ <p><h1>Welcome to your A/Bingo dashboard!</h1>
3
+
4
+ See <a href="http://www.bingocardcreator.com/abingo">the official site</a> for documentation.
5
+ I encourage you to customize this page to fit your needs. See /vendor/plugins/abingo/views for
6
+ the view templates. Hack them to pieces -- please!
7
+ </p>
8
+
9
+ <p>
10
+ <% if flash[:notice] %>
11
+ <span style="color: green;"><%= flash[:notice] %> </span>
12
+ <% end %>
13
+ <h2>All Experiments</h2>
14
+
15
+ <% @experiments.each do |experiment| %>
16
+ <%= render :partial => "dashboard/experiment", :locals => {:experiment => experiment} %>
17
+ <br/>
18
+ <% end %>
19
+ </p>
20
+ </div>
@@ -0,0 +1,49 @@
1
+ #This module exists entirely to save finger strain for programmers.
2
+ #It is designed to be included in your ApplicationController.
3
+ #
4
+ #See abingo.rb for descriptions of what these do.
5
+
6
+ module AbingoSugar
7
+
8
+ def ab_test(test_name, alternatives = nil, options = {})
9
+ if (Abingo.options[:enable_specification] && !params[test_name].nil?)
10
+ choice = params[test_name]
11
+ elsif (Abingo.options[:enable_override_in_session] && !session[test_name].nil?)
12
+ choice = session[test_name]
13
+ elsif (alternatives.nil?)
14
+ choice = Abingo.flip(test_name)
15
+ else
16
+ choice = Abingo.test(test_name, alternatives, options)
17
+ end
18
+
19
+ if block_given?
20
+ yield(choice)
21
+ else
22
+ choice
23
+ end
24
+ end
25
+
26
+ def bingo!(test_name, options = {})
27
+ Abingo.bingo!(test_name, options)
28
+ end
29
+
30
+ #Mark the user as a human.
31
+ def abingo_mark_human
32
+ textual_result = "1"
33
+ begin
34
+ a = params[:a].to_i
35
+ b = params[:b].to_i
36
+ c = params[:c].to_i
37
+ if (request.method == :post && (a + b == c))
38
+ Abingo.human!
39
+ else
40
+ textual_result = "0"
41
+ end
42
+ rescue #If a bot doesn't pass a, b, or c, to_i will fail. This scarfs up the exception, to save it from polluting our logs.
43
+ textual_result = "0"
44
+ end
45
+ render :text => textual_result, :layout => false #Not actually used by browser
46
+
47
+ end
48
+
49
+ end
@@ -0,0 +1,45 @@
1
+ #Gives you easy syntax to use ABingo in your views.
2
+
3
+ module AbingoViewHelper
4
+
5
+ def ab_test(test_name, alternatives = nil, options = {}, &block)
6
+
7
+ if (Abingo.options[:enable_specification] && !params[test_name].nil?)
8
+ choice = params[test_name]
9
+ elsif (Abingo.options[:enable_override_in_session] && !session[test_name].nil?)
10
+ choice = session[test_name]
11
+ elsif (alternatives.nil?)
12
+ choice = Abingo.flip(test_name)
13
+ else
14
+ choice = Abingo.test(test_name, alternatives, options)
15
+ end
16
+
17
+ if block
18
+ content_tag = capture(choice, &block)
19
+ unless Rails::VERSION::MAJOR >= 3
20
+ block_called_from_erb?(block) ? concat(content_tag) : content_tag
21
+ else
22
+ content_tag
23
+ end
24
+ else
25
+ choice
26
+ end
27
+ end
28
+
29
+ def bingo!(test_name, options = {})
30
+ Abingo.bingo!(test_name, options)
31
+ end
32
+
33
+ #This causes an AJAX post against the URL. That URL should call Abingo.human!
34
+ #This guarantees that anyone calling Abingo.human! is capable of at least minimal Javascript execution, and thus is (probably) not a robot.
35
+ def include_humanizing_javascript(url = "/abingo_mark_human", style = :prototype)
36
+ script = nil
37
+ if (style == :prototype)
38
+ script = "var a=Math.floor(Math.random()*11); var b=Math.floor(Math.random()*11);var x=new Ajax.Request('#{url}', {parameters:{a: a, b: b, c: a+b}})"
39
+ elsif (style == :jquery)
40
+ script = "var a=Math.floor(Math.random()*11); var b=Math.floor(Math.random()*11);var x=jQuery.post('#{url}', {a: a, b: b, c: a+b})"
41
+ end
42
+ script.nil? ? "" : %Q|<script type="text/javascript">#{script}</script>|
43
+ end
44
+
45
+ end