module Split class Experiment attr_accessor :name attr_accessor :alternative_names attr_accessor :winner attr_accessor :version def initialize(name, *alternative_names) @name = name.to_s @alternative_names = alternative_names.map do |alternative| Split::Alternative.new(alternative, name) end.map(&:name) @version = (Split.redis.get("#{name.to_s}:version").to_i || 0) end def winner if w = Split.redis.hget(:experiment_winner, name) return Split::Alternative.find(w, name) else nil end end def control alternatives.first end def reset_winner Split.redis.hdel(:experiment_winner, name) end def winner=(winner_name) Split.redis.hset(:experiment_winner, name, winner_name.to_s) end def alternatives @alternative_names.map {|a| Split::Alternative.find_or_create(a, name)} end def next_alternative winner || random_alternative end def random_alternative weights = alternatives.map(&:weight) total = weights.inject(0.0) {|t,w| t+w} point = rand * total alternatives.zip(weights).each do |n,w| return n if w >= point point -= w end end def version @version ||= 0 end def increment_version @version += 1 Split.redis.set("#{name}:version", @version) end def key if version.to_i > 0 "#{name}:#{version}" else name end end def reset alternatives.each(&:reset) reset_winner increment_version end def delete alternatives.each(&:delete) reset_winner Split.redis.srem(:experiments, name) Split.redis.del(name) increment_version end def new_record? !Split.redis.exists(name) end def save if new_record? Split.redis.sadd(:experiments, name) @alternative_names.reverse.each {|a| Split.redis.lpush(name, a) } end end def self.load_alternatives_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 def self.all Array(Split.redis.smembers(:experiments)).map {|e| find(e)} end def self.find(name) if Split.redis.exists(name) self.new(name, *load_alternatives_for(name)) else raise 'Experiment not found' end end 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} } else raise InvalidArgument, 'You must declare at least 2 alternatives' end end alts = initialize_alternatives(alternatives, name) if Split.redis.exists(name) if load_alternatives_for(name) == alts.map(&:name) experiment = self.new(name, *load_alternatives_for(name)) else exp = self.new(name, *load_alternatives_for(name)) exp.reset exp.alternatives.each(&:delete) experiment = self.new(name, *alternatives) experiment.save end else experiment = self.new(name, *alternatives) experiment.save end return experiment end def self.initialize_alternatives(alternatives, name) if alternatives.reject {|a| Split::Alternative.valid? a}.any? raise InvalidArgument, 'Alternatives must be strings' end alternatives.map do |alternative| Split::Alternative.new(alternative, name) end end end end