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 +4 -4
- data/.yardopts +1 -0
- data/CHANGELOG.md +4 -0
- data/Gemfile +17 -0
- data/README.md +13 -12
- data/Rakefile +1 -0
- data/lib/pecorino/cached_throttle.rb +6 -2
- data/lib/pecorino/throttle.rb +12 -11
- data/lib/pecorino/version.rb +1 -1
- data/lib/pecorino.rb +2 -2
- data/rbi/pecorino.rbi +25 -21
- data/rbi/pecorino.rbs +791 -0
- metadata +5 -5
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c0e0c8fa1a6bc734dc645b5b021264eda84bcadd1bec8da65c533a33cd243fb6
|
4
|
+
data.tar.gz: c6c053d8ed02f8e180786d4614c607087ca76b016b8acac35b35a3b3cb544391
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
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-#{
|
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
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
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-#{
|
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-#{
|
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,
|
234
|
-
|
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
|
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
@@ -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?
|
data/lib/pecorino/throttle.rb
CHANGED
@@ -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-#{
|
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
|
104
|
-
# @see
|
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
|
-
#
|
135
|
-
#
|
136
|
+
# t.request!
|
137
|
+
# Note.create!(note_params)
|
136
138
|
# rescue Pecorino::Throttle::Throttled => e
|
137
|
-
#
|
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
|
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?
|
data/lib/pecorino/version.rb
CHANGED
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
|
-
# @
|
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.
|
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
|
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
|
503
|
+
# _@param_ `leaky_bucket_options` — Options for {Pecorino::LeakyBucket.new}
|
507
504
|
#
|
508
|
-
# _@see_ `
|
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
|
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
|
-
#
|
544
|
-
#
|
541
|
+
# t.request!
|
542
|
+
# Note.create!(note_params)
|
545
543
|
# rescue Pecorino::Throttle::Throttled => e
|
546
|
-
#
|
544
|
+
# [429, {"Retry-After" => e.retry_after.to_s}, []]
|
547
545
|
# end
|
548
546
|
# ```
|
549
|
-
sig { params(n:
|
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
|
-
# _@
|
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:
|
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-#{
|
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.
|
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.
|
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: []
|