Skip to content

Commit b197e2a

Browse files
committed
feature: add flexibility to persistent types
https://jira.railsc.ru/browse/CRM-4532
1 parent edce26d commit b197e2a

File tree

7 files changed

+196
-68
lines changed

7 files changed

+196
-68
lines changed

Gemfile

+1
Original file line numberDiff line numberDiff line change
@@ -2,5 +2,6 @@ source 'https://rubygems.org'
22

33
gem 'activerecord-postgres-hstore', require: false
44
gem 'simple_hstore_accessor', '~> 0.2', require: false
5+
gem 'pg', '~> 0.11'
56

67
gemspec

lib/redis_counters/dumpers/destination.rb

+19-1
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ class Destination
1818
extend Forwardable
1919
include ::RedisCounters::Dumpers::Dsl::Destination
2020

21+
VALUE_DELIMITER = ','.freeze
22+
2123
# Ссылка на родительский движек - дампер.
2224
attr_accessor :engine
2325

@@ -76,6 +78,9 @@ class Destination
7678
# Returns String
7779
attr_accessor :matching_expr
7880

81+
# Разделитель значений, String.
82+
attr_accessor :value_delimiter
83+
7984
def initialize(engine)
8085
@engine = engine
8186
@fields_map = HashWithIndifferentAccess.new
@@ -174,7 +179,16 @@ def full_fields_map
174179
end
175180

176181
def updating_expression
177-
increment_fields.map { |field| "#{field} = COALESCE(target.#{field}, 0) + source.#{field}" }.join(', ')
182+
increment_fields.map do |field|
183+
case model.columns_hash[field.to_s].type
184+
when :datetime, :date
185+
"#{field} = source.#{field}"
186+
when :string
187+
"#{field} = array_to_string(ARRAY[source.#{field}, target.#{field}], '#{delimiter}')"
188+
else
189+
"#{field} = COALESCE(target.#{field}, 0) + source.#{field}"
190+
end
191+
end.join(', ')
178192
end
179193

180194
def matching_expression
@@ -197,6 +211,10 @@ def source_conditions_expression
197211

198212
"WHERE #{source_conditions.map { |source_condition| "(#{source_condition})" }.join(' AND ')}"
199213
end
214+
215+
def delimiter
216+
value_delimiter || VALUE_DELIMITER
217+
end
200218
end
201219
end
202220
end

lib/redis_counters/dumpers/dsl/destination.rb

+1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ class Configuration < ::RedisCounters::Dumpers::Dsl::Base
1313

1414
setter :model
1515
setter :matching_expr
16+
setter :value_delimiter
1617

1718
varags_setter :fields
1819
varags_setter :key_fields

redis_counters-dumpers.gemspec

-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,6 @@ Gem::Specification.new do |spec|
1818

1919
spec.add_runtime_dependency 'activesupport', '>= 3.0', '< 5'
2020
spec.add_runtime_dependency 'activerecord', '>= 3.0'
21-
spec.add_runtime_dependency 'pg'
2221
spec.add_runtime_dependency 'redis', '>= 3.0'
2322
spec.add_runtime_dependency 'redis-namespace', '>= 1.3'
2423
spec.add_runtime_dependency 'callbacks_rb', '>= 0.0.1'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
class RealtimeStat < ActiveRecord::Base
2+
end

spec/internal/db/schema.rb

+8
Original file line numberDiff line numberDiff line change
@@ -52,4 +52,12 @@
5252
t.integer :value, null: false, default: 0
5353
t.string :payload
5454
end
55+
56+
create_table :realtime_stats do |t|
57+
t.integer :record_id, null: false
58+
t.integer :column_id, null: false
59+
t.integer :hits, null: false, default: 0
60+
t.timestamp :date
61+
t.string :params
62+
end
5563
end

spec/lib/redis_counters/dumpers/engine_spec.rb

+165-66
Original file line numberDiff line numberDiff line change
@@ -65,100 +65,199 @@
6565

6666
describe '#process!' do
6767
context 'when increment_fields specified' do
68-
before do
69-
counter.increment(date: prev_date_s, record_id: 1, column_id: 100, subject: '', params: '')
70-
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
71-
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
72-
counter.increment(date: prev_date_s, record_id: 2, column_id: 100, subject: nil, params: '')
68+
context 'without source conditions' do
69+
before do
70+
counter.increment(date: prev_date_s, record_id: 1, column_id: 100, subject: '', params: '')
71+
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
72+
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
73+
counter.increment(date: prev_date_s, record_id: 2, column_id: 100, subject: nil, params: '')
7374

74-
params = {a: 1}.stringify_keys.to_s[1..-2]
75-
counter.increment(date: prev_date_s, record_id: 3, column_id: 300, subject: nil, params: params)
75+
params = {a: 1}.stringify_keys.to_s[1..-2]
76+
counter.increment(date: prev_date_s, record_id: 3, column_id: 300, subject: nil, params: params)
7677

77-
dumper.process!(counter, date: prev_date)
78+
dumper.process!(counter, date: prev_date)
7879

79-
counter.increment(date: date_s, record_id: 1, column_id: 100, subject: '', params: '')
80-
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
81-
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
82-
counter.increment(date: date_s, record_id: 2, column_id: 100, subject: nil, params: '')
80+
counter.increment(date: date_s, record_id: 1, column_id: 100, subject: '', params: '')
81+
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
82+
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
83+
counter.increment(date: date_s, record_id: 2, column_id: 100, subject: nil, params: '')
8384

84-
dumper.process!(counter, date: date)
85-
end
85+
dumper.process!(counter, date: date)
86+
end
8687

87-
it { expect(StatsByDay.count).to eq 7 }
88-
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: prev_date).first.hits).to eq 1 }
89-
it { expect(StatsByDay.where(record_id: 1, column_id: 200, date: prev_date).first.hits).to eq 2 }
90-
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: prev_date).first.hits).to eq 1 }
91-
it { expect(StatsByDay.where(record_id: 3, column_id: 300, date: prev_date).first.params).to eq("a" => "1") }
92-
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: date).first.hits).to eq 1 }
93-
it { expect(StatsByDay.where(record_id: 1, column_id: 200, date: date).first.hits).to eq 2 }
94-
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: date).first.hits).to eq 1 }
88+
it { expect(StatsByDay.count).to eq 7 }
89+
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: prev_date).first.hits).to eq 1 }
90+
it { expect(StatsByDay.where(record_id: 1, column_id: 200, date: prev_date).first.hits).to eq 2 }
91+
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: prev_date).first.hits).to eq 1 }
92+
it { expect(StatsByDay.where(record_id: 3, column_id: 300, date: prev_date).first.params).to eq("a" => "1") }
93+
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: date).first.hits).to eq 1 }
94+
it { expect(StatsByDay.where(record_id: 1, column_id: 200, date: date).first.hits).to eq 2 }
95+
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: date).first.hits).to eq 1 }
9596

96-
it { expect(StatsTotal.count).to eq 4 }
97-
it { expect(StatsTotal.where(record_id: 1, column_id: 100).first.hits).to eq 2 }
98-
it { expect(StatsTotal.where(record_id: 1, column_id: 200).first.hits).to eq 4 }
99-
it { expect(StatsTotal.where(record_id: 2, column_id: 100).first.hits).to eq 2 }
97+
it { expect(StatsTotal.count).to eq 4 }
98+
it { expect(StatsTotal.where(record_id: 1, column_id: 100).first.hits).to eq 2 }
99+
it { expect(StatsTotal.where(record_id: 1, column_id: 200).first.hits).to eq 4 }
100+
it { expect(StatsTotal.where(record_id: 2, column_id: 100).first.hits).to eq 2 }
100101

101-
it { expect(StatsAggTotal.count).to eq 3 }
102-
it { expect(StatsAggTotal.where(record_id: 1).first.hits).to eq 6 }
103-
it { expect(StatsAggTotal.where(record_id: 2).first.hits).to eq 2 }
102+
it { expect(StatsAggTotal.count).to eq 3 }
103+
it { expect(StatsAggTotal.where(record_id: 1).first.hits).to eq 6 }
104+
it { expect(StatsAggTotal.where(record_id: 2).first.hits).to eq 2 }
105+
end
104106

105107
context 'with source conditions' do
108+
context 'when incremented field class is integer' do
109+
let(:dumper) do
110+
RedisCounters::Dumpers::Engine.build do
111+
name :stats_totals
112+
fields record_id: :integer,
113+
column_id: :integer,
114+
value: :integer,
115+
date: :date
116+
117+
destination do
118+
model StatsByDay
119+
take :record_id, :column_id, :hits, :date
120+
key_fields :record_id, :column_id, :date
121+
increment_fields :hits
122+
map :hits, to: :value
123+
condition 'target.date = :date'
124+
source_condition 'column_id = 100'
125+
end
126+
127+
destination do
128+
model StatsTotal
129+
take :record_id, :column_id, :hits
130+
key_fields :record_id, :column_id
131+
increment_fields :hits
132+
map :hits, to: :value
133+
source_condition 'column_id = 100'
134+
end
135+
136+
destination do
137+
model StatsAggTotal
138+
take :record_id, :hits
139+
key_fields :record_id
140+
increment_fields :hits
141+
map :hits, to: 'sum(value)'
142+
group_by :record_id
143+
source_condition 'column_id = 100'
144+
end
145+
146+
on_before_merge do |dumper, _connection|
147+
dumper.common_params = {date: dumper.args[:date].strftime('%Y-%m-%d')}
148+
end
149+
end
150+
end
151+
152+
before do
153+
counter.increment(date: prev_date_s, record_id: 1, column_id: 100, subject: '', params: '')
154+
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
155+
counter.increment(date: prev_date_s, record_id: 1, column_id: 200, subject: '', params: '')
156+
counter.increment(date: prev_date_s, record_id: 2, column_id: 100, subject: nil, params: '')
157+
158+
params = {a: 1}.stringify_keys.to_s[1..-2]
159+
counter.increment(date: prev_date_s, record_id: 3, column_id: 300, subject: nil, params: params)
160+
161+
dumper.process!(counter, date: prev_date)
162+
163+
counter.increment(date: date_s, record_id: 1, column_id: 100, subject: '', params: '')
164+
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
165+
counter.increment(date: date_s, record_id: 1, column_id: 200, subject: '', params: '')
166+
counter.increment(date: date_s, record_id: 2, column_id: 100, subject: nil, params: '')
167+
168+
dumper.process!(counter, date: date)
169+
end
170+
171+
it { expect(StatsByDay.count).to eq 4 }
172+
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: prev_date).first.hits).to eq 1 }
173+
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: prev_date).first.hits).to eq 1 }
174+
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: date).first.hits).to eq 1 }
175+
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: date).first.hits).to eq 1 }
176+
177+
it { expect(StatsTotal.count).to eq 2 }
178+
it { expect(StatsTotal.where(record_id: 1, column_id: 100).first.hits).to eq 2 }
179+
it { expect(StatsTotal.where(record_id: 2, column_id: 100).first.hits).to eq 2 }
180+
181+
it { expect(StatsAggTotal.count).to eq 2 }
182+
it { expect(StatsAggTotal.where(record_id: 1).first.hits).to eq 2 }
183+
it { expect(StatsAggTotal.where(record_id: 2).first.hits).to eq 2 }
184+
end
185+
end
186+
187+
context 'when incremented field class is string' do
106188
let(:dumper) do
107189
RedisCounters::Dumpers::Engine.build do
108-
name :stats_totals
190+
name :realtime_stats
109191
fields record_id: :integer,
110192
column_id: :integer,
111193
value: :integer,
112-
date: :date
194+
params: :string,
195+
date: :timestamp
113196

114197
destination do
115-
model StatsByDay
116-
take :record_id, :column_id, :hits, :date
117-
key_fields :record_id, :column_id, :date
118-
increment_fields :hits
198+
model RealtimeStat
199+
take :record_id, :column_id, :date, :hits, :params
200+
key_fields :record_id, :column_id
201+
increment_fields :hits, :params
202+
value_delimiter '; '
119203
map :hits, to: :value
120-
condition 'target.date = :date'
121-
source_condition 'column_id = 100'
204+
condition 'target.date::date = :date::date'
122205
end
206+
end
207+
end
208+
209+
before do
210+
counter.increment(date: date, record_id: 1, column_id: 100, subject: '', params: 'abc')
211+
dumper.common_params = {date: date, params: 'abc'}
212+
dumper.process!(counter, date: date)
213+
214+
counter.increment(date: date, record_id: 1, column_id: 100, subject: '', params: 'xyz')
215+
dumper.common_params = {date: date, params: 'xyz'}
216+
dumper.process!(counter, date: date)
217+
end
218+
219+
it do
220+
expect(RealtimeStat.count).to eq 1
221+
expect(RealtimeStat.first.params).to eq 'xyz; abc'
222+
end
223+
end
224+
225+
context 'when incremented field class is date or time' do
226+
let(:current_time) { Date.today.to_time }
227+
let(:dumper) do
228+
RedisCounters::Dumpers::Engine.build do
229+
name :realtime_stats
230+
fields record_id: :integer,
231+
column_id: :integer,
232+
value: :integer,
233+
params: :string,
234+
date: :timestamp
123235

124236
destination do
125-
model StatsTotal
126-
take :record_id, :column_id, :hits
237+
model RealtimeStat
238+
take :record_id, :column_id, :hits, :params, :date
127239
key_fields :record_id, :column_id
128-
increment_fields :hits
240+
increment_fields :hits, :date
129241
map :hits, to: :value
130-
source_condition 'column_id = 100'
131-
end
132-
133-
destination do
134-
model StatsAggTotal
135-
take :record_id, :hits
136-
key_fields :record_id
137-
increment_fields :hits
138-
map :hits, to: 'sum(value)'
139-
group_by :record_id
140-
source_condition 'column_id = 100'
141-
end
142-
143-
on_before_merge do |dumper, _connection|
144-
dumper.common_params = {date: dumper.args[:date].strftime('%Y-%m-%d')}
242+
condition 'target.date::date = :date::date'
145243
end
146244
end
147245
end
148246

149-
it { expect(StatsByDay.count).to eq 4 }
150-
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: prev_date).first.hits).to eq 1 }
151-
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: prev_date).first.hits).to eq 1 }
152-
it { expect(StatsByDay.where(record_id: 1, column_id: 100, date: date).first.hits).to eq 1 }
153-
it { expect(StatsByDay.where(record_id: 2, column_id: 100, date: date).first.hits).to eq 1 }
247+
before do
248+
counter.increment(date: date, record_id: 1, column_id: 100, subject: '', params: '')
249+
dumper.common_params = {date: current_time}
250+
dumper.process!(counter, date: date)
154251

155-
it { expect(StatsTotal.count).to eq 2 }
156-
it { expect(StatsTotal.where(record_id: 1, column_id: 100).first.hits).to eq 2 }
157-
it { expect(StatsTotal.where(record_id: 2, column_id: 100).first.hits).to eq 2 }
252+
counter.increment(date: date, record_id: 1, column_id: 100, subject: '', params: '')
253+
dumper.common_params = {date: current_time}
254+
dumper.process!(counter, date: date)
255+
end
158256

159-
it { expect(StatsAggTotal.count).to eq 2 }
160-
it { expect(StatsAggTotal.where(record_id: 1).first.hits).to eq 2 }
161-
it { expect(StatsAggTotal.where(record_id: 2).first.hits).to eq 2 }
257+
it 'update incremented date' do
258+
expect(RealtimeStat.count).to eq 1
259+
expect(RealtimeStat.first.date).to eq current_time
260+
end
162261
end
163262
end
164263

0 commit comments

Comments
 (0)