All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.boot.autoconfigure.amqp.RabbitProperties Maven / Gradle / Ivy

There is a newer version: 3.2.5
Show newest version
/*
 * Copyright 2012-2020 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 org.springframework.boot.autoconfigure.amqp;

import java.time.Duration;
import java.time.temporal.ChronoUnit;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;

import org.springframework.amqp.core.AcknowledgeMode;
import org.springframework.amqp.rabbit.connection.AbstractConnectionFactory.AddressShuffleMode;
import org.springframework.amqp.rabbit.connection.CachingConnectionFactory.CacheMode;
import org.springframework.amqp.rabbit.connection.CachingConnectionFactory.ConfirmType;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.convert.DurationUnit;
import org.springframework.util.CollectionUtils;
import org.springframework.util.StringUtils;

/**
 * Configuration properties for Rabbit.
 *
 * @author Greg Turnquist
 * @author Dave Syer
 * @author Stephane Nicoll
 * @author Andy Wilkinson
 * @author Josh Thornhill
 * @author Gary Russell
 * @author Artsiom Yudovin
 * @author Franjo Zilic
 * @since 1.0.0
 */
@ConfigurationProperties(prefix = "spring.rabbitmq")
public class RabbitProperties {

	private static final int DEFAULT_PORT = 5672;

	private static final int DEFAULT_PORT_SECURE = 5671;

	/**
	 * RabbitMQ host. Ignored if an address is set.
	 */
	private String host = "localhost";

	/**
	 * RabbitMQ port. Ignored if an address is set. Default to 5672, or 5671 if SSL is
	 * enabled.
	 */
	private Integer port;

	/**
	 * Login user to authenticate to the broker.
	 */
	private String username = "guest";

	/**
	 * Login to authenticate against the broker.
	 */
	private String password = "guest";

	/**
	 * SSL configuration.
	 */
	private final Ssl ssl = new Ssl();

	/**
	 * Virtual host to use when connecting to the broker.
	 */
	private String virtualHost;

	/**
	 * Comma-separated list of addresses to which the client should connect. When set, the
	 * host and port are ignored.
	 */
	private String addresses;

	/**
	 * Mode used to shuffle configured addresses.
	 */
	private AddressShuffleMode addressShuffleMode = AddressShuffleMode.NONE;

	/**
	 * Requested heartbeat timeout; zero for none. If a duration suffix is not specified,
	 * seconds will be used.
	 */
	@DurationUnit(ChronoUnit.SECONDS)
	private Duration requestedHeartbeat;

	/**
	 * Number of channels per connection requested by the client. Use 0 for unlimited.
	 */
	private int requestedChannelMax = 2047;

	/**
	 * Whether to enable publisher returns.
	 */
	private boolean publisherReturns;

	/**
	 * Type of publisher confirms to use.
	 */
	private ConfirmType publisherConfirmType;

	/**
	 * Connection timeout. Set it to zero to wait forever.
	 */
	private Duration connectionTimeout;

	/**
	 * Continuation timeout for RPC calls in channels. Set it to zero to wait forever.
	 */
	private Duration channelRpcTimeout = Duration.ofMinutes(10);

	/**
	 * Cache configuration.
	 */
	private final Cache cache = new Cache();

	/**
	 * Listener container configuration.
	 */
	private final Listener listener = new Listener();

	private final Template template = new Template();

	private List
parsedAddresses; public String getHost() { return this.host; } /** * Returns the host from the first address, or the configured host if no addresses * have been set. * @return the host * @see #setAddresses(String) * @see #getHost() */ public String determineHost() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { return getHost(); } return this.parsedAddresses.get(0).host; } public void setHost(String host) { this.host = host; } public Integer getPort() { return this.port; } /** * Returns the port from the first address, or the configured port if no addresses * have been set. * @return the port * @see #setAddresses(String) * @see #getPort() */ public int determinePort() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { Integer port = getPort(); if (port != null) { return port; } return (Optional.ofNullable(getSsl().getEnabled()).orElse(false)) ? DEFAULT_PORT_SECURE : DEFAULT_PORT; } return this.parsedAddresses.get(0).port; } public void setPort(Integer port) { this.port = port; } public String getAddresses() { return this.addresses; } /** * Returns the comma-separated addresses or a single address ({@code host:port}) * created from the configured host and port if no addresses have been set. * @return the addresses */ public String determineAddresses() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { return this.host + ":" + determinePort(); } List addressStrings = new ArrayList<>(); for (Address parsedAddress : this.parsedAddresses) { addressStrings.add(parsedAddress.host + ":" + parsedAddress.port); } return StringUtils.collectionToCommaDelimitedString(addressStrings); } public void setAddresses(String addresses) { this.addresses = addresses; this.parsedAddresses = parseAddresses(addresses); } private List
parseAddresses(String addresses) { List
parsedAddresses = new ArrayList<>(); for (String address : StringUtils.commaDelimitedListToStringArray(addresses)) { parsedAddresses.add(new Address(address, Optional.ofNullable(getSsl().getEnabled()).orElse(false))); } return parsedAddresses; } public String getUsername() { return this.username; } /** * If addresses have been set and the first address has a username it is returned. * Otherwise returns the result of calling {@code getUsername()}. * @return the username * @see #setAddresses(String) * @see #getUsername() */ public String determineUsername() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { return this.username; } Address address = this.parsedAddresses.get(0); return (address.username != null) ? address.username : this.username; } public void setUsername(String username) { this.username = username; } public String getPassword() { return this.password; } /** * If addresses have been set and the first address has a password it is returned. * Otherwise returns the result of calling {@code getPassword()}. * @return the password or {@code null} * @see #setAddresses(String) * @see #getPassword() */ public String determinePassword() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { return getPassword(); } Address address = this.parsedAddresses.get(0); return (address.password != null) ? address.password : getPassword(); } public void setPassword(String password) { this.password = password; } public Ssl getSsl() { return this.ssl; } public String getVirtualHost() { return this.virtualHost; } /** * If addresses have been set and the first address has a virtual host it is returned. * Otherwise returns the result of calling {@code getVirtualHost()}. * @return the virtual host or {@code null} * @see #setAddresses(String) * @see #getVirtualHost() */ public String determineVirtualHost() { if (CollectionUtils.isEmpty(this.parsedAddresses)) { return getVirtualHost(); } Address address = this.parsedAddresses.get(0); return (address.virtualHost != null) ? address.virtualHost : getVirtualHost(); } public void setVirtualHost(String virtualHost) { this.virtualHost = StringUtils.hasText(virtualHost) ? virtualHost : "/"; } public AddressShuffleMode getAddressShuffleMode() { return this.addressShuffleMode; } public void setAddressShuffleMode(AddressShuffleMode addressShuffleMode) { this.addressShuffleMode = addressShuffleMode; } public Duration getRequestedHeartbeat() { return this.requestedHeartbeat; } public void setRequestedHeartbeat(Duration requestedHeartbeat) { this.requestedHeartbeat = requestedHeartbeat; } public int getRequestedChannelMax() { return this.requestedChannelMax; } public void setRequestedChannelMax(int requestedChannelMax) { this.requestedChannelMax = requestedChannelMax; } public boolean isPublisherReturns() { return this.publisherReturns; } public void setPublisherReturns(boolean publisherReturns) { this.publisherReturns = publisherReturns; } public Duration getConnectionTimeout() { return this.connectionTimeout; } public void setPublisherConfirmType(ConfirmType publisherConfirmType) { this.publisherConfirmType = publisherConfirmType; } public ConfirmType getPublisherConfirmType() { return this.publisherConfirmType; } public void setConnectionTimeout(Duration connectionTimeout) { this.connectionTimeout = connectionTimeout; } public Duration getChannelRpcTimeout() { return this.channelRpcTimeout; } public void setChannelRpcTimeout(Duration channelRpcTimeout) { this.channelRpcTimeout = channelRpcTimeout; } public Cache getCache() { return this.cache; } public Listener getListener() { return this.listener; } public Template getTemplate() { return this.template; } public class Ssl { /** * Whether to enable SSL support. Determined automatically if an address is * provided with the protocol (amqp:// vs. amqps://). */ private Boolean enabled; /** * Path to the key store that holds the SSL certificate. */ private String keyStore; /** * Key store type. */ private String keyStoreType = "PKCS12"; /** * Password used to access the key store. */ private String keyStorePassword; /** * Trust store that holds SSL certificates. */ private String trustStore; /** * Trust store type. */ private String trustStoreType = "JKS"; /** * Password used to access the trust store. */ private String trustStorePassword; /** * SSL algorithm to use. By default, configured by the Rabbit client library. */ private String algorithm; /** * Whether to enable server side certificate validation. */ private boolean validateServerCertificate = true; /** * Whether to enable hostname verification. */ private boolean verifyHostname = true; public Boolean getEnabled() { return this.enabled; } /** * Returns whether SSL is enabled from the first address, or the configured ssl * enabled flag if no addresses have been set. * @return whether ssl is enabled * @see #setAddresses(String) * @see #getEnabled() () */ public boolean determineEnabled() { boolean defaultEnabled = Optional.ofNullable(getEnabled()).orElse(false); if (CollectionUtils.isEmpty(RabbitProperties.this.parsedAddresses)) { return defaultEnabled; } Address address = RabbitProperties.this.parsedAddresses.get(0); return address.determineSslEnabled(defaultEnabled); } public void setEnabled(Boolean enabled) { this.enabled = enabled; } public String getKeyStore() { return this.keyStore; } public void setKeyStore(String keyStore) { this.keyStore = keyStore; } public String getKeyStoreType() { return this.keyStoreType; } public void setKeyStoreType(String keyStoreType) { this.keyStoreType = keyStoreType; } public String getKeyStorePassword() { return this.keyStorePassword; } public void setKeyStorePassword(String keyStorePassword) { this.keyStorePassword = keyStorePassword; } public String getTrustStore() { return this.trustStore; } public void setTrustStore(String trustStore) { this.trustStore = trustStore; } public String getTrustStoreType() { return this.trustStoreType; } public void setTrustStoreType(String trustStoreType) { this.trustStoreType = trustStoreType; } public String getTrustStorePassword() { return this.trustStorePassword; } public void setTrustStorePassword(String trustStorePassword) { this.trustStorePassword = trustStorePassword; } public String getAlgorithm() { return this.algorithm; } public void setAlgorithm(String sslAlgorithm) { this.algorithm = sslAlgorithm; } public boolean isValidateServerCertificate() { return this.validateServerCertificate; } public void setValidateServerCertificate(boolean validateServerCertificate) { this.validateServerCertificate = validateServerCertificate; } public boolean getVerifyHostname() { return this.verifyHostname; } public void setVerifyHostname(boolean verifyHostname) { this.verifyHostname = verifyHostname; } } public static class Cache { private final Channel channel = new Channel(); private final Connection connection = new Connection(); public Channel getChannel() { return this.channel; } public Connection getConnection() { return this.connection; } public static class Channel { /** * Number of channels to retain in the cache. When "check-timeout" > 0, max * channels per connection. */ private Integer size; /** * Duration to wait to obtain a channel if the cache size has been reached. If * 0, always create a new channel. */ private Duration checkoutTimeout; public Integer getSize() { return this.size; } public void setSize(Integer size) { this.size = size; } public Duration getCheckoutTimeout() { return this.checkoutTimeout; } public void setCheckoutTimeout(Duration checkoutTimeout) { this.checkoutTimeout = checkoutTimeout; } } public static class Connection { /** * Connection factory cache mode. */ private CacheMode mode = CacheMode.CHANNEL; /** * Number of connections to cache. Only applies when mode is CONNECTION. */ private Integer size; public CacheMode getMode() { return this.mode; } public void setMode(CacheMode mode) { this.mode = mode; } public Integer getSize() { return this.size; } public void setSize(Integer size) { this.size = size; } } } public enum ContainerType { /** * Container where the RabbitMQ consumer dispatches messages to an invoker thread. */ SIMPLE, /** * Container where the listener is invoked directly on the RabbitMQ consumer * thread. */ DIRECT } public static class Listener { /** * Listener container type. */ private ContainerType type = ContainerType.SIMPLE; private final SimpleContainer simple = new SimpleContainer(); private final DirectContainer direct = new DirectContainer(); public ContainerType getType() { return this.type; } public void setType(ContainerType containerType) { this.type = containerType; } public SimpleContainer getSimple() { return this.simple; } public DirectContainer getDirect() { return this.direct; } } public abstract static class AmqpContainer { /** * Whether to start the container automatically on startup. */ private boolean autoStartup = true; /** * Acknowledge mode of container. */ private AcknowledgeMode acknowledgeMode; /** * Maximum number of unacknowledged messages that can be outstanding at each * consumer. */ private Integer prefetch; /** * Whether rejected deliveries are re-queued by default. */ private Boolean defaultRequeueRejected; /** * How often idle container events should be published. */ private Duration idleEventInterval; /** * Whether the container should present batched messages as discrete messages or * call the listener with the batch. */ private boolean deBatchingEnabled = true; /** * Optional properties for a retry interceptor. */ private final ListenerRetry retry = new ListenerRetry(); public boolean isAutoStartup() { return this.autoStartup; } public void setAutoStartup(boolean autoStartup) { this.autoStartup = autoStartup; } public AcknowledgeMode getAcknowledgeMode() { return this.acknowledgeMode; } public void setAcknowledgeMode(AcknowledgeMode acknowledgeMode) { this.acknowledgeMode = acknowledgeMode; } public Integer getPrefetch() { return this.prefetch; } public void setPrefetch(Integer prefetch) { this.prefetch = prefetch; } public Boolean getDefaultRequeueRejected() { return this.defaultRequeueRejected; } public void setDefaultRequeueRejected(Boolean defaultRequeueRejected) { this.defaultRequeueRejected = defaultRequeueRejected; } public Duration getIdleEventInterval() { return this.idleEventInterval; } public void setIdleEventInterval(Duration idleEventInterval) { this.idleEventInterval = idleEventInterval; } public abstract boolean isMissingQueuesFatal(); public boolean isDeBatchingEnabled() { return this.deBatchingEnabled; } public void setDeBatchingEnabled(boolean deBatchingEnabled) { this.deBatchingEnabled = deBatchingEnabled; } public ListenerRetry getRetry() { return this.retry; } } /** * Configuration properties for {@code SimpleMessageListenerContainer}. */ public static class SimpleContainer extends AmqpContainer { /** * Minimum number of listener invoker threads. */ private Integer concurrency; /** * Maximum number of listener invoker threads. */ private Integer maxConcurrency; /** * Batch size, expressed as the number of physical messages, to be used by the * container. */ private Integer batchSize; /** * Whether to fail if the queues declared by the container are not available on * the broker and/or whether to stop the container if one or more queues are * deleted at runtime. */ private boolean missingQueuesFatal = true; /** * Whether the container creates a batch of messages based on the * 'receive-timeout' and 'batch-size'. Coerces 'de-batching-enabled' to true to * include the contents of a producer created batch in the batch as discrete * records. */ private boolean consumerBatchEnabled; public Integer getConcurrency() { return this.concurrency; } public void setConcurrency(Integer concurrency) { this.concurrency = concurrency; } public Integer getMaxConcurrency() { return this.maxConcurrency; } public void setMaxConcurrency(Integer maxConcurrency) { this.maxConcurrency = maxConcurrency; } public Integer getBatchSize() { return this.batchSize; } public void setBatchSize(Integer batchSize) { this.batchSize = batchSize; } @Override public boolean isMissingQueuesFatal() { return this.missingQueuesFatal; } public void setMissingQueuesFatal(boolean missingQueuesFatal) { this.missingQueuesFatal = missingQueuesFatal; } public boolean isConsumerBatchEnabled() { return this.consumerBatchEnabled; } public void setConsumerBatchEnabled(boolean consumerBatchEnabled) { this.consumerBatchEnabled = consumerBatchEnabled; } } /** * Configuration properties for {@code DirectMessageListenerContainer}. */ public static class DirectContainer extends AmqpContainer { /** * Number of consumers per queue. */ private Integer consumersPerQueue; /** * Whether to fail if the queues declared by the container are not available on * the broker. */ private boolean missingQueuesFatal = false; public Integer getConsumersPerQueue() { return this.consumersPerQueue; } public void setConsumersPerQueue(Integer consumersPerQueue) { this.consumersPerQueue = consumersPerQueue; } @Override public boolean isMissingQueuesFatal() { return this.missingQueuesFatal; } public void setMissingQueuesFatal(boolean missingQueuesFatal) { this.missingQueuesFatal = missingQueuesFatal; } } public static class Template { private final Retry retry = new Retry(); /** * Whether to enable mandatory messages. */ private Boolean mandatory; /** * Timeout for `receive()` operations. */ private Duration receiveTimeout; /** * Timeout for `sendAndReceive()` operations. */ private Duration replyTimeout; /** * Name of the default exchange to use for send operations. */ private String exchange = ""; /** * Value of a default routing key to use for send operations. */ private String routingKey = ""; /** * Name of the default queue to receive messages from when none is specified * explicitly. */ private String defaultReceiveQueue; public Retry getRetry() { return this.retry; } public Boolean getMandatory() { return this.mandatory; } public void setMandatory(Boolean mandatory) { this.mandatory = mandatory; } public Duration getReceiveTimeout() { return this.receiveTimeout; } public void setReceiveTimeout(Duration receiveTimeout) { this.receiveTimeout = receiveTimeout; } public Duration getReplyTimeout() { return this.replyTimeout; } public void setReplyTimeout(Duration replyTimeout) { this.replyTimeout = replyTimeout; } public String getExchange() { return this.exchange; } public void setExchange(String exchange) { this.exchange = exchange; } public String getRoutingKey() { return this.routingKey; } public void setRoutingKey(String routingKey) { this.routingKey = routingKey; } public String getDefaultReceiveQueue() { return this.defaultReceiveQueue; } public void setDefaultReceiveQueue(String defaultReceiveQueue) { this.defaultReceiveQueue = defaultReceiveQueue; } } public static class Retry { /** * Whether publishing retries are enabled. */ private boolean enabled; /** * Maximum number of attempts to deliver a message. */ private int maxAttempts = 3; /** * Duration between the first and second attempt to deliver a message. */ private Duration initialInterval = Duration.ofMillis(1000); /** * Multiplier to apply to the previous retry interval. */ private double multiplier = 1.0; /** * Maximum duration between attempts. */ private Duration maxInterval = Duration.ofMillis(10000); public boolean isEnabled() { return this.enabled; } public void setEnabled(boolean enabled) { this.enabled = enabled; } public int getMaxAttempts() { return this.maxAttempts; } public void setMaxAttempts(int maxAttempts) { this.maxAttempts = maxAttempts; } public Duration getInitialInterval() { return this.initialInterval; } public void setInitialInterval(Duration initialInterval) { this.initialInterval = initialInterval; } public double getMultiplier() { return this.multiplier; } public void setMultiplier(double multiplier) { this.multiplier = multiplier; } public Duration getMaxInterval() { return this.maxInterval; } public void setMaxInterval(Duration maxInterval) { this.maxInterval = maxInterval; } } public static class ListenerRetry extends Retry { /** * Whether retries are stateless or stateful. */ private boolean stateless = true; public boolean isStateless() { return this.stateless; } public void setStateless(boolean stateless) { this.stateless = stateless; } } private static final class Address { private static final String PREFIX_AMQP = "amqp://"; private static final String PREFIX_AMQP_SECURE = "amqps://"; private String host; private int port; private String username; private String password; private String virtualHost; private Boolean secureConnection; private Address(String input, boolean sslEnabled) { input = input.trim(); input = trimPrefix(input); input = parseUsernameAndPassword(input); input = parseVirtualHost(input); parseHostAndPort(input, sslEnabled); } private String trimPrefix(String input) { if (input.startsWith(PREFIX_AMQP_SECURE)) { this.secureConnection = true; return input.substring(PREFIX_AMQP_SECURE.length()); } if (input.startsWith(PREFIX_AMQP)) { this.secureConnection = false; return input.substring(PREFIX_AMQP.length()); } return input; } private String parseUsernameAndPassword(String input) { if (input.contains("@")) { String[] split = StringUtils.split(input, "@"); String creds = split[0]; input = split[1]; split = StringUtils.split(creds, ":"); this.username = split[0]; if (split.length > 0) { this.password = split[1]; } } return input; } private String parseVirtualHost(String input) { int hostIndex = input.indexOf('/'); if (hostIndex >= 0) { this.virtualHost = input.substring(hostIndex + 1); if (this.virtualHost.isEmpty()) { this.virtualHost = "/"; } input = input.substring(0, hostIndex); } return input; } private void parseHostAndPort(String input, boolean sslEnabled) { int portIndex = input.indexOf(':'); if (portIndex == -1) { this.host = input; this.port = (determineSslEnabled(sslEnabled)) ? DEFAULT_PORT_SECURE : DEFAULT_PORT; } else { this.host = input.substring(0, portIndex); this.port = Integer.parseInt(input.substring(portIndex + 1)); } } private boolean determineSslEnabled(boolean sslEnabled) { return (this.secureConnection != null) ? this.secureConnection : sslEnabled; } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy