uWSGI provides the ability to dynamically scale the number of running workers
via pluggable algorithms. Use uwsgi --cheaper-algos-list
to get the list
of available algorithms.
To enable cheaper mode add the cheaper = N
option to the uWSGI
configuration file, where N is the minimum number of workers uWSGI can run. The
cheaper
value must be lower than the maximum number of configured workers
(workers
or processes
option).
# set cheaper algorithm to use, if not set default will be used
cheaper-algo = spare
# minimum number of workers to keep at all times
cheaper = 2
# number of workers to spawn at startup
cheaper-initial = 5
# maximum number of workers that can be spawned
workers = 10
# how many workers should be spawned at a time
cheaper-step = 1
This configuration will tell uWSGI to run up to 10 workers under load. If the
app is idle uWSGI will stop workers but it will always leave at least 2 of them
running. With cheaper-initial
you can control how many workers should be
spawned at startup. If your average load requires more than minimum number of
workers you can have them spawned right away and then "cheaped" (killed off) if
load is low enough. When the cheaper algorithm decides that it needs more
workers it will spawn cheaper-step
of them. This is useful if you have a
high maximum number of workers -- in the event of a sudden load spike it would
otherwise take a lot of time to spawn enough workers one by one.
Note: Worker is notified by uWSGI before being cheaped. Worker should finish until
timeout is reached (this is configured by worker-reload-mercy
configuration parameter).
Otherwise, uWSGI kills the worker. Killing worker in the middle of serving request can cause
errors or partial responses to the client.
Starting with 1.9.16 rss memory limits can be set to stop cheaper spawning new workers if process count limit was not reached, but total sum of rss memory used by all workers reached given limit.
# soft limit will prevent cheaper from spawning new workers
# if workers total rss memory is equal or higher
# we use 128MB soft limit below (values are in bytes)
cheaper-rss-limit-soft = 134217728
# hard limit will force cheaper to cheap single worker
# if workers total rss memory is equal or higher
# we use 160MB hard limit below (values are in bytes)
cheaper-rss-limit-hard = 167772160
Notes:
- Hard limit is optional, soft limit alone can be used.
- Hard value must be higher then soft value, both values shouldn't be too close to each other.
- Hard value should be soft + at least average worker memory usage for given app.
- Soft value is the limiter for cheaper, it won't spawn more workers, but already running workers memory usage might grow, to handle that reload-on-rss can be set too. To set unbreakable barrier for app memory usage cgroups are recommended.
This is the default algorithm. If all workers are busy for
cheaper_overload
seconds then uWSGI will spawn new workers. When the load
is gone it will begin stopping processes one at a time.
This algorithm is similar to spare, but suitable for large scale by increase workers faster and decrease workers slower.
When number of idle workers is smaller than number specified by cheaper
, it spawns (cheaper
-
number of idle workers) workers. Maximum workers spawned at once can be limited by cheaper-step
.
For example, cheaper
is 4, there are 2 idle workers and cheaper-step
is 1, it spawns 1 worker.
When number of idle workers is larger than cheaper
, it increments internal counter.
When number of idle workers is smaller than or equal to cheaper
, reset the counter.
When the counter is equal to cheaper-idle
, cheap one worker and reset the counter.
Sample configuration:
workers = 64 # maximum number of workers
cheaper-algo = spare2
cheaper = 8 # tries to keep 8 idle workers
cheaper-initial = 8 # starts with minimal workers
cheaper-step = 4 # spawn at most 4 workers at once
cheaper-idle = 60 # cheap one worker per minute while idle
Note
backlog
is only available on Linux and only on TCP sockets (not UNIX domain sockets).
If the socket's listen queue has more than cheaper_overload
requests
waiting to be processed, uWSGI will spawn new workers. If the backlog is lower
it will begin killing processes one at a time.
Note
This algorithm is optional, it is only available if the cheaper_busyness
plugin is compiled and loaded.
This plugin implements an algorithm which adds or removes workers based on average utilization for a given time period. It's goal is to keep more workers than the minimum needed available at any given time, so the app will always have capacity for new requests. If you want to run only minimum number of workers then use the spare or backlog algorithms.
This plugin primarily is used because the way spare and backlog plugins work
causes very aggressive scaling behavior. If you set a low cheaper
value
(for example 1), then uWSGI will keep only 1 worker running and spawn new
workers only when that running worker is overloaded. If an app requires more
workers, then uWSGI will be spawning and stopping workers all the time. Only
during times of very low load the would the minimum number of workers be
enough.
The Busyness algorithm tries to do the opposite: spawn as many workers as needed and stop some of them only when there is a good chance that they are not needed. This should lead to a more stable worker count and much less respawns. Since for most of the time we have more worker capacity than actually needed, average application response times should be lower than with other plugins.
Options:
Specifies the window, in seconds, for tracking the average busyness of workers. Example:
cheaper-overload = 30
This option will check busyness every 30 seconds. If during the last 30 seconds
all workers were running for 3 seconds and idle for the remaining 27 seconds
the calculated busyness will be 10% (3/30). This value will decide how fast
uWSGI can respond to load spikes. New workers will be spawned at most every
cheaper-overload
seconds (unless you are running uWSGI on Linux -- see
cheaper-busyness-backlog-alert
for details).
If you want to react to load spikes faster, keep this value low so busyness is
calculated more often. Keep in mind this may cause workers to be
started/stopped more often than required since every minor spike may spawn new
workers. With a high cheaper-overload
value the worker count will change
much less since longer cycles will eat all short spikes of load and extreme
values.
Default is 3, for busyness plugin it's best to use higher value (10-30).
How many workers to spawn when the algorithm decides they are needed. Default is 1.
The number of workers to be started when starting the application. After the app is started the algorithm can stop or start workers if needed.
This is the maximum busyness we allow. Every time the calculated busyness for
last cheaper-overload
seconds is higher than this value, uWSGI will spawn
cheaper-step
new workers. Default is 50.
This is minimum busyness. If current busyness is below this value, the app is considered as being in an "idle cycl" and uWSGI will start counting them. Once we reach needed number of idle cycles uWSGI will kill one worker. Default is 25.
This option tells uWSGI how many idle cycles we need before stopping a worker. After reaching this limit uWSGI will stop a worker and reset this counter.
For example:
cheaper-overload = 10
cheaper-busyness-multiplier = 20
cheaper-busyness-min = 25
If average worker busyness is under 25% for 20 checks in a row, executed every
10 seconds (total of 200 seconds), one worker will be stopped. The idle cycles
counter will be reset if average busyness jumps above cheaper-busyness-max
and we spawn new workers. If during idle cycle counting the average busyness
jumps above cheaper-busyness-min
but still below cheaper-busyness-max
,
then the idle cycles counter is adjusted and we need to wait extra one idle
cycle. If during idle cycle counting the average busyness jumps above
cheaper-busyness-min
but still below cheaper-busyness-max
three times
in a row, then the idle cycle counter is reset.
uWSGI will automatically tune the number of idle cycles needed to stop worker
when worker is stopped due to enough idle cycles and then spawned back to fast
(less than the same time we need to cheap worker), then we will increment the
cheaper-busyness-multiplier
value this value. Default is 1.
Example:
cheaper-overload = 10
cheaper-busyness-multiplier = 20
cheaper-busyness-min = 25
cheaper-busyness-penalty = 2
If average worker busyness is under 25% for 20 checks in a row, executed every
10 seconds (total 200 seconds), one worker will be stopped. If new worker is
spawned in less than 200 seconds (counting from the time when we spawned the
last worker before it), the cheaper-busyness-multiplier
value will be
incremented up to 22 (20+2). Now we will need to wait 220 seconds (22*10) to
cheap another worker. This option is used to prevent workers from being
started and stopped all the time since once we stop one worker, busyness might
jump up enough to hit cheaper-busyness-max
. Without this, or if tuned
poorly, we can get into a stop/start feedback loop .
This option enables debug logs from the cheaper_busyness
plugin.
This option is only available on Linux. It is used to allow quick response to
load spikes even with high cheaper-overload
values. On every uWSGI master
cycle (default 1 second) the current listen queue is checked. If it is higher
than this value, an emergency worker is spawned. When using this option it is
safe to use high cheaper-overload
values to have smoother scaling of worker
count. Default is 33.
This option is only available on Linux. It works just like
cheaper-busyness-multiplier
, except it is used only for emergency workers
spawned when listen queue was higher than cheaper-busyness-backlog-alert
.
Emergency workers are spawned in case of big load spike to prevent currently running workers from being overloaded. Sometimes load spike are random and short which can spawn a lot of emergency workers. In such cases we would need to wait several cycles before reaping those workers. This provides an alternate multiplier to reap these processes faster. Default is 3.
This option is only available on Linux. It sets the number of emergency workers
spawned when listen queue is higher than cheaper-busyness-backlog-alert
.
Defaults to 1.
This option is only available on Linux. It will spawn new emergency workers if the request listen queue is > 0 for more than N seconds. It is used to protect the server from the corner case where there is only a single worker running and the worker is handling a long running request. If uWSGI receives new requests they would stay in the request queue until that long running request is completed. With this option we can detect such a condition and spawn new worker to prevent queued requests from being timed out. Default is 60.
Experiment with settings, there is no one golden rule of what values should be used for everyone. Test and pick values that are best for you. Monitoring uWSGI stats (via Carbon, for instance) will make it easy to decide on good values.
Don't expect busyness to be constant. it will change frequently. In the end, real users interact with your apps in very random way. It's recommended to use longer --cheaper-overload values (>=30) to have less spikes.
If you want to run some benchmarks with this plugin, you should use tools that add randomness to the work load
With a low number of workers (2-3) starting new worker or stopping one might affect busyness a lot, if You have 2 workers with busyness of 50%, than stopping one of them will increase busyness to 100%. Keep that in mind when picking min and max levels, with only few workers running most of the time max should be more than double of min, otherwise every time one worker is stopped it might increase busyness to above max level.
With a low number of workers (1-4) and default settings expect this plugin will keep average busyness below the minimum level; adjust levels to compensate for this.
With a higher number of workers required to handle load, worker count should stabilize somewhere near minimum busyness level, jumping a little bit around this value
When experimenting with this plugin it is advised to enable
--cheaper-busyness-verbose
to get an idea of what it is doing. An example log follows.# These messages are logged at startup to show current settings [busyness] settings: min=20%, max=60%, overload=20, multiplier=15, respawn penalty=3 [busyness] backlog alert is set to 33 request(s) # With --cheaper-busyness-verbose enabled You can monitor calculated busyness [busyness] worker nr 1 20s average busyness is at 11% [busyness] worker nr 2 20s average busyness is at 11% [busyness] worker nr 3 20s average busyness is at 20% [busyness] 20s average busyness of 3 worker(s) is at 14% # Average busyness is under 20%, we start counting idle cycles # we have overload=20 and multiplier=15 so we need to wait 300 seconds before we can stop worker # cycle we just had was counted as idle so we need to wait another 280 seconds # 1 missing second below is just from rounding, master cycle is every 1 second but it also takes some time, this is normal [busyness] need to wait 279 more second(s) to cheap worker # We waited long enough and we can stop one worker [busyness] worker nr 1 20s average busyness is at 6% [busyness] worker nr 2 20s average busyness is at 22% [busyness] worker nr 3 20s average busyness is at 19% [busyness] 20s average busyness of 3 worker(s) is at 15% [busyness] 20s average busyness is at 15%, cheap one of 3 running workers # After stopping one worker average busyness is now higher, which is no surprise [busyness] worker nr 2 20s average busyness is at 36% [busyness] worker nr 3 20s average busyness is at 24% [busyness] 20s average busyness of 2 worker(s) is at 30% # 30% is above our minimum (20%), but it's still far from our maximum (60%) # since this is not idle cycle uWSGI will ignore it when counting when to stop worker [busyness] 20s average busyness is at 30%, 1 non-idle cycle(s), adjusting cheaper timer # After a while our average busyness is still low enough, so we stop another worker [busyness] 20s average busyness is at 3%, cheap one of 2 running workers # With only one worker running we won't see per worker busyness since it's the same as total average [busyness] 20s average busyness of 1 worker(s) is at 16% [busyness] 20s average busyness of 1 worker(s) is at 17% # Shortly after stopping second worker and with only one running we have load spike that is enough to hit our maximum level # this was just few cycles after stopping worker so uWSGI will increase multiplier # now we need to wait extra 3 cycles before stopping worker [busyness] worker(s) respawned to fast, increasing cheaper multiplier to 18 (+3) # Initially we needed to wait only 300 seconds, now we need to have 360 subsequent seconds when workers busyness is below minimum level # 10*20 + 3*20 = 360 [busyness] worker nr 1 20s average busyness is at 9% [busyness] worker nr 2 20s average busyness is at 17% [busyness] worker nr 3 20s average busyness is at 17% [busyness] worker nr 4 20s average busyness is at 21% [busyness] 20s average busyness of 4 worker(s) is at 16% [busyness] need to wait 339 more second(s) to cheap worker