pgtk 0.31.1 → 0.31.3

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a15b5d36eab5407bca78c591056966cee9b90167e5bfa0cc98fa36984801622a
4
- data.tar.gz: bd201e64d30a88836337d86b4b3d202b708109de1676ce5d83593df8f4e4ace4
3
+ metadata.gz: ce069bed0f29101349d2617ad2ae7c51f4f03e6b6a4b800d717bf4ac75bb533f
4
+ data.tar.gz: 81e34a18fdf26fea4545e33f5c554af8b17f74f75e0a8efe450dea16fa13df49
5
5
  SHA512:
6
- metadata.gz: 8b58b179b7c85ff2183d7c909da288f9a4be5c67bbe8040350c8706098bac8ccbfe05010d638f28db67cf70b1434744abc6667008f1fcc32bdc6aaa3a4cb1fd8
7
- data.tar.gz: 0da90a8d6b61c07125bab269833149217f625caadbfbb8e12363ac84f84efa36e321fb6ec18f5b12c0b1bb9dfdab42ad74086cb6300b9fb0b122e2a52e9384cf
6
+ metadata.gz: fea29eaaef7f045eec87fd114d5ba55782d50332333169ab30012bbf9e221dd2c3ef1be57af808f4aa1986b5356e27a8b25ce144c52bcaad3ad8329772eb81a2
7
+ data.tar.gz: c84f1647c650b98ff6e34e752184a922ec74c285c55dcfa26714c50280a09bb5cf6bf3ff6e40dee0c5f03a39aa0aa30396ad7ffd30b05859d981eb67c2697437
data/Gemfile.lock CHANGED
@@ -142,9 +142,9 @@ GEM
142
142
  unicode-emoji (~> 4.1)
143
143
  unicode-emoji (4.2.0)
144
144
  waitutil (0.2.1)
145
- xcop (0.8.0)
145
+ xcop (0.9.0)
146
146
  differ (~> 0.1.2)
147
- nokogiri (~> 1.10)
147
+ nokogiri (~> 1.8)
148
148
  rainbow (~> 3.0)
149
149
  slop (~> 4.4)
150
150
  yard (0.9.43)
data/lib/pgtk/pool.rb CHANGED
@@ -343,16 +343,27 @@ class Pgtk::Pool
343
343
 
344
344
  def connect
345
345
  conn = @pool.pop
346
+ conn = renew(conn) if dead?(conn)
346
347
  begin
347
348
  yield(conn)
348
349
  rescue StandardError => e
349
- conn = renew(conn)
350
+ begin
351
+ conn = renew(conn)
352
+ rescue StandardError => re
353
+ @log.warn("Failed to renew connection after #{e.message}: #{re.message}")
354
+ end
350
355
  raise(e)
351
356
  ensure
352
357
  @pool.push(conn)
353
358
  end
354
359
  end
355
360
 
361
+ def dead?(conn)
362
+ conn.finished? || conn.status == PG::Constants::CONNECTION_BAD
363
+ rescue StandardError
364
+ true
365
+ end
366
+
356
367
  def renew(conn)
357
368
  begin
358
369
  conn.close unless conn.finished?
data/lib/pgtk/stash.rb CHANGED
@@ -57,7 +57,7 @@ class Pgtk::Stash
57
57
  # @param [Concurrent::ReentrantReadWriteLock] entrance Read-write lock for thread-safe access
58
58
  def initialize(
59
59
  pool,
60
- stash: { queries: {}, tables: {} },
60
+ stash: { queries: {}, tables: {}, table_mod: {} },
61
61
  loog: Loog::NULL,
62
62
  entrance: Concurrent::ReentrantReadWriteLock.new,
63
63
  refill: 16,
@@ -71,6 +71,7 @@ class Pgtk::Stash
71
71
  )
72
72
  @pool = pool
73
73
  @stash = stash
74
+ @stash[:table_mod] ||= {}
74
75
  @loog = loog
75
76
  @entrance = entrance
76
77
  @refill = refill
@@ -216,11 +217,13 @@ class Pgtk::Stash
216
217
  tables = pure.scan(ALTS_RE).flatten
217
218
  tables.uniq!
218
219
  ret = @pool.exec(pure, params, result)
220
+ now = Time.now
219
221
  @entrance.with_write_lock do
220
222
  tables.each do |t|
223
+ @stash[:table_mod][t] = now
221
224
  @stash[:tables][t]&.each do |q|
222
225
  @stash[:queries][q]&.each_key do |key|
223
- @stash[:queries][q][key][:stale] = Time.now
226
+ @stash[:queries][q][key][:stale] = now
224
227
  end
225
228
  end
226
229
  end
@@ -233,16 +236,17 @@ class Pgtk::Stash
233
236
  ret = @stash.dig(:queries, pure, key, :ret)
234
237
  if ret.nil? || @stash.dig(:queries, pure, key, :stale)
235
238
  mark = @stash.dig(:queries, pure, key, :stale)
239
+ tables = pure.scan(/(?<=^|\s)(?:FROM|JOIN) ([a-z_]+)(?=\s|;|$)/).flatten
240
+ tables.uniq!
241
+ marks = tables.to_h { |t| [t, @stash[:table_mod][t]] }
236
242
  ret = @pool.exec(pure, params, result)
237
- cache(pure, key, params, result, ret, mark) unless pure.include?(' NOW() ')
243
+ cache(pure, key, params, result, ret, mark, tables, marks) unless pure.include?(' NOW() ')
238
244
  end
239
245
  bump(pure, key) if @stash.dig(:queries, pure, key)
240
246
  ret
241
247
  end
242
248
 
243
- def cache(pure, key, params, result, ret, mark)
244
- tables = pure.scan(/(?<=^|\s)(?:FROM|JOIN) ([a-z_]+)(?=\s|;|$)/).flatten
245
- tables.uniq!
249
+ def cache(pure, key, params, result, ret, mark, tables, marks)
246
250
  raise(ArgumentError, "No tables at #{pure.inspect}") if tables.empty?
247
251
  @entrance.with_write_lock do
248
252
  tables.each do |t|
@@ -252,8 +256,9 @@ class Pgtk::Stash
252
256
  @stash[:queries][pure] ||= {}
253
257
  existing = @stash[:queries][pure][key]
254
258
  stale = existing && existing[:stale]
259
+ stillborn = tables.any? { |t| (cur = @stash[:table_mod][t]) && cur != marks[t] }
255
260
  entry = { ret:, params:, result:, used: Time.now }
256
- entry[:stale] = stale if stale && stale != mark
261
+ entry[:stale] = stale == mark ? Time.now : stale if (stale && stale != mark) || stillborn
257
262
  @stash[:queries][pure][key] = entry
258
263
  end
259
264
  end
@@ -294,7 +299,7 @@ class Pgtk::Stash
294
299
  m = @stash[:queries][q].values.map { |h| h[:used] }.min
295
300
  next unless m
296
301
  @stash[:queries][q].delete_if { |_, h| h[:used] == m }
297
- @stash[:queries].delete_if { |_, kk| kk.empty? }
302
+ evict(q) if @stash[:queries][q].empty?
298
303
  end
299
304
  end
300
305
  end
@@ -306,12 +311,18 @@ class Pgtk::Stash
306
311
  @entrance.with_write_lock do
307
312
  @stash[:queries].each_key do |q|
308
313
  @stash[:queries][q].delete_if { |_, h| h[:used] < Time.now - @retire }
309
- @stash[:queries].delete_if { |_, kk| kk.empty? }
314
+ evict(q) if @stash[:queries][q].empty?
310
315
  end
311
316
  end
312
317
  end
313
318
  end
314
319
 
320
+ def evict(query)
321
+ @stash[:queries].delete(query)
322
+ @stash[:tables].each_value { |list| list.delete(query) }
323
+ @stash[:tables].delete_if { |_, list| list.empty? }
324
+ end
325
+
315
326
  def refiller!
316
327
  Concurrent::TimerTask.execute(execution_interval: @refill, executor: @tpool) do
317
328
  ranked.each { |q| replenish(q) }
@@ -328,16 +339,22 @@ class Pgtk::Stash
328
339
  end
329
340
 
330
341
  def replenish(query)
331
- @entrance.with_write_lock { @stash[:queries][query].keys }.each do |k|
332
- next unless @stash[:queries][query][k][:stale]
333
- next if @stash[:queries][query][k][:stale] > Time.now - @delay
342
+ snapshot =
343
+ @entrance.with_read_lock do
344
+ @stash[:queries][query]&.filter_map do |k, h|
345
+ next unless h[:stale]
346
+ next if h[:stale] > Time.now - @delay
347
+ [k, h[:params], h[:result], h[:stale]]
348
+ end
349
+ end
350
+ return unless snapshot
351
+ snapshot.each do |k, params, result, mark|
334
352
  next if @tpool.queue_length >= @maxqueue
335
353
  @tpool.post do
336
- h = @stash[:queries][query][k]
337
- mark = h[:stale]
338
- ret = @pool.exec(query, h[:params], h[:result])
354
+ ret = @pool.exec(query, params, result)
339
355
  @entrance.with_write_lock do
340
- h = @stash[:queries][query][k]
356
+ h = @stash[:queries][query]&.dig(k)
357
+ next unless h
341
358
  if h[:stale] == mark
342
359
  h[:ret] = ret
343
360
  h.delete(:stale)
data/lib/pgtk/version.rb CHANGED
@@ -10,5 +10,5 @@ require_relative '../pgtk'
10
10
  # Copyright:: Copyright (c) 2019-2026 Yegor Bugayenko
11
11
  # License:: MIT
12
12
  module Pgtk
13
- VERSION = '0.31.1' unless defined?(VERSION)
13
+ VERSION = '0.31.3' unless defined?(VERSION)
14
14
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: pgtk
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.31.1
4
+ version: 0.31.3
5
5
  platform: ruby
6
6
  authors:
7
7
  - Yegor Bugayenko