-
Notifications
You must be signed in to change notification settings - Fork 459
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
#switch! always establish connection leads to threads leak #565
Comments
this is found on a delayed_job worker, working off massive queue of jobs. and 10,000 jobs will cause 10,000 threads leaks |
@FLemon We have run into this as well, which in my development environment on macOS actually causes the process to run out of Threads (default max threads seems to be much lower in macOS than linux, but haven't found exactly where that's set). What's worked for us is adding As you can see, setting the frequency to 0 causes the threaded part of the code to be skipped. This is the first time I've really dug into ActiveRecord internals, so I may have some of these details wrong, but this is the explanation I've come up with so far. Anytime Apartment establishes a connection, a new connection pool seems to be created, and the connection pool creates this Reaper thread. There appears to also be a memory leak of connection pools, at least when Another interesting thing to note is that if you have excluded models, another connection pool is created for each one. In our case, we have 15 excluded models, which means every request was creating 17 leaked threads (15 for excluded models, 1 for the elevator switching to the desired tenant, and 1 more when the elevator switched back to the default tenant). That causes the thread count to max out very quickly. Considering these excluded models are all accessing the same "global" database, I would have expected them to share the connection pool, but this does not seem to be the case. The excluded model issue, however, only appears to happen in https://github.com/influitive/apartment/blob/development/lib/apartment/railtie.rb#L29 I looked this config block up, and it says:
https://guides.rubyonrails.org/configuring.html#initialization-events So, assuming I'm correct in that a connection pool is created for every excluded model, and that there is a memory leak of those connection pools, then every request/job in the development environment is leaking an additional N connection pools, where N is the number of excluded models. When reaping is enabled, this also results in N leaked threads. |
@adamkrone although i'm using this gem with config.use_schemas = false, I also experienced the issue with excluded models using one connection for each model. It can also cause issues with associations since Rails tries to create associated records in different transactions/connections and they will fail. Here is one way to fix this, in case it helps someone:
At least with mysql2 + use_schemas = false, this will make those models share the same connection. |
we end up switch to using schemas, that stops AR from creating new thread for a DB switch, as all schemas are inside the same DB |
just to clarify, is this only an issue with |
this is caused by a rails bug which causes a thread leak in the Reaper (as some of you had already understood), when a new connection pool is created. |
Steps to reproduce
top -Hc
Apartment::Tenant.switch('tenant_name') { p '' }
Expected behavior
cache or somehow reuse already established connection when doing
#switch
, to avoid threads leakActual behavior
observe threads increased by 2, one for switching to, one for switching back, regardless the value of switching to and switching back are the same.
System configuration
Database: postgresSQL 9.6
Apartment version: 2.1
Apartment config (in
config/initializers/apartment.rb
or so):use_schemas
:true
Rails (or ActiveRecord) version: 5.1.6
Ruby version: 2.5
The text was updated successfully, but these errors were encountered: