Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.apache.camel.component.aws2.sqs.Sqs2Consumer Maven / Gradle / Ivy
/*
* 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.camel.component.aws2.sqs;
import java.io.Closeable;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Queue;
import java.util.UUID;
import java.util.concurrent.Callable;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.regex.Pattern;
import org.apache.camel.AsyncCallback;
import org.apache.camel.Exchange;
import org.apache.camel.ExchangePattern;
import org.apache.camel.ExchangePropertyKey;
import org.apache.camel.Message;
import org.apache.camel.Processor;
import org.apache.camel.clock.Clock;
import org.apache.camel.spi.ExecutorServiceManager;
import org.apache.camel.spi.HeaderFilterStrategy;
import org.apache.camel.spi.ScheduledPollConsumerScheduler;
import org.apache.camel.spi.Synchronization;
import org.apache.camel.spi.ThreadPoolProfile;
import org.apache.camel.support.DefaultScheduledPollConsumerScheduler;
import org.apache.camel.support.ScheduledBatchPollingConsumer;
import org.apache.camel.util.CastUtils;
import org.apache.camel.util.ObjectHelper;
import org.apache.camel.util.URISupport;
import org.apache.commons.io.function.IOConsumer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import software.amazon.awssdk.core.exception.SdkException;
import software.amazon.awssdk.services.sqs.SqsClient;
import software.amazon.awssdk.services.sqs.model.ChangeMessageVisibilityBatchRequest;
import software.amazon.awssdk.services.sqs.model.ChangeMessageVisibilityBatchRequestEntry;
import software.amazon.awssdk.services.sqs.model.DeleteMessageRequest;
import software.amazon.awssdk.services.sqs.model.MessageAttributeValue;
import software.amazon.awssdk.services.sqs.model.MessageNotInflightException;
import software.amazon.awssdk.services.sqs.model.MessageSystemAttributeName;
import software.amazon.awssdk.services.sqs.model.QueueDeletedRecentlyException;
import software.amazon.awssdk.services.sqs.model.QueueDoesNotExistException;
import software.amazon.awssdk.services.sqs.model.ReceiptHandleIsInvalidException;
import software.amazon.awssdk.services.sqs.model.ReceiveMessageRequest;
import software.amazon.awssdk.services.sqs.model.SqsException;
import static java.util.Collections.emptyList;
import static java.util.Collections.unmodifiableList;
import static java.util.Comparator.comparing;
/**
* A Consumer of messages from the Amazon Web Service Simple Queue Service AWS
* SQS
*/
public class Sqs2Consumer extends ScheduledBatchPollingConsumer {
private static final Logger LOG = LoggerFactory.getLogger(Sqs2Consumer.class);
private TimeoutExtender timeoutExtender;
private ScheduledFuture> scheduledFuture;
private ScheduledExecutorService scheduledExecutor;
private PollingTask pollingTask;
private final String sqsConsumerToString;
public Sqs2Consumer(Sqs2Endpoint endpoint, Processor processor) {
super(endpoint, processor);
sqsConsumerToString = "SqsConsumer[%s]".formatted(URISupport.sanitizeUri(endpoint.getEndpointUri()));
}
@Override
protected int poll() throws Exception {
// must reset for each poll
shutdownRunningTask = null;
pendingExchanges = 0;
List messages = pollingTask.call();
// okay we have some response from aws so lets mark the consumer as ready
forceConsumerAsReady();
Queue exchanges = createExchanges(messages);
return processBatch(CastUtils.cast(exchanges));
}
protected Queue createExchanges(List messages) {
if (LOG.isTraceEnabled()) {
LOG.trace("Received {} messages in this poll", messages.size());
}
Queue answer = new LinkedList<>();
for (software.amazon.awssdk.services.sqs.model.Message message : messages) {
Exchange exchange = createExchange(message);
answer.add(exchange);
}
return answer;
}
@Override
public int processBatch(Queue exchanges) throws Exception {
int total = exchanges.size();
for (int index = 0; index < total && isBatchAllowed(); index++) {
// only loop if we are started (allowed to run)
final Exchange exchange = ObjectHelper.cast(Exchange.class, exchanges.poll());
// add current index and total as properties
exchange.setProperty(ExchangePropertyKey.BATCH_INDEX, index);
exchange.setProperty(ExchangePropertyKey.BATCH_SIZE, total);
exchange.setProperty(ExchangePropertyKey.BATCH_COMPLETE, index == total - 1);
// update pending number of exchanges
pendingExchanges = total - index - 1;
if (this.timeoutExtender != null) {
timeoutExtender.add(exchange);
}
// add on completion to handle after work when the exchange is done
exchange.getExchangeExtension().addOnCompletion(new Synchronization() {
@Override
public void onComplete(Exchange exchange) {
processCommit(exchange);
}
@Override
public void onFailure(Exchange exchange) {
processRollback(exchange);
}
@Override
public String toString() {
return "SqsConsumerOnCompletion";
}
});
// use default consumer callback
AsyncCallback cb = defaultConsumerCallback(exchange, true);
getAsyncProcessor().process(exchange, cb);
}
return total;
}
/**
* Strategy to delete the message after being processed.
*
* @param exchange the exchange
*/
protected void processCommit(Exchange exchange) {
try {
if (shouldDelete(exchange)) {
String receiptHandle = exchange.getIn().getHeader(Sqs2Constants.RECEIPT_HANDLE, String.class);
DeleteMessageRequest.Builder deleteRequest
= DeleteMessageRequest.builder().queueUrl(getQueueUrl()).receiptHandle(receiptHandle);
LOG.trace("Deleting message with receipt handle {}...", receiptHandle);
getClient().deleteMessage(deleteRequest.build());
LOG.trace("Deleted message with receipt handle {}...", receiptHandle);
}
} catch (SdkException e) {
getExceptionHandler().handleException("Error occurred during deleting message. This exception is ignored.",
exchange, e);
}
}
private boolean shouldDelete(Exchange exchange) {
boolean shouldDeleteByFilter = exchange.getProperty(Sqs2Constants.SQS_DELETE_FILTERED) != null
&& getConfiguration().isDeleteIfFiltered() && passedThroughFilter(exchange);
return getConfiguration().isDeleteAfterRead() || shouldDeleteByFilter;
}
private static boolean passedThroughFilter(Exchange exchange) {
return exchange.getProperty(Sqs2Constants.SQS_DELETE_FILTERED, false, Boolean.class);
}
/**
* Strategy when processing the exchange failed.
*
* @param exchange the exchange
*/
protected void processRollback(Exchange exchange) {
Exception cause = exchange.getException();
if (cause != null) {
getExceptionHandler().handleException(
"Error during processing exchange. Will attempt to process the message on next poll.", exchange, cause);
}
}
protected Sqs2Configuration getConfiguration() {
return getEndpoint().getConfiguration();
}
protected SqsClient getClient() {
return getEndpoint().getClient();
}
protected String getQueueUrl() {
return getEndpoint().getQueueUrl();
}
@Override
public Sqs2Endpoint getEndpoint() {
return (Sqs2Endpoint) super.getEndpoint();
}
public Exchange createExchange(software.amazon.awssdk.services.sqs.model.Message msg) {
return createExchange(getEndpoint().getExchangePattern(), msg);
}
private Exchange createExchange(ExchangePattern pattern, software.amazon.awssdk.services.sqs.model.Message msg) {
Exchange exchange = createExchange(true);
exchange.setPattern(pattern);
Message message = exchange.getIn();
message.setBody(msg.body());
message.setHeaders(new HashMap<>(msg.attributesAsStrings()));
message.setHeader(Sqs2Constants.MESSAGE_ID, msg.messageId());
message.setHeader(Sqs2Constants.MD5_OF_BODY, msg.md5OfBody());
message.setHeader(Sqs2Constants.RECEIPT_HANDLE, msg.receiptHandle());
message.setHeader(Sqs2Constants.ATTRIBUTES, msg.attributes());
message.setHeader(Sqs2Constants.MESSAGE_ATTRIBUTES, msg.messageAttributes());
// Need to apply the SqsHeaderFilterStrategy this time
HeaderFilterStrategy headerFilterStrategy = getEndpoint().getHeaderFilterStrategy();
// add all sqs message attributes as camel message headers so that
// knowledge of the Sqs class MessageAttributeValue will not leak to the
// client
for (Map.Entry entry : msg.messageAttributes().entrySet()) {
String header = entry.getKey();
Object value = Sqs2MessageHelper.fromMessageAttributeValue(entry.getValue());
if (!headerFilterStrategy.applyFilterToExternalHeaders(header, value, exchange)) {
message.setHeader(header, value);
}
}
return exchange;
}
@Override
public String toString() {
return sqsConsumerToString;
}
@Override
protected void afterConfigureScheduler(ScheduledPollConsumerScheduler scheduler, boolean newScheduler) {
if (newScheduler && scheduler instanceof DefaultScheduledPollConsumerScheduler defaultScheduledPollConsumerScheduler) {
defaultScheduledPollConsumerScheduler.setConcurrentConsumers(getConfiguration().getConcurrentConsumers());
// if using concurrent consumers then resize pool to be at least
// same size
int poolSize = Math.max(defaultScheduledPollConsumerScheduler.getPoolSize(),
getConfiguration().getConcurrentConsumers());
defaultScheduledPollConsumerScheduler.setPoolSize(poolSize);
}
}
@Override
protected void doStart() throws Exception {
pollingTask = new PollingTask(getEndpoint());
// start scheduler first
if (getConfiguration().isExtendMessageVisibility() && scheduledExecutor == null) {
ThreadPoolProfile profile = new ThreadPoolProfile("SqsTimeoutExtender");
profile.setPoolSize(1);
profile.setAllowCoreThreadTimeOut(false);
// the max queue is set to be unbound as there is no way to register
// the required size. If using the Thread EIP, then the max queue
// size is equal to maxQueueSize of the consumer thread EIP+max
// thread count+consumer-thread.
// The consumer would block when this limit was reached. It is safe
// to set this queue to unbound as it will be limited by the
// consumer.
profile.setMaxQueueSize(-1);
this.scheduledExecutor = getEndpoint().getCamelContext().getExecutorServiceManager().newScheduledThreadPool(this,
"SqsTimeoutExtender", profile);
Integer visibilityTimeout = getConfiguration().getVisibilityTimeout();
if (visibilityTimeout != null && visibilityTimeout > 0) {
int delay = visibilityTimeout;
int repeatSeconds = (int) (visibilityTimeout.doubleValue() * 1.5);
this.timeoutExtender = new TimeoutExtender(repeatSeconds);
if (LOG.isDebugEnabled()) {
LOG.debug(
"Scheduled TimeoutExtender task to start after {} delay, and run with {}/{} period/repeat (seconds)",
delay, delay, repeatSeconds);
}
this.scheduledFuture
= scheduledExecutor.scheduleAtFixedRate(this.timeoutExtender, delay, delay, TimeUnit.SECONDS);
}
}
super.doStart();
}
@Override
protected void doShutdown() throws Exception {
if (timeoutExtender != null) {
timeoutExtender.cancel();
timeoutExtender = null;
}
if (scheduledFuture != null) {
scheduledFuture.cancel(true);
scheduledFuture = null;
}
if (scheduledExecutor != null) {
getEndpoint().getCamelContext().getExecutorServiceManager().shutdownNow(scheduledExecutor);
scheduledExecutor = null;
}
if (pollingTask != null) {
pollingTask.close();
pollingTask = null;
}
super.doShutdown();
}
private class TimeoutExtender implements Runnable {
private static final int MAX_REQUESTS = 10;
private final int repeatSeconds;
private final AtomicBoolean run = new AtomicBoolean(true);
private final Map entries = new ConcurrentHashMap<>();
TimeoutExtender(int repeatSeconds) {
this.repeatSeconds = repeatSeconds;
}
public void add(Exchange exchange) {
exchange.getExchangeExtension().addOnCompletion(new Synchronization() {
@Override
public void onComplete(Exchange exchange) {
remove(exchange);
}
@Override
public void onFailure(Exchange exchange) {
remove(exchange);
}
private void remove(Exchange exchange) {
LOG.trace("Removing exchangeId {} from the TimeoutExtender, processing done",
exchange.getExchangeId());
entries.remove(exchange.getExchangeId());
}
});
ChangeMessageVisibilityBatchRequestEntry entry
= ChangeMessageVisibilityBatchRequestEntry.builder()
.id(exchange.getExchangeId()).visibilityTimeout(repeatSeconds)
.receiptHandle(exchange.getIn().getHeader(Sqs2Constants.RECEIPT_HANDLE, String.class))
.build();
entries.put(exchange.getExchangeId(), entry);
}
public void cancel() {
// cancel by setting to no longer run
run.set(false);
}
@Override
public void run() {
if (run.get()) {
Queue entryQueue = new LinkedList<>(entries.values());
while (!entryQueue.isEmpty()) {
List batchEntries = new LinkedList<>();
// up to 10 requests can be sent with each ChangeMessageVisibilityBatch action
while (!entryQueue.isEmpty() && batchEntries.size() < MAX_REQUESTS) {
batchEntries.add(entryQueue.poll());
}
ChangeMessageVisibilityBatchRequest request
= ChangeMessageVisibilityBatchRequest.builder().queueUrl(getQueueUrl()).entries(batchEntries)
.build();
try {
LOG.trace("Extending visibility window by {} seconds for request entries {}", repeatSeconds,
batchEntries);
getEndpoint().getClient().changeMessageVisibilityBatch(request);
LOG.debug("Extended visibility window for request entries {}", batchEntries);
} catch (MessageNotInflightException | ReceiptHandleIsInvalidException e) {
// Ignore.
} catch (SqsException e) {
if (e.getMessage()
.contains("Message does not exist or is not available for visibility timeout change")) {
// Ignore.
} else {
logException(e, batchEntries);
}
} catch (Exception e) {
logException(e, batchEntries);
}
}
}
}
private void logException(Exception e, List entries) {
LOG.warn("Extending visibility window failed for entries {}"
+ ". Will not attempt to extend visibility further. This exception will be ignored.",
entries, e);
}
}
/**
* Task responsible for polling the messages from Amazon SQS server.
*
* Depending on the configuration, the polling may involve sending one or more receive requests in a single task
* call. The number of send requests depends on the {@link Sqs2Endpoint#getMaxMessagesPerPoll()} configuration. The
* Amazon SQS receive API has upper limit of maximum 10 messages that can be fetched with a single request. To
* enable handling greater number of messages fetched per poll, multiple requests are being send asynchronously and
* then joined together.
*
* To preserver the ordering, an optional {@link Sqs2Configuration#getSortAttributeName()} can be configured. When
* specified, all messages collected from the concurrent requests are being sorted using this attribute.
*
* In addition to that, the task is also responsible for handling auto-creation of the SQS queue, when its missing.
* The queue is created when receive request returns an error about the missing queue and the
* {@link Sqs2Configuration#isAutoCreateQueue()} is enabled. In such case, the queue will be created and the task
* will return empty list of messages.
*
* If the queue creation fails with an error related to recently deleted queue, the queue creation will be postponed
* for at least 30 seconds. To prevent task from blocking the consumer thread, the 30 second timeout is being
* checked in each task call. If the scheduled time for queue auto-creation was not reached yet, the task will
* simply return empty list of messages. Once the scheduled time is reached, another queue creation attempt will be
* made.
*/
private static class PollingTask implements Callable>, Closeable {
/**
* The maximum number of messages that can be requested in a single request to AWS SQS.
*/
private static final int MAX_NUMBER_OF_MESSAGES_PER_REQUEST = 10;
/**
* The time to wait before re-creating recently deleted queue.
*/
private static final long RECENTLY_DELETED_QUEUE_BACKOFF_TIME_MS = 30_000L;
private static final Pattern COMMA_SEPARATED_PATTERN = Pattern.compile(",");
/**
* A scheduled time for queue auto-creation, measured with {@link Clock#elapsed()} value. The value of
*
*
* 0
*
*
* means there is no schedule.
*/
private final AtomicLong queueAutoCreationScheduleTime = new AtomicLong(0L);
private final Lock lock = new ReentrantLock();
private final AtomicBoolean closed = new AtomicBoolean();
private final Clock clock;
private final SqsClient sqsClient;
private final ExecutorService requestExecutor;
private final ExecutorServiceManager executorServiceManager;
private final IOConsumer createQueueOperation;
private final String queueName;
private final String queueUrl;
private final int maxMessagesPerPoll;
private final Integer visibilityTimeout;
private final Integer waitTimeSeconds;
private final Collection attributeNames;
private final Collection messageAttributeNames;
private final int numberOfRequestsPerPoll;
private final boolean queueAutoCreationEnabled;
private final MessageSystemAttributeName sortAttributeName;
@SuppressWarnings("resource")
private PollingTask(Sqs2Endpoint endpoint) {
clock = endpoint.getClock();
sqsClient = endpoint.getClient();
executorServiceManager = endpoint.getCamelContext().getExecutorServiceManager();
createQueueOperation = endpoint::createQueue;
queueName = endpoint.getConfiguration().getQueueName();
queueUrl = endpoint.getQueueUrl();
visibilityTimeout = endpoint.getConfiguration().getVisibilityTimeout();
waitTimeSeconds = endpoint.getConfiguration().getWaitTimeSeconds();
messageAttributeNames = splitCommaSeparatedValues(endpoint.getConfiguration().getMessageAttributeNames());
sortAttributeName = getSortAttributeName(endpoint.getConfiguration());
attributeNames = getAttributeNames(endpoint.getConfiguration(), sortAttributeName);
queueAutoCreationEnabled = endpoint.getConfiguration().isAutoCreateQueue();
maxMessagesPerPoll = Math.max(1, endpoint.getMaxMessagesPerPoll());
numberOfRequestsPerPoll = computeNumberOfRequestPerPoll(maxMessagesPerPoll);
requestExecutor = executorServiceManager.newFixedThreadPool(this,
"%s[%s]".formatted(getClass().getSimpleName(), queueName),
Math.min(numberOfRequestsPerPoll, Math.max(1, endpoint.getConfiguration().getConcurrentRequestLimit())));
}
@Override
public void close() {
closed.set(true);
executorServiceManager.shutdownNow(requestExecutor);
}
@Override
public List call() throws IOException {
if (isClosed() || processScheduledQueueAutoCreation()) {
return emptyList();
}
final PollingContext context = new PollingContext();
final List messages = poll(context);
if (context.errorCount() == numberOfRequestsPerPoll) {
if (context.errorCount() == 1) {
context.rethrowIfFirstErrorIsRuntimeException();
throw new IOException("Error while polling", context.firstError());
}
throw new IOException(
("Error while polling - all %s requests resulted in an error, "
+ "please check the logs for more details").formatted(numberOfRequestsPerPoll));
}
return messages;
}
private List poll(final PollingContext pollContext)
throws IOException {
if (numberOfRequestsPerPoll == 1) {
return poll(maxMessagesPerPoll, pollContext);
}
int remaining = maxMessagesPerPoll;
try {
CompletableFuture> future
= CompletableFuture.completedFuture(emptyList());
while (remaining > 0) {
int numberOfMessages = Math.min(remaining, MAX_NUMBER_OF_MESSAGES_PER_REQUEST);
future = mergeResults(future,
CompletableFuture.supplyAsync(() -> poll(numberOfMessages, pollContext), requestExecutor));
remaining -= MAX_NUMBER_OF_MESSAGES_PER_REQUEST;
}
return future.thenApply(this::sortIfNeeded).get();
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
LOG.debug("Polling interrupted", e);
return emptyList();
} catch (ExecutionException e) {
throw new IOException("Error while polling", e.getCause());
}
}
private List poll(int maxNumberOfMessages, PollingContext context) {
if (context.isQueueMissing()) {
// if one of the request encountered a missing queue error the remaining requests
// should be ignored, even if the queue is automatically created it will be empty
// so there is no reason for immediate polling after creation
return emptyList();
}
try {
return sqsClient.receiveMessage(createReceiveRequest(maxNumberOfMessages)).messages();
} catch (QueueDoesNotExistException e) {
return handleMissingQueueError(context, e);
} catch (Exception e) {
LOG.error("Error while polling", e);
context.firePollingError(e);
return emptyList();
}
}
private List handleMissingQueueError(
PollingContext context, QueueDoesNotExistException error) {
if (context.isQueueMissing()) {
// if the context is flagged with missing queue
// it means another thread is handling the error
return emptyList();
}
final UUID requestId = UUID.randomUUID();
context.fireQueueMissing(requestId);
if (queueAutoCreationEnabled) {
createQueue(requestId, context);
return emptyList();
}
LOG.error("Error while polling {} queue does not exists", queueName, error);
context.firePollingError(error);
return emptyList();
}
private ReceiveMessageRequest createReceiveRequest(int maxNumberOfMessages) {
ReceiveMessageRequest.Builder requestBuilder = ReceiveMessageRequest.builder()
.queueUrl(queueUrl)
.maxNumberOfMessages(maxNumberOfMessages)
.visibilityTimeout(visibilityTimeout)
.waitTimeSeconds(waitTimeSeconds);
if (!attributeNames.isEmpty()) {
requestBuilder.messageSystemAttributeNames(attributeNames);
}
if (!messageAttributeNames.isEmpty()) {
requestBuilder.messageAttributeNames(messageAttributeNames);
}
LOG.trace("Receiving messages with request [{}]...", requestBuilder);
return requestBuilder.build();
}
private void createQueue(UUID requestId, PollingContext context) {
lock.lock();
try {
if (isClosed() || context.isMissingQueueHandledInAnotherRequest(requestId)) {
// the missing queue error can be thrown by multiple threads
// the first thread that is handling the error should prevent other threads
// from repeating the logic
// as the operation is synchronized, the other threads should wait and then
// check if it wasn't handled already
return;
}
try {
createQueueOperation.accept(sqsClient);
} catch (QueueDeletedRecentlyException e) {
LOG.debug("Queue recently deleted, will retry after at least 30 seconds on next polling request.", e);
scheduleQueueAutoCreation();
} catch (Exception e) {
LOG.error("Error while creating queue.", e);
context.firePollingError(e);
}
} finally {
lock.unlock();
}
}
private boolean processScheduledQueueAutoCreation() throws IOException {
long scheduleTimeMs = queueAutoCreationScheduleTime.get();
if (scheduleTimeMs == 0) {
// queue creation is not scheduled - ignoring
return false;
}
long elapsedTimeMillis = clock.elapsed();
if (scheduleTimeMs > elapsedTimeMillis) {
LOG.debug("{}ms remaining until queue auto-creation is triggered", scheduleTimeMs - elapsedTimeMillis);
return true;
}
final PollingContext context = new PollingContext();
createQueue(UUID.randomUUID(), context);
if (context.hasErrors()) {
context.rethrowIfFirstErrorIsRuntimeException();
throw new IOException("Error while creating %s queue".formatted(queueName), context.firstError());
}
cancelScheduledQueueAutoCreation();
return true;
}
private void scheduleQueueAutoCreation() {
queueAutoCreationScheduleTime.set(clock.elapsed() + RECENTLY_DELETED_QUEUE_BACKOFF_TIME_MS);
}
private void cancelScheduledQueueAutoCreation() {
queueAutoCreationScheduleTime.set(0);
}
private boolean isClosed() {
return closed.get();
}
private static List splitCommaSeparatedValues(String value) {
if (value == null || value.isEmpty()) {
return emptyList();
}
return COMMA_SEPARATED_PATTERN.splitAsStream(value).map(String::trim).filter(it -> !it.isEmpty()).toList();
}
private static Optional parseMessageSystemAttributeName(String attribute) {
if (attribute == null || attribute.isEmpty()) {
return Optional.empty();
}
MessageSystemAttributeName result = MessageSystemAttributeName.fromValue(attribute);
if (result == MessageSystemAttributeName.UNKNOWN_TO_SDK_VERSION) {
LOG.warn("Unsupported attribute name '{}' use one of {}", attribute, MessageSystemAttributeName.knownValues());
return Optional.empty();
}
return Optional.of(result);
}
private static MessageSystemAttributeName getSortAttributeName(Sqs2Configuration configuration) {
return parseMessageSystemAttributeName(configuration.getSortAttributeName())
.filter(attribute -> {
if (attribute == MessageSystemAttributeName.ALL) {
LOG.warn("The {} attribute cannot be used for sorting the received messages",
MessageSystemAttributeName.ALL);
return false;
}
return true;
})
.orElse(null);
}
private static List getAttributeNames(
Sqs2Configuration configuration, MessageSystemAttributeName sortAttributeName) {
List result = new ArrayList<>();
for (String attributeName : splitCommaSeparatedValues(configuration.getAttributeNames())) {
parseMessageSystemAttributeName(attributeName)
.filter(it -> !result.contains(it))
.ifPresent(result::add);
}
if (sortAttributeName != null && !result.contains(MessageSystemAttributeName.ALL)
&& !result.contains(sortAttributeName)) {
result.add(sortAttributeName);
}
return unmodifiableList(result);
}
private static int computeNumberOfRequestPerPoll(int maxMessagesPerPoll) {
return (int) Math.ceil((double) Math.max(1, maxMessagesPerPoll) / MAX_NUMBER_OF_MESSAGES_PER_REQUEST);
}
private static CompletableFuture> mergeResults(
CompletableFuture> future1, CompletableFuture> future2) {
return future1.thenCombine(future2, (messages1, messages2) -> {
final List allMessages = new ArrayList<>(messages1);
allMessages.addAll(messages2);
return allMessages;
});
}
private List sortIfNeeded(
List messages) {
if (LOG.isTraceEnabled()) {
LOG.trace("Received {} messages in {} requests", messages.size(), numberOfRequestsPerPoll);
}
if (sortAttributeName != null) {
return messages.stream()
.sorted(comparing(message -> message.attributes().getOrDefault(sortAttributeName, "")))
.toList();
}
return messages;
}
}
private record PollingContext(AtomicReference missingQueueHandlerRequestId, Queue errors) {
private PollingContext() {
this(new AtomicReference<>(), new ConcurrentLinkedQueue<>());
}
PollingContext {
Objects.requireNonNull(missingQueueHandlerRequestId);
Objects.requireNonNull(errors);
}
private void fireQueueMissing(UUID requestId) {
missingQueueHandlerRequestId.compareAndSet(null, requestId);
}
private void firePollingError(Exception error) {
errors.offer(error);
}
private boolean isQueueMissing() {
return missingQueueHandlerRequestId.get() != null;
}
private boolean isMissingQueueHandledInAnotherRequest(UUID requestId) {
UUID handlingRequestId = missingQueueHandlerRequestId.get();
return handlingRequestId != null && !requestId.equals(handlingRequestId);
}
private boolean hasErrors() {
return !errors.isEmpty();
}
private int errorCount() {
return errors.size();
}
private Exception firstError() {
return errors.peek();
}
private void rethrowIfFirstErrorIsRuntimeException() {
if (firstError() instanceof RuntimeException runtimeError) {
throw runtimeError;
}
}
}
}