dk.cloudcreate.essentials.components.queue.postgresql.PostgresqlDurableQueues Maven / Gradle / Ivy
/*
* Copyright 2021-2024 the original author or authors.
*
* Licensed 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
*
* https://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 dk.cloudcreate.essentials.components.queue.postgresql;
import com.fasterxml.jackson.annotation.*;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.json.JsonMapper;
import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import dk.cloudcreate.essentials.components.foundation.json.*;
import dk.cloudcreate.essentials.components.foundation.messaging.queue.*;
import dk.cloudcreate.essentials.components.foundation.messaging.queue.operations.*;
import dk.cloudcreate.essentials.components.foundation.postgresql.*;
import dk.cloudcreate.essentials.components.foundation.transaction.*;
import dk.cloudcreate.essentials.components.foundation.transaction.jdbi.*;
import dk.cloudcreate.essentials.components.queue.postgresql.jdbi.*;
import dk.cloudcreate.essentials.jackson.immutable.EssentialsImmutableJacksonModule;
import dk.cloudcreate.essentials.jackson.types.EssentialTypesJacksonModule;
import dk.cloudcreate.essentials.reactive.*;
import dk.cloudcreate.essentials.shared.Exceptions;
import dk.cloudcreate.essentials.shared.collections.Lists;
import dk.cloudcreate.essentials.shared.interceptor.InterceptorChain;
import dk.cloudcreate.essentials.shared.reflection.Classes;
import org.jdbi.v3.core.Handle;
import org.jdbi.v3.core.mapper.RowMapper;
import org.jdbi.v3.core.statement.StatementContext;
import org.slf4j.*;
import java.sql.*;
import java.time.*;
import java.util.*;
import java.util.concurrent.*;
import java.util.function.*;
import java.util.stream.Collectors;
import static dk.cloudcreate.essentials.shared.FailFast.*;
import static dk.cloudcreate.essentials.shared.MessageFormatter.NamedArgumentBinding.arg;
import static dk.cloudcreate.essentials.shared.MessageFormatter.*;
import static dk.cloudcreate.essentials.shared.interceptor.DefaultInterceptorChain.sortInterceptorsByOrder;
import static dk.cloudcreate.essentials.shared.interceptor.InterceptorChain.newInterceptorChainForOperation;
/**
* Postgresql version of the {@link DurableQueues} concept.
* Works together with {@link UnitOfWorkFactory} in order to support queuing message together with business logic (such as failing to handle an Event, etc.)
*
* Security
* {@link DurableQueues} allows the user of the component to override the {@link #getSharedQueueTableName()}, which is the name of the table that will contain all messages (across all {@link QueueName}'s)
* Note:
* To support customization of storage table name, the {@code sharedQueueTableName} will be directly used in constructing SQL statements
* through string concatenation, which exposes the component to SQL injection attacks.
*
* Security Note:
* It is the responsibility of the user of this component to sanitize the {@code sharedQueueTableName}
* to ensure the security of all the SQL statements generated by this component.
* The {@link PostgresqlDurableQueues} component will
* call the {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} method to validate the table name as a first line of defense.
* The {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} provides an initial layer of defense against SQL injection by applying naming conventions intended to reduce the risk of malicious input.
* However, Essentials components as well as {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} does not offer exhaustive protection, nor does it assure the complete security of the resulting SQL against SQL injection threats.
* The responsibility for implementing protective measures against SQL Injection lies exclusively with the users/developers using the Essentials components and its supporting classes.
* Users must ensure thorough sanitization and validation of API input parameters, column, table, and index names.
* Insufficient attention to these practices may leave the application vulnerable to SQL injection, potentially endangering the security and integrity of the database.
* The responsibility for implementing protective measures against SQL Injection lies exclusively with the users/developers using the Essentials components and its supporting classes.
* Users must ensure thorough sanitization and validation of API input parameters, column, table, and index names.
* Insufficient attention to these practices may leave the application vulnerable to SQL injection, potentially endangering the security and integrity of the database.
*
* It is highly recommended that the {@code sharedQueueTableName} value is only derived from a controlled and trusted source.
* To mitigate the risk of SQL injection attacks, external or untrusted inputs should never directly provide the {@code sharedQueueTableName} value.
*/
public final class PostgresqlDurableQueues implements DurableQueues {
private static final Logger log = LoggerFactory.getLogger(PostgresqlDurableQueues.class);
public static final String DEFAULT_DURABLE_QUEUES_TABLE_NAME = "durable_queues";
private static final Object NO_PAYLOAD = new Object();
private final HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory;
private final JSONSerializer jsonSerializer;
private final String sharedQueueTableName;
private final ConcurrentMap durableQueueConsumers = new ConcurrentHashMap<>();
private final QueuedMessageRowMapper queuedMessageMapper;
private final List interceptors = new ArrayList<>();
private final Optional> multiTableChangeListener;
private final Function queuePollingOptimizerFactory;
private final TransactionalMode transactionalMode;
/**
* Only used if {@link #transactionalMode} has value {@link TransactionalMode#SingleOperationTransaction}
*/
private int messageHandlingTimeoutMs;
/**
* Contains the timestamp of the last performed {@link #resetMessagesStuckBeingDelivered(QueueName)} check
* Only used if {@link #transactionalMode} has value {@link TransactionalMode#SingleOperationTransaction}
*/
protected ConcurrentMap lastResetStuckMessagesCheckTimestamps = new ConcurrentHashMap<>();
private volatile boolean started;
public static PostgresqlDurableQueuesBuilder builder() {
return new PostgresqlDurableQueuesBuilder();
}
/**
* Create {@link DurableQueues} with sharedQueueTableName: {@value DEFAULT_DURABLE_QUEUES_TABLE_NAME} and the default {@link JacksonJSONSerializer} using {@link #createDefaultObjectMapper()}
* configuration
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory) {
this(unitOfWorkFactory,
new JacksonJSONSerializer(createDefaultObjectMapper()),
DEFAULT_DURABLE_QUEUES_TABLE_NAME,
null,
null);
}
/**
* Create {@link DurableQueues} with sharedQueueTableName: {@value DEFAULT_DURABLE_QUEUES_TABLE_NAME} and the default {@link JacksonJSONSerializer} using {@link #createDefaultObjectMapper()}
* configuration
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
* @param queuePollingOptimizerFactory optional {@link QueuePollingOptimizer} factory that creates a {@link QueuePollingOptimizer} per {@link ConsumeFromQueue} command -
* if set to null {@link #createQueuePollingOptimizerFor(ConsumeFromQueue)} is used instead
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory,
Function queuePollingOptimizerFactory) {
this(unitOfWorkFactory,
new JacksonJSONSerializer(createDefaultObjectMapper()),
DEFAULT_DURABLE_QUEUES_TABLE_NAME,
null,
queuePollingOptimizerFactory);
}
/**
* Create {@link DurableQueues} with custom jsonSerializer with sharedQueueTableName: {@value DEFAULT_DURABLE_QUEUES_TABLE_NAME}
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
* @param jsonSerializer the {@link JSONSerializer} that is used to serialize/deserialize message payloads
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory,
JSONSerializer jsonSerializer) {
this(unitOfWorkFactory,
jsonSerializer,
DEFAULT_DURABLE_QUEUES_TABLE_NAME,
null,
null);
}
/**
* Create {@link DurableQueues} with custom jsonSerializer with sharedQueueTableName: {@value DEFAULT_DURABLE_QUEUES_TABLE_NAME}
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
* @param jsonSerializer the {@link JSONSerializer} that is used to serialize/deserialize message payloads
* @param queuePollingOptimizerFactory optional {@link QueuePollingOptimizer} factory that creates a {@link QueuePollingOptimizer} per {@link ConsumeFromQueue} command -
* if set to null {@link #createQueuePollingOptimizerFor(ConsumeFromQueue)} is used instead
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory,
JSONSerializer jsonSerializer,
Function queuePollingOptimizerFactory) {
this(unitOfWorkFactory,
jsonSerializer,
DEFAULT_DURABLE_QUEUES_TABLE_NAME,
null,
queuePollingOptimizerFactory);
}
/**
* Create {@link DurableQueues} with custom jsonSerializer and sharedQueueTableName
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
* @param jsonSerializer the {@link JSONSerializer} that is used to serialize/deserialize message payloads
* @param sharedQueueTableName the name of the table that will contain all messages (across all {@link QueueName}'s)
* Note:
* To support customization of storage table name, the {@code sharedQueueTableName} will be directly used in constructing SQL statements
* through string concatenation, which exposes the component to SQL injection attacks.
*
* Security Note:
* It is the responsibility of the user of this component to sanitize the {@code sharedQueueTableName}
* to ensure the security of all the SQL statements generated by this component.
* The {@link PostgresqlDurableQueues} component will
* call the {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} method to validate the table name as a first line of defense.
* The {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} provides an initial layer of defense against SQL injection by applying naming conventions intended to reduce the risk of malicious input.
* However, Essentials components as well as {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} does not offer exhaustive protection, nor does it assure the complete security of the resulting SQL against SQL injection threats.
* The responsibility for implementing protective measures against SQL Injection lies exclusively with the users/developers using the Essentials components and its supporting classes.
* Users must ensure thorough sanitization and validation of API input parameters, column, table, and index names.
* Insufficient attention to these practices may leave the application vulnerable to SQL injection, potentially endangering the security and integrity of the database.
*
* It is highly recommended that the {@code sharedQueueTableName} value is only derived from a controlled and trusted source.
* To mitigate the risk of SQL injection attacks, external or untrusted inputs should never directly provide the {@code sharedQueueTableName} value.
* @param multiTableChangeListener optional {@link MultiTableChangeListener} that allows {@link PostgresqlDurableQueues} to use {@link QueuePollingOptimizer}
* @param queuePollingOptimizerFactory optional {@link QueuePollingOptimizer} factory that creates a {@link QueuePollingOptimizer} per {@link ConsumeFromQueue} command -
* if set to null {@link #createQueuePollingOptimizerFor(ConsumeFromQueue)} is used instead
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory,
JSONSerializer jsonSerializer,
String sharedQueueTableName,
MultiTableChangeListener multiTableChangeListener,
Function queuePollingOptimizerFactory) {
this(unitOfWorkFactory,
jsonSerializer,
sharedQueueTableName,
multiTableChangeListener,
queuePollingOptimizerFactory,
TransactionalMode.FullyTransactional,
null);
}
/**
* Create {@link DurableQueues} with custom jsonSerializer and sharedQueueTableName
*
*
* @param unitOfWorkFactory the {@link UnitOfWorkFactory} needed to access the database
* @param jsonSerializer the {@link JSONSerializer} that is used to serialize/deserialize message payloads
* @param sharedQueueTableName the name of the table that will contain all messages (across all {@link QueueName}'s)
* Note:
* To support customization of storage table name, the {@code sharedQueueTableName} will be directly used in constructing SQL statements
* through string concatenation, which exposes the component to SQL injection attacks.
*
* Security Note:
* It is the responsibility of the user of this component to sanitize the {@code sharedQueueTableName}
* to ensure the security of all the SQL statements generated by this component.
* The {@link PostgresqlDurableQueues} component will
* call the {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} method to validate the table name as a first line of defense.
* The {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} provides an initial layer of defense against SQL injection by applying naming conventions intended to reduce the risk of malicious input.
* However, Essentials components as well as {@link PostgresqlUtil#checkIsValidTableOrColumnName(String)} does not offer exhaustive protection, nor does it assure the complete security of the resulting SQL against SQL injection threats.
* The responsibility for implementing protective measures against SQL Injection lies exclusively with the users/developers using the Essentials components and its supporting classes.
* Users must ensure thorough sanitization and validation of API input parameters, column, table, and index names.
* Insufficient attention to these practices may leave the application vulnerable to SQL injection, potentially endangering the security and integrity of the database.
*
* It is highly recommended that the {@code sharedQueueTableName} value is only derived from a controlled and trusted source.
* To mitigate the risk of SQL injection attacks, external or untrusted inputs should never directly provide the {@code sharedQueueTableName} value.
* @param multiTableChangeListener optional {@link MultiTableChangeListener} that allows {@link PostgresqlDurableQueues} to use {@link QueuePollingOptimizer}
* @param queuePollingOptimizerFactory optional {@link QueuePollingOptimizer} factory that creates a {@link QueuePollingOptimizer} per {@link ConsumeFromQueue} command -
* if set to null {@link #createQueuePollingOptimizerFor(ConsumeFromQueue)} is used instead
* @param transactionalMode The {@link TransactionalMode} for this {@link DurableQueues} instance. If set to {@link TransactionalMode#SingleOperationTransaction}
* then the consumer MUST call the {@link DurableQueues#acknowledgeMessageAsHandled(AcknowledgeMessageAsHandled)} explicitly in a new {@link UnitOfWork}
* @param messageHandlingTimeout Only required if transactionalMode
is {@link TransactionalMode#SingleOperationTransaction}.
* The parameter defines the timeout for messages being delivered, but haven't yet been acknowledged.
* After this timeout the message delivery will be reset and the message will again be a candidate for delivery
*/
public PostgresqlDurableQueues(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory,
JSONSerializer jsonSerializer,
String sharedQueueTableName,
MultiTableChangeListener multiTableChangeListener,
Function queuePollingOptimizerFactory,
TransactionalMode transactionalMode,
Duration messageHandlingTimeout) {
this.unitOfWorkFactory = requireNonNull(unitOfWorkFactory, "No unitOfWorkFactory instance provided");
this.jsonSerializer = requireNonNull(jsonSerializer, "No jsonSerializer");
this.sharedQueueTableName = requireNonNull(sharedQueueTableName, "No sharedQueueTableName provided").toLowerCase(Locale.ROOT);
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
this.queuedMessageMapper = new QueuedMessageRowMapper();
this.multiTableChangeListener = Optional.ofNullable(multiTableChangeListener);
this.queuePollingOptimizerFactory = queuePollingOptimizerFactory != null ? queuePollingOptimizerFactory : this::createQueuePollingOptimizerFor;
this.transactionalMode = requireNonNull(transactionalMode, "No transactionalMode instance provided");
if (transactionalMode == TransactionalMode.SingleOperationTransaction) {
messageHandlingTimeoutMs = (int) requireNonNull(messageHandlingTimeout, "No messageHandlingTimeout provided").toMillis();
addInterceptor(new SingleOperationTransactionDurableQueuesInterceptor(unitOfWorkFactory));
}
initializeQueueTables();
}
private void initializeQueueTables() {
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
unitOfWorkFactory.usingUnitOfWork(handleAwareUnitOfWork -> {
handleAwareUnitOfWork.handle().getJdbi().registerArgument(new QueueNameArgumentFactory());
handleAwareUnitOfWork.handle().getJdbi().registerColumnMapper(new QueueNameColumnMapper());
handleAwareUnitOfWork.handle().getJdbi().registerArgument(new QueueEntryIdArgumentFactory());
handleAwareUnitOfWork.handle().getJdbi().registerColumnMapper(new QueueEntryIdColumnMapper());
handleAwareUnitOfWork.handle().execute(bind("CREATE TABLE IF NOT EXISTS {:tableName} (\n" +
" id TEXT PRIMARY KEY,\n" +
" queue_name TEXT NOT NULL,\n" +
" message_payload JSONB NOT NULL,\n" +
" message_payload_type TEXT NOT NULL,\n" +
" added_ts TIMESTAMPTZ NOT NULL,\n" +
" next_delivery_ts TIMESTAMPTZ,\n" +
" delivery_ts TIMESTAMPTZ DEFAULT NULL,\n" +
" total_attempts INTEGER DEFAULT 0,\n" +
" redelivery_attempts INTEGER DEFAULT 0,\n" +
" last_delivery_error TEXT DEFAULT NULL,\n" +
" is_being_delivered BOOLEAN DEFAULT FALSE,\n" +
" is_dead_letter_message BOOLEAN NOT NULL DEFAULT FALSE,\n" +
" meta_data JSONB DEFAULT NULL,\n" +
" delivery_mode TEXT NOT NULL,\n" +
" key TEXT DEFAULT NULL,\n" +
" key_order BIGINT DEFAULT -1\n" +
")",
arg("tableName", sharedQueueTableName))
);
log.info("Ensured Durable Queues table '{}' exists", sharedQueueTableName);
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_queue_name ON {:tableName} (queue_name)",
handleAwareUnitOfWork.handle());
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_next_delivery_ts ON {:tableName} (next_delivery_ts)",
handleAwareUnitOfWork.handle());
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_is_dead_letter_message ON {:tableName} (is_dead_letter_message)",
handleAwareUnitOfWork.handle());
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_is_being_delivered ON {:tableName} (is_being_delivered)",
handleAwareUnitOfWork.handle());
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_ordered_msg ON {:tableName} (queue_name, key, key_order)",
handleAwareUnitOfWork.handle());
createIndex("CREATE INDEX IF NOT EXISTS idx_{:tableName}_next_msg ON {:tableName} (queue_name, is_dead_letter_message, is_being_delivered, next_delivery_ts)",
handleAwareUnitOfWork.handle());
multiTableChangeListener.ifPresent(listener -> {
ListenNotify.addChangeNotificationTriggerToTable(handleAwareUnitOfWork.handle(),
sharedQueueTableName,
List.of(ListenNotify.SqlOperation.INSERT, ListenNotify.SqlOperation.UPDATE),
"id", "queue_name", "added_ts", "next_delivery_ts", "delivery_ts", "is_dead_letter_message", "is_being_delivered");
});
});
}
private void createIndex(String indexStatement, Handle handle) {
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
handle.execute(bind(indexStatement,
arg("tableName", sharedQueueTableName))
);
}
private void dropIndex(String indexStatement, Handle handle) {
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
handle.execute(bind(indexStatement,
arg("tableName", sharedQueueTableName))
);
}
/**
* The name of the shared table where all queue messages are stored
*
* @return the name of the shared table where all queue messages are stored
*/
public final String getSharedQueueTableName() {
return sharedQueueTableName;
}
public final List getInterceptors() {
return Collections.unmodifiableList(interceptors);
}
@Override
public final void start() {
if (!started) {
started = true;
log.info("Starting");
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
interceptors.forEach(durableQueuesInterceptor -> durableQueuesInterceptor.setDurableQueues(this));
sortInterceptorsByOrder(interceptors);
durableQueueConsumers.values().forEach(PostgresqlDurableQueueConsumer::start);
multiTableChangeListener.ifPresent(listener -> {
listener.listenToNotificationsFor(sharedQueueTableName,
QueueTableNotification.class);
listener.getEventBus().addAsyncSubscriber(new AnnotatedEventHandler() {
@Handler
void handle(QueueTableNotification e) {
try {
log.trace("[{}:{}] Received QueueMessage notification",
e.queueName,
e.id);
var queueName = QueueName.of(e.queueName);
durableQueueConsumers.values()
.stream()
.filter(durableQueueConsumer -> durableQueueConsumer.queueName.equals(queueName))
.forEach(durableQueueConsumer -> {
durableQueueConsumer.messageAdded(new DefaultQueuedMessage(QueueEntryId.of(String.valueOf(e.id)),
queueName,
Message.of(NO_PAYLOAD),
e.addedTimestamp,
e.nextDeliveryTimestamp,
e.deliveryTimestamp,
null,
-1,
-1,
e.isDeadLetterMessage,
e.isBeingDelivered));
});
} catch (Exception ex) {
log.error("Error occurred while handling notification", ex);
}
}
});
});
log.info("Started");
}
}
@Override
public final void stop() {
if (started) {
log.info("Stopping");
PostgresqlUtil.checkIsValidTableOrColumnName(sharedQueueTableName);
durableQueueConsumers.values().forEach(PostgresqlDurableQueueConsumer::stop);
multiTableChangeListener.ifPresent(listener -> {
listener.unlistenToNotificationsFor(sharedQueueTableName);
});
started = false;
log.info("Stopped");
}
}
@Override
public final boolean isStarted() {
return started;
}
@Override
public final TransactionalMode getTransactionalMode() {
return transactionalMode;
}
@Override
public final Optional> getUnitOfWorkFactory() {
return Optional.ofNullable(unitOfWorkFactory);
}
@Override
public final Set getQueueNames() {
var consumerQueueNames = durableQueueConsumers.keySet();
var dbQueueNames = unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle()
.createQuery(bind("SELECT distinct queue_name FROM {:tableName}",
arg("tableName", sharedQueueTableName)))
.mapTo(QueueName.class)
.set());
dbQueueNames.addAll(consumerQueueNames);
return dbQueueNames;
}
@Override
public final DurableQueueConsumer consumeFromQueue(ConsumeFromQueue operation) {
requireNonNull(operation, "No operation provided");
if (durableQueueConsumers.containsKey(operation.queueName)) {
throw new DurableQueueException("There is already an DurableConsumer for this queue", operation.queueName);
}
operation.validate();
return durableQueueConsumers.computeIfAbsent(operation.queueName, _queueName -> {
PostgresqlDurableQueueConsumer consumer = (PostgresqlDurableQueueConsumer) newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var pollingIntervalMs = operation.getPollingInterval().toMillis();
var queuePollingOptimizer = multiTableChangeListener.map(_ignore -> queuePollingOptimizerFactory.apply(operation))
.orElseGet(QueuePollingOptimizer::None);
return (DurableQueueConsumer) new PostgresqlDurableQueueConsumer(operation,
unitOfWorkFactory,
this,
this::removeQueueConsumer,
pollingIntervalMs,
queuePollingOptimizer);
}).proceed();
if (started) {
consumer.start();
}
return consumer;
});
}
/**
* Override this method to provide another {@link QueuePollingOptimizer} than the default {@link QueuePollingOptimizer.SimpleQueuePollingOptimizer}
* Only called if the {@link PostgresqlDurableQueues} is configured with a {@link MultiTableChangeListener}
*
* @param operation the operation for which the {@link QueuePollingOptimizer} will be responsible
* @return the {@link QueuePollingOptimizer}
*/
protected QueuePollingOptimizer createQueuePollingOptimizerFor(ConsumeFromQueue operation) {
var pollingIntervalMs = operation.getPollingInterval().toMillis();
return new QueuePollingOptimizer.SimpleQueuePollingOptimizer(operation,
(long) (pollingIntervalMs * 0.5d),
pollingIntervalMs * 20);
}
final void removeQueueConsumer(DurableQueueConsumer durableQueueConsumer) {
requireNonNull(durableQueueConsumer, "You must provide a durableQueueConsumer");
requireFalse(durableQueueConsumer.isStarted(), msg("Cannot remove DurableQueueConsumer '{}' since it's started!", durableQueueConsumer.queueName()));
var operation = new StopConsumingFromQueue(durableQueueConsumer);
try {
newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> (DurableQueueConsumer) durableQueueConsumers.remove(durableQueueConsumer.queueName()))
.proceed();
} catch (Exception e) {
log.error(msg("Failed to perform {}", operation), e);
}
}
@Override
public final QueueEntryId queueMessage(QueueMessage operation) {
requireNonNull(operation, "You must provide a QueueMessage instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> queueMessage(operation.queueName,
operation.getMessage(),
false,
operation.getCauseOfEnqueuing(),
operation.getDeliveryDelay()))
.proceed();
}
@Override
public final QueueEntryId queueMessageAsDeadLetterMessage(QueueMessageAsDeadLetterMessage operation) {
requireNonNull(operation, "You must provide a QueueMessageAsDeadLetterMessage instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> queueMessage(operation.queueName,
operation.getMessage(),
true,
Optional.ofNullable(operation.getCauseOfError()),
Optional.empty()))
.proceed();
}
protected final QueueEntryId queueMessage(QueueName queueName,
Message message,
boolean isDeadLetterMessage,
Optional causeOfEnqueuing,
Optional deliveryDelay) {
requireNonNull(queueName, "You must provide a queueName");
requireNonNull(message, "You must provide a message");
requireNonNull(causeOfEnqueuing, "You must provide a causeOfEnqueuing option");
requireNonNull(deliveryDelay, "You must provide a deliveryDelay option");
var queueEntryId = QueueEntryId.random();
var addedTimestamp = Instant.now();
var nextDeliveryTimestamp = isDeadLetterMessage ? null : addedTimestamp.plus(deliveryDelay.orElse(Duration.ZERO));
var isOrderedMessage = message instanceof OrderedMessage;
log.trace("[{}:{}] Queuing {}{}message{} with nextDeliveryTimestamp {}",
queueName,
queueEntryId,
isDeadLetterMessage ? "Dead Letter " : "",
isOrderedMessage ? "Ordered " : "",
isOrderedMessage ? msg(" {}:{}", ((OrderedMessage) message).getKey(), ((OrderedMessage) message).getOrder()) : "",
nextDeliveryTimestamp);
String jsonPayload;
try {
jsonPayload = jsonSerializer.serialize(message.getPayload());
} catch (JSONSerializationException e) {
throw new DurableQueueException(msg("Failed to serialize message payload of type", message.getPayload().getClass().getName()), e, queueName);
}
if (transactionalMode == TransactionalMode.FullyTransactional) {
unitOfWorkFactory.getRequiredUnitOfWork();
}
// TODO: Future improvement: If queueing an OrderedMessage check if another OrderMessage related to the same key and a lower order is already marked as a dead letter message,
// in which case this message can be queued directly as a dead letter message
unitOfWorkFactory.usingUnitOfWork(unitOfWork -> {
var update = unitOfWork.handle().createUpdate(bind("INSERT INTO {:tableName} (\n" +
" id,\n" +
" queue_name,\n" +
" message_payload,\n" +
" message_payload_type,\n" +
" added_ts,\n" +
" next_delivery_ts,\n" +
" last_delivery_error,\n" +
" is_dead_letter_message,\n" +
" meta_data,\n" +
" delivery_mode,\n" +
" key,\n" +
" key_order\n" +
" ) VALUES (\n" +
" :id,\n" +
" :queueName,\n" +
" :message_payload::jsonb,\n" +
" :message_payload_type,\n" +
" :addedTimestamp,\n" +
" :nextDeliveryTimestamp,\n" +
" :lastDeliveryError,\n" +
" :isDeadLetterMessage,\n" +
" :metaData::jsonb,\n" +
" :deliveryMode,\n" +
" :key,\n" +
" :order\n" +
" )",
arg("tableName", sharedQueueTableName)))
.bind("id", queueEntryId)
.bind("queueName", queueName)
.bind("message_payload", jsonPayload)
.bind("message_payload_type", message.getPayload().getClass().getName())
.bind("addedTimestamp", addedTimestamp)
.bind("nextDeliveryTimestamp", nextDeliveryTimestamp)
.bind("isDeadLetterMessage", isDeadLetterMessage);
if (message instanceof OrderedMessage) {
var orderedMessage = (OrderedMessage) message;
requireNonNull(orderedMessage.getKey(), "An OrderedMessage requires a non null key");
requireTrue(orderedMessage.getOrder() >= 0, "An OrderedMessage requires an order >= 0");
update.bind("deliveryMode", QueuedMessage.DeliveryMode.IN_ORDER)
.bind("key", orderedMessage.getKey())
.bind("order", orderedMessage.getOrder());
} else {
update.bind("deliveryMode", QueuedMessage.DeliveryMode.NORMAL)
.bindNull("key", Types.VARCHAR)
.bind("order", -1L);
}
try {
var jsonMetaData = jsonSerializer.serialize(message.getMetaData());
update.bind("metaData", jsonMetaData);
} catch (JSONSerializationException e) {
throw new DurableQueueException("Failed to serialize message meta-data", e, queueName);
}
if (causeOfEnqueuing.isPresent()) {
update.bind("lastDeliveryError", causeOfEnqueuing.map(Exceptions::getStackTrace).get());
} else {
update.bindNull("lastDeliveryError", Types.VARCHAR);
}
var numberOfRowsUpdated = update.execute();
if (numberOfRowsUpdated == 0) {
throw new DurableQueueException("Failed to insert message", queueName);
}
});
log.debug("[{}:{}] Queued {}{}message{} with nextDeliveryTimestamp {}",
queueName,
queueEntryId,
isDeadLetterMessage ? "Dead Letter " : "",
isOrderedMessage ? "Ordered " : "",
isOrderedMessage ? msg(" {}:{}", ((OrderedMessage) message).getKey(), ((OrderedMessage) message).getOrder()) : "",
nextDeliveryTimestamp);
return queueEntryId;
}
@Override
public final List queueMessages(QueueMessages operation) {
requireNonNull(operation, "You must provide a QueueMessages instance");
operation.validate();
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var queueName = operation.getQueueName();
var deliveryDelay = operation.getDeliveryDelay();
var messages = operation.getMessages();
var addedTimestamp = OffsetDateTime.now(Clock.systemUTC());
var nextDeliveryTimestamp = addedTimestamp.plus(deliveryDelay.orElse(Duration.ZERO));
var batch = unitOfWorkFactory.getRequiredUnitOfWork().handle().prepareBatch(bind("INSERT INTO {:tableName} (\n" +
" id,\n" +
" queue_name,\n" +
" message_payload,\n" +
" message_payload_type,\n" +
" added_ts,\n" +
" next_delivery_ts,\n" +
" last_delivery_error,\n" +
" is_dead_letter_message,\n" +
" is_being_delivered,\n" +
" meta_data,\n" +
" delivery_mode,\n" +
" key,\n" +
" key_order\n" +
" ) VALUES (\n" +
" :id,\n" +
" :queueName,\n" +
" :message_payload::jsonb,\n" +
" :message_payload_type,\n" +
" :addedTimestamp,\n" +
" :nextDeliveryTimestamp,\n" +
" :lastDeliveryError,\n" +
" :isDeadLetterMessage,\n" +
" :isBeingDelivered,\n" +
" :metaData::jsonb,\n" +
" :deliveryMode,\n" +
" :key,\n" +
" :order\n" +
" )",
arg("tableName", sharedQueueTableName)));
var queueEntryIds = Lists.toIndexedStream(messages).map(indexedMessage -> {
var message = indexedMessage._2;
String jsonPayload;
try {
jsonPayload = jsonSerializer.serialize(message.getPayload());
} catch (JSONSerializationException e) {
throw new DurableQueueException(msg("Failed to serialize message payload of type", message.getPayload().getClass().getName()), e, queueName);
}
var queueEntryId = QueueEntryId.random();
batch.bind("id", queueEntryId)
.bind("queueName", queueName)
.bind("message_payload", jsonPayload)
.bind("message_payload_type", message.getPayload().getClass().getName())
.bind("addedTimestamp", addedTimestamp)
.bind("nextDeliveryTimestamp", nextDeliveryTimestamp)
.bind("isDeadLetterMessage", false)
.bind("isBeingDelivered", false)
.bindNull("lastDeliveryError", Types.VARCHAR);
if (message instanceof OrderedMessage) {
var orderedMessage = (OrderedMessage) message;
requireNonNull(orderedMessage.getKey(), msg("[Index: {}] - OrderedMessage requires a non null key", indexedMessage._1));
requireTrue(orderedMessage.getOrder() >= 0, msg("[Index: {}] - OrderedMessage requires an order >= 0", indexedMessage._1));
batch.bind("deliveryMode", QueuedMessage.DeliveryMode.IN_ORDER)
.bind("key", orderedMessage.getKey())
.bind("order", orderedMessage.getOrder());
} else {
batch.bind("deliveryMode", QueuedMessage.DeliveryMode.NORMAL)
.bindNull("key", Types.VARCHAR)
.bind("order", -1L);
}
try {
var jsonMetaData = jsonSerializer.serialize(message.getMetaData());
batch.bind("metaData", jsonMetaData);
} catch (JSONSerializationException e) {
throw new DurableQueueException("Failed to serialize message meta-data", e, queueName);
}
batch.add();
return queueEntryId;
}).collect(Collectors.toList());
var numberOfRowsUpdated = Arrays.stream(batch.execute())
.reduce(Integer::sum).orElse(0);
if (numberOfRowsUpdated != messages.size()) {
throw new DurableQueueException(msg("Attempted to queue {} messages but only inserted {} messages", messages.size(), numberOfRowsUpdated),
queueName);
}
log.debug("[{}] Queued {} Messages with nextDeliveryTimestamp {} and entry-id's: {}",
queueName,
messages.size(),
nextDeliveryTimestamp,
queueEntryIds);
return queueEntryIds;
}).proceed();
}
@Override
public final Optional retryMessage(RetryMessage operation) {
requireNonNull(operation, "You must provide a RetryMessage instance");
operation.validate();
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var nextDeliveryTimestamp = OffsetDateTime.now(Clock.systemUTC()).plus(operation.getDeliveryDelay());
var result = unitOfWorkFactory.getRequiredUnitOfWork().handle().createQuery(bind("UPDATE {:tableName} SET\n" +
" next_delivery_ts = :nextDeliveryTimestamp,\n" +
" last_delivery_error = :lastDeliveryError,\n" +
" redelivery_attempts = redelivery_attempts + 1,\n" +
" is_being_delivered = FALSE,\n" +
" delivery_ts = NULL\n" +
" WHERE id = :id\n" +
" RETURNING *",
arg("tableName", sharedQueueTableName)))
.bind("nextDeliveryTimestamp", nextDeliveryTimestamp)
.bind("lastDeliveryError", Exceptions.getStackTrace(operation.getCauseForRetry()))
.bind("id", operation.queueEntryId)
.map(queuedMessageMapper)
.findOne();
if (result.isPresent()) {
log.debug("Marked Message with id '{}' for Retry at {}. Message entry after update: {}", operation.queueEntryId, nextDeliveryTimestamp, result.get());
return result;
} else {
log.error("Failed to Mark Message with id '{}' for Retry", operation.queueEntryId);
return Optional.empty();
}
}).proceed();
}
@Override
public final Optional markAsDeadLetterMessage(MarkAsDeadLetterMessage operation) {
requireNonNull(operation, "You must provide a MarkAsDeadLetterMessage instance");
operation.validate();
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var result = unitOfWorkFactory.getRequiredUnitOfWork().handle().createQuery(bind("UPDATE {:tableName} SET\n" +
" next_delivery_ts = NULL,\n" +
" last_delivery_error = :lastDeliveryError,\n" +
" is_dead_letter_message = TRUE,\n" +
" is_being_delivered = FALSE,\n" +
" delivery_ts = NULL\n" +
" WHERE id = :id AND is_dead_letter_message = FALSE\n" +
" RETURNING *",
arg("tableName", sharedQueueTableName)))
.bind("lastDeliveryError", Exceptions.getStackTrace(operation.getCauseForBeingMarkedAsDeadLetter()))
.bind("id", operation.queueEntryId)
.map(queuedMessageMapper)
.findOne();
if (result.isPresent()) {
log.debug("Marked message with id '{}' as Dead Letter Message. Message entry after update: {}", operation.queueEntryId, result.get());
return result;
} else {
log.error("Failed to Mark as Message message with id '{}' as Dead Letter Message", operation.queueEntryId);
return Optional.empty();
}
}).proceed();
}
@Override
public final Optional resurrectDeadLetterMessage(ResurrectDeadLetterMessage operation) {
requireNonNull(operation, "You must provide a ResurrectDeadLetterMessage instance");
operation.validate();
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var nextDeliveryTimestamp = OffsetDateTime.now(Clock.systemUTC()).plus(operation.getDeliveryDelay());
var result = unitOfWorkFactory.getRequiredUnitOfWork().handle().createQuery(bind("UPDATE {:tableName} SET\n" +
" next_delivery_ts = :nextDeliveryTimestamp,\n" +
" is_dead_letter_message = FALSE\n" +
" WHERE id = :id AND " +
"is_dead_letter_message = TRUE\n" +
" RETURNING *",
arg("tableName", sharedQueueTableName)))
.bind("nextDeliveryTimestamp", nextDeliveryTimestamp)
.bind("id", operation.queueEntryId)
.map(queuedMessageMapper)
.findOne();
if (result.isPresent()) {
var updateResult = result.get();
var isOrderedMessage = updateResult.getDeliveryMode() == QueuedMessage.DeliveryMode.IN_ORDER;
log.debug("[{}] Resurrected Dead Letter {}Message with id '{}' {} and nextDeliveryTimestamp: {}. Message entry after update: {}",
updateResult.getQueueName(),
isOrderedMessage ? "Ordered " : "",
operation.getQueueEntryId(),
isOrderedMessage ? "(key: " + ((OrderedMessage) updateResult).getKey() + ", order: " + ((OrderedMessage) updateResult).getOrder() + ")" : "",
nextDeliveryTimestamp,
updateResult);
return result;
} else {
log.error("Failed to resurrect Dead Letter Message with id '{}'", operation.queueEntryId);
return Optional.empty();
}
}).proceed();
}
@Override
public final boolean acknowledgeMessageAsHandled(AcknowledgeMessageAsHandled operation) {
requireNonNull(operation, "You must provide a AcknowledgeMessageAsHandled instance");
return unitOfWorkFactory.withUnitOfWork(() -> newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
log.debug("Acknowledging-Message-As-Handled regarding Message with id '{}'", operation.queueEntryId);
return deleteMessage(new DeleteMessage(operation.queueEntryId));
})
.proceed());
}
@Override
public final boolean deleteMessage(DeleteMessage operation) {
requireNonNull(operation, "You must provide a DeleteMessage instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
var rowsUpdated = unitOfWorkFactory.getRequiredUnitOfWork().handle().createUpdate(bind("DELETE FROM {:tableName} WHERE id = :id",
arg("tableName", sharedQueueTableName)))
.bind("id", operation.queueEntryId)
.execute();
if (rowsUpdated == 1) {
log.debug("Deleted Message with id '{}'", operation.queueEntryId);
return true;
} else {
log.error("Couldn't Delete Message with id '{}' - it may already have been deleted", operation.queueEntryId);
return false;
}
}).proceed();
}
@Override
public final Optional getNextMessageReadyForDelivery(GetNextMessageReadyForDelivery operation) {
requireNonNull(operation, "You must specify a GetNextMessageReadyForDelivery instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> {
resetMessagesStuckBeingDelivered(operation.queueName);
var now = OffsetDateTime.now(Clock.systemUTC());
var excludeKeysLimitSql = "";
var excludedKeys = operation.getExcludeOrderedMessagesWithKey() != null ? operation.getExcludeOrderedMessagesWithKey() : List.of();
if (!excludedKeys.isEmpty()) {
excludeKeysLimitSql = " AND key NOT IN ()\n";
}
var sql = bind("WITH queued_message_ready_for_delivery AS (\n" +
" SELECT id FROM {:tableName} q1 \n" +
" WHERE\n" +
" queue_name = :queueName AND\n" +
" is_dead_letter_message = FALSE AND\n" +
" is_being_delivered = FALSE AND\n" +
" next_delivery_ts <= :now AND\n" +
" NOT EXISTS (SELECT 1 FROM {:tableName} q2 WHERE q2.key = q1.key AND q2.queue_name = q1.queue_name AND q2.key_order < q1.key_order)\n" +
excludeKeysLimitSql +
" ORDER BY key_order ASC, next_delivery_ts ASC\n" + // TODO: Future improvement: Allow the user to specify if key_order or next_delivery_ts should have the highest priority
" LIMIT 1\n" +
" FOR UPDATE SKIP LOCKED\n" +
" )\n" +
" UPDATE {:tableName} queued_message SET\n" +
" total_attempts = total_attempts + 1,\n" +
" next_delivery_ts = NULL,\n" +
" is_being_delivered = TRUE,\n" +
" delivery_ts = :now\n" +
" FROM queued_message_ready_for_delivery\n" +
" WHERE queued_message.id = queued_message_ready_for_delivery.id\n" +
" RETURNING\n" +
" queued_message.id,\n" +
" queued_message.queue_name,\n" +
" queued_message.message_payload,\n" +
" queued_message.message_payload_type,\n" +
" queued_message.added_ts,\n" +
" queued_message.next_delivery_ts,\n" +
" queued_message.delivery_ts,\n" +
" queued_message.last_delivery_error,\n" +
" queued_message.total_attempts,\n" +
" queued_message.redelivery_attempts,\n" +
" queued_message.is_dead_letter_message,\n" +
" queued_message.is_being_delivered,\n" +
" queued_message.meta_data,\n" +
" queued_message.delivery_mode,\n" +
" queued_message.key,\n" +
" queued_message.key_order",
arg("tableName", sharedQueueTableName));
var query = unitOfWorkFactory.getRequiredUnitOfWork().handle().createQuery(sql)
.bind("queueName", operation.queueName)
.bind("now", now);
if (!excludedKeys.isEmpty()) {
query.bindList("excludedKeys", excludedKeys);
}
return query
.map(queuedMessageMapper)
.findOne();
}).proceed();
}
/**
* This operation will scan for messages that has been marked as {@link QueuedMessage#isBeingDelivered()} for longer
* than {@link #messageHandlingTimeoutMs}
* All messages found will have {@link QueuedMessage#isBeingDelivered()} and {@link QueuedMessage#getDeliveryTimestamp()}
* reset
* Only relevant for when using {@link TransactionalMode#SingleOperationTransaction}
*
* @param queueName the queue for which we're looking for messages stuck being marked as {@link QueuedMessage#isBeingDelivered()}
*/
protected final void resetMessagesStuckBeingDelivered(QueueName queueName) {
// Reset stuck messages
if (transactionalMode == TransactionalMode.SingleOperationTransaction) {
var now = Instant.now();
var lastStuckMessageResetTimestamp = lastResetStuckMessagesCheckTimestamps.get(queueName);
if (lastStuckMessageResetTimestamp == null || Duration.between(now, lastStuckMessageResetTimestamp).abs().toMillis() > messageHandlingTimeoutMs) {
if (log.isDebugEnabled()) {
log.debug("[{}] Looking for messages stuck marked as isBeingDelivered. Last check was performed: {}", queueName, lastStuckMessageResetTimestamp);
}
var numberOfChanges = unitOfWorkFactory.getRequiredUnitOfWork().handle().createUpdate(bind("UPDATE {:tableName} SET\n" +
" is_being_delivered = FALSE,\n" +
" delivery_ts = NULL,\n" +
" redelivery_attempts = redelivery_attempts + 1,\n" +
" next_delivery_ts = :now,\n" +
" last_delivery_error = :error\n" +
" WHERE is_being_delivered = TRUE\n" +
" AND delivery_ts <= :threshold\n",
arg("tableName", sharedQueueTableName)))
.bind("threshold", now.minusMillis(messageHandlingTimeoutMs))
.bind("error", "Handler Processing of the Message was determined to have Timed Out")
.bind("now", now)
.execute();
if (numberOfChanges > 0) {
log.debug("[{}] Reset {} messages stuck marked as isBeingDelivered", queueName, numberOfChanges);
} else {
log.debug("[{}] Didn't find any messages being stuck marked as isBeingDelivered", queueName);
}
lastResetStuckMessagesCheckTimestamps.put(queueName, now);
}
}
}
@Override
public final boolean hasMessagesQueuedFor(QueueName queueName) {
return getTotalMessagesQueuedFor(queueName) > 0;
}
@Override
public final long getTotalMessagesQueuedFor(GetTotalMessagesQueuedFor operation) {
requireNonNull(operation, "You must specify a GetTotalMessagesQueuedFor instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createQuery(bind("SELECT count(*) FROM {:tableName} \n" +
" WHERE \n" +
" queue_name = :queueName AND\n" +
" is_dead_letter_message = FALSE",
arg("tableName", sharedQueueTableName)))
.bind("queueName", operation.queueName)
.mapTo(Long.class)
.one()))
.proceed();
}
@Override
public final long getTotalDeadLetterMessagesQueuedFor(GetTotalDeadLetterMessagesQueuedFor operation) {
requireNonNull(operation, "You must specify a GetTotalDeadLetterMessagesQueuedFor instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createQuery(bind("SELECT count(*) FROM {:tableName} \n" +
" WHERE \n" +
" queue_name = :queueName AND\n" +
" is_dead_letter_message = TRUE",
arg("tableName", sharedQueueTableName)))
.bind("queueName", operation.queueName)
.mapTo(Long.class)
.one()))
.proceed();
}
@Override
public final int purgeQueue(PurgeQueue operation) {
requireNonNull(operation, "You must specify a PurgeQueue instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createUpdate(bind("DELETE FROM {:tableName} WHERE queue_name = :queueName",
arg("tableName", sharedQueueTableName)))
.bind("queueName", operation.queueName)
.execute()))
.proceed();
}
@Override
public final List queryForMessagesSoonReadyForDelivery(QueueName queueName, Instant withNextDeliveryTimestampAfter, int maxNumberOfMessagesToReturn) {
requireNonNull(queueName, "No queueName provided");
requireNonNull(withNextDeliveryTimestampAfter, "No withNextDeliveryTimestampAfter provided");
return unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createQuery(bind("SELECT id, added_ts, next_delivery_ts FROM {:tableName} \n" +
" WHERE queue_name = :queueName\n" +
" AND is_dead_letter_message = FALSE\n" +
" AND is_being_delivered = FALSE\n" +
" AND next_delivery_ts > :now\n" +
" ORDER BY next_delivery_ts ASC\n" +
" LIMIT :pageSize",
arg("tableName", sharedQueueTableName)))
.bind("queueName", requireNonNull(queueName, "No QueueName provided"))
.bind("now", withNextDeliveryTimestampAfter)
.bind("pageSize", maxNumberOfMessagesToReturn)
.map((rs, ctx) -> new NextQueuedMessage(QueueEntryId.of(rs.getString("id")),
queueName,
rs.getObject("added_ts", OffsetDateTime.class).toInstant(),
rs.getObject("next_delivery_ts", OffsetDateTime.class).toInstant()))
.list());
}
@Override
public final List getQueuedMessages(GetQueuedMessages operation) {
requireNonNull(operation, "You must specify a GetQueuedMessages instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> queryQueuedMessages(operation.queueName, operation.getQueueingSortOrder(), IncludeMessages.QUEUED_MESSAGES, operation.getStartIndex(), operation.getPageSize()))
.proceed();
}
@Override
public final List getDeadLetterMessages(GetDeadLetterMessages operation) {
requireNonNull(operation, "You must specify a GetDeadLetterMessages instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> queryQueuedMessages(operation.queueName, operation.getQueueingSortOrder(), IncludeMessages.DEAD_LETTER_MESSAGES, operation.getStartIndex(), operation.getPageSize()))
.proceed();
}
protected enum IncludeMessages {
ALL, DEAD_LETTER_MESSAGES, QUEUED_MESSAGES
}
protected final List queryQueuedMessages(QueueName queueName, QueueingSortOrder queueingSortOrder, IncludeMessages includeMessages, long startIndex, long pageSize) {
requireNonNull(queueName, "No queueName provided");
requireNonNull(queueingSortOrder, "No queueingOrder provided");
requireNonNull(includeMessages, "No includeMessages provided");
Supplier resolveIncludeMessagesSql = () -> {
switch (includeMessages) {
case ALL:
return "";
case DEAD_LETTER_MESSAGES:
return "AND is_dead_letter_message = TRUE\n";
case QUEUED_MESSAGES:
return "AND is_dead_letter_message = FALSE\n";
default:
throw new IllegalArgumentException("Unsupported IncludeMessages value: " + includeMessages);
}
};
return unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createQuery(bind("SELECT * FROM {:tableName} \n" +
" WHERE queue_name = :queueName\n" +
"{:includeMessages}" +
" LIMIT :pageSize \n" +
" OFFSET :offset",
arg("tableName", sharedQueueTableName),
arg("includeMessages", resolveIncludeMessagesSql.get())))
.bind("queueName", requireNonNull(queueName, "No QueueName provided"))
.bind("offset", startIndex)
.bind("pageSize", pageSize)
.map(queuedMessageMapper)
.list());
}
@Override
public final DurableQueues addInterceptor(DurableQueuesInterceptor interceptor) {
requireNonNull(interceptor, "No interceptor provided");
log.info("Adding interceptor: {}", interceptor);
interceptor.setDurableQueues(this);
interceptors.add(interceptor);
sortInterceptorsByOrder(interceptors);
return this;
}
@Override
public final DurableQueues removeInterceptor(DurableQueuesInterceptor interceptor) {
requireNonNull(interceptor, "No interceptor provided");
log.info("Removing interceptor: {}", interceptor);
interceptors.remove(interceptor);
sortInterceptorsByOrder(interceptors);
return this;
}
@Override
public final Optional getQueueNameFor(QueueEntryId queueEntryId) {
return unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle()
.createQuery(bind("SELECT queue_name FROM {:tableName} WHERE \n" +
" id = :id",
arg("tableName", sharedQueueTableName)))
.bind("id", requireNonNull(queueEntryId, "No queueEntryId provided"))
.mapTo(QueueName.class)
.findOne());
}
@Override
public final Optional getDeadLetterMessage(GetDeadLetterMessage operation) {
requireNonNull(operation, "You must specify a GetDeadLetterMessage instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> getQueuedMessage(operation.queueEntryId, true))
.proceed();
}
@Override
public final Optional getQueuedMessage(GetQueuedMessage operation) {
requireNonNull(operation, "You must specify a GetQueuedMessage instance");
return newInterceptorChainForOperation(operation,
interceptors,
(interceptor, interceptorChain) -> interceptor.intercept(operation, interceptorChain),
() -> getQueuedMessage(operation.queueEntryId, false))
.proceed();
}
private Optional getQueuedMessage(QueueEntryId queueEntryId, boolean isDeadLetterMessage) {
return unitOfWorkFactory.withUnitOfWork(handleAwareUnitOfWork -> handleAwareUnitOfWork.handle().createQuery(bind("SELECT * FROM {:tableName} WHERE \n" +
" id = :id AND\n" +
" is_dead_letter_message = :isDeadLetterMessage",
arg("tableName", sharedQueueTableName)))
.bind("id", requireNonNull(queueEntryId, "No queueEntryId provided"))
.bind("isDeadLetterMessage", isDeadLetterMessage)
.map(queuedMessageMapper)
.findOne());
}
private Object deserializeMessagePayload(QueueName queueName, String messagePayload, String messagePayloadType) {
requireNonNull(queueName, "No queueName provided");
requireNonNull(messagePayload, "No messagePayload provided");
requireNonNull(messagePayloadType, "No messagePayloadType provided");
try {
return jsonSerializer.deserialize(messagePayload, Classes.forName(messagePayloadType));
} catch (Throwable e) {
throw new DurableQueueException(msg("Failed to deserialize message payload of type {}", messagePayloadType), e, queueName);
}
}
private MessageMetaData deserializeMessageMetadata(QueueName queueName, String metaData) {
requireNonNull(queueName, "No queueName provided");
requireNonNull(metaData, "No messagePayload provided");
try {
return jsonSerializer.deserialize(metaData, MessageMetaData.class);
} catch (Throwable e) {
throw new DurableQueueException(msg("Failed to deserialize message meta-data"), e, queueName);
}
}
/**
* Default {@link ObjectMapper} supporting {@link Jdk8Module}, {@link JavaTimeModule}, {@link EssentialTypesJacksonModule} and {@link EssentialsImmutableJacksonModule}, which
* is used together with the {@link JacksonJSONSerializer}
*
* @return the default {@link ObjectMapper}
*/
public static ObjectMapper createDefaultObjectMapper() {
var objectMapper = JsonMapper.builder()
.disable(MapperFeature.AUTO_DETECT_GETTERS)
.disable(MapperFeature.AUTO_DETECT_IS_GETTERS)
.disable(MapperFeature.AUTO_DETECT_SETTERS)
.disable(MapperFeature.DEFAULT_VIEW_INCLUSION)
.disable(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS)
.disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)
.disable(SerializationFeature.FAIL_ON_EMPTY_BEANS)
.enable(MapperFeature.AUTO_DETECT_CREATORS)
.enable(MapperFeature.AUTO_DETECT_FIELDS)
.enable(MapperFeature.PROPAGATE_TRANSIENT_MARKER)
.addModule(new Jdk8Module())
.addModule(new JavaTimeModule())
.addModule(new EssentialTypesJacksonModule())
.addModule(new EssentialsImmutableJacksonModule())
.build();
objectMapper.setVisibility(objectMapper.getSerializationConfig().getDefaultVisibilityChecker()
.withGetterVisibility(JsonAutoDetect.Visibility.NONE)
.withSetterVisibility(JsonAutoDetect.Visibility.NONE)
.withFieldVisibility(JsonAutoDetect.Visibility.ANY)
.withCreatorVisibility(JsonAutoDetect.Visibility.ANY));
return objectMapper;
}
private static class SingleOperationTransactionDurableQueuesInterceptor implements DurableQueuesInterceptor {
private final HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory;
private DurableQueues durableQueues;
public SingleOperationTransactionDurableQueuesInterceptor(HandleAwareUnitOfWorkFactory extends HandleAwareUnitOfWork> unitOfWorkFactory) {
this.unitOfWorkFactory = unitOfWorkFactory;
}
@Override
public void setDurableQueues(DurableQueues durableQueues) {
this.durableQueues = requireNonNull(durableQueues, "No durableQueues instance provided");
}
@Override
public Optional intercept(GetDeadLetterMessage operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public Optional intercept(GetQueuedMessage operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public DurableQueueConsumer intercept(ConsumeFromQueue operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public DurableQueueConsumer intercept(StopConsumingFromQueue operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public QueueEntryId intercept(QueueMessage operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public QueueEntryId intercept(QueueMessageAsDeadLetterMessage operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public List intercept(QueueMessages operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public Optional intercept(RetryMessage operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public Optional intercept(MarkAsDeadLetterMessage operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public Optional intercept(ResurrectDeadLetterMessage operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public boolean intercept(AcknowledgeMessageAsHandled operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public boolean intercept(DeleteMessage operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public Optional intercept(GetNextMessageReadyForDelivery operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public long intercept(GetTotalMessagesQueuedFor operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public List intercept(GetQueuedMessages operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public List intercept(GetDeadLetterMessages operation, InterceptorChain, DurableQueuesInterceptor> interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
@Override
public int intercept(PurgeQueue operation, InterceptorChain interceptorChain) {
return unitOfWorkFactory.withUnitOfWork(interceptorChain::proceed);
}
}
private class QueuedMessageRowMapper implements RowMapper {
public QueuedMessageRowMapper() {
}
@Override
public QueuedMessage map(ResultSet rs, StatementContext ctx) throws SQLException {
var queueName = QueueName.of(rs.getString("queue_name"));
var messagePayload = PostgresqlDurableQueues.this.deserializeMessagePayload(queueName, rs.getString("message_payload"), rs.getString("message_payload_type"));
MessageMetaData messageMetaData = null;
var metaDataColumnValue = rs.getString("meta_data");
if (metaDataColumnValue != null) {
messageMetaData = PostgresqlDurableQueues.this.deserializeMessageMetadata(queueName, metaDataColumnValue);
} else {
messageMetaData = new MessageMetaData();
}
var deliveryMode = QueuedMessage.DeliveryMode.valueOf(rs.getString("delivery_mode"));
Message message = null;
switch (deliveryMode) {
case NORMAL:
message = new Message(messagePayload,
messageMetaData);
break;
case IN_ORDER:
message = new OrderedMessage(messagePayload,
rs.getString("key"),
rs.getLong("key_order"),
messageMetaData);
break;
default:
throw new IllegalStateException(msg("Unsupported deliveryMode '{}'", deliveryMode));
}
return new DefaultQueuedMessage(QueueEntryId.of(rs.getString("id")),
queueName,
message,
rs.getObject("added_ts", OffsetDateTime.class),
rs.getObject("next_delivery_ts", OffsetDateTime.class),
rs.getObject("delivery_ts", OffsetDateTime.class),
rs.getString("last_delivery_error"),
rs.getInt("total_attempts"),
rs.getInt("redelivery_attempts"),
rs.getBoolean("is_dead_letter_message"),
rs.getBoolean("is_being_delivered"));
}
}
public static class QueueTableNotification extends TableChangeNotification {
@JsonProperty("id")
private String id;
@JsonProperty("queue_name")
private String queueName;
@JsonProperty("added_ts")
private OffsetDateTime addedTimestamp;
@JsonProperty("next_delivery_ts")
private OffsetDateTime nextDeliveryTimestamp;
@JsonProperty("delivery_ts")
private OffsetDateTime deliveryTimestamp;
@JsonProperty("is_dead_letter_message")
private boolean isDeadLetterMessage;
@JsonProperty("is_being_delivered")
private boolean isBeingDelivered;
public QueueTableNotification() {
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy