Class: Split::Experiment

Inherits:
Object
  • Object
show all
Defined in:
lib/split/experiment.rb

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(name, options = {}) ⇒ Experiment

Returns a new instance of Experiment.



7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
# File 'lib/split/experiment.rb', line 7

def initialize(name, options = {})      
  options = {
      :resettable => true,
    }.merge(options)
   
  @name = name.to_s 
  @alternatives = options[:alternatives]  if !options[:alternatives].nil?
  
  if !options[:algorithm].nil?    
    @algorithm = options[:algorithm].is_a?(String) ? options[:algorithm].constantize : options[:algorithm]
  end
  
  if !options[:resettable].nil?
    @resettable = options[:resettable].is_a?(String) ? options[:resettable] == 'true' : options[:resettable]
  end
  
  if !options[:alternative_names].nil? 
    @alternatives = options[:alternative_names].map do |alternative|
                      Split::Alternative.new(alternative, name)
                    end
  end
  
  
end

Instance Attribute Details

#algorithmObject



32
33
34
# File 'lib/split/experiment.rb', line 32

def algorithm
  @algorithm ||= Split.configuration.algorithm
end

#nameObject

Returns the value of attribute name.



3
4
5
# File 'lib/split/experiment.rb', line 3

def name
  @name
end

#resettableObject

Returns the value of attribute resettable.



5
6
7
# File 'lib/split/experiment.rb', line 5

def resettable
  @resettable
end

Class Method Details

.allObject



198
199
200
# File 'lib/split/experiment.rb', line 198

def self.all
  Array(all_experiment_names_from_redis + all_experiment_names_from_configuration).map {|e| find(e)}
end

.all_experiment_names_from_configurationObject



206
207
208
# File 'lib/split/experiment.rb', line 206

def self.all_experiment_names_from_configuration
  Split.configuration.experiments ? Split.configuration.experiments.keys : []
end

.all_experiment_names_from_redisObject



202
203
204
# File 'lib/split/experiment.rb', line 202

def self.all_experiment_names_from_redis
  Split.redis.smembers(:experiments)
end

.experiment_config_key(name) ⇒ Object



194
195
196
# File 'lib/split/experiment.rb', line 194

def self.experiment_config_key(name)
  "experiment_configurations/#{name}"
end

.find(name) ⇒ Object



211
212
213
214
215
216
217
218
219
220
# File 'lib/split/experiment.rb', line 211

def self.find(name)
  if Split.configuration.experiment_for(name)
    obj = load_from_configuration(name)
  elsif Split.redis.exists(name)
    obj = load_from_redis(name)
  else
    obj = nil
  end
  obj
end

.find_or_create(key, *alternatives) ⇒ Object



222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
# File 'lib/split/experiment.rb', line 222

def self.find_or_create(key, *alternatives)
  name = key.to_s.split(':')[0]

  if alternatives.length == 1
    if alternatives[0].is_a? Hash
      alternatives = alternatives[0].map{|k,v| {k => v} }
    end
  end

  alts = initialize_alternatives(alternatives, name)

  if Split.redis.exists(name)
    existing_alternatives = load_alternatives_for(name)
    if existing_alternatives == alts.map(&:name)
      experiment = self.new(name, :alternative_names => alternatives)
    else
      exp = self.new(name, :alternative_names => existing_alternatives)
      exp.reset
      exp.alternatives.each(&:delete)
      experiment = self.new(name, :alternative_names =>alternatives)
      experiment.save
    end
  else
    experiment = self.new(name, :alternative_names => alternatives)
    experiment.save
  end
  return experiment

end

.initialize_alternatives(alternatives, name) ⇒ Object



252
253
254
255
256
257
258
259
260
261
# File 'lib/split/experiment.rb', line 252

def self.initialize_alternatives(alternatives, name)

  unless alternatives.all? { |a| Split::Alternative.valid?(a) }
    raise ArgumentError, 'Alternatives must be strings'
  end

  alternatives.map do |alternative|
    Split::Alternative.new(alternative, name)
  end
end

.load_alternatives_for(name) ⇒ Object



150
151
152
153
154
155
156
# File 'lib/split/experiment.rb', line 150

def self.load_alternatives_for(name)
  if Split.configuration.experiment_for(name)
    load_alternatives_from_configuration_for(name)
  else
    load_alternatives_from_redis_for(name)
  end
end

.load_alternatives_from_configuration_for(name) ⇒ Object

Raises:

  • (ArgumentError)


158
159
160
161
162
163
164
165
166
# File 'lib/split/experiment.rb', line 158

def self.load_alternatives_from_configuration_for(name)
  alts = Split.configuration.experiment_for(name)[:alternatives]
  raise ArgumentError, "Experiment configuration is missing :alternatives array" if alts.nil?
  if alts.is_a?(Hash)
    alts.keys
  else
    alts.flatten
  end
end

.load_alternatives_from_redis_for(name) ⇒ Object



168
169
170
171
172
173
174
175
176
177
178
# File 'lib/split/experiment.rb', line 168

def self.load_alternatives_from_redis_for(name)
  case Split.redis.type(name)
  when 'set' # convert legacy sets to lists
    alts = Split.redis.smembers(name)
    Split.redis.del(name)
    alts.reverse.each {|a| Split.redis.lpush(name, a) }
    Split.redis.lrange(name, 0, -1)
  else
    Split.redis.lrange(name, 0, -1)
  end
end

.load_from_configuration(name) ⇒ Object



180
181
182
183
184
185
# File 'lib/split/experiment.rb', line 180

def self.load_from_configuration(name)
  exp_config = Split.configuration.experiment_for(name) || {}
  self.new(name, :alternative_names => load_alternatives_for(name), 
                       :resettable => exp_config[:resettable], 
                       :algorithm => exp_config[:algorithm])
end

.load_from_redis(name) ⇒ Object



187
188
189
190
191
192
# File 'lib/split/experiment.rb', line 187

def self.load_from_redis(name)
  exp_config = Split.redis.hgetall(experiment_config_key(name))
  self.new(name, :alternative_names => load_alternatives_for(name),
                  :resettable => exp_config['resettable'], 
                  :algorithm => exp_config['algorithm'])     
end

Instance Method Details

#==(obj) ⇒ Object



36
37
38
# File 'lib/split/experiment.rb', line 36

def ==(obj)
  self.name == obj.name
end

#[](name) ⇒ Object



69
70
71
# File 'lib/split/experiment.rb', line 69

def [](name)
  alternatives.find{|a| a.name == name} 
end

#alternative_namesObject



77
78
79
# File 'lib/split/experiment.rb', line 77

def alternative_names
  @alternatives.map(&:name)
end

#alternativesObject



73
74
75
# File 'lib/split/experiment.rb', line 73

def alternatives
  @alternatives.dup
end

#controlObject



52
53
54
# File 'lib/split/experiment.rb', line 52

def control
  alternatives.first
end

#deleteObject



123
124
125
126
127
128
129
# File 'lib/split/experiment.rb', line 123

def delete
  alternatives.each(&:delete)
  reset_winner
  Split.redis.srem(:experiments, name)
  Split.redis.del(name)
  increment_version
end

#finished_keyObject



109
110
111
# File 'lib/split/experiment.rb', line 109

def finished_key
  "#{key}:finished"
end

#increment_versionObject



97
98
99
# File 'lib/split/experiment.rb', line 97

def increment_version
  @version = Split.redis.incr("#{name}:version")
end

#keyObject



101
102
103
104
105
106
107
# File 'lib/split/experiment.rb', line 101

def key
  if version.to_i > 0
    "#{name}:#{version}"
  else
    name
  end
end

#new_record?Boolean

Returns:

  • (Boolean)


131
132
133
# File 'lib/split/experiment.rb', line 131

def new_record?
  !Split.redis.exists(name)
end

#next_alternativeObject



81
82
83
# File 'lib/split/experiment.rb', line 81

def next_alternative
  winner || random_alternative
end

#participant_countObject



48
49
50
# File 'lib/split/experiment.rb', line 48

def participant_count
  alternatives.inject(0){|sum,a| sum + a.participant_count}
end

#random_alternativeObject



85
86
87
88
89
90
91
# File 'lib/split/experiment.rb', line 85

def random_alternative
  if alternatives.length > 1
    Split.configuration.algorithm.choose_alternative(self)
  else
    alternatives.first
  end
end

#resetObject



117
118
119
120
121
# File 'lib/split/experiment.rb', line 117

def reset
  alternatives.each(&:reset)
  reset_winner
  increment_version
end

#reset_winnerObject



56
57
58
# File 'lib/split/experiment.rb', line 56

def reset_winner
  Split.redis.hdel(:experiment_winner, name)
end

#resettable?Boolean

Returns:

  • (Boolean)


113
114
115
# File 'lib/split/experiment.rb', line 113

def resettable?
  resettable
end

#saveObject



135
136
137
138
139
140
141
142
143
144
145
146
147
148
# File 'lib/split/experiment.rb', line 135

def save
  if new_record?
    Split.redis.sadd(:experiments, name)
    Split.redis.hset(:experiment_start_times, @name, Time.now)
    @alternatives.reverse.each {|a| Split.redis.lpush(name, a.name) }
  else
    Split.redis.del(name)
    @alternatives.reverse.each {|a| Split.redis.lpush(name, a.name) }
  end
  config_key = Split::Experiment.experiment_config_key(name)
  Split.redis.hset(config_key, :resettable, resettable)
  Split.redis.hset(config_key, :algorithm, algorithm.to_s)
  self
end

#start_timeObject



64
65
66
67
# File 'lib/split/experiment.rb', line 64

def start_time
  t = Split.redis.hget(:experiment_start_times, @name)
  Time.parse(t) if t
end

#versionObject



93
94
95
# File 'lib/split/experiment.rb', line 93

def version
  @version ||= (Split.redis.get("#{name.to_s}:version").to_i || 0)
end

#winnerObject



40
41
42
43
44
45
46
# File 'lib/split/experiment.rb', line 40

def winner
  if w = Split.redis.hget(:experiment_winner, name)
    Split::Alternative.new(w, name)
  else
    nil
  end
end

#winner=(winner_name) ⇒ Object



60
61
62
# File 'lib/split/experiment.rb', line 60

def winner=(winner_name)
  Split.redis.hset(:experiment_winner, name, winner_name.to_s)
end