counter_culture 3.8.1 → 3.9.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.circleci/config.yml +37 -1
- data/Appraisals +2 -1
- data/CHANGELOG.md +12 -1
- data/README.md +34 -1
- data/counter_culture.gemspec +1 -1
- data/gemfiles/rails_5.2.gemfile +1 -1
- data/gemfiles/rails_6.0.gemfile +1 -1
- data/gemfiles/rails_6.1.gemfile +1 -1
- data/gemfiles/rails_7.0.gemfile +1 -1
- data/gemfiles/rails_7.1.gemfile +1 -1
- data/gemfiles/rails_7.2.gemfile +1 -1
- data/gemfiles/rails_8.0.gemfile +7 -0
- data/lib/counter_culture/configuration.rb +34 -0
- data/lib/counter_culture/reconciler.rb +11 -6
- data/lib/counter_culture/version.rb +1 -1
- data/lib/counter_culture/with_connection.rb +26 -12
- data/lib/counter_culture.rb +1 -0
- data/lib/generators/counter_culture_generator.rb +2 -2
- metadata +5 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: f93d3d3e415f5d92e3870caf67067cdb75a6a76f68542b82c9ec2411987a0e4d
|
4
|
+
data.tar.gz: 89cf5171b549aab368a1dd87117afd429bf557d93f198e220fbe3b2a24e81f20
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 17b50c40207f615d0018c142062e536c3da69bebab209d4facb63ee03273363d97f83675d173f3096f98c68393f374e39aae0dba0db6364de9bc3c83d38af160
|
7
|
+
data.tar.gz: d7bef1c4214a30fa9888843c35fe2f074ea89e78b9341b5ed458471934576dea9ef8bf12fac72bd549a532fc50d34b9e327c4f2b943e7965720f3b12bef503b6
|
data/.circleci/config.yml
CHANGED
@@ -49,7 +49,7 @@ workflows:
|
|
49
49
|
matrix:
|
50
50
|
parameters:
|
51
51
|
ruby-version: ["2.6", "2.7", "3.0", "3.1", "3.2", "3.3"]
|
52
|
-
rails-version: ["5.2", "6.0", "6.1", "7.0", "7.1", "7.2"]
|
52
|
+
rails-version: ["5.2", "6.0", "6.1", "7.0", "7.1", "7.2", "8.0"]
|
53
53
|
database: ["postgresql", "sqlite3", "mysql2"]
|
54
54
|
exclude:
|
55
55
|
- ruby-version: "3.0"
|
@@ -133,3 +133,39 @@ workflows:
|
|
133
133
|
- ruby-version: "3.0"
|
134
134
|
rails-version: "7.2"
|
135
135
|
database: "mysql2"
|
136
|
+
- ruby-version: "2.6"
|
137
|
+
rails-version: "8.0"
|
138
|
+
database: "postgresql"
|
139
|
+
- ruby-version: "2.6"
|
140
|
+
rails-version: "8.0"
|
141
|
+
database: "sqlite3"
|
142
|
+
- ruby-version: "2.6"
|
143
|
+
rails-version: "8.0"
|
144
|
+
database: "mysql2"
|
145
|
+
- ruby-version: "2.7"
|
146
|
+
rails-version: "8.0"
|
147
|
+
database: "postgresql"
|
148
|
+
- ruby-version: "2.7"
|
149
|
+
rails-version: "8.0"
|
150
|
+
database: "sqlite3"
|
151
|
+
- ruby-version: "2.7"
|
152
|
+
rails-version: "8.0"
|
153
|
+
database: "mysql2"
|
154
|
+
- ruby-version: "3.0"
|
155
|
+
rails-version: "8.0"
|
156
|
+
database: "postgresql"
|
157
|
+
- ruby-version: "3.0"
|
158
|
+
rails-version: "8.0"
|
159
|
+
database: "sqlite3"
|
160
|
+
- ruby-version: "3.0"
|
161
|
+
rails-version: "8.0"
|
162
|
+
database: "mysql2"
|
163
|
+
- ruby-version: "3.1"
|
164
|
+
rails-version: "8.0"
|
165
|
+
database: "postgresql"
|
166
|
+
- ruby-version: "3.1"
|
167
|
+
rails-version: "8.0"
|
168
|
+
database: "sqlite3"
|
169
|
+
- ruby-version: "3.1"
|
170
|
+
rails-version: "8.0"
|
171
|
+
database: "mysql2"
|
data/Appraisals
CHANGED
@@ -5,9 +5,10 @@
|
|
5
5
|
7.0
|
6
6
|
7.1
|
7
7
|
7.2
|
8
|
+
8.0
|
8
9
|
].each do |rails_version|
|
9
10
|
appraise "rails-#{rails_version}" do
|
10
|
-
gem '
|
11
|
+
gem 'activerecord', "~> #{rails_version}.0"
|
11
12
|
if Gem::Version.new(rails_version) < Gem::Version.new("7.2")
|
12
13
|
gem 'sqlite3', "~> 1.4"
|
13
14
|
end
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,14 @@
|
|
1
|
+
## 3.9.0 (March 20, 2025)
|
2
|
+
|
3
|
+
New features:
|
4
|
+
- Add global configuration for using a read replica (#405)
|
5
|
+
|
6
|
+
## 3.8.2 (October 21, 2024)
|
7
|
+
|
8
|
+
Bugfixes:
|
9
|
+
- Use ActiveRecord version, not Rails version, to make it possible to use `counter_culture_fix_counts` without Rails
|
10
|
+
- Tests no longer require Rails, making sure we do not introduce this regression again
|
11
|
+
|
1
12
|
## 3.8.1 (October 18, 2024)
|
2
13
|
|
3
14
|
Bugfixes:
|
@@ -268,7 +279,7 @@ Bugfixes:
|
|
268
279
|
## 1.8.1 (September 5, 2017)
|
269
280
|
|
270
281
|
Improvements:
|
271
|
-
- Use ActiveRecord version, not Rails version, in `Reconciler`,
|
282
|
+
- Use ActiveRecord version, not Rails version, in `Reconciler`, making it possible to use `counter_culture_fix_counts` without Rails
|
272
283
|
|
273
284
|
## 1.8.0 (August 30, 2017)
|
274
285
|
|
data/README.md
CHANGED
@@ -7,7 +7,7 @@ Turbo-charged counter caches for your Rails app. Huge improvements over the Rail
|
|
7
7
|
* Supports dynamic column names, making it possible to split up the counter cache for different types of objects
|
8
8
|
* Can keep a running count, or a running total
|
9
9
|
|
10
|
-
Tested against Ruby 2.6, 2.7, 3.0, 3.1, 3.2 and 3.3 and against the latest patch releases of Rails 5.2, 6.0, 6.1, 7.0
|
10
|
+
Tested against Ruby 2.6, 2.7, 3.0, 3.1, 3.2 and 3.3 and against the latest patch releases of Rails 5.2, 6.0, 6.1, 7.0, 7.1, 7.2 and 8.0.
|
11
11
|
|
12
12
|
Please note that -- unlike Rails' built-in counter-caches -- counter_culture does not currently change the behavior of the `.size` method on ActiveRecord associations. If you want to avoid a database query and read the cached value, please use the attribute name containing the counter cache directly.
|
13
13
|
|
@@ -618,6 +618,39 @@ Image.counter_culture_fix_counts(polymorphic_classes: Product)
|
|
618
618
|
Image.counter_culture_fix_counts(polymorphic_classes: [Product, Employee])
|
619
619
|
```
|
620
620
|
|
621
|
+
## Using Read Replicas
|
622
|
+
|
623
|
+
When using `counter_culture_fix_counts`, you can configure counter_culture to use read replicas for counting operations, which can help reduce load on your primary database. Write operations will still use the primary database.
|
624
|
+
|
625
|
+
### Global Configuration
|
626
|
+
|
627
|
+
You can enable read replica support globally:
|
628
|
+
|
629
|
+
```ruby
|
630
|
+
# config/initializers/counter_culture.rb
|
631
|
+
CounterCulture.configure do |config|
|
632
|
+
config.use_read_replica = true
|
633
|
+
end
|
634
|
+
```
|
635
|
+
|
636
|
+
### Per-Call Configuration
|
637
|
+
|
638
|
+
You can also specify read replica usage per call (this takes precedence over global configuration):
|
639
|
+
|
640
|
+
```ruby
|
641
|
+
Model.counter_culture_fix_counts(
|
642
|
+
db_connection_builder: proc { |reading, block|
|
643
|
+
if reading
|
644
|
+
ApplicationRecord.connected_to(role: :reading, &block)
|
645
|
+
else
|
646
|
+
ApplicationRecord.connected_to(role: :writing, &block)
|
647
|
+
end
|
648
|
+
}
|
649
|
+
)
|
650
|
+
```
|
651
|
+
|
652
|
+
Note: This feature requires Rails 7.1 or higher for full read replica support. On older versions, it will fall back to using the primary database.
|
653
|
+
|
621
654
|
## Contributing to counter_culture
|
622
655
|
|
623
656
|
* Check out the latest master to make sure the feature hasn't been implemented or the bug hasn't been fixed yet.
|
data/counter_culture.gemspec
CHANGED
@@ -36,7 +36,7 @@ Gem::Specification.new do |spec|
|
|
36
36
|
spec.add_development_dependency 'paper_trail'
|
37
37
|
spec.add_development_dependency 'paranoia'
|
38
38
|
spec.add_development_dependency 'after_commit_action'
|
39
|
-
spec.add_development_dependency '
|
39
|
+
spec.add_development_dependency 'activerecord', '>= 4.2'
|
40
40
|
spec.add_development_dependency 'rake', '>= 10.0'
|
41
41
|
spec.add_development_dependency 'rdoc', ">= 6.3.1"
|
42
42
|
spec.add_development_dependency 'rspec', '~> 3.0'
|
data/gemfiles/rails_5.2.gemfile
CHANGED
data/gemfiles/rails_6.0.gemfile
CHANGED
data/gemfiles/rails_6.1.gemfile
CHANGED
data/gemfiles/rails_7.0.gemfile
CHANGED
data/gemfiles/rails_7.1.gemfile
CHANGED
data/gemfiles/rails_7.2.gemfile
CHANGED
@@ -0,0 +1,34 @@
|
|
1
|
+
module CounterCulture
|
2
|
+
def self.configuration
|
3
|
+
@configuration ||= Configuration.new
|
4
|
+
end
|
5
|
+
|
6
|
+
def self.configure
|
7
|
+
yield(configuration)
|
8
|
+
end
|
9
|
+
|
10
|
+
def self.reset_configuration
|
11
|
+
@configuration = Configuration.new
|
12
|
+
end
|
13
|
+
|
14
|
+
class Configuration
|
15
|
+
attr_reader :use_read_replica
|
16
|
+
|
17
|
+
def initialize
|
18
|
+
@use_read_replica = false
|
19
|
+
end
|
20
|
+
|
21
|
+
def use_read_replica=(value)
|
22
|
+
if value && !rails_supports_read_replica?
|
23
|
+
raise "Counter Culture's read replica support requires Rails 7.1 or higher"
|
24
|
+
end
|
25
|
+
@use_read_replica = value
|
26
|
+
end
|
27
|
+
|
28
|
+
private
|
29
|
+
|
30
|
+
def rails_supports_read_replica?
|
31
|
+
Gem::Requirement.new('>= 7.1.0').satisfied_by?(ActiveRecord.version)
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
@@ -66,6 +66,7 @@ module CounterCulture
|
|
66
66
|
@counter, @options, = counter, options
|
67
67
|
@relation_class = relation_class
|
68
68
|
@changes_holder = changes_holder
|
69
|
+
@connection_handler = WithConnection.new(relation_class)
|
69
70
|
end
|
70
71
|
|
71
72
|
def perform
|
@@ -181,17 +182,17 @@ module CounterCulture
|
|
181
182
|
def log(message)
|
182
183
|
return unless log?
|
183
184
|
|
184
|
-
|
185
|
+
ActiveRecord::Base.logger.info(message)
|
185
186
|
end
|
186
187
|
|
187
188
|
def log_without_newline(message)
|
188
189
|
return unless log?
|
189
190
|
|
190
|
-
|
191
|
+
ActiveRecord::Base.logger << message if ActiveRecord::Base.logger.info?
|
191
192
|
end
|
192
193
|
|
193
194
|
def log?
|
194
|
-
options[:verbose] &&
|
195
|
+
options[:verbose] && ActiveRecord::Base.logger
|
195
196
|
end
|
196
197
|
|
197
198
|
# keep track of what we fixed, e.g. for a notification email
|
@@ -314,7 +315,7 @@ module CounterCulture
|
|
314
315
|
# using Postgres with schema-namespaced tables. But then it's required,
|
315
316
|
# and otherwise it's just a no-op, so why not do it?
|
316
317
|
def quote_table_name(table_name)
|
317
|
-
|
318
|
+
@connection_handler.call do |connection|
|
318
319
|
connection.quote_table_name(table_name)
|
319
320
|
end
|
320
321
|
end
|
@@ -331,7 +332,9 @@ module CounterCulture
|
|
331
332
|
if builder = options[:db_connection_builder]
|
332
333
|
builder.call(true, block)
|
333
334
|
else
|
334
|
-
|
335
|
+
@connection_handler.call(reading: true) do |_conn|
|
336
|
+
yield
|
337
|
+
end
|
335
338
|
end
|
336
339
|
end
|
337
340
|
|
@@ -339,7 +342,9 @@ module CounterCulture
|
|
339
342
|
if builder = options[:db_connection_builder]
|
340
343
|
builder.call(false, block)
|
341
344
|
else
|
342
|
-
|
345
|
+
@connection_handler.call(reading: false) do |_conn|
|
346
|
+
yield
|
347
|
+
end
|
343
348
|
end
|
344
349
|
end
|
345
350
|
end
|
@@ -6,28 +6,42 @@ module CounterCulture
|
|
6
6
|
|
7
7
|
attr_reader :recipient
|
8
8
|
|
9
|
-
def call
|
10
|
-
if
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
yield connection
|
9
|
+
def call(reading: false)
|
10
|
+
if rails_7_1_or_greater?
|
11
|
+
use_read_replica = CounterCulture.configuration.use_read_replica && reading
|
12
|
+
role = use_read_replica ? :reading : :writing
|
13
|
+
|
14
|
+
ActiveRecord::Base.connected_to(role: role) do
|
15
|
+
yield_with_connection { |conn| yield conn }
|
17
16
|
end
|
18
17
|
else
|
19
|
-
|
18
|
+
# For older Rails versions, just use normal connection
|
19
|
+
yield_with_connection { |conn| yield conn }
|
20
20
|
end
|
21
21
|
end
|
22
22
|
|
23
23
|
private
|
24
24
|
|
25
|
-
def
|
26
|
-
|
25
|
+
def yield_with_connection
|
26
|
+
if rails_7_2_or_greater?
|
27
|
+
recipient.with_connection { |conn| yield conn }
|
28
|
+
elsif rails_7_1?
|
29
|
+
recipient.connection_pool.with_connection { |conn| yield conn }
|
30
|
+
else
|
31
|
+
yield recipient.connection
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def rails_7_1_or_greater?
|
36
|
+
Gem::Requirement.new('>= 7.1.0').satisfied_by?(ActiveRecord.version)
|
27
37
|
end
|
28
38
|
|
29
39
|
def rails_7_2_or_greater?
|
30
|
-
Gem::Requirement.new('>= 7.2.0').satisfied_by?(
|
40
|
+
Gem::Requirement.new('>= 7.2.0').satisfied_by?(ActiveRecord.version)
|
41
|
+
end
|
42
|
+
|
43
|
+
def rails_7_1?
|
44
|
+
Gem::Requirement.new('~> 7.1.0').satisfied_by?(ActiveRecord.version)
|
31
45
|
end
|
32
46
|
end
|
33
47
|
end
|
data/lib/counter_culture.rb
CHANGED
@@ -3,6 +3,7 @@ require 'active_support/lazy_load_hooks'
|
|
3
3
|
|
4
4
|
require 'counter_culture/version'
|
5
5
|
require 'counter_culture/extensions'
|
6
|
+
require 'counter_culture/configuration'
|
6
7
|
require 'counter_culture/counter'
|
7
8
|
require 'counter_culture/reconciler'
|
8
9
|
require 'counter_culture/skip_updates'
|
@@ -27,8 +27,8 @@ class CounterCultureGenerator < ActiveRecord::Generators::Base
|
|
27
27
|
end
|
28
28
|
|
29
29
|
def migration_version
|
30
|
-
if
|
31
|
-
"[#{
|
30
|
+
if ActiveRecord.version >= Gem::Version.new('5.0.0')
|
31
|
+
"[#{ActiveRecord::VERSION::MAJOR}.#{ActiveRecord::VERSION::MINOR}]"
|
32
32
|
end
|
33
33
|
end
|
34
34
|
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: counter_culture
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 3.
|
4
|
+
version: 3.9.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Magnus von Koeller
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2025-03-20 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: activerecord
|
@@ -151,7 +151,7 @@ dependencies:
|
|
151
151
|
- !ruby/object:Gem::Version
|
152
152
|
version: '0'
|
153
153
|
- !ruby/object:Gem::Dependency
|
154
|
-
name:
|
154
|
+
name: activerecord
|
155
155
|
requirement: !ruby/object:Gem::Requirement
|
156
156
|
requirements:
|
157
157
|
- - ">="
|
@@ -319,7 +319,9 @@ files:
|
|
319
319
|
- gemfiles/rails_7.0.gemfile
|
320
320
|
- gemfiles/rails_7.1.gemfile
|
321
321
|
- gemfiles/rails_7.2.gemfile
|
322
|
+
- gemfiles/rails_8.0.gemfile
|
322
323
|
- lib/counter_culture.rb
|
324
|
+
- lib/counter_culture/configuration.rb
|
323
325
|
- lib/counter_culture/counter.rb
|
324
326
|
- lib/counter_culture/extensions.rb
|
325
327
|
- lib/counter_culture/reconciler.rb
|