io.nats.client.Connection Maven / Gradle / Ivy
// Copyright 2015-2018 The NATS 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:
//
// 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 io.nats.client;
import io.nats.client.api.ServerInfo;
import io.nats.client.impl.Headers;
import java.io.IOException;
import java.net.InetAddress;
import java.time.Duration;
import java.util.Collection;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.TimeoutException;
/**
* The Connection class is at the heart of the NATS Java client. Fundamentally a connection represents
* a single network connection to the NATS server.
*
* Each connection you create will result in the creation of a single socket and several threads:
*
* - A reader thread for taking data off the socket
*
- A writer thread for putting data onto the socket
*
- A timer thread for a few maintenance timers
*
- A dispatch thread to handle request/reply traffic
*
*
* The connection has a {@link Connection.Status status} which can be checked using the {@link #getStatus() getStatus}
* method or watched using a {@link ConnectionListener ConnectionListener}.
*
*
Connections, by default, are configured to try to reconnect to the server if there is a network failure up to
* {@link Options#DEFAULT_MAX_RECONNECT times}. You can configure this behavior in the {@link Options Options}.
* Moreover, the options allows you to control whether reconnect happens in the same order every time, and the time
* to wait if trying to reconnect to the same server over and over.
*
*
The list of servers used for connecting is provided by the {@link Options Options}. The list of servers used
* during reconnect can be an expanded list. This expansion comes from the connections most recent server. For example,
* if you connect to serverA, it can tell the connection "i know about serverB and serverC". If serverA goes down
* the client library will try to connect to serverA, serverB and serverC. Now, if the library connects to serverB, it may tell the client
* "i know about serverB and serverE". The client's list of servers, available from {@link #getServers() getServers()}
* will now be serverA from the initial connect, serverB and serverE, the reference to serverC is lost.
*
*
When a connection is {@link #close() closed} the thread and socket resources are cleaned up.
*
*
All outgoing messages are sent through the connection object using one of the two
* {@link #publish(String, byte[]) publish} methods or the {@link #request(String, byte[]) request} method.
* When publishing you can specify a reply to subject which can be retrieved by the receiver to respond.
* The request method will handle this behavior itself, but it relies on getting the value out of a Future
* so may be less flexible than publish with replyTo set.
*
*
Messages can be received in two ways. You can create a Subscription which will allow you to read messages
* synchronously using the {@link Subscription#nextMessage(Duration) nextMessage} method or you can create a
* {@link Dispatcher Dispatcher}. The Dispatcher will create a thread to listen for messages on one or more subscriptions.
* The Dispatcher groups a set of subscriptions into a single listener thread that calls application code
* for each messages.
*
*
Applications can use the {@link #flush(Duration) flush} method to check that published messages have
* made it to the server. However, this method initiates a round trip to the server and waits for the response so
* it should be used sparingly.
*
*
The connection provides two listeners via the Options. The {@link ConnectionListener ConnectionListener}
* can be used to listen for lifecycle events. This listener is required for
* {@link Nats#connectAsynchronously(Options, boolean) connectAsynchronously}, but otherwise optional. The
* {@link ErrorListener ErrorListener} provides three callback opportunities including slow consumers, error
* messages from the server and exceptions handled by the client library. These listeners can only be set at creation time
* using the {@link Options options}.
*
*
Note: The publish methods take an array of bytes. These arrays will not be copied. This design choice
* is based on the common case of strings or objects being converted to bytes. Once a client can be sure a message was received by
* the NATS server it is theoretically possible to reuse that byte array, but this pattern should be treated as advanced and only used
* after thorough testing.
*/
public interface Connection extends AutoCloseable {
enum Status {
/**
* The {@code Connection} is not connected.
*/
DISCONNECTED,
/**
* The {@code Connection} is currently connected.
*/
CONNECTED,
/**
* The {@code Connection} is currently closed.
*/
CLOSED,
/**
* The {@code Connection} is currently attempting to reconnect to a server from its server list.
*/
RECONNECTING,
/**
* The {@code Connection} is currently connecting to a server for the first
* time.
*/
CONNECTING;
}
/**
* Send a message to the specified subject. The message body will
* not be copied. The expected usage with string content is something
* like:
*
*
* nc = Nats.connect()
* nc.publish("destination", "message".getBytes("UTF-8"))
*
*
* where the sender creates a byte array immediately before calling publish.
* See {@link #publish(String, String, byte[]) publish()} for more details on
* publish during reconnect.
*
* @param subject the subject to send the message to
* @param body the message body
* @throws IllegalStateException if the reconnect buffer is exceeded
*/
void publish(String subject, byte[] body);
/**
* Send a message to the specified subject. The message body will
* not be copied. The expected usage with string content is something
* like:
*
*
* nc = Nats.connect()
* Headers h = new Headers().put("key", "value");
* nc.publish("destination", h, "message".getBytes("UTF-8"))
*
*
* where the sender creates a byte array immediately before calling publish.
* See {@link #publish(String, String, byte[]) publish()} for more details on
* publish during reconnect.
*
* @param subject the subject to send the message to
* @param headers Optional headers to publish with the message.
* @param body the message body
* @throws IllegalStateException if the reconnect buffer is exceeded
*/
void publish(String subject, Headers headers, byte[] body);
/**
* Send a request to the specified subject, providing a replyTo subject. The
* message body will not be copied. The expected usage with
* string content is something like:
*
*
* nc = Nats.connect()
* nc.publish("destination", "reply-to", "message".getBytes("UTF-8"))
*
*
* where the sender creates a byte array immediately before calling publish.
*
* During reconnect the client will try to buffer messages. The buffer size is set
* in the connect options, see {@link Options.Builder#reconnectBufferSize(long) reconnectBufferSize()}
* with a default value of {@link Options#DEFAULT_RECONNECT_BUF_SIZE 8 * 1024 * 1024} bytes.
* If the buffer is exceeded an IllegalStateException is thrown. Applications should use
* this exception as a signal to wait for reconnect before continuing.
*
* @param subject the subject to send the message to
* @param replyTo the subject the receiver should send the response to
* @param body the message body
* @throws IllegalStateException if the reconnect buffer is exceeded
*/
void publish(String subject, String replyTo, byte[] body);
/**
* Send a request to the specified subject, providing a replyTo subject. The
* message body will not be copied. The expected usage with
* string content is something like:
*
*
* nc = Nats.connect()
* Headers h = new Headers().put("key", "value");
* nc.publish("destination", "reply-to", h, "message".getBytes("UTF-8"))
*
*
* where the sender creates a byte array immediately before calling publish.
*
* During reconnect the client will try to buffer messages. The buffer size is set
* in the connect options, see {@link Options.Builder#reconnectBufferSize(long) reconnectBufferSize()}
* with a default value of {@link Options#DEFAULT_RECONNECT_BUF_SIZE 8 * 1024 * 1024} bytes.
* If the buffer is exceeded an IllegalStateException is thrown. Applications should use
* this exception as a signal to wait for reconnect before continuing.
*
* @param subject the subject to send the message to
* @param replyTo the subject the receiver should send the response to
* @param headers Optional headers to publish with the message.
* @param body the message body
* @throws IllegalStateException if the reconnect buffer is exceeded
*/
void publish(String subject, String replyTo, Headers headers, byte[] body);
/**
* Send a message to the specified subject. The message body will
* not be copied. The expected usage with string content is something
* like:
*
*
* nc = Nats.connect()
* nc.publish(NatsMessage.builder()...build())
*
*
* where the sender creates a byte array immediately before calling publish.
* See {@link #publish(String, String, byte[]) publish()} for more details on
* publish during reconnect.
*
* @param message the message
* @throws IllegalStateException if the reconnect buffer is exceeded
*/
void publish(Message message);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* @param subject the subject for the service that will handle the request
* @param body the content of the message
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture request(String subject, byte[] body);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* @param subject the subject for the service that will handle the request
* @param headers Optional headers to publish with the message.
* @param body the content of the message
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture request(String subject, Headers headers, byte[] body);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* @param subject the subject for the service that will handle the request
* @param body the content of the message
* @param timeout the time to wait for a response
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture requestWithTimeout(String subject, byte[] body, Duration timeout);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* @param subject the subject for the service that will handle the request
* @param body the content of the message
* @param headers Optional headers to publish with the message.
* @param timeout the time to wait for a response
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture requestWithTimeout(String subject, Headers headers, byte[] body, Duration timeout);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* The Message object allows you to set a replyTo, but in requests,
* the replyTo is reserved for internal use as the address for the
* server to respond to the client with the consumer's reply.
*
* @param message the message
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture request(Message message);
/**
* Send a request. The returned future will be completed when the
* response comes back.
*
* The Message object allows you to set a replyTo, but in requests,
* the replyTo is reserved for internal use as the address for the
* server to respond to the client with the consumer's reply.
*
* @param message the message
* @param timeout the time to wait for a response
* @return a Future for the response, which may be cancelled on error or timed out
*/
CompletableFuture requestWithTimeout(Message message, Duration timeout);
/**
* Send a request and returns the reply or null. This version of request is equivalent
* to calling get on the future returned from {@link #request(String, byte[]) request()} with
* the timeout and handling the ExecutionException and TimeoutException.
*
* @param subject the subject for the service that will handle the request
* @param body the content of the message
* @param timeout the time to wait for a response
* @return the reply message or null if the timeout is reached
* @throws InterruptedException if one is thrown while waiting, in order to propagate it up
*/
Message request(String subject, byte[] body, Duration timeout) throws InterruptedException;
/**
* Send a request and returns the reply or null. This version of request is equivalent
* to calling get on the future returned from {@link #request(String, byte[]) request()} with
* the timeout and handling the ExecutionException and TimeoutException.
*
* @param subject the subject for the service that will handle the request
* @param headers Optional headers to publish with the message.
* @param body the content of the message
* @param timeout the time to wait for a response
* @return the reply message or null if the timeout is reached
* @throws InterruptedException if one is thrown while waiting, in order to propagate it up
*/
Message request(String subject, Headers headers, byte[] body, Duration timeout) throws InterruptedException;
/**
* Send a request and returns the reply or null. This version of request is equivalent
* to calling get on the future returned from {@link #request(String, byte[]) request()} with
* the timeout and handling the ExecutionException and TimeoutException.
*
* The Message object allows you to set a replyTo, but in requests,
* the replyTo is reserved for internal use as the address for the
* server to respond to the client with the consumer's reply.
*
* @param message the message
* @param timeout the time to wait for a response
* @return the reply message or null if the timeout is reached
* @throws InterruptedException if one is thrown while waiting, in order to propagate it up
*/
Message request(Message message, Duration timeout) throws InterruptedException;
/**
* Create a synchronous subscription to the specified subject.
*
* Use the {@link io.nats.client.Subscription#nextMessage(Duration) nextMessage}
* method to read messages for this subscription.
*
*
See {@link #createDispatcher(MessageHandler) createDispatcher} for
* information about creating an asynchronous subscription with callbacks.
*
*
As of 2.6.1 this method will throw an IllegalArgumentException if the subject contains whitespace.
*
* @param subject the subject to subscribe to
* @return an object representing the subscription
*/
Subscription subscribe(String subject);
/**
* Create a synchronous subscription to the specified subject and queue.
*
*
Use the {@link Subscription#nextMessage(Duration) nextMessage} method to read
* messages for this subscription.
*
*
See {@link #createDispatcher(MessageHandler) createDispatcher} for
* information about creating an asynchronous subscription with callbacks.
*
*
As of 2.6.1 this method will throw an IllegalArgumentException if either string contains whitespace.
*
* @param subject the subject to subscribe to
* @param queueName the queue group to join
* @return an object representing the subscription
*/
Subscription subscribe(String subject, String queueName);
/**
* Create a {@code Dispatcher} for this connection. The dispatcher can group one
* or more subscriptions into a single callback thread. All messages go to the
* same {@code MessageHandler}.
*
*
Use the Dispatcher's {@link Dispatcher#subscribe(String)} and
* {@link Dispatcher#subscribe(String, String)} methods to add subscriptions.
*
*
* nc = Nats.connect()
* d = nc.createDispatcher((m) -> System.out.println(m)).subscribe("hello");
*
*
* @param handler The target for the messages
* @return a new Dispatcher
*/
Dispatcher createDispatcher(MessageHandler handler);
/**
* Convenience method to create a dispatcher with no default handler. Only used
* with JetStream push subscriptions that require specific handlers per subscription.
*
* @return a new Dispatcher
*/
Dispatcher createDispatcher();
/**
* Close a dispatcher. This will unsubscribe any subscriptions and stop the delivery thread.
*
* Once closed the dispatcher will throw an exception on subsequent subscribe or unsubscribe calls.
*
* @param dispatcher the dispatcher to close
*/
void closeDispatcher(Dispatcher dispatcher);
/**
* Attach another ConnectionListener.
*
*
The ConnectionListener will only receive Connection events arriving after it has been attached. When
* a Connection event is raised, the invocation order and parallelism of multiple ConnectionListeners is not
* specified.
*
* @param connectionListener the ConnectionListener to attach
*/
void addConnectionListener(ConnectionListener connectionListener);
/**
* Detach a ConnectionListioner. This will cease delivery of any further Connection events to this instance.
*
* @param connectionListener the ConnectionListener to detach
*/
void removeConnectionListener(ConnectionListener connectionListener);
/**
* Flush the connection's buffer of outgoing messages, including sending a
* protocol message to and from the server. Passing null is equivalent to
* passing 0, which will wait forever.
* If called while the connection is closed, this method will immediately
* throw a TimeoutException, regardless of the timeout.
* If called while the connection is disconnected due to network issues this
* method will wait for up to the timeout for a reconnect or close.
*
* @param timeout The time to wait for the flush to succeed, pass 0 to wait
* forever.
* @throws TimeoutException if the timeout is exceeded
* @throws InterruptedException if the underlying thread is interrupted
*/
void flush(Duration timeout) throws TimeoutException, InterruptedException;
/**
* Drain tells the connection to process in flight messages before closing.
* Drain initially drains all the consumers, stopping incoming messages.
* Next, publishing is halted and a flush call is used to insure all published
* messages have reached the server.
* Finally, the connection is closed.
* In order to drain subscribers, an unsub protocol message is sent to the server followed by a flush.
* These two steps occur before drain returns. The remaining steps occur in a background thread.
* This method tries to manage the timeout properly, so that if the timeout is 1 second, and the flush
* takes 100ms, the remaining steps have 900ms in the background thread.
* The connection will try to let all messages be drained, but when the timeout is reached
* the connection is closed and any outstanding dispatcher threads are interrupted.
* A future allows this call to be treated as synchronous or asynchronous as
* needed by the application. The value of the future will be true if all the subscriptions
* were drained in the timeout, and false otherwise. The future completes after the connection
* is closed, so any connection handler notifications will happen before the future completes.
*
* @param timeout The time to wait for the drain to succeed, pass 0 to wait
* forever. Drain involves moving messages to and from the server
* so a very short timeout is not recommended. If the timeout is reached before
* the drain completes, the connection is simply closed, which can result in message
* loss.
* @return A future that can be used to check if the drain has completed
* @throws InterruptedException if the thread is interrupted
* @throws TimeoutException if the initial flush times out
*/
CompletableFuture drain(Duration timeout) throws TimeoutException, InterruptedException;
/**
* Close the connection and release all blocking calls like {@link #flush flush}
* and {@link Subscription#nextMessage(Duration) nextMessage}.
* If close() is called after {@link #drain(Duration) drain} it will wait up to the connection timeout
* to return, but it will not initiate a close. The drain takes precedence and will initiate the close.
*
* @throws InterruptedException if the thread, or one owned by the connection is interrupted during the close
*/
void close() throws InterruptedException ;
/**
* Returns the connections current status.
*
* @return the connection's status
*/
Status getStatus();
/**
* MaxPayload returns the size limit that a message payload can have. This is
* set by the server configuration and delivered to the client upon connect.
*
* @return the maximum size of a message payload
*/
long getMaxPayload();
/**
* Return the list of known server urls, including additional servers discovered
* after a connection has been established.
*
* @return this connection's list of known server URLs
*/
Collection getServers();
/**
* @return a wrapper for useful statistics about the connection
*/
Statistics getStatistics();
/**
* @return the read-only options used to create this connection
*/
Options getOptions();
/**
* @return the server information such as id, client info, etc.
*/
ServerInfo getServerInfo();
/**
* @return the url used for the current connection, or null if disconnected
*/
String getConnectedUrl();
/**
* @return the InetAddress of client as known by the NATS server, otherwise null.
*/
InetAddress getClientInetAddress();
/**
* @return the error text from the last error sent by the server to this client
*/
String getLastError();
/**
* Clear the last error from the server
*/
void clearLastError();
/**
* @return a new inbox subject, can be used for directed replies from
* subscribers. These are guaranteed to be unique, but can be shared and subscribed
* to by others.
*/
String createInbox();
/**
* Immediately flushes the underlying connection buffer if the connection is valid.
* @throws IOException the connection flush fails
*/
void flushBuffer() throws IOException;
/**
* Forces reconnect behavior. Stops the current connection including the reading and writing,
* copies already queued outgoing messages, and then begins the reconnect logic.
* Does not flush. Does not force close the connection. See {@link #forceReconnect(ForceReconnectOptions)}.
* @throws IOException the forceReconnect fails
* @throws InterruptedException the connection is not connected
*/
void forceReconnect() throws IOException, InterruptedException;
/**
* Forces reconnect behavior. Stops the current connection including the reading and writing,
* copies already queued outgoing messages, and then begins the reconnect logic.
* @param options options for how the forceReconnect works
* @throws IOException the forceReconnect fails
* @throws InterruptedException the connection is not connected
*/
void forceReconnect(ForceReconnectOptions options) throws IOException, InterruptedException;
/**
* Calculates the round trip time between this client and the server.
* @return the RTT as a duration
* @throws IOException various IO exception such as timeout or interruption
*/
Duration RTT() throws IOException;
/**
* Get a stream context for a specific stream.
* @param streamName the stream for the context
* @return a StreamContext instance.
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
* @throws JetStreamApiException the request had an error related to the data
*/
StreamContext getStreamContext(String streamName) throws IOException, JetStreamApiException;
/**
* Get a stream context for a specific stream.
* @param streamName the stream for the context
* @param options JetStream options.
* @return a StreamContext instance.
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
* @throws JetStreamApiException the request had an error related to the data
*/
StreamContext getStreamContext(String streamName, JetStreamOptions options) throws IOException, JetStreamApiException;
/**
* Get a consumer context for a specific named stream and specific named consumer.
* Verifies that the stream and consumer exist.
* @param streamName the name of the stream
* @param consumerName the name of the consumer
* @return a ConsumerContext object
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
* @throws JetStreamApiException the request had an error related to the data
*/
ConsumerContext getConsumerContext(String streamName, String consumerName) throws IOException, JetStreamApiException;
/**
* Get a consumer context for a specific named stream and specific named consumer.
* Verifies that the stream and consumer exist.
* @param streamName the name of the stream
* @param consumerName the name of the consumer
* @param options JetStream options.
* @return a ConsumerContext object
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
* @throws JetStreamApiException the request had an error related to the data
*/
ConsumerContext getConsumerContext(String streamName, String consumerName, JetStreamOptions options) throws IOException, JetStreamApiException;
/**
* Gets a context for publishing and subscribing to subjects backed by Jetstream streams
* and consumers.
* @return a JetStream instance.
* @throws IOException various IO exception such as timeout or interruption
*/
JetStream jetStream() throws IOException;
/**
* Gets a context for publishing and subscribing to subjects backed by Jetstream streams
* and consumers.
* @param options JetStream options.
* @return a JetStream instance.
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
*/
JetStream jetStream(JetStreamOptions options) throws IOException;
/**
* Gets a context for managing Jetstream streams
* and consumers.
* @return a JetStreamManagement instance.
* @throws IOException various IO exception such as timeout or interruption
*/
JetStreamManagement jetStreamManagement() throws IOException;
/**
* Gets a context for managing Jetstream streams
* and consumers.
* @param options JetStream options.
* @return a JetStreamManagement instance.
* @throws IOException covers various communication issues with the NATS
* server such as timeout or interruption
*/
JetStreamManagement jetStreamManagement(JetStreamOptions options) throws IOException;
/**
* Gets a context for working with a Key Value bucket
* @param bucketName the bucket name
* @return a KeyValue instance.
* @throws IOException various IO exception such as timeout or interruption
*/
KeyValue keyValue(String bucketName) throws IOException;
/**
* Gets a context for working with an Key Value bucket
* @param bucketName the bucket name
* @param options KeyValue options.
* @return a KeyValue instance.
* @throws IOException various IO exception such as timeout or interruption
*/
KeyValue keyValue(String bucketName, KeyValueOptions options) throws IOException;
/**
* Gets a context for managing Key Value buckets
* @return a KeyValueManagement instance.
* @throws IOException various IO exception such as timeout or interruption
*/
KeyValueManagement keyValueManagement() throws IOException;
/**
* Gets a context for managing Key Value buckets
* @param options KeyValue options.
* @return a KeyValueManagement instance.
* @throws IOException various IO exception such as timeout or interruption
*/
KeyValueManagement keyValueManagement(KeyValueOptions options) throws IOException;
/**
* Gets a context for working with an Object Store.
* @param bucketName the bucket name
* @return an ObjectStore instance.
* @throws IOException various IO exception such as timeout or interruption
*/
ObjectStore objectStore(String bucketName) throws IOException;
/**
* Gets a context for working with an Object Store.
* @param bucketName the bucket name
* @param options ObjectStore options.
* @return an ObjectStore instance.
* @throws IOException various IO exception such as timeout or interruption
*/
ObjectStore objectStore(String bucketName, ObjectStoreOptions options) throws IOException;
/**
* Gets a context for managing Object Stores
* @return an ObjectStoreManagement instance.
* @throws IOException various IO exception such as timeout or interruption
*/
ObjectStoreManagement objectStoreManagement() throws IOException;
/**
* Gets a context for managing Object Stores
* @param options ObjectStore options.
* @return a ObjectStoreManagement instance.
* @throws IOException various IO exception such as timeout or interruption
*/
ObjectStoreManagement objectStoreManagement(ObjectStoreOptions options) throws IOException;
}