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.
Files changed (285) hide show
  1. checksums.yaml +5 -5
  2. data/.codeclimate.yml +1 -0
  3. data/.github/FUNDING.yml +1 -0
  4. data/.github/dependabot.yml +6 -0
  5. data/.github/workflows/ci.yml +110 -0
  6. data/.github/workflows/examples.yml +105 -0
  7. data/.github/workflows/release.yml +54 -0
  8. data/.rspec +1 -0
  9. data/CLAUDE.md +87 -0
  10. data/Changelog.md +2 -215
  11. data/Dockerfile +1 -1
  12. data/Gemfile +28 -20
  13. data/README.md +72 -62
  14. data/Rakefile +13 -3
  15. data/benchmark/enabled_ips.rb +10 -0
  16. data/benchmark/enabled_multiple_actors_ips.rb +20 -0
  17. data/benchmark/enabled_profile.rb +20 -0
  18. data/benchmark/instrumentation_ips.rb +21 -0
  19. data/benchmark/typecast_ips.rb +27 -0
  20. data/docker-compose.yml +37 -34
  21. data/docs/DockerCompose.md +0 -1
  22. data/docs/README.md +1 -0
  23. data/docs/images/banner.jpg +0 -0
  24. data/docs/images/flipper_cloud.png +0 -0
  25. data/examples/api/basic.ru +18 -0
  26. data/examples/api/custom_memoized.ru +36 -0
  27. data/examples/api/memoized.ru +42 -0
  28. data/examples/basic.rb +1 -12
  29. data/examples/cloud/app.ru +12 -0
  30. data/examples/cloud/backoff_policy.rb +13 -0
  31. data/examples/cloud/basic.rb +22 -0
  32. data/examples/cloud/cloud_setup.rb +20 -0
  33. data/examples/cloud/forked.rb +36 -0
  34. data/examples/cloud/import.rb +17 -0
  35. data/examples/cloud/poll_interval/README.md +111 -0
  36. data/examples/cloud/poll_interval/client.rb +108 -0
  37. data/examples/cloud/poll_interval/server.rb +98 -0
  38. data/examples/cloud/threaded.rb +33 -0
  39. data/examples/configuring_default.rb +2 -5
  40. data/examples/dsl.rb +10 -35
  41. data/examples/enabled_for_actor.rb +10 -15
  42. data/examples/expressions.rb +237 -0
  43. data/examples/group.rb +3 -6
  44. data/examples/group_dynamic_lookup.rb +5 -19
  45. data/examples/group_with_members.rb +4 -14
  46. data/examples/importing.rb +1 -1
  47. data/examples/individual_actor.rb +2 -5
  48. data/examples/instrumentation.rb +2 -2
  49. data/examples/instrumentation_last_accessed_at.rb +38 -0
  50. data/examples/memoizing.rb +35 -0
  51. data/examples/mirroring.rb +59 -0
  52. data/examples/percentage_of_actors.rb +6 -16
  53. data/examples/percentage_of_actors_enabled_check.rb +7 -10
  54. data/examples/percentage_of_actors_group.rb +5 -18
  55. data/examples/percentage_of_time.rb +3 -6
  56. data/examples/strict.rb +18 -0
  57. data/exe/flipper +5 -0
  58. data/flipper-cloud.gemspec +19 -0
  59. data/flipper.gemspec +10 -7
  60. data/lib/flipper/actor.rb +10 -3
  61. data/lib/flipper/adapter.rb +50 -8
  62. data/lib/flipper/adapter_builder.rb +44 -0
  63. data/lib/flipper/adapters/actor_limit.rb +54 -0
  64. data/lib/flipper/adapters/cache_base.rb +161 -0
  65. data/lib/flipper/adapters/dual_write.rb +63 -0
  66. data/lib/flipper/adapters/failover.rb +85 -0
  67. data/lib/flipper/adapters/failsafe.rb +72 -0
  68. data/lib/flipper/adapters/http/client.rb +64 -7
  69. data/lib/flipper/adapters/http/error.rb +19 -1
  70. data/lib/flipper/adapters/http.rb +97 -43
  71. data/lib/flipper/adapters/instrumented.rb +47 -26
  72. data/lib/flipper/adapters/memoizable.rb +44 -40
  73. data/lib/flipper/adapters/memory.rb +75 -111
  74. data/lib/flipper/adapters/operation_logger.rb +22 -78
  75. data/lib/flipper/adapters/poll/poller.rb +2 -0
  76. data/lib/flipper/adapters/poll.rb +52 -0
  77. data/lib/flipper/adapters/pstore.rb +27 -17
  78. data/lib/flipper/adapters/read_only.rb +8 -41
  79. data/lib/flipper/adapters/strict.rb +45 -0
  80. data/lib/flipper/adapters/sync/feature_synchronizer.rb +14 -1
  81. data/lib/flipper/adapters/sync/interval_synchronizer.rb +2 -7
  82. data/lib/flipper/adapters/sync/synchronizer.rb +13 -6
  83. data/lib/flipper/adapters/sync.rb +23 -29
  84. data/lib/flipper/adapters/wrapper.rb +54 -0
  85. data/lib/flipper/cli.rb +314 -0
  86. data/lib/flipper/cloud/configuration.rb +271 -0
  87. data/lib/flipper/cloud/dsl.rb +27 -0
  88. data/lib/flipper/cloud/message_verifier.rb +95 -0
  89. data/lib/flipper/cloud/middleware.rb +63 -0
  90. data/lib/flipper/cloud/migrate.rb +71 -0
  91. data/lib/flipper/cloud/routes.rb +14 -0
  92. data/lib/flipper/cloud/telemetry/backoff_policy.rb +96 -0
  93. data/lib/flipper/cloud/telemetry/instrumenter.rb +22 -0
  94. data/lib/flipper/cloud/telemetry/metric.rb +39 -0
  95. data/lib/flipper/cloud/telemetry/metric_storage.rb +30 -0
  96. data/lib/flipper/cloud/telemetry/submitter.rb +100 -0
  97. data/lib/flipper/cloud/telemetry.rb +191 -0
  98. data/lib/flipper/cloud.rb +54 -0
  99. data/lib/flipper/configuration.rb +54 -7
  100. data/lib/flipper/dsl.rb +58 -47
  101. data/lib/flipper/engine.rb +102 -0
  102. data/lib/flipper/errors.rb +3 -21
  103. data/lib/flipper/export.rb +24 -0
  104. data/lib/flipper/exporter.rb +17 -0
  105. data/lib/flipper/exporters/json/export.rb +32 -0
  106. data/lib/flipper/exporters/json/v1.rb +33 -0
  107. data/lib/flipper/expression/builder.rb +73 -0
  108. data/lib/flipper/expression/constant.rb +25 -0
  109. data/lib/flipper/expression.rb +71 -0
  110. data/lib/flipper/expressions/all.rb +9 -0
  111. data/lib/flipper/expressions/any.rb +9 -0
  112. data/lib/flipper/expressions/boolean.rb +9 -0
  113. data/lib/flipper/expressions/comparable.rb +13 -0
  114. data/lib/flipper/expressions/equal.rb +9 -0
  115. data/lib/flipper/expressions/feature_enabled.rb +34 -0
  116. data/lib/flipper/expressions/greater_than.rb +9 -0
  117. data/lib/flipper/expressions/greater_than_or_equal_to.rb +9 -0
  118. data/lib/flipper/expressions/less_than.rb +9 -0
  119. data/lib/flipper/expressions/less_than_or_equal_to.rb +9 -0
  120. data/lib/flipper/expressions/not_equal.rb +9 -0
  121. data/lib/flipper/expressions/now.rb +9 -0
  122. data/lib/flipper/expressions/number.rb +9 -0
  123. data/lib/flipper/expressions/percentage.rb +9 -0
  124. data/lib/flipper/expressions/percentage_of_actors.rb +12 -0
  125. data/lib/flipper/expressions/property.rb +9 -0
  126. data/lib/flipper/expressions/random.rb +9 -0
  127. data/lib/flipper/expressions/string.rb +9 -0
  128. data/lib/flipper/expressions/time.rb +16 -0
  129. data/lib/flipper/feature.rb +95 -28
  130. data/lib/flipper/feature_check_context.rb +10 -6
  131. data/lib/flipper/gate.rb +13 -11
  132. data/lib/flipper/gate_values.rb +5 -18
  133. data/lib/flipper/gates/actor.rb +10 -17
  134. data/lib/flipper/gates/boolean.rb +1 -1
  135. data/lib/flipper/gates/expression.rb +75 -0
  136. data/lib/flipper/gates/group.rb +5 -7
  137. data/lib/flipper/gates/percentage_of_actors.rb +10 -13
  138. data/lib/flipper/gates/percentage_of_time.rb +1 -2
  139. data/lib/flipper/identifier.rb +17 -0
  140. data/lib/flipper/instrumentation/log_subscriber.rb +35 -8
  141. data/lib/flipper/instrumentation/statsd.rb +4 -2
  142. data/lib/flipper/instrumentation/statsd_subscriber.rb +2 -4
  143. data/lib/flipper/instrumentation/subscriber.rb +8 -5
  144. data/lib/flipper/instrumenters/memory.rb +6 -2
  145. data/lib/flipper/metadata.rb +8 -1
  146. data/lib/flipper/middleware/memoizer.rb +46 -27
  147. data/lib/flipper/middleware/setup_env.rb +13 -3
  148. data/lib/flipper/model/active_record.rb +23 -0
  149. data/lib/flipper/poller.rb +157 -0
  150. data/lib/flipper/serializers/gzip.rb +22 -0
  151. data/lib/flipper/serializers/json.rb +17 -0
  152. data/lib/flipper/spec/shared_adapter_specs.rb +122 -56
  153. data/lib/flipper/test/shared_adapter_test.rb +120 -52
  154. data/lib/flipper/test_help.rb +43 -0
  155. data/lib/flipper/typecast.rb +59 -18
  156. data/lib/flipper/types/actor.rb +19 -13
  157. data/lib/flipper/types/group.rb +12 -5
  158. data/lib/flipper/types/percentage.rb +1 -1
  159. data/lib/flipper/version.rb +11 -1
  160. data/lib/flipper.rb +71 -12
  161. data/lib/generators/flipper/setup_generator.rb +68 -0
  162. data/lib/generators/flipper/templates/initializer.rb +45 -0
  163. data/lib/generators/flipper/templates/update/migrations/01_create_flipper_tables.rb.erb +22 -0
  164. data/lib/generators/flipper/templates/update/migrations/02_change_flipper_gates_value_to_text.rb.erb +18 -0
  165. data/lib/generators/flipper/update_generator.rb +35 -0
  166. data/package-lock.json +41 -0
  167. data/package.json +10 -0
  168. data/spec/fixtures/environment.rb +1 -0
  169. data/spec/fixtures/flipper_pstore_1679087600.json +46 -0
  170. data/spec/flipper/actor_spec.rb +10 -2
  171. data/spec/flipper/adapter_builder_spec.rb +72 -0
  172. data/spec/flipper/adapter_spec.rb +52 -6
  173. data/spec/flipper/adapters/actor_limit_spec.rb +75 -0
  174. data/spec/flipper/adapters/dual_write_spec.rb +82 -0
  175. data/spec/flipper/adapters/failover_spec.rb +141 -0
  176. data/spec/flipper/adapters/failsafe_spec.rb +58 -0
  177. data/spec/flipper/adapters/http/client_spec.rb +61 -0
  178. data/spec/flipper/adapters/http_spec.rb +402 -65
  179. data/spec/flipper/adapters/instrumented_spec.rb +31 -13
  180. data/spec/flipper/adapters/memoizable_spec.rb +51 -33
  181. data/spec/flipper/adapters/memory_spec.rb +33 -5
  182. data/spec/flipper/adapters/operation_logger_spec.rb +38 -12
  183. data/spec/flipper/adapters/poll_spec.rb +41 -0
  184. data/spec/flipper/adapters/pstore_spec.rb +0 -2
  185. data/spec/flipper/adapters/read_only_spec.rb +32 -18
  186. data/spec/flipper/adapters/strict_spec.rb +64 -0
  187. data/spec/flipper/adapters/sync/feature_synchronizer_spec.rb +39 -1
  188. data/spec/flipper/adapters/sync/interval_synchronizer_spec.rb +4 -5
  189. data/spec/flipper/adapters/sync/synchronizer_spec.rb +87 -1
  190. data/spec/flipper/adapters/sync_spec.rb +17 -6
  191. data/spec/flipper/cli_spec.rb +217 -0
  192. data/spec/flipper/cloud/configuration_spec.rb +257 -0
  193. data/spec/flipper/cloud/dsl_spec.rb +90 -0
  194. data/spec/flipper/cloud/message_verifier_spec.rb +104 -0
  195. data/spec/flipper/cloud/middleware_spec.rb +307 -0
  196. data/spec/flipper/cloud/migrate_spec.rb +160 -0
  197. data/spec/flipper/cloud/telemetry/backoff_policy_spec.rb +107 -0
  198. data/spec/flipper/cloud/telemetry/metric_spec.rb +87 -0
  199. data/spec/flipper/cloud/telemetry/metric_storage_spec.rb +58 -0
  200. data/spec/flipper/cloud/telemetry/submitter_spec.rb +145 -0
  201. data/spec/flipper/cloud/telemetry_spec.rb +208 -0
  202. data/spec/flipper/cloud_spec.rb +186 -0
  203. data/spec/flipper/configuration_spec.rb +37 -3
  204. data/spec/flipper/dsl_spec.rb +67 -80
  205. data/spec/flipper/engine_spec.rb +374 -0
  206. data/spec/flipper/export_spec.rb +13 -0
  207. data/spec/flipper/exporter_spec.rb +16 -0
  208. data/spec/flipper/exporters/json/export_spec.rb +60 -0
  209. data/spec/flipper/exporters/json/v1_spec.rb +33 -0
  210. data/spec/flipper/expression/builder_spec.rb +248 -0
  211. data/spec/flipper/expression_spec.rb +188 -0
  212. data/spec/flipper/expressions/all_spec.rb +15 -0
  213. data/spec/flipper/expressions/any_spec.rb +15 -0
  214. data/spec/flipper/expressions/boolean_spec.rb +15 -0
  215. data/spec/flipper/expressions/equal_spec.rb +24 -0
  216. data/spec/flipper/expressions/greater_than_or_equal_to_spec.rb +28 -0
  217. data/spec/flipper/expressions/greater_than_spec.rb +28 -0
  218. data/spec/flipper/expressions/less_than_or_equal_to_spec.rb +28 -0
  219. data/spec/flipper/expressions/less_than_spec.rb +32 -0
  220. data/spec/flipper/expressions/not_equal_spec.rb +15 -0
  221. data/spec/flipper/expressions/now_spec.rb +11 -0
  222. data/spec/flipper/expressions/number_spec.rb +21 -0
  223. data/spec/flipper/expressions/percentage_of_actors_spec.rb +20 -0
  224. data/spec/flipper/expressions/percentage_spec.rb +15 -0
  225. data/spec/flipper/expressions/property_spec.rb +13 -0
  226. data/spec/flipper/expressions/random_spec.rb +9 -0
  227. data/spec/flipper/expressions/string_spec.rb +11 -0
  228. data/spec/flipper/expressions/time_spec.rb +29 -0
  229. data/spec/flipper/feature_check_context_spec.rb +18 -20
  230. data/spec/flipper/feature_spec.rb +461 -48
  231. data/spec/flipper/gate_spec.rb +0 -2
  232. data/spec/flipper/gate_values_spec.rb +2 -34
  233. data/spec/flipper/gates/actor_spec.rb +0 -2
  234. data/spec/flipper/gates/boolean_spec.rb +1 -3
  235. data/spec/flipper/gates/expression_spec.rb +190 -0
  236. data/spec/flipper/gates/group_spec.rb +2 -5
  237. data/spec/flipper/gates/percentage_of_actors_spec.rb +61 -7
  238. data/spec/flipper/gates/percentage_of_time_spec.rb +2 -4
  239. data/spec/flipper/identifier_spec.rb +12 -0
  240. data/spec/flipper/instrumentation/log_subscriber_spec.rb +24 -7
  241. data/spec/flipper/instrumentation/statsd_subscriber_spec.rb +26 -3
  242. data/spec/flipper/instrumenters/memory_spec.rb +18 -1
  243. data/spec/flipper/instrumenters/noop_spec.rb +14 -8
  244. data/spec/flipper/middleware/memoizer_spec.rb +199 -62
  245. data/spec/flipper/middleware/setup_env_spec.rb +23 -5
  246. data/spec/flipper/model/active_record_spec.rb +72 -0
  247. data/spec/flipper/poller_spec.rb +390 -0
  248. data/spec/flipper/registry_spec.rb +0 -1
  249. data/spec/flipper/serializers/gzip_spec.rb +13 -0
  250. data/spec/flipper/serializers/json_spec.rb +13 -0
  251. data/spec/flipper/typecast_spec.rb +121 -7
  252. data/spec/flipper/types/actor_spec.rb +63 -47
  253. data/spec/flipper/types/boolean_spec.rb +0 -1
  254. data/spec/flipper/types/group_spec.rb +24 -3
  255. data/spec/flipper/types/percentage_of_actors_spec.rb +0 -1
  256. data/spec/flipper/types/percentage_of_time_spec.rb +0 -1
  257. data/spec/flipper/types/percentage_spec.rb +0 -1
  258. data/spec/{integration_spec.rb → flipper_integration_spec.rb} +301 -59
  259. data/spec/flipper_spec.rb +123 -29
  260. data/spec/{helper.rb → spec_helper.rb} +23 -21
  261. data/spec/support/actor_names.yml +1 -0
  262. data/spec/support/descriptions.yml +1 -0
  263. data/spec/support/fail_on_output.rb +8 -0
  264. data/spec/support/fake_backoff_policy.rb +15 -0
  265. data/spec/support/skippable.rb +18 -0
  266. data/spec/support/spec_helpers.rb +53 -6
  267. data/test/adapters/actor_limit_test.rb +20 -0
  268. data/test/test_helper.rb +2 -1
  269. data/test_rails/generators/flipper/setup_generator_test.rb +69 -0
  270. data/test_rails/generators/flipper/update_generator_test.rb +96 -0
  271. data/test_rails/helper.rb +31 -0
  272. data/test_rails/system/test_help_test.rb +52 -0
  273. metadata +200 -82
  274. data/.rubocop.yml +0 -54
  275. data/.rubocop_todo.yml +0 -199
  276. data/docs/Adapters.md +0 -124
  277. data/docs/Caveats.md +0 -4
  278. data/docs/Gates.md +0 -167
  279. data/docs/Instrumentation.md +0 -27
  280. data/docs/Optimization.md +0 -114
  281. data/docs/api/README.md +0 -849
  282. data/docs/http/README.md +0 -35
  283. data/docs/read-only/README.md +0 -21
  284. data/examples/example_setup.rb +0 -8
  285. 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 File.expand_path('../example_setup', __FILE__)
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.default do
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 File.expand_path('../example_setup', __FILE__)
2
-
1
+ require 'bundler/setup'
3
2
  require 'flipper'
4
3
 
5
- adapter = Flipper::Adapters::Memory.new
6
- flipper = Flipper.new(adapter)
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: #{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 "flipper.enabled? :stats, person: #{flipper.enabled? :stats, person}"
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 = flipper[:stats]\n\n"
32
- stats = flipper[: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
- flipper.enable :stats
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
- flipper.disable :stats
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 File.expand_path('../example_setup', __FILE__)
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
- flipper[:search].enable
36
- flipper[:stats].enable_actor user1
37
- flipper[:pro_stats].enable_percentage_of_actors 50
38
- flipper[:tweets].enable_group :admins
39
- flipper[:posts].enable_actor user2
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 flipper.features.select { |feature| feature.enabled?(user1) }.map(&:name)
42
- pp flipper.features.select { |feature| feature.enabled?(user2) }.map(&:name)
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