Skip to content

Commit

Permalink
Merge branch 'main' into batch_listener_not_support_observation
Browse files Browse the repository at this point in the history
  • Loading branch information
sobychacko authored Feb 21, 2024
2 parents 6400003 + 7a53b01 commit 7b859cb
Show file tree
Hide file tree
Showing 8 changed files with 112 additions and 92 deletions.
6 changes: 3 additions & 3 deletions build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -63,13 +63,13 @@ ext {
kotlinCoroutinesVersion = '1.7.3'
log4jVersion = '2.22.1'
micrometerDocsVersion = '1.0.2'
micrometerVersion = '1.13.0-M1'
micrometerTracingVersion = '1.3.0-M1'
micrometerVersion = '1.13.0-SNAPSHOT'
micrometerTracingVersion = '1.3.0-SNAPSHOT'
mockitoVersion = '5.8.0'
reactorVersion = '2023.0.3'
scalaVersion = '2.13'
springBootVersion = '3.2.2' // docs module
springDataVersion = '2024.0.0-M1'
springDataVersion = '2024.0.0-SNAPSHOT'
springRetryVersion = '2.0.5'
springVersion = '6.1.4'
zookeeperVersion = '3.8.3'
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,7 @@ In either case, you should NOT perform any seeks on the consumer because the con

Starting with version 2.8, the legacy `ErrorHandler` and `BatchErrorHandler` interfaces have been superseded by a new `CommonErrorHandler`.
These error handlers can handle errors for both record and batch listeners, allowing a single listener container factory to create containers for both types of listener.
`CommonErrorHandler` implementations to replace most legacy framework error handler implementations are provided and the legacy error handlers deprecated.
The legacy interfaces are still supported by listener containers and listener container factories; they will be deprecated in a future release.
`CommonErrorHandler` implementations to replace most legacy framework error handler implementations are provided.

See xref:kafka/annotation-error-handling.adoc#migrating-legacy-eh[Migrating Custom Legacy Error Handler Implementations to `CommonErrorHandler`] for information to migrate custom error handlers to `CommonErrorHandler`.

Expand Down Expand Up @@ -425,7 +424,7 @@ To replace any `BatchErrorHandler` implementation, you should implement `handleB
You should also implement `handleOtherException()` - to handle exceptions that occur outside the scope of record processing (e.g. consumer errors).

[[after-rollback]]
== After-rollback Processor
== After Rollback Processor

When using transactions, if the listener throws an exception (and an error handler, if present, throws an exception), the transaction is rolled back.
By default, any unprocessed records (including the failed record) are re-fetched on the next poll.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,14 +30,18 @@
See the JavaDocs for `ContainerProperties.AssignmentCommitOption` for more information about the available options.

|[[asyncAcks]]<<asyncAcks,`asyncAcks`>>
|false
|`false`
|Enable out-of-order commits (see xref:kafka/receiving-messages/ooo-commits.adoc[Manually Committing Offsets]); the consumer is paused and commits are deferred until gaps are filled.

|[[authExceptionRetryInterval]]<<authExceptionRetryInterval,`authExceptionRetryInterval`>>
|`null`
|When not null, a `Duration` to sleep between polls when an `AuthenticationException` or `AuthorizationException` is thrown by the Kafka client.
When null, such exceptions are considered fatal and the container will stop.

|[[batchRecoverAfterRollback]]<<batchRecoverAfterRollback,`batchRecoverAfterRollback`>>
|`false`
|Set to `true` to enable batch recovery, See xref:kafka/annotation-error-handling.adoc#after-rollback[After Rollback Processor].

|[[clientId]]<<clientId,`clientId`>>
|(empty string)
|A prefix for the `client.id` consumer property.
Expand All @@ -57,10 +61,6 @@ Useful when the consumer code cannot determine that an `ErrorHandlingDeserialize
|`null`
|When present and `syncCommits` is `false` a callback invoked after the commit completes.

|[[offsetAndMetadataProvider]]<<offsetAndMetadataProvider,`offsetAndMetadataProvider`>>
|`null`
|A provider for `OffsetAndMetadata`; by default, the provider creates an offset and metadata with empty metadata. The provider gives a way to customize the metadata.

|[[commitLogLevel]]<<commitLogLevel,`commitLogLevel`>>
|DEBUG
|The logging level for logs pertaining to committing offsets.
Expand All @@ -69,15 +69,15 @@ Useful when the consumer code cannot determine that an `ErrorHandlingDeserialize
|`null`
|A rebalance listener; see xref:kafka/receiving-messages/rebalance-listeners.adoc[Rebalancing Listeners].

|[[consumerStartTimout]]<<consumerStartTimout,`consumerStartTimout`>>
|[[commitRetries]]<<commitRetries,`commitRetries`>>
|3
|Set the number of retries `RetriableCommitFailedException` when using `syncCommits` set to true.
Default 3 (4-attempt total).

|[[consumerStartTimeout]]<<consumerStartTimeout,`consumerStartTimeout`>>
|30s
|The time to wait for the consumer to start before logging an error; this might happen if, say, you use a task executor with insufficient threads.

|[[consumerTaskExecutor]]<<consumerTaskExecutor,`consumerTaskExecutor`>>
|`SimpleAsyncTaskExecutor`
|A task executor to run the consumer threads.
The default executor creates threads named `<name>-C-n`; with the `KafkaMessageListenerContainer`, the name is the bean name; with the `ConcurrentMessageListenerContainer` the name is the bean name suffixed with `-n` where n is incremented for each child container.

|[[deliveryAttemptHeader]]<<deliveryAttemptHeader,`deliveryAttemptHeader`>>
|`false`
|See xref:kafka/annotation-error-handling.adoc#delivery-header[Delivery Attempts Header].
Expand Down Expand Up @@ -123,9 +123,14 @@ Also see `idleBeforeDataMultiplier`.
|None
|Used to override any arbitrary consumer properties configured on the consumer factory.

|[[listenerTaskExecutor]]<<listenerTaskExecutor,`listenerTaskExecutor`>>
|`SimpleAsyncTaskExecutor`
|A task executor to run the consumer threads.
The default executor creates threads named `<name>-C-n`; with the `KafkaMessageListenerContainer`, the name is the bean name; with the `ConcurrentMessageListenerContainer` the name is the bean name suffixed with `-n` where n is incremented for each child container.

|[[logContainerConfig]]<<logContainerConfig,`logContainerConfig`>>
|`false`
|Set to true to log at INFO level all container properties.
|Set to `true` to log at INFO level all container properties.

|[[messageListener]]<<messageListener,`messageListener`>>
|`null`
Expand All @@ -145,7 +150,7 @@ Also see `idleBeforeDataMultiplier`.

|[[missingTopicsFatal]]<<missingTopicsFatal,`missingTopicsFatal`>>
|`false`
|When true prevents the container from starting if the confifgured topic(s) are not present on the broker.
|When true prevents the container from starting if the configured topic(s) are not present on the broker.

|[[monitorInterval]]<<monitorInterval,`monitorInterval`>>
|30s
Expand All @@ -157,9 +162,21 @@ See `noPollThreshold` and `pollTimeout`.
|Multiplied by `pollTimeOut` to determine whether to publish a `NonResponsiveConsumerEvent`.
See `monitorInterval`.

|[[observationConvention]]<<observationConvention,`observationConvention`>>
|`null`
|When set, add dynamic tags to the timers and traces, based on information in the consumer records.

|[[observationEnabled]]<<observationEnabled,`observationEnabled`>>
|`false`
|Set to `true` to enable observation via Micrometer.

|[[offsetAndMetadataProvider]]<<offsetAndMetadataProvider,`offsetAndMetadataProvider`>>
|`null`
|A provider for `OffsetAndMetadata`; by default, the provider creates an offset and metadata with empty metadata. The provider gives a way to customize the metadata.

|[[onlyLogRecordMetadata]]<<onlyLogRecordMetadata,`onlyLogRecordMetadata`>>
|`false`
|Set to false to log the complete consumer record (in error, debug logs etc) instead of just `topic-partition@offset`.
|Set to `false` to log the complete consumer record (in error, debug logs etc.) instead of just `topic-partition@offset`.

|[[pauseImmediate]]<<pauseImmediate,`pauseImmediate`>>
|`false`
Expand Down Expand Up @@ -256,14 +273,6 @@ See xref:kafka/annotation-error-handling.adoc#error-handlers[Container Error Han
|`ContainerProperties`
|The container properties instance.

|[[errorHandler]]<<errorHandler,`errorHandler`>>
|See desc.
|Deprecated - see `commonErrorHandler`.

|[[genericErrorHandler]]<<genericErrorHandler,`genericErrorHandler`>>
|See desc.
|Deprecated - see `commonErrorHandler`.

|[[groupId2]]<<groupId2,`groupId`>>
|See desc.
|The `containerProperties.groupId`, if present, otherwise the `group.id` property from the consumer factory.
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ NOTE: With the concurrent container, timers are created for each thread and the
[[monitoring-kafkatemplate-performance]]
== Monitoring KafkaTemplate Performance

Starting with version 2.5, the template will automatically create and update Micrometer `Timer`+++s for send operations, if `Micrometer` is detected on the classpath, and a single `MeterRegistry` is present in the application context.
Starting with version 2.5, the template will automatically create and update Micrometer `Timer`+++s+++ for send operations, if `Micrometer` is detected on the classpath, and a single `MeterRegistry` is present in the application context.
The timers can be disabled by setting the template's `micrometerEnabled` property to `false`.

Two timers are maintained - one for successful calls to the listener and one for failures.
Expand Down Expand Up @@ -111,6 +111,6 @@ Starting with version 3.0.6, you can add dynamic tags to the timers and traces,
To do so, add a custom `KafkaListenerObservationConvention` and/or `KafkaTemplateObservationConvention` to the listener container properties or `KafkaTemplate` respectively.
The `record` property in both observation contexts contains the `ConsumerRecord` or `ProducerRecord` respectively.

The sender and receiver contexts' `remoteServiceName` properties are set to the Kafka `clusterId` property; this is retrieved by a `KafkaAdmin`.
The sender and receiver contexts `remoteServiceName` properties are set to the Kafka `clusterId` property; this is retrieved by a `KafkaAdmin`.
If, for some reason - perhaps lack of admin permissions, you cannot retrieve the cluster id, starting with version 3.1, you can set a manual `clusterId` on the `KafkaAdmin` and inject it into `KafkaTemplate` s and listener containers.
When it is `null` (default), the admin will invoke the `describeCluster` admin operation to retrieve it from the broker.
Original file line number Diff line number Diff line change
Expand Up @@ -13,11 +13,11 @@ Starting with version 2.1.5, you can call `isPauseRequested()` to see if `pause(
However, the consumers might not have actually paused yet.
`isConsumerPaused()` returns true if all `Consumer` instances have actually paused.

In addition (also since 2.1.5), `ConsumerPausedEvent` and `ConsumerResumedEvent` instances are published with the container as the `source` property and the `TopicPartition` instances involved in the `partitions` property.
In addition(also since 2.1.5), `ConsumerPausedEvent` and `ConsumerResumedEvent` instances are published with the container as the `source` property and the `TopicPartition` instances involved in the `partitions` property.

Starting with version 2.9, a new container property `pauseImmediate`, when set to true, causes the pause to take effect after the current record is processed.
By default, the pause takes effect when all of the records from the previous poll have been processed.
See <<pauseImmediate>>.
By default, the pause takes effect when all the records from the previous poll have been processed.
See xref:kafka/container-props.adoc#pauseImmediate[pauseImmediate].

The following simple Spring Boot application demonstrates by using the container registry to get a reference to a `@KafkaListener` method's container and pausing or resuming its consumers as well as receiving the corresponding events:

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1059,6 +1059,7 @@ public String toString() {
+ "\n ackMode=" + this.ackMode
+ "\n ackCount=" + this.ackCount
+ "\n ackTime=" + this.ackTime
+ "\n consumerStartTimeout=" + this.consumerStartTimeout
+ "\n messageListener=" + this.messageListener
+ (this.listenerTaskExecutor != null
? "\n listenerTaskExecutor=" + this.listenerTaskExecutor
Expand All @@ -1074,16 +1075,21 @@ public String toString() {
+ "\n monitorInterval=" + this.monitorInterval
+ (this.scheduler != null ? "\n scheduler=" + this.scheduler : "")
+ "\n noPollThreshold=" + this.noPollThreshold
+ "\n pauseImmediate=" + this.pauseImmediate
+ "\n pollTimeoutWhilePaused=" + this.pollTimeoutWhilePaused
+ "\n subBatchPerPartition=" + this.subBatchPerPartition
+ "\n assignmentCommitOption=" + this.assignmentCommitOption
+ "\n deliveryAttemptHeader=" + this.deliveryAttemptHeader
+ "\n batchRecoverAfterRollback=" + this.batchRecoverAfterRollback
+ "\n eosMode=" + this.eosMode
+ "\n transactionDefinition=" + this.transactionDefinition
+ "\n stopContainerWhenFenced=" + this.stopContainerWhenFenced
+ "\n stopImmediate=" + this.stopImmediate
+ "\n asyncAcks=" + this.asyncAcks
+ "\n logContainerConfig=" + this.logContainerConfig
+ "\n missingTopicsFatal=" + this.missingTopicsFatal
+ "\n idleBeforeDataMultiplier=" + this.idleBeforeDataMultiplier
+ "\n idleBetweenPolls=" + this.idleBetweenPolls
+ "\n micrometerEnabled=" + this.micrometerEnabled
+ "\n observationEnabled=" + this.observationEnabled
+ (this.observationConvention != null
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3091,36 +3091,36 @@ private void processSeeks() {
traceSeek(offset);
try {
SeekPosition position = offset.getPosition();
TopicPartition topicPartition = offset.getTopicPartition();
Long whereTo = offset.getOffset();
if (position == null) {
if (offset.isRelativeToCurrent()) {
whereTo += this.consumer.position(offset.getTopicPartition());
whereTo += this.consumer.position(topicPartition);
whereTo = Math.max(whereTo, 0);
}
this.consumer.seek(offset.getTopicPartition(), whereTo);
this.consumer.seek(topicPartition, whereTo);
}
else if (position.equals(SeekPosition.BEGINNING)) {
this.consumer.seekToBeginning(Collections.singletonList(offset.getTopicPartition()));
if (whereTo != null) {
this.consumer.seek(offset.getTopicPartition(), whereTo);
}
}
else if (position.equals(SeekPosition.TIMESTAMP)) {
else if (SeekPosition.TIMESTAMP.equals(position)) {
// possible late addition since the grouped processing above
Map<TopicPartition, OffsetAndTimestamp> offsetsForTimes = this.consumer
.offsetsForTimes(
Collections.singletonMap(offset.getTopicPartition(), offset.getOffset()));
Collections.singletonMap(topicPartition, offset.getOffset()));
offsetsForTimes.forEach((tp, ot) -> {
if (ot != null) {
this.consumer.seek(tp, ot.offset());
}
});
}
else {
this.consumer.seekToEnd(Collections.singletonList(offset.getTopicPartition()));
if (SeekPosition.BEGINNING.equals(position)) {
this.consumer.seekToBeginning(Collections.singletonList(topicPartition));
}
else {
this.consumer.seekToEnd(Collections.singletonList(topicPartition));
}
if (whereTo != null) {
whereTo += this.consumer.position(offset.getTopicPartition());
this.consumer.seek(offset.getTopicPartition(), whereTo);
whereTo += this.consumer.position(topicPartition);
this.consumer.seek(topicPartition, whereTo);
}
}
}
Expand Down Expand Up @@ -3354,7 +3354,7 @@ public void seekToEnd(Collection<TopicPartition> partitions) {
@Override
public void seekRelative(String topic, int partition, long offset, boolean toCurrent) {
if (toCurrent) {
this.seeks.add(new TopicPartitionOffset(topic, partition, offset, toCurrent));
this.seeks.add(new TopicPartitionOffset(topic, partition, offset, true));
}
else if (offset >= 0) {
this.seeks.add(new TopicPartitionOffset(topic, partition, offset, SeekPosition.BEGINNING));
Expand Down
Loading

0 comments on commit 7b859cb

Please sign in to comment.