current tenant is lost when creating a new fiber
Closed this issue · 4 comments
Steps to reproduce
Apartment::Tenant.switch! 'db1'
puts Apartment::Tenant.current # prints "db1"
Fiber.new { puts Apartment::Tenant.current }.resume # prints "public"
puts (1..2).lazy.map { Apartment::Tenant.current }.peek # prints "public"
User.limit(2)
.find_each(batch_size: 1)
.lazy
.map { puts Apartment::Tenant.current }.next # prints "public"
puts Apartment::Tenant.current # prints "db1"
Expected behavior
All of the above should print "db1".
Actual behavior
Inside a new fiber, the fiber local variable :apartment_adapter
no longer exists, so it creates a new one, set to the default tenant. Since enumerators are implemented using fibers, some enumerator methods use the default tenant instead of the tenant of the parent fiber.
Apartment::Tenant.adapter
was probably meant to use thread local variables, but Ruby 1.9.2 changed the meaning of Thread#[]
and Thread#[]=
to operate on fiber local variables instead of thread local variables. Instead Thread#thread_variable_get
and Thread#thread_variable_set
should be used.
System configuration
- Database: (Tell us what database and its version you use.)
Postgres 12.6 - Apartment version:
2.9.0 use_schemas
:true
- Rails (or ActiveRecord) version:
6.1.4.4 - Ruby version:
2.7.5
Hey @derikson, I'm not a contributor but I will tell you my opinion.
I'd say it's expected behavior, I faced the same case on my project and we weren't surprised to see a public tenant under a new thread even though it was not public in a parent.
Hi @derikson, I don't have a use case for fibers so for me it's harder to make an informed decision. I do have a use case where I switch the DB connection from a writer to a reader db and I have the apartment setting the same tenant on the new connection.
All of this to say, I do understand your concern and maybe we can make it configurable.
I think what @derikson presented here should be considered a bug according to his code. lazy
shouldn't cause us to lose the current tenant.
@rpbaltazar I'm no longer working on a project that uses this gem, so I no longer have a use case that this solves. If nothing else, this bug report might help someone who is getting data from different tenants mixed together track down the cause.