Skip to content

Commit 0ac43aa

Browse files
authored
Support rails 7.2 (citusdata#239)
* Add rails 7.2 to Appraisals * The key of _reflections is a symbol since Rails 7.2. related: rails/rails#51726 `_reflect_on_association` is also a private API and will may be broken, but it works for both String and Symbol, absorbing type differences, so use this one. * cached_find_by_statement requires two positional arguments in Rails 7.2. related: rails/rails#51725 * build_arel requires `connection` as arguments in Rails 7.2. related: rails/rails#51725 * Introduced #attributes_for_inspect in Rails 7.2 By default, only `id` is output, so set `:all` to output all attributes for testing. related: rails/rails#49765 * Overrides #reset_primary_key to modify @primary_key. ar-multitenant expects that #primary_key returns single column instead of composite primary_keys for backward compatibilities. Previously overwriting #primary_key to return single pk, but since the timing for writing @primary_key has changed since Rails 7.2, this way is no longer available. Instead, overwriting `#reset_primary_key`, which handles calculating `@primary_key`. * fix rubocop related: citusdata#239 (comment)
1 parent 28487fb commit 0ac43aa

File tree

7 files changed

+49
-18
lines changed

7 files changed

+49
-18
lines changed

.github/workflows/active-record-multi-tenant-tests.yml

+3
Original file line numberDiff line numberDiff line change
@@ -47,15 +47,18 @@ jobs:
4747
- '3.0'
4848
- '3.1'
4949
- '3.2'
50+
- '3.3'
5051
appraisal:
5152
- rails-6.0
5253
- rails-6.1
5354
- rails-7.0
5455
- rails-7.1
56+
- rails-7.2
5557
- active-record-6.0
5658
- active-record-6.1
5759
- active-record-7.0
5860
- active-record-7.1
61+
- active-record-7.2
5962
citus_version:
6063
- '10'
6164
- '11'

Appraisals

+8
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,10 @@ appraise 'rails-7.1' do
1616
gem 'rails', '~> 7.1.0'
1717
end
1818

19+
appraise 'rails-7.2' do
20+
gem 'rails', '~> 7.2.0'
21+
end
22+
1923
appraise 'active-record-6.0' do
2024
gem 'activerecord', '~> 6.0.3'
2125
end
@@ -31,3 +35,7 @@ end
3135
appraise 'active-record-7.1' do
3236
gem 'activerecord', '~> 7.1.0'
3337
end
38+
39+
appraise 'active-record-7.2' do
40+
gem 'activerecord', '~> 7.2.0'
41+
end

lib/activerecord-multi-tenant/habtm.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ def has_and_belongs_to_many_with_tenant(name, scope = nil, **options, &extension
1212
# rubocop:enable Naming/PredicateName
1313
has_and_belongs_to_many_without_tenant(name, scope, **options, &extension)
1414

15-
middle_reflection = _reflections[name.to_s].through_reflection
15+
middle_reflection = _reflect_on_association(name.to_s).through_reflection
1616
join_model = middle_reflection.klass
1717

1818
# get tenant_enabled from options and if it is not set, set it to false

lib/activerecord-multi-tenant/model_extensions.rb

+7-12
Original file line numberDiff line numberDiff line change
@@ -42,20 +42,15 @@ def partition_key
4242
.try(:instance_variable_get, :@partition_key)
4343
end
4444

45-
# Avoid primary_key errors when using composite primary keys (e.g. id, tenant_id)
46-
def primary_key
47-
if defined?(PRIMARY_KEY_NOT_SET) ? !PRIMARY_KEY_NOT_SET.equal?(@primary_key) : @primary_key
48-
return @primary_key
49-
end
50-
45+
def reset_primary_key
5146
primary_object_keys = Array.wrap(connection.schema_cache.primary_keys(table_name)) - [partition_key]
5247

53-
@primary_key = if primary_object_keys.size == 1
54-
primary_object_keys.first
55-
elsif table_name &&
56-
connection.schema_cache.columns_hash(table_name).include?(DEFAULT_ID_FIELD)
57-
DEFAULT_ID_FIELD
58-
end
48+
self.primary_key = if primary_object_keys.size == 1
49+
primary_object_keys.first
50+
elsif table_name &&
51+
connection.schema_cache.columns_hash(table_name).include?(DEFAULT_ID_FIELD)
52+
DEFAULT_ID_FIELD
53+
end
5954
end
6055

6156
def inherited(subclass)

lib/activerecord-multi-tenant/query_rewriter.rb

+11-4
Original file line numberDiff line numberDiff line change
@@ -375,11 +375,18 @@ def relations_from_node_join(node_join)
375375
ActiveRecord::QueryMethods.prepend(MultiTenant::QueryMethodsExtensions)
376376

377377
module MultiTenantFindBy
378-
def cached_find_by_statement(key, &block)
379-
return super unless respond_to?(:scoped_by_tenant?) && scoped_by_tenant?
378+
if ActiveRecord::VERSION::MAJOR >= 7 && ActiveRecord::VERSION::MINOR >= 2
379+
def cached_find_by_statement(connection, key, &block)
380+
return super unless respond_to?(:scoped_by_tenant?) && scoped_by_tenant?
380381

381-
key = Array.wrap(key) + [MultiTenant.current_tenant_id.to_s]
382-
super
382+
super(connection, Array.wrap(key) + [MultiTenant.current_tenant_id.to_s], &block)
383+
end
384+
else
385+
def cached_find_by_statement(key, &block)
386+
return super unless respond_to?(:scoped_by_tenant?) && scoped_by_tenant?
387+
388+
super(Array.wrap(key) + [MultiTenant.current_tenant_id.to_s], &block)
389+
end
383390
end
384391
end
385392

lib/activerecord-multi-tenant/relation_extension.rb

+8-1
Original file line numberDiff line numberDiff line change
@@ -37,7 +37,14 @@ def generate_in_condition_subquery
3737
tenant_id = MultiTenant.current_tenant_id
3838

3939
# Build an Arel query
40-
arel = eager_loading? ? apply_join_dependency.arel : build_arel
40+
arel = if eager_loading?
41+
apply_join_dependency.arel
42+
elsif ActiveRecord::VERSION::MAJOR >= 7 && ActiveRecord::VERSION::MINOR >= 2
43+
build_arel(klass.connection)
44+
else
45+
build_arel
46+
end
47+
4148
arel.source.left = table
4249

4350
# If the tenant ID is present and the tenant key is a column in the model,

spec/activerecord-multi-tenant/model_extensions_spec.rb

+11
Original file line numberDiff line numberDiff line change
@@ -132,6 +132,17 @@ def self.name
132132
end
133133

134134
describe 'inspect method filters senstive column values' do
135+
if ActiveRecord::VERSION::MAJOR >= 7 && ActiveRecord::VERSION::MINOR >= 2
136+
# related: https://github.com/rails/rails/pull/49765
137+
around do |example|
138+
prev = Account.attributes_for_inspect
139+
Account.attributes_for_inspect = :all
140+
example.run
141+
ensure
142+
Account.attributes_for_inspect = prev
143+
end
144+
end
145+
135146
it 'filters senstive value' do
136147
account = Account.new(name: 'foo', password: 'baz')
137148
expect(account.inspect).to eq '#<Account id: nil, name: nil, subdomain: nil, domain: nil, password: [FILTERED]>'

0 commit comments

Comments
 (0)