4
4
5
5
package akka .persistence .r2dbc .internal
6
6
7
+ import java .time .Clock
8
+
7
9
import scala .collection .immutable
8
10
import java .time .Instant
9
11
import java .time .{ Duration => JDuration }
12
+
10
13
import scala .annotation .tailrec
11
14
import scala .concurrent .ExecutionContext
12
15
import scala .concurrent .Future
13
16
import scala .concurrent .duration .Duration
14
17
import scala .concurrent .duration .FiniteDuration
18
+
15
19
import akka .NotUsed
16
20
import akka .annotation .InternalApi
17
21
import akka .persistence .query .Offset
@@ -42,7 +46,11 @@ import org.slf4j.Logger
42
46
backtrackingExpectFiltered = 0 ,
43
47
buckets = Buckets .empty,
44
48
previous = TimestampOffset .Zero ,
45
- previousBacktracking = TimestampOffset .Zero )
49
+ previousBacktracking = TimestampOffset .Zero ,
50
+ startTimestamp = Instant .EPOCH ,
51
+ startWallClock = Instant .EPOCH ,
52
+ currentQueryWallClock = Instant .EPOCH ,
53
+ previousQueryWallClock = Instant .EPOCH )
46
54
}
47
55
48
56
final case class QueryState (
@@ -57,24 +65,32 @@ import org.slf4j.Logger
57
65
backtrackingExpectFiltered : Int ,
58
66
buckets : Buckets ,
59
67
previous : TimestampOffset ,
60
- previousBacktracking : TimestampOffset ) {
68
+ previousBacktracking : TimestampOffset ,
69
+ startTimestamp : Instant ,
70
+ startWallClock : Instant ,
71
+ currentQueryWallClock : Instant ,
72
+ previousQueryWallClock : Instant ) {
61
73
62
74
def backtracking : Boolean = backtrackingCount > 0
63
75
64
76
def currentOffset : TimestampOffset =
65
77
if (backtracking) latestBacktracking
66
78
else latest
67
79
68
- def nextQueryFromTimestamp : Instant =
69
- if (backtracking) latestBacktracking.timestamp
70
- else latest.timestamp
80
+ def nextQueryFromTimestamp (backtrackingWindow : JDuration ): Instant =
81
+ if (backtracking && latest.timestamp.minus(backtrackingWindow).isAfter(latestBacktracking.timestamp))
82
+ latest.timestamp.minus(backtrackingWindow)
83
+ else if (backtracking)
84
+ latestBacktracking.timestamp
85
+ else
86
+ latest.timestamp
71
87
72
88
def nextQueryFromSeqNr : Option [Long ] =
73
89
if (backtracking) highestSeenSeqNr(previousBacktracking, latestBacktracking)
74
90
else highestSeenSeqNr(previous, latest)
75
91
76
- def nextQueryToTimestamp (atLeastNumberOfEvents : Int ): Option [Instant ] = {
77
- buckets.findTimeForLimit(nextQueryFromTimestamp, atLeastNumberOfEvents) match {
92
+ def nextQueryToTimestamp (backtrackingWindow : JDuration , atLeastNumberOfEvents : Int ): Option [Instant ] = {
93
+ buckets.findTimeForLimit(nextQueryFromTimestamp(backtrackingWindow) , atLeastNumberOfEvents) match {
78
94
case Some (t) =>
79
95
if (backtracking)
80
96
if (t.isAfter(latest.timestamp)) Some (latest.timestamp) else Some (t)
@@ -208,15 +224,18 @@ import org.slf4j.Logger
208
224
dao : BySliceQuery .Dao [Row ],
209
225
createEnvelope : (TimestampOffset , Row ) => Envelope ,
210
226
extractOffset : Envelope => TimestampOffset ,
227
+ createHeartbeat : Instant => Option [Envelope ],
228
+ clock : Clock ,
211
229
settings : R2dbcSettings ,
212
230
log : Logger )(implicit val ec : ExecutionContext ) {
213
231
import BySliceQuery ._
214
232
import TimestampOffset .toTimestampOffset
215
233
216
234
private val backtrackingWindow = JDuration .ofMillis(settings.querySettings.backtrackingWindow.toMillis)
217
235
private val halfBacktrackingWindow = backtrackingWindow.dividedBy(2 )
218
- private val firstBacktrackingQueryWindow =
219
- backtrackingWindow.plus(JDuration .ofMillis(settings.querySettings.backtrackingBehindCurrentTime.toMillis))
236
+ private val backtrackingBehindCurrentTime =
237
+ JDuration .ofMillis(settings.querySettings.backtrackingBehindCurrentTime.toMillis)
238
+ private val firstBacktrackingQueryWindow = backtrackingWindow.plus(backtrackingBehindCurrentTime)
220
239
private val eventBucketCountInterval = JDuration .ofSeconds(60 )
221
240
222
241
def currentBySlices (
@@ -228,8 +247,12 @@ import org.slf4j.Logger
228
247
filterEventsBeforeSnapshots : (String , Long , String ) => Boolean = (_, _, _) => true ): Source [Envelope , NotUsed ] = {
229
248
val initialOffset = toTimestampOffset(offset)
230
249
231
- def nextOffset (state : QueryState , envelope : Envelope ): QueryState =
232
- state.copy(latest = extractOffset(envelope), rowCount = state.rowCount + 1 )
250
+ def nextOffset (state : QueryState , envelope : Envelope ): QueryState = {
251
+ if (EnvelopeOrigin .isHeartbeatEvent(envelope))
252
+ state
253
+ else
254
+ state.copy(latest = extractOffset(envelope), rowCount = state.rowCount + 1 )
255
+ }
233
256
234
257
def nextQuery (state : QueryState , endTimestamp : Instant ): (QueryState , Option [Source [Envelope , NotUsed ]]) = {
235
258
// Note that we can't know how many events with the same timestamp that are filtered out
@@ -241,7 +264,7 @@ import org.slf4j.Logger
241
264
val fromTimestamp = state.latest.timestamp
242
265
val fromSeqNr = highestSeenSeqNr(state.previous, state.latest)
243
266
244
- val toTimestamp = newState.nextQueryToTimestamp(settings.querySettings.bufferSize) match {
267
+ val toTimestamp = newState.nextQueryToTimestamp(backtrackingWindow, settings.querySettings.bufferSize) match {
245
268
case Some (t) =>
246
269
if (t.isBefore(endTimestamp)) t else endTimestamp
247
270
case None =>
@@ -333,45 +356,49 @@ import org.slf4j.Logger
333
356
initialOffset.timestamp)
334
357
335
358
def nextOffset (state : QueryState , envelope : Envelope ): QueryState = {
336
- val offset = extractOffset(envelope)
337
- if (state.backtracking) {
338
- if (offset.timestamp.isBefore(state.latestBacktracking.timestamp))
339
- throw new IllegalArgumentException (
340
- s " Unexpected offset [ $offset] before latestBacktracking [ ${state.latestBacktracking}]. " )
341
-
342
- val newSeenCount =
343
- if (offset.timestamp == state.latestBacktracking.timestamp &&
344
- highestSeenSeqNr(state.previousBacktracking, offset) ==
345
- highestSeenSeqNr(state.previousBacktracking, state.latestBacktracking))
346
- state.latestBacktrackingSeenCount + 1
347
- else 1
348
-
349
- state.copy(
350
- latestBacktracking = offset,
351
- latestBacktrackingSeenCount = newSeenCount,
352
- rowCount = state.rowCount + 1 )
359
+ if (EnvelopeOrigin .isHeartbeatEvent(envelope))
360
+ state
361
+ else {
362
+ val offset = extractOffset(envelope)
363
+ if (state.backtracking) {
364
+ if (offset.timestamp.isBefore(state.latestBacktracking.timestamp))
365
+ throw new IllegalArgumentException (
366
+ s " Unexpected offset [ $offset] before latestBacktracking [ ${state.latestBacktracking}]. " )
367
+
368
+ val newSeenCount =
369
+ if (offset.timestamp == state.latestBacktracking.timestamp &&
370
+ highestSeenSeqNr(state.previousBacktracking, offset) ==
371
+ highestSeenSeqNr(state.previousBacktracking, state.latestBacktracking))
372
+ state.latestBacktrackingSeenCount + 1
373
+ else 1
353
374
354
- } else {
355
- if (offset.timestamp.isBefore(state.latest.timestamp))
356
- throw new IllegalArgumentException (s " Unexpected offset [ $offset] before latest [ ${state.latest}]. " )
375
+ state.copy(
376
+ latestBacktracking = offset,
377
+ latestBacktrackingSeenCount = newSeenCount,
378
+ rowCount = state.rowCount + 1 )
357
379
358
- if (log.isDebugEnabled()) {
359
- if (state.latestBacktracking.seen.nonEmpty &&
360
- offset.timestamp.isAfter(state.latestBacktracking.timestamp.plus(firstBacktrackingQueryWindow)))
361
- log.debug(
362
- " {} next offset is outside the backtracking window, latestBacktracking: [{}], offset: [{}]" ,
363
- logPrefix,
364
- state.latestBacktracking,
365
- offset)
366
- }
380
+ } else {
381
+ if (offset.timestamp.isBefore(state.latest.timestamp))
382
+ throw new IllegalArgumentException (s " Unexpected offset [ $offset] before latest [ ${state.latest}]. " )
383
+
384
+ if (log.isDebugEnabled()) {
385
+ if (state.latestBacktracking.seen.nonEmpty &&
386
+ offset.timestamp.isAfter(state.latestBacktracking.timestamp.plus(firstBacktrackingQueryWindow)))
387
+ log.debug(
388
+ " {} next offset is outside the backtracking window, latestBacktracking: [{}], offset: [{}]" ,
389
+ logPrefix,
390
+ state.latestBacktracking,
391
+ offset)
392
+ }
367
393
368
- state.copy(latest = offset, rowCount = state.rowCount + 1 )
394
+ state.copy(latest = offset, rowCount = state.rowCount + 1 )
395
+ }
369
396
}
370
397
}
371
398
372
399
def delayNextQuery (state : QueryState ): Option [FiniteDuration ] = {
373
400
if (switchFromBacktracking(state)) {
374
- // switch from from backtracking immediately
401
+ // switch from backtracking immediately
375
402
None
376
403
} else {
377
404
val delay = ContinuousQuery .adjustNextDelay(
@@ -398,20 +425,38 @@ import org.slf4j.Logger
398
425
state.backtracking && state.rowCount < settings.querySettings.bufferSize - state.backtrackingExpectFiltered
399
426
}
400
427
428
+ def switchToBacktracking (state : QueryState , newIdleCount : Long ): Boolean = {
429
+ // Note that when starting the query with offset = NoOffset it will switch to backtracking immediately after
430
+ // the first normal query because between(latestBacktracking.timestamp, latest.timestamp) > halfBacktrackingWindow
431
+
432
+ val qSettings = settings.querySettings
433
+
434
+ def disableBacktrackingWhenFarBehindCurrentWallClockTime : Boolean = {
435
+ val aheadOfInitial =
436
+ initialOffset == TimestampOffset .Zero || state.latestBacktracking.timestamp.isAfter(initialOffset.timestamp)
437
+ val previousTimestamp =
438
+ if (state.previous == TimestampOffset .Zero ) state.latest.timestamp else state.previous.timestamp
439
+ aheadOfInitial &&
440
+ previousTimestamp.isBefore(clock.instant().minus(firstBacktrackingQueryWindow))
441
+ }
442
+
443
+ qSettings.backtrackingEnabled &&
444
+ ! state.backtracking &&
445
+ state.latest != TimestampOffset .Zero &&
446
+ ! disableBacktrackingWhenFarBehindCurrentWallClockTime &&
447
+ (newIdleCount >= 5 ||
448
+ state.rowCountSinceBacktracking + state.rowCount >= qSettings.bufferSize * 3 ||
449
+ JDuration
450
+ .between(state.latestBacktracking.timestamp, state.latest.timestamp)
451
+ .compareTo(halfBacktrackingWindow) > 0 )
452
+ }
453
+
401
454
def nextQuery (state : QueryState ): (QueryState , Option [Source [Envelope , NotUsed ]]) = {
402
455
val newIdleCount = if (state.rowCount == 0 ) state.idleCount + 1 else 0
456
+ // only start tracking query wall clock (for heartbeats) after initial backtracking query
457
+ val newQueryWallClock = if (state.latestBacktracking != TimestampOffset .Zero ) clock.instant() else Instant .EPOCH
403
458
val newState =
404
- if (settings.querySettings.backtrackingEnabled && ! state.backtracking && state.latest != TimestampOffset .Zero &&
405
- (newIdleCount >= 5 ||
406
- state.rowCountSinceBacktracking + state.rowCount >= settings.querySettings.bufferSize * 3 ||
407
- JDuration
408
- .between(state.latestBacktracking.timestamp, state.latest.timestamp)
409
- .compareTo(halfBacktrackingWindow) > 0 )) {
410
- // FIXME config for newIdleCount >= 5 and maybe something like `newIdleCount % 5 == 0`
411
-
412
- // Note that when starting the query with offset = NoOffset it will switch to backtracking immediately after
413
- // the first normal query because between(latestBacktracking.timestamp, latest.timestamp) > halfBacktrackingWindow
414
-
459
+ if (switchToBacktracking(state, newIdleCount)) {
415
460
// switching to backtracking
416
461
val fromOffset =
417
462
if (state.latestBacktracking == TimestampOffset .Zero )
@@ -426,15 +471,19 @@ import org.slf4j.Logger
426
471
idleCount = newIdleCount,
427
472
backtrackingCount = 1 ,
428
473
latestBacktracking = fromOffset,
429
- backtrackingExpectFiltered = state.latestBacktrackingSeenCount)
474
+ backtrackingExpectFiltered = state.latestBacktrackingSeenCount,
475
+ currentQueryWallClock = newQueryWallClock,
476
+ previousQueryWallClock = state.currentQueryWallClock)
430
477
} else if (switchFromBacktracking(state)) {
431
- // switch from backtracking
478
+ // switching from backtracking
432
479
state.copy(
433
480
rowCount = 0 ,
434
481
rowCountSinceBacktracking = 0 ,
435
482
queryCount = state.queryCount + 1 ,
436
483
idleCount = newIdleCount,
437
- backtrackingCount = 0 )
484
+ backtrackingCount = 0 ,
485
+ currentQueryWallClock = newQueryWallClock,
486
+ previousQueryWallClock = state.currentQueryWallClock)
438
487
} else {
439
488
// continue
440
489
val newBacktrackingCount = if (state.backtracking) state.backtrackingCount + 1 else 0
@@ -444,16 +493,18 @@ import org.slf4j.Logger
444
493
queryCount = state.queryCount + 1 ,
445
494
idleCount = newIdleCount,
446
495
backtrackingCount = newBacktrackingCount,
447
- backtrackingExpectFiltered = state.latestBacktrackingSeenCount)
496
+ backtrackingExpectFiltered = state.latestBacktrackingSeenCount,
497
+ currentQueryWallClock = newQueryWallClock,
498
+ previousQueryWallClock = state.currentQueryWallClock)
448
499
}
449
500
450
501
val behindCurrentTime =
451
502
if (newState.backtracking) settings.querySettings.backtrackingBehindCurrentTime
452
503
else settings.querySettings.behindCurrentTime
453
504
454
- val fromTimestamp = newState.nextQueryFromTimestamp
505
+ val fromTimestamp = newState.nextQueryFromTimestamp(backtrackingWindow)
455
506
val fromSeqNr = newState.nextQueryFromSeqNr
456
- val toTimestamp = newState.nextQueryToTimestamp(settings.querySettings.bufferSize)
507
+ val toTimestamp = newState.nextQueryToTimestamp(backtrackingWindow, settings.querySettings.bufferSize)
457
508
458
509
if (log.isDebugEnabled()) {
459
510
val backtrackingInfo =
@@ -501,12 +552,38 @@ import org.slf4j.Logger
501
552
.via(deserializeAndAddOffset(newState.currentOffset)))
502
553
}
503
554
504
- ContinuousQuery [QueryState , Envelope ](
505
- initialState = QueryState .empty.copy(latest = initialOffset),
506
- updateState = nextOffset,
507
- delayNextQuery = delayNextQuery,
508
- nextQuery = nextQuery,
509
- beforeQuery = beforeQuery(logPrefix, entityType, minSlice, maxSlice, _))
555
+ def heartbeat (state : QueryState ): Option [Envelope ] = {
556
+ if (state.idleCount >= 1 && state.previousQueryWallClock != Instant .EPOCH ) {
557
+ // using wall clock to measure duration since the start time (database timestamp) up to idle backtracking limit
558
+ val timestamp = state.startTimestamp.plus(
559
+ JDuration .between(state.startWallClock, state.previousQueryWallClock.minus(backtrackingBehindCurrentTime)))
560
+ createHeartbeat(timestamp)
561
+ } else
562
+ None
563
+ }
564
+
565
+ val nextHeartbeat : QueryState => Option [Envelope ] =
566
+ if (settings.journalPublishEvents) heartbeat else _ => None
567
+
568
+ val currentTimestamp =
569
+ if (settings.useAppTimestamp) Future .successful(InstantFactory .now())
570
+ else dao.currentDbTimestamp(minSlice)
571
+
572
+ Source
573
+ .futureSource[Envelope , NotUsed ] {
574
+ currentTimestamp.map { currentTime =>
575
+ val currentWallClock = clock.instant()
576
+ ContinuousQuery [QueryState , Envelope ](
577
+ initialState = QueryState .empty
578
+ .copy(latest = initialOffset, startTimestamp = currentTime, startWallClock = currentWallClock),
579
+ updateState = nextOffset,
580
+ delayNextQuery = delayNextQuery,
581
+ nextQuery = nextQuery,
582
+ beforeQuery = beforeQuery(logPrefix, entityType, minSlice, maxSlice, _),
583
+ heartbeat = nextHeartbeat)
584
+ }
585
+ }
586
+ .mapMaterializedValue(_ => NotUsed )
510
587
}
511
588
512
589
private def beforeQuery (
0 commit comments