Skip to content

Commit d842d1f

Browse files
authored
Merge pull request #184 from esl/throttle_api
Throttle api
2 parents 360c3a3 + 92ef315 commit d842d1f

15 files changed

+1065
-604
lines changed

.github/workflows/ci.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ jobs:
1313
strategy:
1414
matrix:
1515
otp_vsn: ['27', '26', '25']
16-
rebar_vsn: ['3.23.0']
16+
rebar_vsn: ['3.24.0']
1717
test-type: ['regular', 'integration']
1818
runs-on: 'ubuntu-24.04'
1919
steps:

guides/coordinator.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
## API
22

3-
See `amoc_coordinator`.
3+
See `m:amoc_coordinator`.
44

55
## Description
66

guides/telemetry.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@ This event is raised only on the master node.
6262

6363
```erlang
6464
event_name: [amoc, throttle, rate]
65-
measurements: #{rate := non_neg_integer()}
65+
measurements: #{rate := rate(), interval := interval()}
6666
metadata: #{monotonic_time := integer(), name := atom(), msg => binary()}
6767
```
6868

guides/throttle.md

+23-16
Original file line numberDiff line numberDiff line change
@@ -1,18 +1,22 @@
11
## API
22

3-
See `amoc_throttle`
3+
See `m:amoc_throttle`.
44

55
## Overview
66

77
Amoc throttle is a module that allows limiting the number of users' actions per given interval, no matter how many users there are in a test.
8-
It works in both local and distributed environments, allows for dynamic rate changes during a test and exposes metrics which show the number of requests and executions.
8+
It works in both local and distributed environments, allows for dynamic rate changes during a test and exposes telemetry events showing the number of requests and executions.
99

10-
Amoc throttle allows setting the execution `Rate` per `Interval` or limiting the number of parallel executions when `Interval` is set to `0`.
11-
Each `Rate` is identified with a `Name`.
12-
The rate limiting mechanism allows responding to a request only when it does not exceed the given `Rate`.
13-
Amoc throttle makes sure that the given `Rate` per `Interval` is maintained on a constant level.
10+
Amoc throttle allows to:
11+
12+
- Setting the execution `Rate` per `Interval`, or inversely, the `Interarrival` time between actions.
13+
- Limiting the number of parallel executions when `interval` is set to `0`.
14+
15+
Each throttle is identified with a `Name`.
16+
The rate limiting mechanism allows responding to a request only when it does not exceed the given throttle.
17+
Amoc throttle makes sure that the given throttle is maintained on a constant level.
1418
It prevents bursts of executions which could blurry the results, as they technically produce a desired rate in a given interval.
15-
Because of that, it may happen that the actual `Rate` would be slightly below the demanded rate. However, it will never be exceeded.
19+
Because of that, it may happen that the actual throttle rate would be slightly below the demanded rate. However, it will never be exceeded.
1620

1721
## Examples
1822

@@ -42,18 +46,21 @@ user_loop(Id) ->
4246
user_loop(Id).
4347
```
4448
Here a system should be under a continuous load of 100 messages per minute.
45-
Note that if we used something like `amoc_throttle:run(messages_rate, fun() -> send_message(Id) end)` instead of `amoc_throttle:send_and_wait/2` the system would be flooded with requests.
49+
Note that if we used something like `amoc_throttle:run(messages_rate, fun() -> send_message(Id) end)` instead of `amoc_throttle:wait/1` the system would be flooded with requests.
4650

4751
A test may of course be much more complicated.
4852
For example it can have the load changing in time.
4953
A plan for that can be set for the whole test in `init/1`:
5054
```erlang
5155
init() ->
52-
%% init metrics
5356
amoc_throttle:start(messages_rate, 100),
5457
%% 9 steps of 100 increases in Rate, each lasting one minute
55-
amoc_throttle:change_rate_gradually(messages_rate, 100, 1000, 60000, 60000, 9),
56-
ok.
58+
Gradual = #{from_rate => 100,
59+
to_rate => 1000,
60+
step_count => 9,
61+
step_size => 100,
62+
step_interval => timer:minutes(1)},
63+
amoc_throttle:change_rate_gradually(messages_rate, Gradual).
5764
```
5865

5966
Normal Erlang messages can be used to schedule tasks for users by themselves or by some controller process.
@@ -97,13 +104,13 @@ For a more comprehensive example please refer to the `throttle_test` scenario, w
97104
- `amoc_throttle_controller.erl` - a gen_server which is responsible for reacting to requests, and managing `throttle_processes`.
98105
In a distributed environment an instance of `throttle_controller` runs on every node, and the one running on the master Amoc node stores the state for all nodes.
99106
- `amoc_throttle_process.erl` - gen_server module, implements the logic responsible for limiting the rate.
100-
For every `Name`, a `NoOfProcesses` are created, each responsible for keeping executions at a level proportional to their part of `Rate`.
107+
For every `Name`, a number of processes are created, each responsible for keeping executions at a level proportional to their part of the throttle.
101108

102109
### Distributed environment
103110

104111
#### Metrics
105-
In a distributed environment every Amoc node with a throttle started, exposes metrics showing the numbers of requests and executions.
106-
Those exposed by the master node show the sum of all metrics from all nodes.
112+
In a distributed environment every Amoc node with a throttle started, exposes telemetry events showing the numbers of requests and executions.
113+
Those exposed by the master node show the aggregate of all telemetry events from all nodes.
107114
This allows to quickly see the real rates across the whole system.
108115

109116
#### Workflow
@@ -112,12 +119,12 @@ Then a runner process is spawned on the same node.
112119
Its task will be to execute `Fun` asynchronously.
113120
A random throttle process which is assigned to the `Name` is asked for a permission for asynchronous runner to execute `Fun`.
114121
When the request reaches the master node, where throttle processes reside, the request metric on the master node is updated and the throttle process which got the request starts monitoring the asynchronous runner process.
115-
Then, depending on the system's load and the current rate of executions, the asynchronous runner is allowed to run the `Fun` or compelled to wait, because executing the function would exceed the calculated `Rate` in an `Interval`.
122+
Then, depending on the system's load and the current rate of executions, the asynchronous runner is allowed to run the `Fun` or compelled to wait, because executing the function would exceed the calculated throttle.
116123
When the rate finally allows it, the asynchronous runner gets the permission to run the function from the throttle process.
117124
Both processes increase the metrics which count executions, but for each the metric is assigned to their own node.
118125
Then the asynchronous runner tries to execute `Fun`.
119126
It may succeed or fail, either way it dies and an `'EXIT'` signal is sent to the throttle process.
120-
This way it knows that the execution of a task has ended, and can allow a different process to run its task connected to the same `Name` if the current `Rate` allows it.
127+
This way it knows that the execution of a task has ended, and can allow a different process to run its task connected to the same `Name` if the current throttle allows it.
121128

122129
Below is a graph showing the communication between processes on different nodes described above.
123130
![amoc_throttle_dist](assets/amoc_throttle_dist.svg)

rebar.config

+4-4
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
]}.
55

66
{deps, [
7-
{telemetry, "1.2.1"}
7+
{telemetry, "1.3.0"}
88
]}.
99

1010
{profiles, [
@@ -14,10 +14,10 @@
1414
{meck, "0.9.2"},
1515
{proper, "1.4.0"},
1616
{bbmustache, "1.12.2"},
17-
{wait_helper, "0.2.0"}
17+
{wait_helper, "0.2.1"}
1818
]}
1919
]},
20-
{elvis, [{plugins, [{rebar3_lint, "3.2.3"}]}]}
20+
{elvis, [{plugins, [{rebar3_lint, "3.2.6"}]}]}
2121
]}.
2222

2323
{relx, [
@@ -62,7 +62,7 @@
6262
{'guides/amoc_livebook.livemd', #{title => <<"Livebook tutorial">>}},
6363
{'LICENSE', #{title => <<"License">>}}
6464
]},
65-
{assets, <<"guides/assets">>},
65+
{assets, #{<<"guides/assets">> => <<"assets">>}},
6666
{main, <<"readme">>}
6767
]}.
6868

rebar.lock

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,8 @@
11
{"1.2.0",
2-
[{<<"telemetry">>,{pkg,<<"telemetry">>,<<"1.2.1">>},0}]}.
2+
[{<<"telemetry">>,{pkg,<<"telemetry">>,<<"1.3.0">>},0}]}.
33
[
44
{pkg_hash,[
5-
{<<"telemetry">>, <<"68FDFE8D8F05A8428483A97D7AAB2F268AAFF24B49E0F599FAA091F1D4E7F61C">>}]},
5+
{<<"telemetry">>, <<"FEDEBBAE410D715CF8E7062C96A1EF32EC22E764197F70CDA73D82778D61E7A2">>}]},
66
{pkg_hash_ext,[
7-
{<<"telemetry">>, <<"DAD9CE9D8EFFC621708F99EAC538EF1CBE05D6A874DD741DE2E689C47FEAFED5">>}]}
7+
{<<"telemetry">>, <<"7015FC8919DBE63764F4B4B87A95B7C0996BD539E0D499BE6EC9D7F3875B79E6">>}]}
88
].

0 commit comments

Comments
 (0)