org.apache.kafka.clients.admin.internals.DescribeTransactionsHandler Maven / Gradle / Ivy
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.kafka.clients.admin.internals;
import org.apache.kafka.clients.admin.TransactionDescription;
import org.apache.kafka.clients.admin.TransactionState;
import org.apache.kafka.common.Node;
import org.apache.kafka.common.TopicPartition;
import org.apache.kafka.common.errors.TransactionalIdAuthorizationException;
import org.apache.kafka.common.errors.TransactionalIdNotFoundException;
import org.apache.kafka.common.message.DescribeTransactionsRequestData;
import org.apache.kafka.common.message.DescribeTransactionsResponseData;
import org.apache.kafka.common.protocol.Errors;
import org.apache.kafka.common.requests.AbstractResponse;
import org.apache.kafka.common.requests.DescribeTransactionsRequest;
import org.apache.kafka.common.requests.DescribeTransactionsResponse;
import org.apache.kafka.common.requests.FindCoordinatorRequest;
import org.apache.kafka.common.requests.FindCoordinatorRequest.CoordinatorType;
import org.apache.kafka.common.utils.LogContext;
import org.slf4j.Logger;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.OptionalLong;
import java.util.Set;
import java.util.stream.Collectors;
public class DescribeTransactionsHandler extends AdminApiHandler.Batched {
private final Logger log;
private final AdminApiLookupStrategy lookupStrategy;
public DescribeTransactionsHandler(
LogContext logContext
) {
this.log = logContext.logger(DescribeTransactionsHandler.class);
this.lookupStrategy = new CoordinatorStrategy(CoordinatorType.TRANSACTION, logContext);
}
public static AdminApiFuture.SimpleAdminApiFuture newFuture(
Collection transactionalIds
) {
return AdminApiFuture.forKeys(buildKeySet(transactionalIds));
}
private static Set buildKeySet(Collection transactionalIds) {
return transactionalIds.stream()
.map(CoordinatorKey::byTransactionalId)
.collect(Collectors.toSet());
}
@Override
public String apiName() {
return "describeTransactions";
}
@Override
public AdminApiLookupStrategy lookupStrategy() {
return lookupStrategy;
}
@Override
public DescribeTransactionsRequest.Builder buildBatchedRequest(
int brokerId,
Set keys
) {
DescribeTransactionsRequestData request = new DescribeTransactionsRequestData();
List transactionalIds = keys.stream().map(key -> {
if (key.type != FindCoordinatorRequest.CoordinatorType.TRANSACTION) {
throw new IllegalArgumentException("Invalid group coordinator key " + key +
" when building `DescribeTransaction` request");
}
return key.idValue;
}).collect(Collectors.toList());
request.setTransactionalIds(transactionalIds);
return new DescribeTransactionsRequest.Builder(request);
}
@Override
public ApiResult handleResponse(
Node broker,
Set keys,
AbstractResponse abstractResponse
) {
DescribeTransactionsResponse response = (DescribeTransactionsResponse) abstractResponse;
Map completed = new HashMap<>();
Map failed = new HashMap<>();
List unmapped = new ArrayList<>();
for (DescribeTransactionsResponseData.TransactionState transactionState : response.data().transactionStates()) {
CoordinatorKey transactionalIdKey = CoordinatorKey.byTransactionalId(
transactionState.transactionalId());
if (!keys.contains(transactionalIdKey)) {
log.warn("Response included transactionalId `{}`, which was not requested",
transactionState.transactionalId());
continue;
}
Errors error = Errors.forCode(transactionState.errorCode());
if (error != Errors.NONE) {
handleError(transactionalIdKey, error, failed, unmapped);
continue;
}
OptionalLong transactionStartTimeMs = transactionState.transactionStartTimeMs() < 0 ?
OptionalLong.empty() :
OptionalLong.of(transactionState.transactionStartTimeMs());
completed.put(transactionalIdKey, new TransactionDescription(
broker.id(),
TransactionState.parse(transactionState.transactionState()),
transactionState.producerId(),
transactionState.producerEpoch(),
transactionState.transactionTimeoutMs(),
transactionStartTimeMs,
collectTopicPartitions(transactionState)
));
}
return new ApiResult<>(completed, failed, unmapped);
}
private Set collectTopicPartitions(
DescribeTransactionsResponseData.TransactionState transactionState
) {
Set res = new HashSet<>();
for (DescribeTransactionsResponseData.TopicData topicData : transactionState.topics()) {
String topic = topicData.topic();
for (Integer partitionId : topicData.partitions()) {
res.add(new TopicPartition(topic, partitionId));
}
}
return res;
}
private void handleError(
CoordinatorKey transactionalIdKey,
Errors error,
Map failed,
List unmapped
) {
switch (error) {
case TRANSACTIONAL_ID_AUTHORIZATION_FAILED:
failed.put(transactionalIdKey, new TransactionalIdAuthorizationException(
"DescribeTransactions request for transactionalId `" + transactionalIdKey.idValue + "` " +
"failed due to authorization failure"));
break;
case TRANSACTIONAL_ID_NOT_FOUND:
failed.put(transactionalIdKey, new TransactionalIdNotFoundException(
"DescribeTransactions request for transactionalId `" + transactionalIdKey.idValue + "` " +
"failed because the ID could not be found"));
break;
case COORDINATOR_LOAD_IN_PROGRESS:
// If the coordinator is in the middle of loading, then we just need to retry
log.debug("DescribeTransactions request for transactionalId `{}` failed because the " +
"coordinator is still in the process of loading state. Will retry",
transactionalIdKey.idValue);
break;
case NOT_COORDINATOR:
case COORDINATOR_NOT_AVAILABLE:
// If the coordinator is unavailable or there was a coordinator change, then we unmap
// the key so that we retry the `FindCoordinator` request
unmapped.add(transactionalIdKey);
log.debug("DescribeTransactions request for transactionalId `{}` returned error {}. Will attempt " +
"to find the coordinator again and retry", transactionalIdKey.idValue, error);
break;
default:
failed.put(transactionalIdKey, error.exception("DescribeTransactions request for " +
"transactionalId `" + transactionalIdKey.idValue + "` failed due to unexpected error"));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy