pecorino 0.7.2 → 0.7.3

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: df8db59f2303035498ca51c54787f664fbc000148965e3782e188e26b90fea31
4
- data.tar.gz: 7e67b40c92846045b0e38c8e706622d6114e307b49a8efaaec287db144734fdf
3
+ metadata.gz: c0e0c8fa1a6bc734dc645b5b021264eda84bcadd1bec8da65c533a33cd243fb6
4
+ data.tar.gz: c6c053d8ed02f8e180786d4614c607087ca76b016b8acac35b35a3b3cb544391
5
5
  SHA512:
6
- metadata.gz: 6906f549004f30b57bbf9c0ca9d2a36ba1183acf0aec863563786d9118d8f0a1094d0dc2b6eddcdeb7929be7af1f017df0063a5073a798c74cd2337888a4a3e0
7
- data.tar.gz: d16cc4946a3205488afe0ab00cabc6f8a482e59751468513bec02fa0685ec77676db58b5bcb9d92d7d2fb7631e0a39fe823cfe2ecade824a0e241e45a37510fd
6
+ metadata.gz: 02207b3f3d52aa635a960c3914768237a25da4b4cdd7a5ad5da222adf44e738ac987f5182392d0617c60848c745fb0cc5225e2e2392dff50b6e83f58b703d191
7
+ data.tar.gz: 350d04b9e366d229d87146465574632a00e5e0c88eda7cbc540b494d8ab1d3520d87092f5ae6a6b11f0dd323cc7fd15436728c9b6210fe2264be384b0e9505b4
data/.yardopts ADDED
@@ -0,0 +1 @@
1
+ --markup markdown - README.md CHANGELOG.md LICENSE.txt
data/CHANGELOG.md CHANGED
@@ -1,3 +1,7 @@
1
+ ## 0.7.3
2
+
3
+ - Fix a number of YARD issues and generate both .rbi and .rbs typedefs
4
+
1
5
  ## 0.7.2
2
6
 
3
7
  - Set up a workable test harness for testing on both Rails 8 (Ruby 3.x) and Rails 7 (Ruby 2.x)
data/Gemfile ADDED
@@ -0,0 +1,17 @@
1
+ # frozen_string_literal: true
2
+
3
+ source "https://rubygems.org"
4
+
5
+ ruby ">= 3.0"
6
+ gemspec
7
+
8
+ gem "pg"
9
+ gem "sqlite3"
10
+ gem "activesupport", ">= 8"
11
+ gem "rake", "~> 13.0"
12
+ gem "minitest", "~> 5.0"
13
+ gem "redis", "~> 5", "< 6"
14
+ gem "yard"
15
+ gem "standard"
16
+ gem "sord"
17
+ gem "redcarpet"
data/README.md CHANGED
@@ -31,7 +31,7 @@ Once the installation is done you can use Pecorino to start defining your thrott
31
31
  We call this pattern **prefix usage** - apply throttle before allowing the action to proceed. This is more secure than registering an action after it has taken place.
32
32
 
33
33
  ```ruby
34
- throttle = Pecorino::Throttle.new(key: "password-attempts-#{request.ip}", over_time: 1.minute, capacity: 5, block_for: 30.minutes)
34
+ throttle = Pecorino::Throttle.new(key: "password-attempts-#{the_request.ip}", over_time: 1.minute, capacity: 5, block_for: 30.minutes)
35
35
  throttle.request!
36
36
  ```
37
37
  In a Rails controller you can then rescue from this exception to render the appropriate response:
@@ -119,11 +119,11 @@ class WalletController < ApplicationController
119
119
  end
120
120
 
121
121
  def withdraw
122
- Wallet.transaction do
123
- t = Pecorino::Throttle.new("wallet_#{current_user.id}_max_withdrawal", capacity: 200_00, over_time: 5.minutes)
124
- t.request!(10_00)
125
- current_user.wallet.withdraw(Money.new(10, "EUR"))
126
- end
122
+ Wallet.transaction do
123
+ t = Pecorino::Throttle.new("wallet_#{current_user.id}_max_withdrawal", capacity: 200_00, over_time: 5.minutes)
124
+ t.request!(10_00)
125
+ current_user.wallet.withdraw(Money.new(10, "EUR"))
126
+ end
127
127
  end
128
128
  end
129
129
  ```
@@ -201,7 +201,7 @@ If you are using Redis, you may want to ensure it gets truncated/reset for every
201
201
  If a throttle is triggered, Pecorino sets a "block" record for that throttle key. Any request to that throttle will fail until the block is lifted. If you are getting hammered by requests which are getting throttled, it might be a good idea to install a caching layer which will respond with a "rate limit exceeded" error even before hitting your database - until the moment when the block would be lifted. You can use any [ActiveSupport::Cache::Store](https://api.rubyonrails.org/classes/ActiveSupport/Cache/Store.html) to store your blocks. If you have a fast Rails cache configured, create a wrapped throttle:
202
202
 
203
203
  ```ruby
204
- throttle = Pecorino::Throttle.new(key: "ip-#{request.ip}", capacity: 10, over_time: 2.seconds, block_for: 2.minutes)
204
+ throttle = Pecorino::Throttle.new(key: "ip-#{the_request.ip}", capacity: 10, over_time: 2.seconds, block_for: 2.minutes)
205
205
  cached_throttle = Pecorino::CachedThrottle.new(Rails.cache, throttle)
206
206
  cached_throttle.request!
207
207
  ```
@@ -214,7 +214,7 @@ config.pecorino_throttle_cache = ActiveSupport::Cache::MemoryStore.new
214
214
 
215
215
  # in your controller
216
216
 
217
- throttle = Pecorino::Throttle.new(key: "ip-#{request.ip}", capacity: 10, over_time: 2.seconds, block_for: 2.minutes)
217
+ throttle = Pecorino::Throttle.new(key: "ip-#{the_request.ip}", capacity: 10, over_time: 2.seconds, block_for: 2.minutes)
218
218
  cached_throttle = Pecorino::CachedThrottle.new(Rails.application.config.pecorino_throttle_cache, throttle)
219
219
  cached_throttle.request!
220
220
  ```
@@ -230,15 +230,16 @@ ActiveRecord::Base.connection.execute("ALTER TABLE pecorino_blocks SET UNLOGGED"
230
230
 
231
231
  ## Development
232
232
 
233
- After checking out the repo, set the Gemfile appropriate to your Ruby version and run Rake for tests, lint etc.
234
- Note that it is important to use the appropriate Gemfile per Ruby version and Rails version you want to test with. Due to some dependency shenanigans it is currently not very easy to have a single Gemfile.
233
+ After checking out the repo, run `bundle install` and then do the thing you need to do.
234
+
235
+ **Note:** CI runs other Gemfiles, because we can't test all Ruby versions and Rails versions just by swapping Gemfiles. If you need to debug something with a particular Ruby and Rails version, do this:
235
236
 
236
237
  ```bash
237
- $ rbenv local 2.7.7 && export BUNDLE_GEMFILE=gemfiles/Gemfile_ruby27_rails7 && bundle install
238
+ $ bundle rbenv local 2.7.7 && export BUNDLE_GEMFILE=gemfiles/Gemfile_ruby27_rails7 && bundle install
238
239
  $ bundle exec rake
239
240
  ```
240
241
 
241
- Then proceed to develop as normal. CI will run both the oldest supported dependencies and newest supported dependencies.
242
+ Then proceed as normal. Make sure to unset `BUNDLE_GEMFILE` when you are done. CI will run both the oldest supported dependencies and newest supported dependencies.
242
243
 
243
244
  ## Contributing
244
245
 
data/Rakefile CHANGED
@@ -20,6 +20,7 @@ end
20
20
 
21
21
  task :generate_typedefs do
22
22
  `bundle exec sord rbi/pecorino.rbi`
23
+ `bundle exec sord rbi/pecorino.rbs`
23
24
  end
24
25
 
25
26
  task default: [:test, :standard, :generate_typedefs]
@@ -15,6 +15,10 @@ class Pecorino::CachedThrottle
15
15
  @throttle = throttle
16
16
  end
17
17
 
18
+ # Increments the cached throttle by the given number of tokens. If there is currently a known cached block on that throttle
19
+ # an exception will be raised immediately instead of querying the actual throttle data. Otherwise the call gets forwarded
20
+ # to the underlying throttle.
21
+ #
18
22
  # @see Pecorino::Throttle#request!
19
23
  def request!(n = 1)
20
24
  blocked_state = read_cached_blocked_state
@@ -28,9 +32,9 @@ class Pecorino::CachedThrottle
28
32
  end
29
33
  end
30
34
 
31
- # Returns cached `state` for the throttle if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
35
+ # Returns the cached `state` for the throttle if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
32
36
  #
33
- # @see Pecorino::Throttle#request
37
+ # @see Pecorino::Throttle#request!
34
38
  def request(n = 1)
35
39
  blocked_state = read_cached_blocked_state
36
40
  return blocked_state if blocked_state&.blocked?
@@ -91,7 +91,7 @@ class Pecorino::Throttle
91
91
 
92
92
  # The key for that throttle. Each key defines a unique throttle based on either a given name or
93
93
  # discriminators. If there is a component you want to key your throttle by, include it in the
94
- # `key` keyword argument to the constructor, like `"t-ip-#{request.ip}"`
94
+ # `key` keyword argument to the constructor, like `"t-ip-#{your_rails_request.ip}"`
95
95
  #
96
96
  # @return [String]
97
97
  attr_reader :key
@@ -100,8 +100,8 @@ class Pecorino::Throttle
100
100
  # @param block_for[Numeric] the number of seconds to block any further requests for. Defaults to time it takes
101
101
  # the bucket to leak out to the level of 0
102
102
  # @param adapter[Pecorino::Adapters::BaseAdapter] a compatible adapter
103
- # @param leaky_bucket_options Options for `Pecorino::LeakyBucket.new`
104
- # @see PecorinoLeakyBucket.new
103
+ # @param leaky_bucket_options Options for {Pecorino::LeakyBucket.new}
104
+ # @see Pecorino::LeakyBucket.new
105
105
  def initialize(key:, block_for: nil, adapter: Pecorino.adapter, **leaky_bucket_options)
106
106
  @adapter = adapter
107
107
  leaky_bucket_options.delete(:adapter)
@@ -129,16 +129,16 @@ class Pecorino::Throttle
129
129
  # The exception can be rescued later to provide a 429 response. This method is better
130
130
  # to use before performing the unit of work that the throttle is guarding:
131
131
  #
132
+ # If the method call returns it means that the request is not getting throttled.
133
+ #
132
134
  # @example
133
135
  # begin
134
- # t.request!
135
- # Note.create!(note_params)
136
+ # t.request!
137
+ # Note.create!(note_params)
136
138
  # rescue Pecorino::Throttle::Throttled => e
137
- # [429, {"Retry-After" => e.retry_after.to_s}, []]
139
+ # [429, {"Retry-After" => e.retry_after.to_s}, []]
138
140
  # end
139
- #
140
- # If the method call succeeds it means that the request is not getting throttled.
141
- #
141
+ # @param n [Numeric] how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
142
142
  # @return [State] the state of the throttle after filling up the leaky bucket / trying to pass the block
143
143
  def request!(n = 1)
144
144
  request(n).tap do |state_after|
@@ -156,8 +156,8 @@ class Pecorino::Throttle
156
156
  # Entry.create!(entry_params)
157
157
  # t.request
158
158
  # end
159
- #
160
- # @return [State] the state of the throttle after filling up the leaky bucket / trying to pass the block
159
+ # @param n [Numeric] how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
160
+ # @return [State] the state of the throttle after the attempt to fill up the leaky bucket
161
161
  def request(n = 1)
162
162
  existing_blocked_until = Pecorino::Block.blocked_until(key: @key, adapter: @adapter)
163
163
  return State.new(existing_blocked_until.utc) if existing_blocked_until
@@ -181,6 +181,7 @@ class Pecorino::Throttle
181
181
  # @example
182
182
  # t.throttled { Slack.alert("Things are going wrong") }
183
183
  #
184
+ # @param blk The block to run. Will only run if the throttle accepts the call.
184
185
  # @return [Object] the return value of the block if the block gets executed, or `nil` if the call got throttled
185
186
  def throttled(&blk)
186
187
  return if request(1).blocked?
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Pecorino
4
- VERSION = "0.7.2"
4
+ VERSION = "0.7.3"
5
5
  end
data/lib/pecorino.rb CHANGED
@@ -60,9 +60,9 @@ module Pecorino
60
60
 
61
61
  # Returns the database implementation for setting the values atomically. Since the implementation
62
62
  # differs per database, this method will return a different adapter depending on which database is
63
- # being used
63
+ # being used.
64
64
  #
65
- # @param adapter[Pecorino::Adapters::BaseAdapter]
65
+ # @return [Pecorino::Adapters::BaseAdapter]
66
66
  def self.default_adapter_from_main_database
67
67
  model_class = ActiveRecord::Base
68
68
  adapter_name = model_class.connection.adapter_name
data/rbi/pecorino.rbi CHANGED
@@ -1,6 +1,6 @@
1
1
  # typed: strong
2
2
  module Pecorino
3
- VERSION = T.let("0.7.1", T.untyped)
3
+ VERSION = T.let("0.7.3", T.untyped)
4
4
 
5
5
  # Deletes stale leaky buckets and blocks which have expired. Run this method regularly to
6
6
  # avoid accumulating too many unused rows in your tables.
@@ -36,18 +36,15 @@ module Pecorino
36
36
  sig { returns(Pecorino::Adapters::BaseAdapter) }
37
37
  def self.adapter; end
38
38
 
39
- # sord omit - no YARD return type given, using untyped
40
39
  # Returns the database implementation for setting the values atomically. Since the implementation
41
40
  # differs per database, this method will return a different adapter depending on which database is
42
- # being used
43
- #
44
- # _@param_ `adapter`
45
- sig { returns(T.untyped) }
41
+ # being used.
42
+ sig { returns(Pecorino::Adapters::BaseAdapter) }
46
43
  def self.default_adapter_from_main_database; end
47
44
 
48
45
  module Adapters
49
46
  # An adapter allows Pecorino throttles, leaky buckets and other
50
- # resources to interfact to a data storage backend - a database, usually.
47
+ # resources to interface with a data storage backend - a database, usually.
51
48
  class BaseAdapter
52
49
  # Returns the state of a leaky bucket. The state should be a tuple of two
53
50
  # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
@@ -503,9 +500,9 @@ module Pecorino
503
500
  #
504
501
  # _@param_ `adapter` — a compatible adapter
505
502
  #
506
- # _@param_ `leaky_bucket_options` — Options for `Pecorino::LeakyBucket.new`
503
+ # _@param_ `leaky_bucket_options` — Options for {Pecorino::LeakyBucket.new}
507
504
  #
508
- # _@see_ `PecorinoLeakyBucket.new`
505
+ # _@see_ `Pecorino::LeakyBucket.new`
509
506
  sig do
510
507
  params(
511
508
  key: String,
@@ -526,7 +523,6 @@ module Pecorino
526
523
  sig { params(n_tokens: Float).returns(T::Boolean) }
527
524
  def able_to_accept?(n_tokens = 1); end
528
525
 
529
- # sord omit - no YARD type given for "n", using untyped
530
526
  # Register that a request is being performed. Will raise Throttled
531
527
  # if there is a block in place for that throttle, or if the bucket cannot accept
532
528
  # this fillup and the block has just been installed as a result of this particular request.
@@ -534,28 +530,31 @@ module Pecorino
534
530
  # The exception can be rescued later to provide a 429 response. This method is better
535
531
  # to use before performing the unit of work that the throttle is guarding:
536
532
  #
537
- # If the method call succeeds it means that the request is not getting throttled.
533
+ # If the method call returns it means that the request is not getting throttled.
534
+ #
535
+ # _@param_ `n` — how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
538
536
  #
539
537
  # _@return_ — the state of the throttle after filling up the leaky bucket / trying to pass the block
540
538
  #
541
539
  # ```ruby
542
540
  # begin
543
- # t.request!
544
- # Note.create!(note_params)
541
+ # t.request!
542
+ # Note.create!(note_params)
545
543
  # rescue Pecorino::Throttle::Throttled => e
546
- # [429, {"Retry-After" => e.retry_after.to_s}, []]
544
+ # [429, {"Retry-After" => e.retry_after.to_s}, []]
547
545
  # end
548
546
  # ```
549
- sig { params(n: T.untyped).returns(State) }
547
+ sig { params(n: Numeric).returns(State) }
550
548
  def request!(n = 1); end
551
549
 
552
- # sord omit - no YARD type given for "n", using untyped
553
550
  # Register that a request is being performed. Will not raise any exceptions but return
554
551
  # the time at which the block will be lifted if a block resulted from this request or
555
552
  # was already in effect. Can be used for registering actions which already took place,
556
553
  # but should result in subsequent actions being blocked.
557
554
  #
558
- # _@return_the state of the throttle after filling up the leaky bucket / trying to pass the block
555
+ # _@param_ `n` how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
556
+ #
557
+ # _@return_ — the state of the throttle after the attempt to fill up the leaky bucket
559
558
  #
560
559
  # ```ruby
561
560
  # if t.able_to_accept?
@@ -563,7 +562,7 @@ module Pecorino
563
562
  # t.request
564
563
  # end
565
564
  # ```
566
- sig { params(n: T.untyped).returns(State) }
565
+ sig { params(n: Numeric).returns(State) }
567
566
  def request(n = 1); end
568
567
 
569
568
  # Fillup the throttle with 1 request and then perform the passed block. This is useful to perform actions which should
@@ -571,6 +570,8 @@ module Pecorino
571
570
  # the passed block will be executed. If the throttle is in the blocked state or if the call puts the throttle in
572
571
  # the blocked state the block will not be executed
573
572
  #
573
+ # _@param_ `blk` — The block to run. Will only run if the throttle accepts the call.
574
+ #
574
575
  # _@return_ — the return value of the block if the block gets executed, or `nil` if the call got throttled
575
576
  #
576
577
  # ```ruby
@@ -581,7 +582,7 @@ module Pecorino
581
582
 
582
583
  # The key for that throttle. Each key defines a unique throttle based on either a given name or
583
584
  # discriminators. If there is a component you want to key your throttle by, include it in the
584
- # `key` keyword argument to the constructor, like `"t-ip-#{request.ip}"`
585
+ # `key` keyword argument to the constructor, like `"t-ip-#{your_rails_request.ip}"`
585
586
  sig { returns(String) }
586
587
  attr_reader :key
587
588
 
@@ -835,6 +836,9 @@ module Pecorino
835
836
 
836
837
  # sord omit - no YARD type given for "n", using untyped
837
838
  # sord omit - no YARD return type given, using untyped
839
+ # Increments the cached throttle by the given number of tokens. If there is currently a known cached block on that throttle
840
+ # an exception will be raised immediately instead of querying the actual throttle data. Otherwise the call gets forwarded
841
+ # to the underlying throttle.
838
842
  #
839
843
  # _@see_ `Pecorino::Throttle#request!`
840
844
  sig { params(n: T.untyped).returns(T.untyped) }
@@ -842,9 +846,9 @@ module Pecorino
842
846
 
843
847
  # sord omit - no YARD type given for "n", using untyped
844
848
  # sord omit - no YARD return type given, using untyped
845
- # Returns cached `state` for the throttle if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
849
+ # Returns the cached `state` for the throttle if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
846
850
  #
847
- # _@see_ `Pecorino::Throttle#request`
851
+ # _@see_ `Pecorino::Throttle#request!`
848
852
  sig { params(n: T.untyped).returns(T.untyped) }
849
853
  def request(n = 1); end
850
854
 
data/rbi/pecorino.rbs ADDED
@@ -0,0 +1,791 @@
1
+ module Pecorino
2
+ VERSION: untyped
3
+
4
+ # Deletes stale leaky buckets and blocks which have expired. Run this method regularly to
5
+ # avoid accumulating too many unused rows in your tables.
6
+ #
7
+ # _@return_ — void
8
+ def self.prune!: () -> untyped
9
+
10
+ # sord warn - ActiveRecord::SchemaMigration wasn't able to be resolved to a constant in this project
11
+ # Creates the tables and indexes needed for Pecorino. Call this from your migrations like so:
12
+ #
13
+ # class CreatePecorinoTables < ActiveRecord::Migration[7.0]
14
+ # def change
15
+ # Pecorino.create_tables(self)
16
+ # end
17
+ # end
18
+ #
19
+ # _@param_ `active_record_schema` — the migration through which we will create the tables
20
+ #
21
+ # _@return_ — void
22
+ def self.create_tables: (ActiveRecord::SchemaMigration active_record_schema) -> untyped
23
+
24
+ # Allows assignment of an adapter for storing throttles. Normally this would be a subclass of `Pecorino::Adapters::BaseAdapter`, but
25
+ # you can assign anything you like. Set this in an initializer. By default Pecorino will use the adapter configured from your main
26
+ # database, but you can also create a separate database for it - or use Redis or memory storage.
27
+ #
28
+ # _@param_ `adapter`
29
+ def self.adapter=: (Pecorino::Adapters::BaseAdapter adapter) -> Pecorino::Adapters::BaseAdapter
30
+
31
+ # Returns the currently configured adapter, or the default adapter from the main database
32
+ def self.adapter: () -> Pecorino::Adapters::BaseAdapter
33
+
34
+ # Returns the database implementation for setting the values atomically. Since the implementation
35
+ # differs per database, this method will return a different adapter depending on which database is
36
+ # being used.
37
+ def self.default_adapter_from_main_database: () -> Pecorino::Adapters::BaseAdapter
38
+
39
+ module Adapters
40
+ # An adapter allows Pecorino throttles, leaky buckets and other
41
+ # resources to interface with a data storage backend - a database, usually.
42
+ class BaseAdapter
43
+ # Returns the state of a leaky bucket. The state should be a tuple of two
44
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
45
+ #
46
+ # _@param_ `key` — the key of the leaky bucket
47
+ #
48
+ # _@param_ `capacity` — the capacity of the leaky bucket to limit to
49
+ #
50
+ # _@param_ `leak_rate` — how many tokens leak out of the bucket per second
51
+ def state: (key: String, capacity: Float, leak_rate: Float) -> ::Array[untyped]
52
+
53
+ # Adds tokens to the leaky bucket. The return value is a tuple of two
54
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
55
+ #
56
+ # _@param_ `key` — the key of the leaky bucket
57
+ #
58
+ # _@param_ `capacity` — the capacity of the leaky bucket to limit to
59
+ #
60
+ # _@param_ `leak_rate` — how many tokens leak out of the bucket per second
61
+ #
62
+ # _@param_ `n_tokens` — how many tokens to add
63
+ def add_tokens: (
64
+ key: String,
65
+ capacity: Float,
66
+ leak_rate: Float,
67
+ n_tokens: Float
68
+ ) -> ::Array[untyped]
69
+
70
+ # Adds tokens to the leaky bucket conditionally. If there is capacity, the tokens will
71
+ # be added. If there isn't - the fillup will be rejected. The return value is a triplet of
72
+ # the current level (Float), whether the bucket is now at capacity (Boolean)
73
+ # and whether the fillup was accepted (Boolean)
74
+ #
75
+ # _@param_ `key` — the key of the leaky bucket
76
+ #
77
+ # _@param_ `capacity` — the capacity of the leaky bucket to limit to
78
+ #
79
+ # _@param_ `leak_rate` — how many tokens leak out of the bucket per second
80
+ #
81
+ # _@param_ `n_tokens` — how many tokens to add
82
+ def add_tokens_conditionally: (
83
+ key: String,
84
+ capacity: Float,
85
+ leak_rate: Float,
86
+ n_tokens: Float
87
+ ) -> ::Array[untyped]
88
+
89
+ # sord duck - #to_f looks like a duck type, replacing with untyped
90
+ # sord warn - "Active Support Duration" does not appear to be a type
91
+ # sord omit - no YARD return type given, using untyped
92
+ # Sets a timed block for the given key - this is used when a throttle fires. The return value
93
+ # is not defined - the call should always succeed.
94
+ #
95
+ # _@param_ `key` — the key of the block
96
+ #
97
+ # _@param_ `block_for` — the duration of the block, in seconds
98
+ def set_block: (key: String, block_for: (untyped | SORD_ERROR_ActiveSupportDuration)) -> untyped
99
+
100
+ # sord omit - no YARD return type given, using untyped
101
+ # Returns the time until which a block for a given key is in effect. If there is no block in
102
+ # effect, the method should return `nil`. The return value is either a `Time` or `nil`
103
+ #
104
+ # _@param_ `key` — the key of the block
105
+ def blocked_until: (key: String) -> untyped
106
+
107
+ # Deletes leaky buckets which have an expiry value prior to now and throttle blocks which have
108
+ # now lapsed
109
+ def prune: () -> void
110
+
111
+ # sord omit - no YARD type given for "active_record_schema", using untyped
112
+ # sord omit - no YARD return type given, using untyped
113
+ # Creates the database tables for Pecorino to operate, or initializes other
114
+ # schema-like resources the adapter needs to operate
115
+ def create_tables: (untyped active_record_schema) -> untyped
116
+ end
117
+
118
+ # An adapter for storing Pecorino leaky buckets and blocks in Redis. It uses Lua
119
+ # to enforce atomicity for leaky bucket operations
120
+ class RedisAdapter < Pecorino::Adapters::BaseAdapter
121
+ ADD_TOKENS_SCRIPT: untyped
122
+
123
+ # sord omit - no YARD type given for "redis_connection_or_connection_pool", using untyped
124
+ # sord omit - no YARD type given for "key_prefix:", using untyped
125
+ def initialize: (untyped redis_connection_or_connection_pool, ?key_prefix: untyped) -> void
126
+
127
+ # Returns the state of a leaky bucket. The state should be a tuple of two
128
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
129
+ def state: (key: String, capacity: Float, leak_rate: Float) -> ::Array[untyped]
130
+
131
+ # Adds tokens to the leaky bucket. The return value is a tuple of two
132
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
133
+ def add_tokens: (
134
+ key: String,
135
+ capacity: Float,
136
+ leak_rate: Float,
137
+ n_tokens: Float
138
+ ) -> ::Array[untyped]
139
+
140
+ # Adds tokens to the leaky bucket conditionally. If there is capacity, the tokens will
141
+ # be added. If there isn't - the fillup will be rejected. The return value is a triplet of
142
+ # the current level (Float), whether the bucket is now at capacity (Boolean)
143
+ # and whether the fillup was accepted (Boolean)
144
+ def add_tokens_conditionally: (
145
+ key: String,
146
+ capacity: Float,
147
+ leak_rate: Float,
148
+ n_tokens: Float
149
+ ) -> ::Array[untyped]
150
+
151
+ # sord duck - #to_f looks like a duck type, replacing with untyped
152
+ # sord warn - "Active Support Duration" does not appear to be a type
153
+ # sord omit - no YARD return type given, using untyped
154
+ # Sets a timed block for the given key - this is used when a throttle fires. The return value
155
+ # is not defined - the call should always succeed.
156
+ def set_block: (key: String, block_for: (untyped | SORD_ERROR_ActiveSupportDuration)) -> untyped
157
+
158
+ # sord omit - no YARD return type given, using untyped
159
+ # Returns the time until which a block for a given key is in effect. If there is no block in
160
+ # effect, the method should return `nil`. The return value is either a `Time` or `nil`
161
+ def blocked_until: (key: String) -> untyped
162
+
163
+ # sord omit - no YARD return type given, using untyped
164
+ def with_redis: () -> untyped
165
+
166
+ class RedisScript
167
+ # sord omit - no YARD type given for "script_filename", using untyped
168
+ def initialize: (untyped script_filename) -> void
169
+
170
+ # sord omit - no YARD type given for "redis", using untyped
171
+ # sord omit - no YARD type given for "keys", using untyped
172
+ # sord omit - no YARD type given for "argv", using untyped
173
+ # sord omit - no YARD return type given, using untyped
174
+ def load_and_eval: (untyped redis, untyped keys, untyped argv) -> untyped
175
+ end
176
+ end
177
+
178
+ # A memory store for leaky buckets and blocks
179
+ class MemoryAdapter
180
+ def initialize: () -> void
181
+
182
+ # sord omit - no YARD type given for "key:", using untyped
183
+ # sord omit - no YARD type given for "capacity:", using untyped
184
+ # sord omit - no YARD type given for "leak_rate:", using untyped
185
+ # sord omit - no YARD return type given, using untyped
186
+ # Returns the state of a leaky bucket. The state should be a tuple of two
187
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
188
+ def state: (key: untyped, capacity: untyped, leak_rate: untyped) -> untyped
189
+
190
+ # sord omit - no YARD type given for "key:", using untyped
191
+ # sord omit - no YARD type given for "capacity:", using untyped
192
+ # sord omit - no YARD type given for "leak_rate:", using untyped
193
+ # sord omit - no YARD type given for "n_tokens:", using untyped
194
+ # sord omit - no YARD return type given, using untyped
195
+ # Adds tokens to the leaky bucket. The return value is a tuple of two
196
+ # values: the current level (Float) and whether the bucket is now at capacity (Boolean)
197
+ def add_tokens: (
198
+ key: untyped,
199
+ capacity: untyped,
200
+ leak_rate: untyped,
201
+ n_tokens: untyped
202
+ ) -> untyped
203
+
204
+ # sord omit - no YARD type given for "key:", using untyped
205
+ # sord omit - no YARD type given for "capacity:", using untyped
206
+ # sord omit - no YARD type given for "leak_rate:", using untyped
207
+ # sord omit - no YARD type given for "n_tokens:", using untyped
208
+ # sord omit - no YARD return type given, using untyped
209
+ # Adds tokens to the leaky bucket conditionally. If there is capacity, the tokens will
210
+ # be added. If there isn't - the fillup will be rejected. The return value is a triplet of
211
+ # the current level (Float), whether the bucket is now at capacity (Boolean)
212
+ # and whether the fillup was accepted (Boolean)
213
+ def add_tokens_conditionally: (
214
+ key: untyped,
215
+ capacity: untyped,
216
+ leak_rate: untyped,
217
+ n_tokens: untyped
218
+ ) -> untyped
219
+
220
+ # sord omit - no YARD type given for "key:", using untyped
221
+ # sord omit - no YARD type given for "block_for:", using untyped
222
+ # sord omit - no YARD return type given, using untyped
223
+ # Sets a timed block for the given key - this is used when a throttle fires. The return value
224
+ # is not defined - the call should always succeed.
225
+ def set_block: (key: untyped, block_for: untyped) -> untyped
226
+
227
+ # sord omit - no YARD type given for "key:", using untyped
228
+ # sord omit - no YARD return type given, using untyped
229
+ # Returns the time until which a block for a given key is in effect. If there is no block in
230
+ # effect, the method should return `nil`. The return value is either a `Time` or `nil`
231
+ def blocked_until: (key: untyped) -> untyped
232
+
233
+ # sord omit - no YARD return type given, using untyped
234
+ # Deletes leaky buckets which have an expiry value prior to now and throttle blocks which have
235
+ # now lapsed
236
+ def prune: () -> untyped
237
+
238
+ # sord omit - no YARD type given for "active_record_schema", using untyped
239
+ # sord omit - no YARD return type given, using untyped
240
+ # No-op
241
+ def create_tables: (untyped active_record_schema) -> untyped
242
+
243
+ # sord omit - no YARD type given for "key", using untyped
244
+ # sord omit - no YARD type given for "capacity", using untyped
245
+ # sord omit - no YARD type given for "leak_rate", using untyped
246
+ # sord omit - no YARD type given for "n_tokens", using untyped
247
+ # sord omit - no YARD type given for "conditionally", using untyped
248
+ # sord omit - no YARD return type given, using untyped
249
+ def add_tokens_with_lock: (
250
+ untyped key,
251
+ untyped capacity,
252
+ untyped leak_rate,
253
+ untyped n_tokens,
254
+ untyped conditionally
255
+ ) -> untyped
256
+
257
+ # sord omit - no YARD return type given, using untyped
258
+ def get_mono_time: () -> untyped
259
+
260
+ # sord omit - no YARD type given for "min", using untyped
261
+ # sord omit - no YARD type given for "value", using untyped
262
+ # sord omit - no YARD type given for "max", using untyped
263
+ # sord omit - no YARD return type given, using untyped
264
+ def clamp: (untyped min, untyped value, untyped max) -> untyped
265
+
266
+ class KeyedLock
267
+ def initialize: () -> void
268
+
269
+ # sord omit - no YARD type given for "key", using untyped
270
+ # sord omit - no YARD return type given, using untyped
271
+ def lock: (untyped key) -> untyped
272
+
273
+ # sord omit - no YARD type given for "key", using untyped
274
+ # sord omit - no YARD return type given, using untyped
275
+ def unlock: (untyped key) -> untyped
276
+
277
+ # sord omit - no YARD type given for "key", using untyped
278
+ # sord omit - no YARD return type given, using untyped
279
+ def with: (untyped key) -> untyped
280
+ end
281
+ end
282
+
283
+ class SqliteAdapter
284
+ # sord omit - no YARD type given for "model_class", using untyped
285
+ def initialize: (untyped model_class) -> void
286
+
287
+ # sord omit - no YARD type given for "key:", using untyped
288
+ # sord omit - no YARD type given for "capacity:", using untyped
289
+ # sord omit - no YARD type given for "leak_rate:", using untyped
290
+ # sord omit - no YARD return type given, using untyped
291
+ def state: (key: untyped, capacity: untyped, leak_rate: untyped) -> untyped
292
+
293
+ # sord omit - no YARD type given for "key:", using untyped
294
+ # sord omit - no YARD type given for "capacity:", using untyped
295
+ # sord omit - no YARD type given for "leak_rate:", using untyped
296
+ # sord omit - no YARD type given for "n_tokens:", using untyped
297
+ # sord omit - no YARD return type given, using untyped
298
+ def add_tokens: (
299
+ key: untyped,
300
+ capacity: untyped,
301
+ leak_rate: untyped,
302
+ n_tokens: untyped
303
+ ) -> untyped
304
+
305
+ # sord omit - no YARD type given for "key:", using untyped
306
+ # sord omit - no YARD type given for "capacity:", using untyped
307
+ # sord omit - no YARD type given for "leak_rate:", using untyped
308
+ # sord omit - no YARD type given for "n_tokens:", using untyped
309
+ # sord omit - no YARD return type given, using untyped
310
+ def add_tokens_conditionally: (
311
+ key: untyped,
312
+ capacity: untyped,
313
+ leak_rate: untyped,
314
+ n_tokens: untyped
315
+ ) -> untyped
316
+
317
+ # sord omit - no YARD type given for "key:", using untyped
318
+ # sord omit - no YARD type given for "block_for:", using untyped
319
+ # sord omit - no YARD return type given, using untyped
320
+ def set_block: (key: untyped, block_for: untyped) -> untyped
321
+
322
+ # sord omit - no YARD type given for "key:", using untyped
323
+ # sord omit - no YARD return type given, using untyped
324
+ def blocked_until: (key: untyped) -> untyped
325
+
326
+ # sord omit - no YARD return type given, using untyped
327
+ def prune: () -> untyped
328
+
329
+ # sord omit - no YARD type given for "active_record_schema", using untyped
330
+ # sord omit - no YARD return type given, using untyped
331
+ def create_tables: (untyped active_record_schema) -> untyped
332
+ end
333
+
334
+ class PostgresAdapter
335
+ # sord omit - no YARD type given for "model_class", using untyped
336
+ def initialize: (untyped model_class) -> void
337
+
338
+ # sord omit - no YARD type given for "key:", using untyped
339
+ # sord omit - no YARD type given for "capacity:", using untyped
340
+ # sord omit - no YARD type given for "leak_rate:", using untyped
341
+ # sord omit - no YARD return type given, using untyped
342
+ def state: (key: untyped, capacity: untyped, leak_rate: untyped) -> untyped
343
+
344
+ # sord omit - no YARD type given for "key:", using untyped
345
+ # sord omit - no YARD type given for "capacity:", using untyped
346
+ # sord omit - no YARD type given for "leak_rate:", using untyped
347
+ # sord omit - no YARD type given for "n_tokens:", using untyped
348
+ # sord omit - no YARD return type given, using untyped
349
+ def add_tokens: (
350
+ key: untyped,
351
+ capacity: untyped,
352
+ leak_rate: untyped,
353
+ n_tokens: untyped
354
+ ) -> untyped
355
+
356
+ # sord omit - no YARD type given for "key:", using untyped
357
+ # sord omit - no YARD type given for "capacity:", using untyped
358
+ # sord omit - no YARD type given for "leak_rate:", using untyped
359
+ # sord omit - no YARD type given for "n_tokens:", using untyped
360
+ # sord omit - no YARD return type given, using untyped
361
+ def add_tokens_conditionally: (
362
+ key: untyped,
363
+ capacity: untyped,
364
+ leak_rate: untyped,
365
+ n_tokens: untyped
366
+ ) -> untyped
367
+
368
+ # sord omit - no YARD type given for "key:", using untyped
369
+ # sord omit - no YARD type given for "block_for:", using untyped
370
+ # sord omit - no YARD return type given, using untyped
371
+ def set_block: (key: untyped, block_for: untyped) -> untyped
372
+
373
+ # sord omit - no YARD type given for "key:", using untyped
374
+ # sord omit - no YARD return type given, using untyped
375
+ def blocked_until: (key: untyped) -> untyped
376
+
377
+ # sord omit - no YARD return type given, using untyped
378
+ def prune: () -> untyped
379
+
380
+ # sord omit - no YARD type given for "active_record_schema", using untyped
381
+ # sord omit - no YARD return type given, using untyped
382
+ def create_tables: (untyped active_record_schema) -> untyped
383
+ end
384
+ end
385
+
386
+ # Provides access to Pecorino blocks - same blocks which get set when a throttle triggers. The blocks
387
+ # are just keys in the data store which have an expiry value. This can be useful if you want to restrict
388
+ # access to a resource for an arbitrary timespan.
389
+ class Block
390
+ # Sets a block for the given key. The block will also be seen by the Pecorino::Throttle with the same key
391
+ #
392
+ # _@param_ `key` — the key to set the block for
393
+ #
394
+ # _@param_ `block_for` — the number of seconds or a time interval to block for
395
+ #
396
+ # _@param_ `adapter` — the adapter to set the value in.
397
+ #
398
+ # _@return_ — the time when the block will be released
399
+ def self.set!: (key: String, block_for: Float, ?adapter: Pecorino::Adapters::BaseAdapter) -> Time
400
+
401
+ # Returns the time until a certain block is in effect
402
+ #
403
+ # _@param_ `key` — the key to get the expiry time for
404
+ #
405
+ # _@param_ `adapter` — the adapter to get the value from
406
+ #
407
+ # _@return_ — the time when the block will be released
408
+ def self.blocked_until: (key: String, ?adapter: Pecorino::Adapters::BaseAdapter) -> Time?
409
+ end
410
+
411
+ class Railtie < Rails::Railtie
412
+ end
413
+
414
+ # Provides a throttle with a block based on the `LeakyBucket`. Once a bucket fills up,
415
+ # a block will be installed and an exception will be raised. Once a block is set, no
416
+ # checks will be done on the leaky bucket - any further requests will be refused until
417
+ # the block is lifted. The block time can be arbitrarily higher or lower than the amount
418
+ # of time it takes for the leaky bucket to leak out
419
+ class Throttle
420
+ # _@param_ `key` — the key for both the block record and the leaky bucket
421
+ #
422
+ # _@param_ `block_for` — the number of seconds to block any further requests for. Defaults to time it takes the bucket to leak out to the level of 0
423
+ #
424
+ # _@param_ `adapter` — a compatible adapter
425
+ #
426
+ # _@param_ `leaky_bucket_options` — Options for {Pecorino::LeakyBucket.new}
427
+ #
428
+ # _@see_ `Pecorino::LeakyBucket.new`
429
+ def initialize: (
430
+ key: String,
431
+ ?block_for: Numeric?,
432
+ ?adapter: Pecorino::Adapters::BaseAdapter,
433
+ **untyped leaky_bucket_options
434
+ ) -> void
435
+
436
+ # Tells whether the throttle will let this number of requests pass without raising
437
+ # a Throttled. Note that this is not race-safe. Another request could overflow the bucket
438
+ # after you call `able_to_accept?` but before you call `throttle!`. So before performing
439
+ # the action you still need to call `throttle!`. You may still use `able_to_accept?` to
440
+ # provide better UX to your users before they cause an action that would otherwise throttle.
441
+ #
442
+ # _@param_ `n_tokens`
443
+ def able_to_accept?: (?Float n_tokens) -> bool
444
+
445
+ # Register that a request is being performed. Will raise Throttled
446
+ # if there is a block in place for that throttle, or if the bucket cannot accept
447
+ # this fillup and the block has just been installed as a result of this particular request.
448
+ #
449
+ # The exception can be rescued later to provide a 429 response. This method is better
450
+ # to use before performing the unit of work that the throttle is guarding:
451
+ #
452
+ # If the method call returns it means that the request is not getting throttled.
453
+ #
454
+ # _@param_ `n` — how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
455
+ #
456
+ # _@return_ — the state of the throttle after filling up the leaky bucket / trying to pass the block
457
+ #
458
+ # ```ruby
459
+ # begin
460
+ # t.request!
461
+ # Note.create!(note_params)
462
+ # rescue Pecorino::Throttle::Throttled => e
463
+ # [429, {"Retry-After" => e.retry_after.to_s}, []]
464
+ # end
465
+ # ```
466
+ def request!: (?Numeric n) -> State
467
+
468
+ # Register that a request is being performed. Will not raise any exceptions but return
469
+ # the time at which the block will be lifted if a block resulted from this request or
470
+ # was already in effect. Can be used for registering actions which already took place,
471
+ # but should result in subsequent actions being blocked.
472
+ #
473
+ # _@param_ `n` — how many tokens to place into the bucket or remove from the bucket. May be fractional or negative.
474
+ #
475
+ # _@return_ — the state of the throttle after the attempt to fill up the leaky bucket
476
+ #
477
+ # ```ruby
478
+ # if t.able_to_accept?
479
+ # Entry.create!(entry_params)
480
+ # t.request
481
+ # end
482
+ # ```
483
+ def request: (?Numeric n) -> State
484
+
485
+ # Fillup the throttle with 1 request and then perform the passed block. This is useful to perform actions which should
486
+ # be rate-limited - alerts, calls to external services and the like. If the call is allowed to proceed,
487
+ # the passed block will be executed. If the throttle is in the blocked state or if the call puts the throttle in
488
+ # the blocked state the block will not be executed
489
+ #
490
+ # _@param_ `blk` — The block to run. Will only run if the throttle accepts the call.
491
+ #
492
+ # _@return_ — the return value of the block if the block gets executed, or `nil` if the call got throttled
493
+ #
494
+ # ```ruby
495
+ # t.throttled { Slack.alert("Things are going wrong") }
496
+ # ```
497
+ def throttled: () -> Object
498
+
499
+ # The key for that throttle. Each key defines a unique throttle based on either a given name or
500
+ # discriminators. If there is a component you want to key your throttle by, include it in the
501
+ # `key` keyword argument to the constructor, like `"t-ip-#{your_rails_request.ip}"`
502
+ attr_reader key: String
503
+
504
+ # The state represents a snapshot of the throttle state in time
505
+ class State
506
+ # sord omit - no YARD type given for "blocked_until", using untyped
507
+ def initialize: (untyped blocked_until) -> void
508
+
509
+ # Tells whether this throttle still is in the blocked state.
510
+ # If the `blocked_until` value lies in the past, the method will
511
+ # return `false` - this is done so that the `State` can be cached.
512
+ def blocked?: () -> bool
513
+
514
+ attr_reader blocked_until: Time
515
+ end
516
+
517
+ # {Pecorino::Throttle} will raise this exception from `request!`. The exception can be used
518
+ # to do matching, for setting appropriate response headers, and for distinguishing between
519
+ # multiple different throttles.
520
+ class Throttled < StandardError
521
+ # sord omit - no YARD type given for "from_throttle", using untyped
522
+ # sord omit - no YARD type given for "state", using untyped
523
+ def initialize: (untyped from_throttle, untyped state) -> void
524
+
525
+ # Returns the `retry_after` value in seconds, suitable for use in an HTTP header
526
+ def retry_after: () -> Integer
527
+
528
+ # Returns the throttle which raised the exception. Can be used to disambiguiate between
529
+ # multiple Throttled exceptions when multiple throttles are applied in a layered fashion:
530
+ #
531
+ # ```ruby
532
+ # begin
533
+ # ip_addr_throttle.request!
534
+ # user_email_throttle.request!
535
+ # db_insert_throttle.request!(n_items_to_insert)
536
+ # rescue Pecorino::Throttled => e
537
+ # deliver_notification(user) if e.throttle == user_email_throttle
538
+ # firewall.ban_ip(ip) if e.throttle == ip_addr_throttle
539
+ # end
540
+ # ```
541
+ attr_reader throttle: Throttle
542
+
543
+ # Returns the throttle state based on which the exception is getting raised. This can
544
+ # be used for caching the exception, because the state can tell when the block will be
545
+ # lifted. This can be used to shift the throttle verification into a faster layer of the
546
+ # system (like a blocklist in a firewall) or caching the state in an upstream cache. A block
547
+ # in Pecorino is set once and is active until expiry. If your service is under an attack
548
+ # and you know that the call is blocked until a certain future time, the block can be
549
+ # lifted up into a faster/cheaper storage destination, like Rails cache:
550
+ #
551
+ # ```ruby
552
+ # begin
553
+ # ip_addr_throttle.request!
554
+ # rescue Pecorino::Throttled => e
555
+ # firewall.ban_ip(request.ip, ttl_seconds: e.state.retry_after)
556
+ # render :rate_limit_exceeded
557
+ # end
558
+ # ```
559
+ #
560
+ # ```ruby
561
+ # state = Rails.cache.read(ip_addr_throttle.key)
562
+ # return render :rate_limit_exceeded if state && state.blocked? # No need to call Pecorino for this
563
+ #
564
+ # begin
565
+ # ip_addr_throttle.request!
566
+ # rescue Pecorino::Throttled => e
567
+ # Rails.cache.write(ip_addr_throttle.key, e.state, expires_in: (e.state.blocked_until - Time.now))
568
+ # render :rate_limit_exceeded
569
+ # end
570
+ # ```
571
+ attr_reader state: Throttle::State
572
+ end
573
+ end
574
+
575
+ # This offers just the leaky bucket implementation with fill control, but without the timed lock.
576
+ # It does not raise any exceptions, it just tracks the state of a leaky bucket in the database.
577
+ #
578
+ # Leak rate is specified directly in tokens per second, instead of specifying the block period.
579
+ # The bucket level is stored and returned as a Float which allows for finer-grained measurement,
580
+ # but more importantly - makes testing from the outside easier.
581
+ #
582
+ # Note that this implementation has a peculiar property: the bucket is only "full" once it overflows.
583
+ # Due to a leak rate just a few microseconds after that moment the bucket is no longer going to be full
584
+ # anymore as it will have leaked some tokens by then. This means that the information about whether a
585
+ # bucket has become full or not gets returned in the bucket `State` struct right after the database
586
+ # update gets executed, and if your code needs to make decisions based on that data it has to use
587
+ # this returned state, not query the leaky bucket again. Specifically:
588
+ #
589
+ # state = bucket.fillup(1) # Record 1 request
590
+ # state.full? #=> true, this is timely information
591
+ #
592
+ # ...is the correct way to perform the check. This, however, is not:
593
+ #
594
+ # bucket.fillup(1)
595
+ # bucket.state.full? #=> false, some time has passed after the topup and some tokens have already leaked
596
+ #
597
+ # The storage use is one DB row per leaky bucket you need to manage (likely - one throttled entity such
598
+ # as a combination of an IP address + the URL you need to procect). The `key` is an arbitrary string you provide.
599
+ class LeakyBucket
600
+ # sord duck - #to_f looks like a duck type, replacing with untyped
601
+ # Creates a new LeakyBucket. The object controls 1 row in the database is
602
+ # specific to the bucket key.
603
+ #
604
+ # _@param_ `key` — the key for the bucket. The key also gets used to derive locking keys, so that operations on a particular bucket are always serialized.
605
+ #
606
+ # _@param_ `leak_rate` — the leak rate of the bucket, in tokens per second. Either `leak_rate` or `over_time` can be used, but not both.
607
+ #
608
+ # _@param_ `over_time` — over how many seconds the bucket will leak out to 0 tokens. The value is assumed to be the number of seconds - or a duration which returns the number of seconds from `to_f`. Either `leak_rate` or `over_time` can be used, but not both.
609
+ #
610
+ # _@param_ `capacity` — how many tokens is the bucket capped at. Filling up the bucket using `fillup()` will add to that number, but the bucket contents will then be capped at this value. So with bucket_capacity set to 12 and a `fillup(14)` the bucket will reach the level of 12, and will then immediately start leaking again.
611
+ #
612
+ # _@param_ `adapter` — a compatible adapter
613
+ def initialize: (
614
+ key: String,
615
+ capacity: Numeric,
616
+ ?adapter: Pecorino::Adapters::BaseAdapter,
617
+ ?leak_rate: Float?,
618
+ ?over_time: untyped
619
+ ) -> void
620
+
621
+ # Places `n` tokens in the bucket. If the bucket has less capacity than `n` tokens, the bucket will be filled to capacity.
622
+ # If the bucket has less capacity than `n` tokens, it will be filled to capacity. If the bucket is already full
623
+ # when the fillup is requested, the bucket stays at capacity.
624
+ #
625
+ # Once tokens are placed, the bucket is set to expire within 2 times the time it would take it to leak to 0,
626
+ # regardless of how many tokens get put in - since the amount of tokens put in the bucket will always be capped
627
+ # to the `capacity:` value you pass to the constructor.
628
+ #
629
+ # _@param_ `n_tokens` — How many tokens to fillup by
630
+ #
631
+ # _@return_ — the state of the bucket after the operation
632
+ def fillup: (Float n_tokens) -> State
633
+
634
+ # Places `n` tokens in the bucket. If the bucket has less capacity than `n` tokens, the fillup will be rejected.
635
+ # This can be used for "exactly once" semantics or just more precise rate limiting. Note that if the bucket has
636
+ # _exactly_ `n` tokens of capacity the fillup will be accepted.
637
+ #
638
+ # Once tokens are placed, the bucket is set to expire within 2 times the time it would take it to leak to 0,
639
+ # regardless of how many tokens get put in - since the amount of tokens put in the bucket will always be capped
640
+ # to the `capacity:` value you pass to the constructor.
641
+ #
642
+ # _@param_ `n_tokens` — How many tokens to fillup by
643
+ #
644
+ # _@return_ — the state of the bucket after the operation and whether the operation succeeded
645
+ #
646
+ # ```ruby
647
+ # withdrawals = LeakyBuket.new(key: "wallet-#{user.id}", capacity: 200, over_time: 1.day)
648
+ # if withdrawals.fillup_conditionally(amount_to_withdraw).accepted?
649
+ # user.wallet.withdraw(amount_to_withdraw)
650
+ # else
651
+ # raise "You need to wait a bit before withdrawing more"
652
+ # end
653
+ # ```
654
+ def fillup_conditionally: (Float n_tokens) -> ConditionalFillupResult
655
+
656
+ # Returns the current state of the bucket, containing the level and whether the bucket is full.
657
+ # Calling this method will not perform any database writes.
658
+ #
659
+ # _@return_ — the snapshotted state of the bucket at time of query
660
+ def state: () -> State
661
+
662
+ # Tells whether the bucket can accept the amount of tokens without overflowing.
663
+ # Calling this method will not perform any database writes. Note that this call is
664
+ # not race-safe - another caller may still overflow the bucket. Before performing
665
+ # your action, you still need to call `fillup()` - but you can preemptively refuse
666
+ # a request if you already know the bucket is full.
667
+ #
668
+ # _@param_ `n_tokens`
669
+ def able_to_accept?: (Float n_tokens) -> bool
670
+
671
+ # sord omit - no YARD type given for :key, using untyped
672
+ # The key (name) of the leaky bucket
673
+ # @return [String]
674
+ attr_reader key: untyped
675
+
676
+ # sord omit - no YARD type given for :leak_rate, using untyped
677
+ # The leak rate (tokens per second) of the bucket
678
+ # @return [Float]
679
+ attr_reader leak_rate: untyped
680
+
681
+ # sord omit - no YARD type given for :capacity, using untyped
682
+ # The capacity of the bucket in tokens
683
+ # @return [Float]
684
+ attr_reader capacity: untyped
685
+
686
+ # Returned from `.state` and `.fillup`
687
+ class State
688
+ # sord omit - no YARD type given for "level", using untyped
689
+ # sord omit - no YARD type given for "is_full", using untyped
690
+ def initialize: (untyped level, untyped is_full) -> void
691
+
692
+ # Tells whether the bucket was detected to be full when the operation on
693
+ # the LeakyBucket was performed.
694
+ def full?: () -> bool
695
+
696
+ # Returns the level of the bucket
697
+ attr_reader level: Float
698
+ end
699
+
700
+ # Same as `State` but also communicates whether the write has been permitted or not. A conditional fillup
701
+ # may refuse a write if it would make the bucket overflow
702
+ class ConditionalFillupResult < Pecorino::LeakyBucket::State
703
+ # sord omit - no YARD type given for "level", using untyped
704
+ # sord omit - no YARD type given for "is_full", using untyped
705
+ # sord omit - no YARD type given for "accepted", using untyped
706
+ def initialize: (untyped level, untyped is_full, untyped accepted) -> void
707
+
708
+ # Tells whether the bucket did accept the requested fillup
709
+ def accepted?: () -> bool
710
+ end
711
+ end
712
+
713
+ # The cached throttles can be used when you want to lift your throttle blocks into
714
+ # a higher-level cache. If you are dealing with clients which are hammering on your
715
+ # throttles a lot, it is useful to have a process-local cache of the timestamp when
716
+ # the blocks that are set are going to expire. If you are running, say, 10 web app
717
+ # containers - and someone is hammering at an endpoint which starts blocking -
718
+ # you don't really need to query your DB for every request. The first request indicated
719
+ # as "blocked" by Pecorino can write a cache entry into a shared in-memory table,
720
+ # and all subsequent calls to the same process can reuse that `blocked_until` value
721
+ # to quickly refuse the request
722
+ class CachedThrottle
723
+ # sord warn - ActiveSupport::Cache::Store wasn't able to be resolved to a constant in this project
724
+ # _@param_ `cache_store` — the store for the cached blocks. We recommend a MemoryStore per-process.
725
+ #
726
+ # _@param_ `throttle` — the throttle to cache
727
+ def initialize: (ActiveSupport::Cache::Store cache_store, Pecorino::Throttle throttle) -> void
728
+
729
+ # sord omit - no YARD type given for "n", using untyped
730
+ # sord omit - no YARD return type given, using untyped
731
+ # Increments the cached throttle by the given number of tokens. If there is currently a known cached block on that throttle
732
+ # an exception will be raised immediately instead of querying the actual throttle data. Otherwise the call gets forwarded
733
+ # to the underlying throttle.
734
+ #
735
+ # _@see_ `Pecorino::Throttle#request!`
736
+ def request!: (?untyped n) -> untyped
737
+
738
+ # sord omit - no YARD type given for "n", using untyped
739
+ # sord omit - no YARD return type given, using untyped
740
+ # Returns the cached `state` for the throttle if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
741
+ #
742
+ # _@see_ `Pecorino::Throttle#request!`
743
+ def request: (?untyped n) -> untyped
744
+
745
+ # sord omit - no YARD type given for "n", using untyped
746
+ # Returns `false` if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
747
+ #
748
+ # _@see_ `Pecorino::Throttle#able_to_accept?`
749
+ def able_to_accept?: (?untyped n) -> bool
750
+
751
+ # sord omit - no YARD return type given, using untyped
752
+ # Does not run the block if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
753
+ #
754
+ # _@see_ `Pecorino::Throttle#throttled`
755
+ def throttled: () -> untyped
756
+
757
+ # sord omit - no YARD return type given, using untyped
758
+ # Returns the key of the throttle
759
+ #
760
+ # _@see_ `Pecorino::Throttle#key`
761
+ def key: () -> untyped
762
+
763
+ # sord omit - no YARD return type given, using untyped
764
+ # Returns `false` if there is a currently active block for that throttle in the cache. Otherwise forwards to underlying throttle.
765
+ #
766
+ # _@see_ `Pecorino::Throttle#able_to_accept?`
767
+ def state: () -> untyped
768
+
769
+ # sord omit - no YARD type given for "state", using untyped
770
+ # sord omit - no YARD return type given, using untyped
771
+ def write_cache_blocked_state: (untyped state) -> untyped
772
+
773
+ # sord omit - no YARD return type given, using untyped
774
+ def read_cached_blocked_state: () -> untyped
775
+ end
776
+
777
+ #
778
+ # Rails generator used for setting up Pecorino in a Rails application.
779
+ # Run it with +bin/rails g pecorino:install+ in your console.
780
+ class InstallGenerator < Rails::Generators::Base
781
+ include ActiveRecord::Generators::Migration
782
+ TEMPLATES: untyped
783
+
784
+ # sord omit - no YARD return type given, using untyped
785
+ # Generates monolithic migration file that contains all database changes.
786
+ def create_migration_file: () -> untyped
787
+
788
+ # sord omit - no YARD return type given, using untyped
789
+ def migration_version: () -> untyped
790
+ end
791
+ end
metadata CHANGED
@@ -1,11 +1,10 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: pecorino
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.7.2
4
+ version: 0.7.3
5
5
  platform: ruby
6
6
  authors:
7
7
  - Julik Tarkhanov
8
- autorequire:
9
8
  bindir: exe
10
9
  cert_chain: []
11
10
  date: 2025-04-01 00:00:00.000000000 Z
@@ -35,8 +34,10 @@ files:
35
34
  - ".github/workflows/ci.yml"
36
35
  - ".gitignore"
37
36
  - ".standard.yml"
37
+ - ".yardopts"
38
38
  - CHANGELOG.md
39
39
  - CODE_OF_CONDUCT.md
40
+ - Gemfile
40
41
  - LICENSE.txt
41
42
  - README.md
42
43
  - Rakefile
@@ -59,6 +60,7 @@ files:
59
60
  - lib/pecorino/version.rb
60
61
  - pecorino.gemspec
61
62
  - rbi/pecorino.rbi
63
+ - rbi/pecorino.rbs
62
64
  - test/adapters/adapter_test_methods.rb
63
65
  - test/adapters/memory_adapter_test.rb
64
66
  - test/adapters/postgres_adapter_test.rb
@@ -77,7 +79,6 @@ metadata:
77
79
  homepage_uri: https://github.com/cheddar-me/pecorino
78
80
  source_code_uri: https://github.com/cheddar-me/pecorino
79
81
  changelog_uri: https://github.com/cheddar-me/pecorino/CHANGELOG.md
80
- post_install_message:
81
82
  rdoc_options: []
82
83
  require_paths:
83
84
  - lib
@@ -92,8 +93,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
92
93
  - !ruby/object:Gem::Version
93
94
  version: '0'
94
95
  requirements: []
95
- rubygems_version: 3.1.6
96
- signing_key:
96
+ rubygems_version: 3.6.2
97
97
  specification_version: 4
98
98
  summary: Database-based rate limiter using leaky buckets
99
99
  test_files: []