config.use_schemas = false with MySql causes intermittent incorrect db selection
Closed this issue · 1 comments
Steps to reproduce
One Tenant database on MySql
Multiple tenant-specific databases on different MySql servers (can all be reproduced on localhost)
Run multiple curl
requests to query a tenant-specific table and return a single row. In our case we're simply setting a request HTTP header to match what's in the apartment config and running 4 in parallel (one against each tenant)
config.use_schemas = false
causes this to repro consistently, we have considered config.use_schemas = true
but need to spread our tenants across many servers
export RAILS_MIN_THREADS=1
export RAILS_MAX_THREADS=5
export WEB_CONCURRENCY=0
The controller code:
def show
tenant_bird = TenantBird.find_by name: params[:name]
sleep_time = rand(250..3000) / 1000.0
# the sleep here is to sim. long-running queries
Rails.logger.debug "#{Process.ppid}.#{Process.pid} #{Thread.current['x-tenant-id']} Will sleep for #{sleep_time} seconds"
ActiveRecord::Base.connection.exec_query("select sleep(#{sleep_time})")
render json: { slept: sleep_time, bird: tenant_bird, process: { pid: Process.pid, ppid: Process.ppid } }
end
default: &default
adapter: mysql2
encoding: utf8
timeout: 30_000
reconnect: false
pool: 100
username: root
password: password
host: localhost
database: router
development:
<<: *default
Expected behavior
Host and database is properly selected and configured for the request.
Actual behavior
Occasional
Apartment::TenantNotFound (Error while connecting to tenant t_103: No connection pool with 'primary' found.):
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:268:in `raise_connect_error!'
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:189:in `rescue in connect_to_new'
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:178:in `connect_to_new'
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:76:in `block in switch!'
activesupport (5.2.6) lib/active_support/callbacks.rb:98:in `run_callbacks'
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:75:in `switch!'
ros-apartment (2.10.0) lib/apartment/adapters/abstract_adapter.rb:88:in `switch'
Now and again we'll see issues connecting to the default / Tenant database too
ActiveRecord::ConnectionNotEstablished (No connection pool with 'Tenant' found.):
↳ config/initializers/apartment.rb:42
↳ /Users/michaeljon/.rvm/gems/ruby-2.6.7@mw/gems/activerecord-5.2.6/lib/active_record/log_subscriber.rb:98
ActiveRecord::ConnectionNotEstablished (No connection pool with 'Tenant' found.):
config/initializers/apartment.rb:42:in `block in <main>'
Looking for tenant by organization_id 102```
## System configuration
* Database: (Tell us what database and its version you use.) MySql 8.0
* Apartment version: 2.10.0
* Apartment config (in `config/initializers/apartment.rb` or so):
```ruby
# frozen_string_literal: true
require 'apartment/elevators/generic'
Apartment.configure do |config|
config.excluded_models = %w[Tenant]
config.tenant_names = lambda {
names = {}
tenants = Tenant.all.each do |t|
names[t.name] ||= {
adapter: 'mysql2',
encoding: 'utf8',
pool: 100,
timeout: 30_000,
reconnect: false,
username: 'root',
password: 'password',
host: t.host,
database: t.db
}
end
names
}
config.use_schemas = false
config.prepend_environment = false
config.active_record_log = false
end
Rails.application.config.middleware.use Apartment::Elevators::Generic, lambda { |request|
tenant_id = request.cookies['tenant_id']
if tenant_id.nil?
Rails.logger.debug 'Request made without tenant_id'
return nil
end
Rails.logger.debug "Looking for tenant by organization_id #{tenant_id}"
tenant = Tenant.find_by organization_id: tenant_id
if tenant.nil?
Rails.logger.debug "Unable to locate tenant by organization_id #{tenant_id}"
return nil
end
Rails.logger.debug "Found tenant #{tenant_id} named #{tenant.name} at #{tenant.host}.#{tenant.db}"
tenant.name
}
-
Rails (or ActiveRecord) version: 5.2.6
-
Ruby version: 2.6.7
Hi @michaeljon ,
thank you for reporting. we don't actively use the mysql adapter so it's harder for me to identify these issues.
Since you have a very clear failing scenario described, could i ask you to create a repository with that failing scenario and how to reproduce the error in that repo? I could then try to help figuring out where the source of the problem is