flipper 0.16.0 → 1.4.0
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/.codeclimate.yml +1 -0
- data/.github/FUNDING.yml +1 -0
- data/.github/dependabot.yml +6 -0
- data/.github/workflows/ci.yml +110 -0
- data/.github/workflows/examples.yml +105 -0
- data/.github/workflows/release.yml +54 -0
- data/.rspec +1 -0
- data/CLAUDE.md +87 -0
- data/Changelog.md +2 -215
- data/Dockerfile +1 -1
- data/Gemfile +28 -20
- data/README.md +72 -62
- data/Rakefile +13 -3
- data/benchmark/enabled_ips.rb +10 -0
- data/benchmark/enabled_multiple_actors_ips.rb +20 -0
- data/benchmark/enabled_profile.rb +20 -0
- data/benchmark/instrumentation_ips.rb +21 -0
- data/benchmark/typecast_ips.rb +27 -0
- data/docker-compose.yml +37 -34
- data/docs/DockerCompose.md +0 -1
- data/docs/README.md +1 -0
- data/docs/images/banner.jpg +0 -0
- data/docs/images/flipper_cloud.png +0 -0
- data/examples/api/basic.ru +18 -0
- data/examples/api/custom_memoized.ru +36 -0
- data/examples/api/memoized.ru +42 -0
- data/examples/basic.rb +1 -12
- data/examples/cloud/app.ru +12 -0
- data/examples/cloud/backoff_policy.rb +13 -0
- data/examples/cloud/basic.rb +22 -0
- data/examples/cloud/cloud_setup.rb +20 -0
- data/examples/cloud/forked.rb +36 -0
- data/examples/cloud/import.rb +17 -0
- data/examples/cloud/poll_interval/README.md +111 -0
- data/examples/cloud/poll_interval/client.rb +108 -0
- data/examples/cloud/poll_interval/server.rb +98 -0
- data/examples/cloud/threaded.rb +33 -0
- data/examples/configuring_default.rb +2 -5
- data/examples/dsl.rb +10 -35
- data/examples/enabled_for_actor.rb +10 -15
- data/examples/expressions.rb +237 -0
- data/examples/group.rb +3 -6
- data/examples/group_dynamic_lookup.rb +5 -19
- data/examples/group_with_members.rb +4 -14
- data/examples/importing.rb +1 -1
- data/examples/individual_actor.rb +2 -5
- data/examples/instrumentation.rb +2 -2
- data/examples/instrumentation_last_accessed_at.rb +38 -0
- data/examples/memoizing.rb +35 -0
- data/examples/mirroring.rb +59 -0
- data/examples/percentage_of_actors.rb +6 -16
- data/examples/percentage_of_actors_enabled_check.rb +7 -10
- data/examples/percentage_of_actors_group.rb +5 -18
- data/examples/percentage_of_time.rb +3 -6
- data/examples/strict.rb +18 -0
- data/exe/flipper +5 -0
- data/flipper-cloud.gemspec +19 -0
- data/flipper.gemspec +10 -7
- data/lib/flipper/actor.rb +10 -3
- data/lib/flipper/adapter.rb +50 -8
- data/lib/flipper/adapter_builder.rb +44 -0
- data/lib/flipper/adapters/actor_limit.rb +54 -0
- data/lib/flipper/adapters/cache_base.rb +161 -0
- data/lib/flipper/adapters/dual_write.rb +63 -0
- data/lib/flipper/adapters/failover.rb +85 -0
- data/lib/flipper/adapters/failsafe.rb +72 -0
- data/lib/flipper/adapters/http/client.rb +64 -7
- data/lib/flipper/adapters/http/error.rb +19 -1
- data/lib/flipper/adapters/http.rb +97 -43
- data/lib/flipper/adapters/instrumented.rb +47 -26
- data/lib/flipper/adapters/memoizable.rb +44 -40
- data/lib/flipper/adapters/memory.rb +75 -111
- data/lib/flipper/adapters/operation_logger.rb +22 -78
- data/lib/flipper/adapters/poll/poller.rb +2 -0
- data/lib/flipper/adapters/poll.rb +52 -0
- data/lib/flipper/adapters/pstore.rb +27 -17
- data/lib/flipper/adapters/read_only.rb +8 -41
- data/lib/flipper/adapters/strict.rb +45 -0
- data/lib/flipper/adapters/sync/feature_synchronizer.rb +14 -1
- data/lib/flipper/adapters/sync/interval_synchronizer.rb +2 -7
- data/lib/flipper/adapters/sync/synchronizer.rb +13 -6
- data/lib/flipper/adapters/sync.rb +23 -29
- data/lib/flipper/adapters/wrapper.rb +54 -0
- data/lib/flipper/cli.rb +314 -0
- data/lib/flipper/cloud/configuration.rb +271 -0
- data/lib/flipper/cloud/dsl.rb +27 -0
- data/lib/flipper/cloud/message_verifier.rb +95 -0
- data/lib/flipper/cloud/middleware.rb +63 -0
- data/lib/flipper/cloud/migrate.rb +71 -0
- data/lib/flipper/cloud/routes.rb +14 -0
- data/lib/flipper/cloud/telemetry/backoff_policy.rb +96 -0
- data/lib/flipper/cloud/telemetry/instrumenter.rb +22 -0
- data/lib/flipper/cloud/telemetry/metric.rb +39 -0
- data/lib/flipper/cloud/telemetry/metric_storage.rb +30 -0
- data/lib/flipper/cloud/telemetry/submitter.rb +100 -0
- data/lib/flipper/cloud/telemetry.rb +191 -0
- data/lib/flipper/cloud.rb +54 -0
- data/lib/flipper/configuration.rb +54 -7
- data/lib/flipper/dsl.rb +58 -47
- data/lib/flipper/engine.rb +102 -0
- data/lib/flipper/errors.rb +3 -21
- data/lib/flipper/export.rb +24 -0
- data/lib/flipper/exporter.rb +17 -0
- data/lib/flipper/exporters/json/export.rb +32 -0
- data/lib/flipper/exporters/json/v1.rb +33 -0
- data/lib/flipper/expression/builder.rb +73 -0
- data/lib/flipper/expression/constant.rb +25 -0
- data/lib/flipper/expression.rb +71 -0
- data/lib/flipper/expressions/all.rb +9 -0
- data/lib/flipper/expressions/any.rb +9 -0
- data/lib/flipper/expressions/boolean.rb +9 -0
- data/lib/flipper/expressions/comparable.rb +13 -0
- data/lib/flipper/expressions/equal.rb +9 -0
- data/lib/flipper/expressions/feature_enabled.rb +34 -0
- data/lib/flipper/expressions/greater_than.rb +9 -0
- data/lib/flipper/expressions/greater_than_or_equal_to.rb +9 -0
- data/lib/flipper/expressions/less_than.rb +9 -0
- data/lib/flipper/expressions/less_than_or_equal_to.rb +9 -0
- data/lib/flipper/expressions/not_equal.rb +9 -0
- data/lib/flipper/expressions/now.rb +9 -0
- data/lib/flipper/expressions/number.rb +9 -0
- data/lib/flipper/expressions/percentage.rb +9 -0
- data/lib/flipper/expressions/percentage_of_actors.rb +12 -0
- data/lib/flipper/expressions/property.rb +9 -0
- data/lib/flipper/expressions/random.rb +9 -0
- data/lib/flipper/expressions/string.rb +9 -0
- data/lib/flipper/expressions/time.rb +16 -0
- data/lib/flipper/feature.rb +95 -28
- data/lib/flipper/feature_check_context.rb +10 -6
- data/lib/flipper/gate.rb +13 -11
- data/lib/flipper/gate_values.rb +5 -18
- data/lib/flipper/gates/actor.rb +10 -17
- data/lib/flipper/gates/boolean.rb +1 -1
- data/lib/flipper/gates/expression.rb +75 -0
- data/lib/flipper/gates/group.rb +5 -7
- data/lib/flipper/gates/percentage_of_actors.rb +10 -13
- data/lib/flipper/gates/percentage_of_time.rb +1 -2
- data/lib/flipper/identifier.rb +17 -0
- data/lib/flipper/instrumentation/log_subscriber.rb +35 -8
- data/lib/flipper/instrumentation/statsd.rb +4 -2
- data/lib/flipper/instrumentation/statsd_subscriber.rb +2 -4
- data/lib/flipper/instrumentation/subscriber.rb +8 -5
- data/lib/flipper/instrumenters/memory.rb +6 -2
- data/lib/flipper/metadata.rb +8 -1
- data/lib/flipper/middleware/memoizer.rb +46 -27
- data/lib/flipper/middleware/setup_env.rb +13 -3
- data/lib/flipper/model/active_record.rb +23 -0
- data/lib/flipper/poller.rb +157 -0
- data/lib/flipper/serializers/gzip.rb +22 -0
- data/lib/flipper/serializers/json.rb +17 -0
- data/lib/flipper/spec/shared_adapter_specs.rb +122 -56
- data/lib/flipper/test/shared_adapter_test.rb +120 -52
- data/lib/flipper/test_help.rb +43 -0
- data/lib/flipper/typecast.rb +59 -18
- data/lib/flipper/types/actor.rb +19 -13
- data/lib/flipper/types/group.rb +12 -5
- data/lib/flipper/types/percentage.rb +1 -1
- data/lib/flipper/version.rb +11 -1
- data/lib/flipper.rb +71 -12
- data/lib/generators/flipper/setup_generator.rb +68 -0
- data/lib/generators/flipper/templates/initializer.rb +45 -0
- data/lib/generators/flipper/templates/update/migrations/01_create_flipper_tables.rb.erb +22 -0
- data/lib/generators/flipper/templates/update/migrations/02_change_flipper_gates_value_to_text.rb.erb +18 -0
- data/lib/generators/flipper/update_generator.rb +35 -0
- data/package-lock.json +41 -0
- data/package.json +10 -0
- data/spec/fixtures/environment.rb +1 -0
- data/spec/fixtures/flipper_pstore_1679087600.json +46 -0
- data/spec/flipper/actor_spec.rb +10 -2
- data/spec/flipper/adapter_builder_spec.rb +72 -0
- data/spec/flipper/adapter_spec.rb +52 -6
- data/spec/flipper/adapters/actor_limit_spec.rb +75 -0
- data/spec/flipper/adapters/dual_write_spec.rb +82 -0
- data/spec/flipper/adapters/failover_spec.rb +141 -0
- data/spec/flipper/adapters/failsafe_spec.rb +58 -0
- data/spec/flipper/adapters/http/client_spec.rb +61 -0
- data/spec/flipper/adapters/http_spec.rb +402 -65
- data/spec/flipper/adapters/instrumented_spec.rb +31 -13
- data/spec/flipper/adapters/memoizable_spec.rb +51 -33
- data/spec/flipper/adapters/memory_spec.rb +33 -5
- data/spec/flipper/adapters/operation_logger_spec.rb +38 -12
- data/spec/flipper/adapters/poll_spec.rb +41 -0
- data/spec/flipper/adapters/pstore_spec.rb +0 -2
- data/spec/flipper/adapters/read_only_spec.rb +32 -18
- data/spec/flipper/adapters/strict_spec.rb +64 -0
- data/spec/flipper/adapters/sync/feature_synchronizer_spec.rb +39 -1
- data/spec/flipper/adapters/sync/interval_synchronizer_spec.rb +4 -5
- data/spec/flipper/adapters/sync/synchronizer_spec.rb +87 -1
- data/spec/flipper/adapters/sync_spec.rb +17 -6
- data/spec/flipper/cli_spec.rb +217 -0
- data/spec/flipper/cloud/configuration_spec.rb +257 -0
- data/spec/flipper/cloud/dsl_spec.rb +90 -0
- data/spec/flipper/cloud/message_verifier_spec.rb +104 -0
- data/spec/flipper/cloud/middleware_spec.rb +307 -0
- data/spec/flipper/cloud/migrate_spec.rb +160 -0
- data/spec/flipper/cloud/telemetry/backoff_policy_spec.rb +107 -0
- data/spec/flipper/cloud/telemetry/metric_spec.rb +87 -0
- data/spec/flipper/cloud/telemetry/metric_storage_spec.rb +58 -0
- data/spec/flipper/cloud/telemetry/submitter_spec.rb +145 -0
- data/spec/flipper/cloud/telemetry_spec.rb +208 -0
- data/spec/flipper/cloud_spec.rb +186 -0
- data/spec/flipper/configuration_spec.rb +37 -3
- data/spec/flipper/dsl_spec.rb +67 -80
- data/spec/flipper/engine_spec.rb +374 -0
- data/spec/flipper/export_spec.rb +13 -0
- data/spec/flipper/exporter_spec.rb +16 -0
- data/spec/flipper/exporters/json/export_spec.rb +60 -0
- data/spec/flipper/exporters/json/v1_spec.rb +33 -0
- data/spec/flipper/expression/builder_spec.rb +248 -0
- data/spec/flipper/expression_spec.rb +188 -0
- data/spec/flipper/expressions/all_spec.rb +15 -0
- data/spec/flipper/expressions/any_spec.rb +15 -0
- data/spec/flipper/expressions/boolean_spec.rb +15 -0
- data/spec/flipper/expressions/equal_spec.rb +24 -0
- data/spec/flipper/expressions/greater_than_or_equal_to_spec.rb +28 -0
- data/spec/flipper/expressions/greater_than_spec.rb +28 -0
- data/spec/flipper/expressions/less_than_or_equal_to_spec.rb +28 -0
- data/spec/flipper/expressions/less_than_spec.rb +32 -0
- data/spec/flipper/expressions/not_equal_spec.rb +15 -0
- data/spec/flipper/expressions/now_spec.rb +11 -0
- data/spec/flipper/expressions/number_spec.rb +21 -0
- data/spec/flipper/expressions/percentage_of_actors_spec.rb +20 -0
- data/spec/flipper/expressions/percentage_spec.rb +15 -0
- data/spec/flipper/expressions/property_spec.rb +13 -0
- data/spec/flipper/expressions/random_spec.rb +9 -0
- data/spec/flipper/expressions/string_spec.rb +11 -0
- data/spec/flipper/expressions/time_spec.rb +29 -0
- data/spec/flipper/feature_check_context_spec.rb +18 -20
- data/spec/flipper/feature_spec.rb +461 -48
- data/spec/flipper/gate_spec.rb +0 -2
- data/spec/flipper/gate_values_spec.rb +2 -34
- data/spec/flipper/gates/actor_spec.rb +0 -2
- data/spec/flipper/gates/boolean_spec.rb +1 -3
- data/spec/flipper/gates/expression_spec.rb +190 -0
- data/spec/flipper/gates/group_spec.rb +2 -5
- data/spec/flipper/gates/percentage_of_actors_spec.rb +61 -7
- data/spec/flipper/gates/percentage_of_time_spec.rb +2 -4
- data/spec/flipper/identifier_spec.rb +12 -0
- data/spec/flipper/instrumentation/log_subscriber_spec.rb +24 -7
- data/spec/flipper/instrumentation/statsd_subscriber_spec.rb +26 -3
- data/spec/flipper/instrumenters/memory_spec.rb +18 -1
- data/spec/flipper/instrumenters/noop_spec.rb +14 -8
- data/spec/flipper/middleware/memoizer_spec.rb +199 -62
- data/spec/flipper/middleware/setup_env_spec.rb +23 -5
- data/spec/flipper/model/active_record_spec.rb +72 -0
- data/spec/flipper/poller_spec.rb +390 -0
- data/spec/flipper/registry_spec.rb +0 -1
- data/spec/flipper/serializers/gzip_spec.rb +13 -0
- data/spec/flipper/serializers/json_spec.rb +13 -0
- data/spec/flipper/typecast_spec.rb +121 -7
- data/spec/flipper/types/actor_spec.rb +63 -47
- data/spec/flipper/types/boolean_spec.rb +0 -1
- data/spec/flipper/types/group_spec.rb +24 -3
- data/spec/flipper/types/percentage_of_actors_spec.rb +0 -1
- data/spec/flipper/types/percentage_of_time_spec.rb +0 -1
- data/spec/flipper/types/percentage_spec.rb +0 -1
- data/spec/{integration_spec.rb → flipper_integration_spec.rb} +301 -59
- data/spec/flipper_spec.rb +123 -29
- data/spec/{helper.rb → spec_helper.rb} +23 -21
- data/spec/support/actor_names.yml +1 -0
- data/spec/support/descriptions.yml +1 -0
- data/spec/support/fail_on_output.rb +8 -0
- data/spec/support/fake_backoff_policy.rb +15 -0
- data/spec/support/skippable.rb +18 -0
- data/spec/support/spec_helpers.rb +53 -6
- data/test/adapters/actor_limit_test.rb +20 -0
- data/test/test_helper.rb +2 -1
- data/test_rails/generators/flipper/setup_generator_test.rb +69 -0
- data/test_rails/generators/flipper/update_generator_test.rb +96 -0
- data/test_rails/helper.rb +31 -0
- data/test_rails/system/test_help_test.rb +52 -0
- metadata +200 -82
- data/.rubocop.yml +0 -54
- data/.rubocop_todo.yml +0 -199
- data/docs/Adapters.md +0 -124
- data/docs/Caveats.md +0 -4
- data/docs/Gates.md +0 -167
- data/docs/Instrumentation.md +0 -27
- data/docs/Optimization.md +0 -114
- data/docs/api/README.md +0 -849
- data/docs/http/README.md +0 -35
- data/docs/read-only/README.md +0 -21
- data/examples/example_setup.rb +0 -8
- data/test/helper.rb +0 -11
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
if ENV["FLIPPER_CLOUD_TOKEN"].nil? || ENV["FLIPPER_CLOUD_TOKEN"].empty?
|
|
2
|
+
warn "FLIPPER_CLOUD_TOKEN missing so skipping cloud example."
|
|
3
|
+
exit
|
|
4
|
+
end
|
|
5
|
+
|
|
6
|
+
matrix_key = if ENV["CI"]
|
|
7
|
+
suffix_rails = ENV["RAILS_VERSION"].split(".").take(2).join
|
|
8
|
+
suffix_ruby = RUBY_VERSION.split(".").take(2).join
|
|
9
|
+
"FLIPPER_CLOUD_TOKEN_#{suffix_ruby}_#{suffix_rails}"
|
|
10
|
+
else
|
|
11
|
+
"FLIPPER_CLOUD_TOKEN"
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
if matrix_token = ENV[matrix_key]
|
|
15
|
+
puts "Using #{matrix_key} for FLIPPER_CLOUD_TOKEN"
|
|
16
|
+
ENV["FLIPPER_CLOUD_TOKEN"] = matrix_token
|
|
17
|
+
else
|
|
18
|
+
warn "Missing #{matrix_key}. Go create an environment in flipper cloud and set #{matrix_key} to the adapter token for that environment in github actions secrets."
|
|
19
|
+
exit 1
|
|
20
|
+
end
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
# Usage (from the repo root):
|
|
2
|
+
# env FLIPPER_CLOUD_TOKEN=<token> bundle exec ruby examples/cloud/threaded.rb
|
|
3
|
+
|
|
4
|
+
require_relative "./cloud_setup"
|
|
5
|
+
require 'bundler/setup'
|
|
6
|
+
require 'flipper/cloud'
|
|
7
|
+
|
|
8
|
+
puts Process.pid
|
|
9
|
+
|
|
10
|
+
# Make a call in the parent process so we can detect forking.
|
|
11
|
+
Flipper.enabled?(:stats)
|
|
12
|
+
|
|
13
|
+
pids = 2.times.map do |n|
|
|
14
|
+
fork {
|
|
15
|
+
# Check every second to see if the feature is enabled
|
|
16
|
+
threads = []
|
|
17
|
+
2.times do
|
|
18
|
+
threads << Thread.new do
|
|
19
|
+
loop do
|
|
20
|
+
sleep rand
|
|
21
|
+
|
|
22
|
+
if Flipper[:stats].enabled?
|
|
23
|
+
puts "#{Process.pid} #{Time.now.to_i} Enabled!"
|
|
24
|
+
else
|
|
25
|
+
puts "#{Process.pid} #{Time.now.to_i} Disabled!"
|
|
26
|
+
end
|
|
27
|
+
end
|
|
28
|
+
end
|
|
29
|
+
end
|
|
30
|
+
threads.map(&:join)
|
|
31
|
+
}
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
pids.each do |pid|
|
|
35
|
+
Process.waitpid pid, 0
|
|
36
|
+
end
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
# Usage (from the repo root):
|
|
2
|
+
# env FLIPPER_CLOUD_TOKEN=<token> bundle exec ruby examples/cloud/import.rb
|
|
3
|
+
|
|
4
|
+
require_relative "./cloud_setup"
|
|
5
|
+
require 'bundler/setup'
|
|
6
|
+
require 'flipper'
|
|
7
|
+
require 'flipper/cloud'
|
|
8
|
+
|
|
9
|
+
Flipper.enable(:test)
|
|
10
|
+
Flipper.enable(:search)
|
|
11
|
+
Flipper.enable_actor(:stats, Flipper::Actor.new("jnunemaker"))
|
|
12
|
+
Flipper.enable_percentage_of_time(:logging, 5)
|
|
13
|
+
|
|
14
|
+
cloud = Flipper::Cloud.new
|
|
15
|
+
|
|
16
|
+
# makes cloud identical to memory flipper
|
|
17
|
+
cloud.import(Flipper)
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
# Poll Interval Dynamic Adjustment Demo
|
|
2
|
+
|
|
3
|
+
This demo shows how the Flipper poller dynamically adjusts its polling interval based on the `poll-interval` header from the server, and how it responds to the `poll-shutdown` header.
|
|
4
|
+
|
|
5
|
+
## Files
|
|
6
|
+
|
|
7
|
+
- `server.rb` - Test server that responds with configurable headers
|
|
8
|
+
- `client.rb` - Client that polls the server and logs interval changes
|
|
9
|
+
- `README.md` - This file
|
|
10
|
+
|
|
11
|
+
## How to Run
|
|
12
|
+
|
|
13
|
+
### Terminal 1: Start the Server
|
|
14
|
+
|
|
15
|
+
```bash
|
|
16
|
+
bundle exec ruby examples/cloud/poll_interval/server.rb
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
The server will start on http://localhost:3000 and show a prompt where you can control what headers to send.
|
|
20
|
+
|
|
21
|
+
### Terminal 2: Start the Client
|
|
22
|
+
|
|
23
|
+
```bash
|
|
24
|
+
bundle exec ruby examples/cloud/poll_interval/client.rb
|
|
25
|
+
```
|
|
26
|
+
|
|
27
|
+
The client will start polling the server every 10 seconds (the minimum) and log all activity.
|
|
28
|
+
|
|
29
|
+
## Testing Scenarios
|
|
30
|
+
|
|
31
|
+
### 1. Change Poll Interval
|
|
32
|
+
|
|
33
|
+
In the **server terminal**, type a number to set the poll interval:
|
|
34
|
+
|
|
35
|
+
```
|
|
36
|
+
> 20
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
In the **client terminal**, you'll see:
|
|
40
|
+
|
|
41
|
+
```
|
|
42
|
+
[HH:MM:SS] WARN: ⚠️ INTERVAL CHANGED: 10.0s → 20.0s
|
|
43
|
+
```
|
|
44
|
+
|
|
45
|
+
The client will now poll every 20 seconds instead of 10.
|
|
46
|
+
|
|
47
|
+
### 2. Try an Invalid Interval (Below Minimum)
|
|
48
|
+
|
|
49
|
+
In the **server terminal**:
|
|
50
|
+
|
|
51
|
+
```
|
|
52
|
+
> 5
|
|
53
|
+
```
|
|
54
|
+
|
|
55
|
+
In the **client terminal**, you'll see a warning:
|
|
56
|
+
|
|
57
|
+
```
|
|
58
|
+
Flipper::Cloud poll interval must be greater than or equal to 10 but was 5.0. Setting interval to 10.
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
The interval will remain at 10 seconds (the minimum).
|
|
62
|
+
|
|
63
|
+
### 3. Trigger Shutdown
|
|
64
|
+
|
|
65
|
+
In the **server terminal**:
|
|
66
|
+
|
|
67
|
+
```
|
|
68
|
+
> shutdown
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
In the **client terminal**, you'll see:
|
|
72
|
+
|
|
73
|
+
```
|
|
74
|
+
[HH:MM:SS] WARN: Shutdown requested by server via poll-shutdown header
|
|
75
|
+
[HH:MM:SS] WARN: Poller stopped
|
|
76
|
+
[HH:MM:SS] WARN: Poller thread is no longer running
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
The poller will stop gracefully.
|
|
80
|
+
|
|
81
|
+
### 4. Reset Headers
|
|
82
|
+
|
|
83
|
+
In the **server terminal**:
|
|
84
|
+
|
|
85
|
+
```
|
|
86
|
+
> reset
|
|
87
|
+
```
|
|
88
|
+
|
|
89
|
+
The server will stop sending special headers. The client will continue with its current interval.
|
|
90
|
+
|
|
91
|
+
## What You'll Learn
|
|
92
|
+
|
|
93
|
+
- How `poll-interval` header dynamically adjusts polling frequency
|
|
94
|
+
- How `poll-shutdown` header gracefully stops the poller
|
|
95
|
+
- How minimum interval enforcement works (10 seconds minimum)
|
|
96
|
+
- How the poller continues working even if the server returns errors
|
|
97
|
+
- Real-time logging of poller events via instrumentation
|
|
98
|
+
|
|
99
|
+
## Implementation Details
|
|
100
|
+
|
|
101
|
+
The poller checks response headers in the `ensure` block of the `sync` method, which means:
|
|
102
|
+
|
|
103
|
+
- Interval adjustments happen even if the sync fails with an error
|
|
104
|
+
- Shutdown signals are never missed, even during failures
|
|
105
|
+
- The poller is resilient to network issues
|
|
106
|
+
|
|
107
|
+
The `interval=` setter handles all validation:
|
|
108
|
+
|
|
109
|
+
- Type conversion via `Flipper::Typecast.to_float`
|
|
110
|
+
- Minimum enforcement (10 seconds)
|
|
111
|
+
- Warning messages for invalid values
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
# Example showing poll interval being dynamically adjusted via poll-interval header
|
|
2
|
+
#
|
|
3
|
+
# Usage:
|
|
4
|
+
# 1. Terminal 1: bundle exec ruby examples/cloud/poll_interval/server.rb
|
|
5
|
+
# 2. Terminal 2: bundle exec ruby examples/cloud/poll_interval/client.rb
|
|
6
|
+
|
|
7
|
+
require 'bundler/setup'
|
|
8
|
+
require 'flipper'
|
|
9
|
+
require 'flipper/adapters/http'
|
|
10
|
+
require 'flipper/poller'
|
|
11
|
+
require 'logger'
|
|
12
|
+
|
|
13
|
+
# Setup logging to show what's happening
|
|
14
|
+
logger = Logger.new(STDOUT)
|
|
15
|
+
logger.level = Logger::INFO
|
|
16
|
+
logger.formatter = proc do |severity, datetime, progname, msg|
|
|
17
|
+
"[#{datetime.strftime('%H:%M:%S')}] #{severity}: #{msg}\n"
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
# Create HTTP adapter pointing to localhost:3000
|
|
21
|
+
http_adapter = Flipper::Adapters::Http.new(url: 'http://localhost:3000/flipper')
|
|
22
|
+
|
|
23
|
+
# Create instrumenter to log poller events
|
|
24
|
+
instrumenter = Module.new do
|
|
25
|
+
def self.instrument(name, payload = {})
|
|
26
|
+
case payload[:operation]
|
|
27
|
+
when :poll
|
|
28
|
+
logger.info "Polling remote adapter..."
|
|
29
|
+
when :shutdown_requested
|
|
30
|
+
logger.warn "Shutdown requested by server via poll-shutdown header"
|
|
31
|
+
when :stop
|
|
32
|
+
logger.warn "Poller stopped"
|
|
33
|
+
when :thread_start
|
|
34
|
+
logger.info "Poller thread started"
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
result = yield if block_given?
|
|
38
|
+
|
|
39
|
+
if payload[:operation] == :poll && result
|
|
40
|
+
logger.info "Poll completed successfully"
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
result
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
def self.logger=(l)
|
|
47
|
+
@logger = l
|
|
48
|
+
end
|
|
49
|
+
|
|
50
|
+
def self.logger
|
|
51
|
+
@logger
|
|
52
|
+
end
|
|
53
|
+
end
|
|
54
|
+
instrumenter.logger = logger
|
|
55
|
+
|
|
56
|
+
# Create poller with custom instrumenter and short initial interval
|
|
57
|
+
poller = Flipper::Poller.new(
|
|
58
|
+
remote_adapter: http_adapter,
|
|
59
|
+
interval: 5, # Start with 5 second interval (will be enforced to 10 minimum)
|
|
60
|
+
instrumenter: instrumenter,
|
|
61
|
+
start_automatically: false,
|
|
62
|
+
shutdown_automatically: false
|
|
63
|
+
)
|
|
64
|
+
|
|
65
|
+
logger.info "Starting poller with interval: #{poller.interval} seconds"
|
|
66
|
+
logger.info "Minimum allowed interval: #{Flipper::Poller::MINIMUM_POLL_INTERVAL} seconds"
|
|
67
|
+
logger.info ""
|
|
68
|
+
logger.info "Server can control polling via response headers:"
|
|
69
|
+
logger.info " - poll-interval: <seconds> (adjust poll frequency)"
|
|
70
|
+
logger.info " - poll-shutdown: true (stop polling)"
|
|
71
|
+
logger.info ""
|
|
72
|
+
|
|
73
|
+
# Track interval changes
|
|
74
|
+
last_interval = poller.interval
|
|
75
|
+
|
|
76
|
+
# Start the poller
|
|
77
|
+
poller.start
|
|
78
|
+
|
|
79
|
+
# Monitor for interval changes and log them
|
|
80
|
+
logger.info "Monitoring poller... (Ctrl+C to exit)"
|
|
81
|
+
logger.info ""
|
|
82
|
+
|
|
83
|
+
begin
|
|
84
|
+
loop do
|
|
85
|
+
sleep 2
|
|
86
|
+
|
|
87
|
+
current_interval = poller.interval
|
|
88
|
+
|
|
89
|
+
# Highlight when it changes
|
|
90
|
+
if current_interval != last_interval
|
|
91
|
+
logger.warn "⚠️ INTERVAL CHANGED: #{last_interval}s → #{current_interval}s"
|
|
92
|
+
last_interval = current_interval
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
# Check if poller thread is still alive
|
|
96
|
+
unless poller.thread&.alive?
|
|
97
|
+
logger.warn "Poller thread is no longer running"
|
|
98
|
+
break
|
|
99
|
+
end
|
|
100
|
+
end
|
|
101
|
+
rescue Interrupt
|
|
102
|
+
logger.info ""
|
|
103
|
+
logger.info "Interrupted by user"
|
|
104
|
+
ensure
|
|
105
|
+
logger.info "Stopping poller..."
|
|
106
|
+
poller.stop
|
|
107
|
+
logger.info "Final interval: #{poller.interval} seconds"
|
|
108
|
+
end
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
# Simple test server for demonstrating poll interval changes
|
|
2
|
+
#
|
|
3
|
+
# Usage:
|
|
4
|
+
# 1. Terminal 1: bundle exec ruby examples/cloud/poll_interval/server.rb
|
|
5
|
+
# 2. Terminal 2: bundle exec ruby examples/cloud/poll_interval/client.rb
|
|
6
|
+
#
|
|
7
|
+
# Commands in server terminal:
|
|
8
|
+
# - Type a number (e.g., "15") to set poll-interval header to that value
|
|
9
|
+
# - Type "shutdown" to send poll-shutdown: true header
|
|
10
|
+
# - Type "reset" to stop sending special headers
|
|
11
|
+
# - Ctrl+C to exit
|
|
12
|
+
|
|
13
|
+
require 'bundler/setup'
|
|
14
|
+
require 'webrick'
|
|
15
|
+
require 'json'
|
|
16
|
+
|
|
17
|
+
# State for what headers to send
|
|
18
|
+
$poll_interval = nil
|
|
19
|
+
$poll_shutdown = false
|
|
20
|
+
|
|
21
|
+
# Thread to handle user input for changing headers
|
|
22
|
+
input_thread = Thread.new do
|
|
23
|
+
puts ""
|
|
24
|
+
puts "=" * 60
|
|
25
|
+
puts "Server Controls:"
|
|
26
|
+
puts " Type a number (e.g., '15') to set poll-interval"
|
|
27
|
+
puts " Type 'shutdown' to trigger poll shutdown"
|
|
28
|
+
puts " Type 'reset' to clear all special headers"
|
|
29
|
+
puts "=" * 60
|
|
30
|
+
puts ""
|
|
31
|
+
|
|
32
|
+
loop do
|
|
33
|
+
print "> "
|
|
34
|
+
input = gets&.chomp
|
|
35
|
+
break if input.nil?
|
|
36
|
+
|
|
37
|
+
case input
|
|
38
|
+
when /^\d+$/
|
|
39
|
+
$poll_interval = input.to_i
|
|
40
|
+
puts "✓ Will send poll-interval: #{$poll_interval}"
|
|
41
|
+
when "shutdown"
|
|
42
|
+
$poll_shutdown = true
|
|
43
|
+
puts "✓ Will send poll-shutdown: true"
|
|
44
|
+
when "reset"
|
|
45
|
+
$poll_interval = nil
|
|
46
|
+
$poll_shutdown = false
|
|
47
|
+
puts "✓ Cleared all special headers"
|
|
48
|
+
else
|
|
49
|
+
puts "Unknown command. Use a number, 'shutdown', or 'reset'"
|
|
50
|
+
end
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
# Setup WEBrick server
|
|
55
|
+
server = WEBrick::HTTPServer.new(
|
|
56
|
+
Port: 3000,
|
|
57
|
+
Logger: WEBrick::Log.new($stdout, WEBrick::Log::INFO),
|
|
58
|
+
AccessLog: [[
|
|
59
|
+
$stdout,
|
|
60
|
+
WEBrick::AccessLog::COMMON_LOG_FORMAT
|
|
61
|
+
]]
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
# Handle GET /flipper/features
|
|
65
|
+
server.mount_proc '/flipper/features' do |req, res|
|
|
66
|
+
# Build response
|
|
67
|
+
response_body = {
|
|
68
|
+
features: []
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
res.status = 200
|
|
72
|
+
res['Content-Type'] = 'application/json'
|
|
73
|
+
res.body = JSON.generate(response_body)
|
|
74
|
+
|
|
75
|
+
# Add special headers if configured
|
|
76
|
+
if $poll_interval
|
|
77
|
+
res['poll-interval'] = $poll_interval.to_s
|
|
78
|
+
puts "→ Sent poll-interval: #{$poll_interval}"
|
|
79
|
+
end
|
|
80
|
+
|
|
81
|
+
if $poll_shutdown
|
|
82
|
+
res['poll-shutdown'] = 'true'
|
|
83
|
+
puts "→ Sent poll-shutdown: true"
|
|
84
|
+
end
|
|
85
|
+
end
|
|
86
|
+
|
|
87
|
+
# Trap interrupt and shutdown gracefully
|
|
88
|
+
trap('INT') do
|
|
89
|
+
puts "\nShutting down server..."
|
|
90
|
+
server.shutdown
|
|
91
|
+
input_thread.kill
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
puts "Server starting on http://localhost:3000"
|
|
95
|
+
puts "Endpoint: GET http://localhost:3000/flipper/features"
|
|
96
|
+
puts ""
|
|
97
|
+
|
|
98
|
+
server.start
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
# Usage (from the repo root):
|
|
2
|
+
# env FLIPPER_CLOUD_TOKEN=<token> bundle exec ruby examples/cloud/threaded.rb
|
|
3
|
+
|
|
4
|
+
require_relative "./cloud_setup"
|
|
5
|
+
require 'bundler/setup'
|
|
6
|
+
require 'flipper/cloud'
|
|
7
|
+
|
|
8
|
+
puts Process.pid
|
|
9
|
+
|
|
10
|
+
Flipper.configure do |config|
|
|
11
|
+
config.default {
|
|
12
|
+
Flipper::Cloud.new(
|
|
13
|
+
local_adapter: config.adapter,
|
|
14
|
+
debug_output: STDOUT,
|
|
15
|
+
)
|
|
16
|
+
}
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
# You might want to do this at some point to see different results:
|
|
20
|
+
# Flipper.enable(:search)
|
|
21
|
+
# Flipper.disable(:stats)
|
|
22
|
+
|
|
23
|
+
# Check every second to see if the feature is enabled
|
|
24
|
+
5.times.map { |i|
|
|
25
|
+
Thread.new {
|
|
26
|
+
loop do
|
|
27
|
+
sleep rand
|
|
28
|
+
|
|
29
|
+
Flipper.enabled?(:stats)
|
|
30
|
+
Flipper.enabled?(:search)
|
|
31
|
+
end
|
|
32
|
+
}
|
|
33
|
+
}.each(&:join)
|
|
@@ -1,12 +1,9 @@
|
|
|
1
|
-
require
|
|
2
|
-
|
|
1
|
+
require 'bundler/setup'
|
|
3
2
|
require 'flipper'
|
|
4
3
|
|
|
5
4
|
# sets up default adapter so Flipper works like Flipper::DSL
|
|
6
5
|
Flipper.configure do |config|
|
|
7
|
-
config.
|
|
8
|
-
Flipper.new Flipper::Adapters::Memory.new
|
|
9
|
-
end
|
|
6
|
+
config.adapter { Flipper::Adapters::Memory.new }
|
|
10
7
|
end
|
|
11
8
|
|
|
12
9
|
puts Flipper.enabled?(:search) # => false
|
data/examples/dsl.rb
CHANGED
|
@@ -1,20 +1,9 @@
|
|
|
1
|
-
require
|
|
2
|
-
|
|
1
|
+
require 'bundler/setup'
|
|
3
2
|
require 'flipper'
|
|
4
3
|
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
# create a thing with an identifier
|
|
9
|
-
class Person
|
|
10
|
-
attr_reader :id
|
|
11
|
-
|
|
12
|
-
def initialize(id)
|
|
13
|
-
@id = id
|
|
14
|
-
end
|
|
15
|
-
|
|
16
|
-
# Must respond to flipper_id
|
|
17
|
-
alias_method :flipper_id, :id
|
|
4
|
+
# create an actor with an identifier
|
|
5
|
+
class Person < Struct.new(:id)
|
|
6
|
+
include Flipper::Identifier
|
|
18
7
|
end
|
|
19
8
|
|
|
20
9
|
person = Person.new(1)
|
|
@@ -22,14 +11,14 @@ person = Person.new(1)
|
|
|
22
11
|
puts "Stats are disabled by default\n\n"
|
|
23
12
|
|
|
24
13
|
# is a feature enabled
|
|
25
|
-
puts "flipper.enabled? :stats: #{
|
|
14
|
+
puts "flipper.enabled? :stats: #{Flipper.enabled? :stats}"
|
|
26
15
|
|
|
27
16
|
# is a feature on or off for a particular person
|
|
28
|
-
puts "
|
|
17
|
+
puts "Flipper.enabled? :stats, person: #{Flipper.enabled? :stats, person}"
|
|
29
18
|
|
|
30
19
|
# get at a feature
|
|
31
|
-
puts "\nYou can also get an individual feature like this:\nstats =
|
|
32
|
-
stats =
|
|
20
|
+
puts "\nYou can also get an individual feature like this:\nstats = Flipper[:stats]\n\n"
|
|
21
|
+
stats = Flipper[:stats]
|
|
33
22
|
|
|
34
23
|
# is that feature enabled
|
|
35
24
|
puts "stats.enabled?: #{stats.enabled?}"
|
|
@@ -39,7 +28,7 @@ puts "stats.enabled? person: #{stats.enabled? person}"
|
|
|
39
28
|
|
|
40
29
|
# enable a feature by name
|
|
41
30
|
puts "\nEnabling stats\n\n"
|
|
42
|
-
|
|
31
|
+
Flipper.enable :stats
|
|
43
32
|
|
|
44
33
|
# or, you can use the feature to enable
|
|
45
34
|
stats.enable
|
|
@@ -49,7 +38,7 @@ puts "stats.enabled? person: #{stats.enabled? person}"
|
|
|
49
38
|
|
|
50
39
|
# oh, no, let's turn this baby off
|
|
51
40
|
puts "\nDisabling stats\n\n"
|
|
52
|
-
|
|
41
|
+
Flipper.disable :stats
|
|
53
42
|
|
|
54
43
|
# or we can disable using feature obviously
|
|
55
44
|
stats.disable
|
|
@@ -58,20 +47,6 @@ puts "stats.enabled?: #{stats.enabled?}"
|
|
|
58
47
|
puts "stats.enabled? person: #{stats.enabled? person}"
|
|
59
48
|
puts
|
|
60
49
|
|
|
61
|
-
# get an instance of the percentage of time type set to 5
|
|
62
|
-
puts flipper.time(5).inspect
|
|
63
|
-
|
|
64
|
-
# get an instance of the percentage of actors type set to 15
|
|
65
|
-
puts flipper.actors(15).inspect
|
|
66
|
-
|
|
67
|
-
# get an instance of an actor using an object that responds to flipper_id
|
|
68
|
-
responds_to_flipper_id = Struct.new(:flipper_id).new(10)
|
|
69
|
-
puts flipper.actor(responds_to_flipper_id).inspect
|
|
70
|
-
|
|
71
|
-
# get an instance of an actor using an object
|
|
72
|
-
thing = Struct.new(:flipper_id).new(22)
|
|
73
|
-
puts flipper.actor(thing).inspect
|
|
74
|
-
|
|
75
50
|
# register a top level group
|
|
76
51
|
admins = Flipper.register(:admins) { |actor|
|
|
77
52
|
actor.respond_to?(:admin?) && actor.admin?
|
|
@@ -1,5 +1,4 @@
|
|
|
1
|
-
require
|
|
2
|
-
|
|
1
|
+
require 'bundler/setup'
|
|
3
2
|
require 'flipper'
|
|
4
3
|
|
|
5
4
|
# Some class that represents what will be trying to do something
|
|
@@ -22,21 +21,17 @@ end
|
|
|
22
21
|
user1 = User.new(1, true)
|
|
23
22
|
user2 = User.new(2, false)
|
|
24
23
|
|
|
25
|
-
# pick an adapter
|
|
26
|
-
adapter = Flipper::Adapters::Memory.new
|
|
27
|
-
|
|
28
|
-
# get a handy dsl instance
|
|
29
|
-
flipper = Flipper.new(adapter)
|
|
30
|
-
|
|
31
24
|
Flipper.register :admins do |actor|
|
|
32
25
|
actor.admin?
|
|
33
26
|
end
|
|
34
27
|
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
28
|
+
Flipper.enable :search
|
|
29
|
+
Flipper.enable_actor :stats, user1
|
|
30
|
+
Flipper.enable_percentage_of_actors :pro_stats, 50
|
|
31
|
+
Flipper.enable_group :tweets, :admins
|
|
32
|
+
Flipper.enable_actor :posts, user2
|
|
40
33
|
|
|
41
|
-
pp
|
|
42
|
-
pp
|
|
34
|
+
pp Flipper.features.select { |feature| feature.enabled?(user1) }.map(&:name).sort
|
|
35
|
+
pp Flipper.features.select { |feature| feature.enabled?(user2) }.map(&:name).sort
|
|
36
|
+
pp Flipper.features.select { |feature| feature.enabled?(user1, user2) }.map(&:name).sort
|
|
37
|
+
pp Flipper.features.select { |feature| feature.enabled?([user2, user1]) }.map(&:name).sort
|