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.
data/spec/spec_helper.rb CHANGED
@@ -1,11 +1,21 @@
1
- $LOAD_PATH.unshift(File.dirname(__FILE__))
2
- $LOAD_PATH.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
3
- require 'rollout'
4
- require 'rspec'
5
- require 'bourne'
6
- require 'redis'
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.mock_with :mocha
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.1.0
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: 2014-12-10 00:00:00.000000000 Z
11
+ date: 2019-11-07 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
- name: rspec
14
+ name: redis
15
15
  requirement: !ruby/object:Gem::Requirement
16
16
  requirements:
17
17
  - - "~>"
18
18
  - !ruby/object:Gem::Version
19
- version: 2.10.0
20
- type: :development
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: 2.10.0
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: 1.0.0
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: 1.0.0
54
+ version: '0'
41
55
  - !ruby/object:Gem::Dependency
42
- name: jeweler
56
+ name: rspec
43
57
  requirement: !ruby/object:Gem::Requirement
44
58
  requirements:
45
59
  - - "~>"
46
60
  - !ruby/object:Gem::Version
47
- version: 1.6.4
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: 1.6.4
68
+ version: '3.0'
55
69
  - !ruby/object:Gem::Dependency
56
- name: bourne
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: '1.0'
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: '1.0'
82
+ version: '0.4'
69
83
  - !ruby/object:Gem::Dependency
70
- name: mocha
84
+ name: rubocop
71
85
  requirement: !ruby/object:Gem::Requirement
72
86
  requirements:
73
- - - '='
87
+ - - "~>"
74
88
  - !ruby/object:Gem::Version
75
- version: 0.9.8
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.9.8
96
+ version: '0.71'
83
97
  - !ruby/object:Gem::Dependency
84
- name: redis
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
- - ".document"
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.rdoc
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/jamesgolick/rollout
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: '0'
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
- rubyforge_project: rollout
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
@@ -1,5 +0,0 @@
1
- README.rdoc
2
- lib/**/*.rb
3
- bin/*
4
- features/**/*.feature
5
- LICENSE
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.
@@ -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
@@ -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