rollout 2.1.0 → 2.4.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/.circleci/config.yml +80 -0
- data/.gitignore +2 -0
- data/.rubocop.yml +11 -0
- data/.travis.yml +13 -3
- data/Gemfile +3 -1
- data/Gemfile.lock +52 -27
- data/README.md +193 -0
- data/Rakefile +7 -0
- data/lib/rollout/version.rb +3 -1
- data/lib/rollout.rb +166 -72
- data/rollout.gemspec +26 -22
- data/spec/rollout_spec.rb +465 -95
- data/spec/spec_helper.rb +17 -7
- metadata +50 -39
- data/.document +0 -5
- data/README.rdoc +0 -130
- data/lib/rollout/legacy.rb +0 -134
- data/misc/check_rollout.rb +0 -16
- data/spec/legacy_spec.rb +0 -221
- /data/{spec/spec.opts → .rspec} +0 -0
data/spec/spec_helper.rb
CHANGED
@@ -1,11 +1,21 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
require '
|
4
|
-
|
5
|
-
|
6
|
-
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'simplecov'
|
4
|
+
|
5
|
+
SimpleCov.start
|
6
|
+
|
7
|
+
require 'bundler/setup'
|
8
|
+
require ENV["USE_REAL_REDIS"] == "true" ? "redis" : "fakeredis"
|
9
|
+
require "rollout"
|
7
10
|
|
8
11
|
RSpec.configure do |config|
|
9
|
-
config.
|
12
|
+
config.example_status_persistence_file_path = '.rspec_status'
|
13
|
+
|
14
|
+
# config.disable_monkey_patching!
|
15
|
+
|
16
|
+
config.expect_with :rspec do |c|
|
17
|
+
c.syntax = :expect
|
18
|
+
end
|
19
|
+
|
10
20
|
config.before { Redis.new.flushdb }
|
11
21
|
end
|
metadata
CHANGED
@@ -1,99 +1,113 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: rollout
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.4.5
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- James Golick
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2019-11-07 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
|
-
name:
|
14
|
+
name: redis
|
15
15
|
requirement: !ruby/object:Gem::Requirement
|
16
16
|
requirements:
|
17
17
|
- - "~>"
|
18
18
|
- !ruby/object:Gem::Version
|
19
|
-
version:
|
20
|
-
type: :
|
19
|
+
version: '4.0'
|
20
|
+
type: :runtime
|
21
21
|
prerelease: false
|
22
22
|
version_requirements: !ruby/object:Gem::Requirement
|
23
23
|
requirements:
|
24
24
|
- - "~>"
|
25
25
|
- !ruby/object:Gem::Version
|
26
|
-
version:
|
26
|
+
version: '4.0'
|
27
27
|
- !ruby/object:Gem::Dependency
|
28
28
|
name: bundler
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - "~>"
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '1.17'
|
34
|
+
type: :development
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - "~>"
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '1.17'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: fakeredis
|
29
43
|
requirement: !ruby/object:Gem::Requirement
|
30
44
|
requirements:
|
31
45
|
- - ">="
|
32
46
|
- !ruby/object:Gem::Version
|
33
|
-
version:
|
47
|
+
version: '0'
|
34
48
|
type: :development
|
35
49
|
prerelease: false
|
36
50
|
version_requirements: !ruby/object:Gem::Requirement
|
37
51
|
requirements:
|
38
52
|
- - ">="
|
39
53
|
- !ruby/object:Gem::Version
|
40
|
-
version:
|
54
|
+
version: '0'
|
41
55
|
- !ruby/object:Gem::Dependency
|
42
|
-
name:
|
56
|
+
name: rspec
|
43
57
|
requirement: !ruby/object:Gem::Requirement
|
44
58
|
requirements:
|
45
59
|
- - "~>"
|
46
60
|
- !ruby/object:Gem::Version
|
47
|
-
version:
|
61
|
+
version: '3.0'
|
48
62
|
type: :development
|
49
63
|
prerelease: false
|
50
64
|
version_requirements: !ruby/object:Gem::Requirement
|
51
65
|
requirements:
|
52
66
|
- - "~>"
|
53
67
|
- !ruby/object:Gem::Version
|
54
|
-
version:
|
68
|
+
version: '3.0'
|
55
69
|
- !ruby/object:Gem::Dependency
|
56
|
-
name:
|
70
|
+
name: rspec_junit_formatter
|
57
71
|
requirement: !ruby/object:Gem::Requirement
|
58
72
|
requirements:
|
59
|
-
- -
|
73
|
+
- - "~>"
|
60
74
|
- !ruby/object:Gem::Version
|
61
|
-
version: '
|
75
|
+
version: '0.4'
|
62
76
|
type: :development
|
63
77
|
prerelease: false
|
64
78
|
version_requirements: !ruby/object:Gem::Requirement
|
65
79
|
requirements:
|
66
|
-
- -
|
80
|
+
- - "~>"
|
67
81
|
- !ruby/object:Gem::Version
|
68
|
-
version: '
|
82
|
+
version: '0.4'
|
69
83
|
- !ruby/object:Gem::Dependency
|
70
|
-
name:
|
84
|
+
name: rubocop
|
71
85
|
requirement: !ruby/object:Gem::Requirement
|
72
86
|
requirements:
|
73
|
-
- -
|
87
|
+
- - "~>"
|
74
88
|
- !ruby/object:Gem::Version
|
75
|
-
version: 0.
|
89
|
+
version: '0.71'
|
76
90
|
type: :development
|
77
91
|
prerelease: false
|
78
92
|
version_requirements: !ruby/object:Gem::Requirement
|
79
93
|
requirements:
|
80
|
-
- -
|
94
|
+
- - "~>"
|
81
95
|
- !ruby/object:Gem::Version
|
82
|
-
version: 0.
|
96
|
+
version: '0.71'
|
83
97
|
- !ruby/object:Gem::Dependency
|
84
|
-
name:
|
98
|
+
name: simplecov
|
85
99
|
requirement: !ruby/object:Gem::Requirement
|
86
100
|
requirements:
|
87
|
-
- - "
|
101
|
+
- - "~>"
|
88
102
|
- !ruby/object:Gem::Version
|
89
|
-
version: '0'
|
103
|
+
version: '0.16'
|
90
104
|
type: :development
|
91
105
|
prerelease: false
|
92
106
|
version_requirements: !ruby/object:Gem::Requirement
|
93
107
|
requirements:
|
94
|
-
- - "
|
108
|
+
- - "~>"
|
95
109
|
- !ruby/object:Gem::Version
|
96
|
-
version: '0'
|
110
|
+
version: '0.16'
|
97
111
|
description: Feature flippers with redis.
|
98
112
|
email:
|
99
113
|
- jamesgolick@gmail.com
|
@@ -101,24 +115,24 @@ executables: []
|
|
101
115
|
extensions: []
|
102
116
|
extra_rdoc_files: []
|
103
117
|
files:
|
104
|
-
- ".
|
118
|
+
- ".circleci/config.yml"
|
105
119
|
- ".gitignore"
|
120
|
+
- ".rspec"
|
121
|
+
- ".rubocop.yml"
|
106
122
|
- ".travis.yml"
|
107
123
|
- Gemfile
|
108
124
|
- Gemfile.lock
|
109
125
|
- LICENSE
|
110
|
-
- README.
|
126
|
+
- README.md
|
127
|
+
- Rakefile
|
111
128
|
- lib/rollout.rb
|
112
|
-
- lib/rollout/legacy.rb
|
113
129
|
- lib/rollout/version.rb
|
114
|
-
- misc/check_rollout.rb
|
115
130
|
- rollout.gemspec
|
116
|
-
- spec/legacy_spec.rb
|
117
131
|
- spec/rollout_spec.rb
|
118
|
-
- spec/spec.opts
|
119
132
|
- spec/spec_helper.rb
|
120
|
-
homepage: https://github.com/
|
121
|
-
licenses:
|
133
|
+
homepage: https://github.com/FetLife/rollout
|
134
|
+
licenses:
|
135
|
+
- MIT
|
122
136
|
metadata: {}
|
123
137
|
post_install_message:
|
124
138
|
rdoc_options: []
|
@@ -128,20 +142,17 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
128
142
|
requirements:
|
129
143
|
- - ">="
|
130
144
|
- !ruby/object:Gem::Version
|
131
|
-
version: '
|
145
|
+
version: '2.3'
|
132
146
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
133
147
|
requirements:
|
134
148
|
- - ">="
|
135
149
|
- !ruby/object:Gem::Version
|
136
150
|
version: '0'
|
137
151
|
requirements: []
|
138
|
-
|
139
|
-
rubygems_version: 2.2.2
|
152
|
+
rubygems_version: 3.0.3
|
140
153
|
signing_key:
|
141
154
|
specification_version: 4
|
142
155
|
summary: Feature flippers with redis.
|
143
156
|
test_files:
|
144
|
-
- spec/legacy_spec.rb
|
145
157
|
- spec/rollout_spec.rb
|
146
|
-
- spec/spec.opts
|
147
158
|
- spec/spec_helper.rb
|
data/.document
DELETED
data/README.rdoc
DELETED
@@ -1,130 +0,0 @@
|
|
1
|
-
= rollout
|
2
|
-
|
3
|
-
Feature flippers.
|
4
|
-
|
5
|
-
{<img src="https://travis-ci.org/FetLife/rollout.svg?branch=master" alt="Build Status" />}[https://travis-ci.org/FetLife/rollout]
|
6
|
-
|
7
|
-
== MAKE SURE TO READ THIS: 2.0 Changes and Migration Path
|
8
|
-
|
9
|
-
As of rollout-2.x, only one key is used per feature for performance reasons. The data format is <tt>percentage|user_id,user_id,...|group,_group...</tt>. This has the effect of making concurrent feature modifications unsafe, but in practice, I doubt this will actually be a problem.
|
10
|
-
|
11
|
-
This also has the effect of rollout no longer being dependent on redis. Just give it something that responds to <tt>set(key,value)</tt> and <tt>get(key)</tt>.
|
12
|
-
|
13
|
-
If you have been using the 1.x format, you can initialize Rollout with <tt>:migrate => true</tt> and it'll do its best to automatically migrate your old features to the new format. There will be some performance impact, but it should be limited and short-lived since each feature only needs to get migrated once.
|
14
|
-
|
15
|
-
== Rollout::Legacy
|
16
|
-
|
17
|
-
If you'd prefer to continue to use the old layout in redis, <tt>Rollout::Legacy</tt> is a copy and paste of the old code :-).
|
18
|
-
|
19
|
-
== Install it
|
20
|
-
|
21
|
-
gem install rollout
|
22
|
-
|
23
|
-
== How it works
|
24
|
-
|
25
|
-
Initialize a rollout object. I assign it to a global var.
|
26
|
-
|
27
|
-
require 'redis'
|
28
|
-
|
29
|
-
$redis = Redis.new
|
30
|
-
$rollout = Rollout.new($redis)
|
31
|
-
|
32
|
-
Check whether a feature is active for a particular user:
|
33
|
-
|
34
|
-
$rollout.active?(:chat, User.first) # => true/false
|
35
|
-
|
36
|
-
Check whether a feature is active globally:
|
37
|
-
|
38
|
-
$rollout.active?(:chat)
|
39
|
-
|
40
|
-
You can activate features using a number of different mechanisms.
|
41
|
-
|
42
|
-
== Groups
|
43
|
-
|
44
|
-
Rollout ships with one group by default: "all", which does exactly what it sounds like.
|
45
|
-
|
46
|
-
You can activate the all group for the chat feature like this:
|
47
|
-
|
48
|
-
$rollout.activate_group(:chat, :all)
|
49
|
-
|
50
|
-
You might also want to define your own groups. We have one for our caretakers:
|
51
|
-
|
52
|
-
$rollout.define_group(:caretakers) do |user|
|
53
|
-
user.caretaker?
|
54
|
-
end
|
55
|
-
|
56
|
-
You can activate multiple groups per feature.
|
57
|
-
|
58
|
-
Deactivate groups like this:
|
59
|
-
|
60
|
-
$rollout.deactivate_group(:chat, :all)
|
61
|
-
|
62
|
-
== Specific Users
|
63
|
-
|
64
|
-
You might want to let a specific user into a beta test or something. If that user isn't part of an existing group, you can let them in specifically:
|
65
|
-
|
66
|
-
$rollout.activate_user(:chat, @user)
|
67
|
-
|
68
|
-
Deactivate them like this:
|
69
|
-
|
70
|
-
$rollout.deactivate_user(:chat, @user)
|
71
|
-
|
72
|
-
== User Percentages
|
73
|
-
|
74
|
-
If you're rolling out a new feature, you might want to test the waters by slowly enabling it for a percentage of your users.
|
75
|
-
|
76
|
-
$rollout.activate_percentage(:chat, 20)
|
77
|
-
|
78
|
-
The algorithm for determining which users get let in is this:
|
79
|
-
|
80
|
-
CRC32(user.id) % 100 < percentage
|
81
|
-
|
82
|
-
So, for 20%, users 0, 1, 10, 11, 20, 21, etc would be allowed in. Those users would remain in as the percentage increases.
|
83
|
-
|
84
|
-
Deactivate all percentages like this:
|
85
|
-
|
86
|
-
$rollout.deactivate_percentage(:chat)
|
87
|
-
|
88
|
-
_Note that activating a feature for 100% of users will also make it active "globally". That is when calling Rollout#active? without a user object._
|
89
|
-
|
90
|
-
== Feature is broken
|
91
|
-
|
92
|
-
Deactivate everybody at once:
|
93
|
-
|
94
|
-
$rollout.deactivate(:chat)
|
95
|
-
|
96
|
-
For many of our features, we keep track of error rates using redis, and deactivate them automatically when a threshold is reached to prevent service failures from cascading. See http://github.com/jamesgolick/degrade for the failure detection code.
|
97
|
-
|
98
|
-
== Namespacing
|
99
|
-
|
100
|
-
Rollout separates its keys from other keys in the data store using the "feature" keyspace.
|
101
|
-
|
102
|
-
If you're using redis, you can namespace keys further to support multiple environments by using the http://github.com/defunkt/redis-namespace gem.
|
103
|
-
|
104
|
-
$ns = Redis::Namespace.new(Rails.env, :redis => $redis)
|
105
|
-
$rollout = Rollout.new($ns)
|
106
|
-
$rollout.activate_group(:chat, :all)
|
107
|
-
|
108
|
-
This example would use the "development:feature:chat:groups" key.
|
109
|
-
|
110
|
-
== misc/check_rollout.rb
|
111
|
-
|
112
|
-
In our infrastructure, rollout obviously allows us to progressively enable new features but we also use it to automatically disable features and services that break or fail to prevent them from causing cascading failures and wiping out our entire system.
|
113
|
-
|
114
|
-
When a feature reaches "maturity" - in other words, expected to be at 100% rollout all the time - we use check_rollout.rb to setup nagios alerts on the rollouts so that we get paged if one of them gets disabled.
|
115
|
-
|
116
|
-
|
117
|
-
== Implementations in other languages
|
118
|
-
|
119
|
-
* Python: http://github.com/asenchi/proclaim
|
120
|
-
* PHP: https://github.com/opensoft/rollout
|
121
|
-
* Clojure: https://github.com/tcrayford/shoutout
|
122
|
-
|
123
|
-
== Contributors
|
124
|
-
|
125
|
-
* James Golick - Creator - https://github.com/jamesgolick
|
126
|
-
* Eric Rafaloff - Maintainer - https://github.com/EricR
|
127
|
-
|
128
|
-
== Copyright
|
129
|
-
|
130
|
-
Copyright (c) 2010-InfinityAndBeyond BitLove, Inc. See LICENSE for details.
|
data/lib/rollout/legacy.rb
DELETED
@@ -1,134 +0,0 @@
|
|
1
|
-
class Rollout
|
2
|
-
class Legacy
|
3
|
-
def initialize(redis)
|
4
|
-
@redis = redis
|
5
|
-
@groups = {"all" => lambda { |user| true }}
|
6
|
-
end
|
7
|
-
|
8
|
-
def activate_globally(feature)
|
9
|
-
@redis.sadd(global_key, feature)
|
10
|
-
end
|
11
|
-
|
12
|
-
def deactivate_globally(feature)
|
13
|
-
@redis.srem(global_key, feature)
|
14
|
-
end
|
15
|
-
|
16
|
-
def activate_group(feature, group)
|
17
|
-
@redis.sadd(group_key(feature), group)
|
18
|
-
end
|
19
|
-
|
20
|
-
def deactivate_group(feature, group)
|
21
|
-
@redis.srem(group_key(feature), group)
|
22
|
-
end
|
23
|
-
|
24
|
-
def deactivate_all(feature)
|
25
|
-
@redis.del(group_key(feature))
|
26
|
-
@redis.del(user_key(feature))
|
27
|
-
@redis.del(percentage_key(feature))
|
28
|
-
deactivate_globally(feature)
|
29
|
-
end
|
30
|
-
|
31
|
-
def activate_user(feature, user)
|
32
|
-
@redis.sadd(user_key(feature), user.id)
|
33
|
-
end
|
34
|
-
|
35
|
-
def deactivate_user(feature, user)
|
36
|
-
@redis.srem(user_key(feature), user.id)
|
37
|
-
end
|
38
|
-
|
39
|
-
def define_group(group, &block)
|
40
|
-
@groups[group.to_s] = block
|
41
|
-
end
|
42
|
-
|
43
|
-
def active?(feature, user = nil)
|
44
|
-
if user
|
45
|
-
active_globally?(feature) ||
|
46
|
-
user_in_active_group?(feature, user) ||
|
47
|
-
user_active?(feature, user) ||
|
48
|
-
user_within_active_percentage?(feature, user)
|
49
|
-
else
|
50
|
-
active_globally?(feature)
|
51
|
-
end
|
52
|
-
end
|
53
|
-
|
54
|
-
def activate_percentage(feature, percentage)
|
55
|
-
@redis.set(percentage_key(feature), percentage)
|
56
|
-
end
|
57
|
-
|
58
|
-
def deactivate_percentage(feature)
|
59
|
-
@redis.del(percentage_key(feature))
|
60
|
-
end
|
61
|
-
|
62
|
-
def info(feature = nil)
|
63
|
-
if feature
|
64
|
-
{
|
65
|
-
:percentage => (active_percentage(feature) || 0).to_i,
|
66
|
-
:groups => active_groups(feature).map { |g| g.to_sym },
|
67
|
-
:users => active_user_ids(feature),
|
68
|
-
:global => active_global_features
|
69
|
-
}
|
70
|
-
else
|
71
|
-
{
|
72
|
-
:global => active_global_features
|
73
|
-
}
|
74
|
-
end
|
75
|
-
end
|
76
|
-
|
77
|
-
private
|
78
|
-
def key(name)
|
79
|
-
"feature:#{name}"
|
80
|
-
end
|
81
|
-
|
82
|
-
def group_key(name)
|
83
|
-
"#{key(name)}:groups"
|
84
|
-
end
|
85
|
-
|
86
|
-
def user_key(name)
|
87
|
-
"#{key(name)}:users"
|
88
|
-
end
|
89
|
-
|
90
|
-
def percentage_key(name)
|
91
|
-
"#{key(name)}:percentage"
|
92
|
-
end
|
93
|
-
|
94
|
-
def global_key
|
95
|
-
"feature:__global__"
|
96
|
-
end
|
97
|
-
|
98
|
-
def active_groups(feature)
|
99
|
-
@redis.smembers(group_key(feature)) || []
|
100
|
-
end
|
101
|
-
|
102
|
-
def active_user_ids(feature)
|
103
|
-
@redis.smembers(user_key(feature)).map { |id| id.to_i }
|
104
|
-
end
|
105
|
-
|
106
|
-
def active_global_features
|
107
|
-
(@redis.smembers(global_key) || []).map(&:to_sym)
|
108
|
-
end
|
109
|
-
|
110
|
-
def active_percentage(feature)
|
111
|
-
@redis.get(percentage_key(feature))
|
112
|
-
end
|
113
|
-
|
114
|
-
def active_globally?(feature)
|
115
|
-
@redis.sismember(global_key, feature)
|
116
|
-
end
|
117
|
-
|
118
|
-
def user_in_active_group?(feature, user)
|
119
|
-
active_groups(feature).any? do |group|
|
120
|
-
@groups.key?(group) && @groups[group].call(user)
|
121
|
-
end
|
122
|
-
end
|
123
|
-
|
124
|
-
def user_active?(feature, user)
|
125
|
-
@redis.sismember(user_key(feature), user.id)
|
126
|
-
end
|
127
|
-
|
128
|
-
def user_within_active_percentage?(feature, user)
|
129
|
-
percentage = active_percentage(feature)
|
130
|
-
return false if percentage.nil?
|
131
|
-
user.id % 100 < percentage.to_i
|
132
|
-
end
|
133
|
-
end
|
134
|
-
end
|
data/misc/check_rollout.rb
DELETED
@@ -1,16 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
|
3
|
-
require "rubygems"
|
4
|
-
require "yajl"
|
5
|
-
require "open-uri"
|
6
|
-
|
7
|
-
output = Yajl::Parser.parse(open(ARGV[0]))
|
8
|
-
percentage = output["percentage"].to_i
|
9
|
-
|
10
|
-
puts Yajl::Encoder.encode(output)
|
11
|
-
|
12
|
-
if percentage == 100
|
13
|
-
exit(0)
|
14
|
-
else
|
15
|
-
exit(2)
|
16
|
-
end
|