io.aeron.Aeron Maven / Gradle / Ivy
Show all versions of aeron-client Show documentation
/*
* Copyright 2014-2019 Real Logic Ltd.
*
* 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.aeron;
import io.aeron.exceptions.AeronException;
import io.aeron.exceptions.DriverTimeoutException;
import io.aeron.logbuffer.FragmentHandler;
import org.agrona.DirectBuffer;
import org.agrona.ErrorHandler;
import org.agrona.IoUtil;
import org.agrona.concurrent.*;
import org.agrona.concurrent.broadcast.BroadcastReceiver;
import org.agrona.concurrent.broadcast.CopyBroadcastReceiver;
import org.agrona.concurrent.ringbuffer.ManyToOneRingBuffer;
import org.agrona.concurrent.ringbuffer.RingBuffer;
import org.agrona.concurrent.status.CountersReader;
import java.io.File;
import java.io.IOException;
import java.io.PrintStream;
import java.nio.MappedByteBuffer;
import java.nio.channels.FileChannel;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import static java.nio.channels.FileChannel.MapMode.READ_WRITE;
import static java.nio.file.StandardOpenOption.READ;
import static java.nio.file.StandardOpenOption.WRITE;
import static java.util.concurrent.atomic.AtomicIntegerFieldUpdater.newUpdater;
import static org.agrona.BitUtil.SIZE_OF_INT;
import static org.agrona.SystemUtil.getDurationInNanos;
/**
* Aeron entry point for communicating to the Media Driver for creating {@link Publication}s and {@link Subscription}s.
* Use an {@link Aeron.Context} to configure the Aeron object.
*
* A client application requires only one Aeron object per Media Driver.
*
* Note: If {@link Aeron.Context#errorHandler(ErrorHandler)} is not set and a {@link DriverTimeoutException}
* occurs then the process will face the wrath of {@link System#exit(int)}.
* See {@link Aeron.Configuration#DEFAULT_ERROR_HANDLER}.
*/
public class Aeron implements AutoCloseable
{
/**
* Used to represent a null value for when some value is not yet set.
*/
public static final int NULL_VALUE = -1;
/**
* Using an integer because there is no support for boolean. 1 is closed, 0 is not closed.
*/
private static final AtomicIntegerFieldUpdater IS_CLOSED_UPDATER = newUpdater(Aeron.class, "isClosed");
@SuppressWarnings("unused") private volatile int isClosed;
private final long clientId;
private final ClientConductor conductor;
private final RingBuffer commandBuffer;
private final AgentInvoker conductorInvoker;
private final AgentRunner conductorRunner;
private final Context ctx;
Aeron(final Context ctx)
{
ctx.conclude();
this.ctx = ctx;
clientId = ctx.clientId();
commandBuffer = ctx.toDriverBuffer();
conductor = new ClientConductor(ctx);
if (ctx.useConductorAgentInvoker())
{
conductorInvoker = new AgentInvoker(ctx.errorHandler(), null, conductor);
conductorRunner = null;
}
else
{
conductorInvoker = null;
conductorRunner = new AgentRunner(ctx.idleStrategy(), ctx.errorHandler(), null, conductor);
}
}
/**
* Create an Aeron instance and connect to the media driver with a default {@link Context}.
*
* Threads required for interacting with the media driver are created and managed within the Aeron instance.
*
* @return the new {@link Aeron} instance connected to the Media Driver.
*/
public static Aeron connect()
{
return connect(new Context());
}
/**
* Create an Aeron instance and connect to the media driver.
*
* Threads required for interacting with the media driver are created and managed within the Aeron instance.
*
* If an exception occurs while trying to establish a connection then the {@link Context#close()} method
* will be called on the passed context.
*
* @param ctx for configuration of the client.
* @return the new {@link Aeron} instance connected to the Media Driver.
*/
public static Aeron connect(final Context ctx)
{
try
{
final Aeron aeron = new Aeron(ctx);
if (ctx.useConductorAgentInvoker())
{
aeron.conductorInvoker.start();
}
else
{
AgentRunner.startOnThread(aeron.conductorRunner, ctx.threadFactory);
}
return aeron;
}
catch (final Exception ex)
{
ctx.close();
throw ex;
}
}
/**
* Print out the values from {@link #countersReader()} which can be useful for debugging.
*
* @param out to where the counters get printed.
*/
public void printCounters(final PrintStream out)
{
final CountersReader counters = countersReader();
counters.forEach((value, id, label) -> out.format("%3d: %,20d - %s%n", id, value, label));
}
/**
* Has the client been closed? If not then the CnC file may not be unmapped.
*
* @return true if the client has been explicitly closed otherwise false.
*/
public boolean isClosed()
{
return 1 == isClosed;
}
/**
* Get the {@link Aeron.Context} that is used by this client.
*
* @return the {@link Aeron.Context} that is use by this client.
*/
public Context context()
{
return ctx;
}
/**
* Get the client identity that has been allocated for communicating with the media driver.
*
* @return the client identity that has been allocated for communicating with the media driver.
*/
public long clientId()
{
return clientId;
}
/**
* Get the {@link AgentInvoker} for the client conductor.
*
* @return the {@link AgentInvoker} for the client conductor.
*/
public AgentInvoker conductorAgentInvoker()
{
return conductorInvoker;
}
/**
* Clean up and release all Aeron client resources and shutdown conductor thread if not using
* {@link Context#useConductorAgentInvoker(boolean)}.
*
* This will close all currently open {@link Publication}s, {@link Subscription}s, and {@link Counter}s created
* from this client.
*/
public void close()
{
if (IS_CLOSED_UPDATER.compareAndSet(this, 0, 1))
{
if (null != conductorRunner)
{
conductorRunner.close();
}
else
{
conductorInvoker.close();
}
}
}
/**
* Add a {@link Publication} for publishing messages to subscribers. The publication returned is threadsafe.
*
* @param channel for sending the messages known to the media layer.
* @param streamId within the channel scope.
* @return a new {@link ConcurrentPublication}.
*/
public ConcurrentPublication addPublication(final String channel, final int streamId)
{
return conductor.addPublication(channel, streamId);
}
/**
* Add an {@link ExclusivePublication} for publishing messages to subscribers from a single thread.
*
* @param channel for sending the messages known to the media layer.
* @param streamId within the channel scope.
* @return a new {@link ExclusivePublication}.
*/
public ExclusivePublication addExclusivePublication(final String channel, final int streamId)
{
return conductor.addExclusivePublication(channel, streamId);
}
/**
* Add a new {@link Subscription} for subscribing to messages from publishers.
*
* The method will set up the {@link Subscription} to use the
* {@link Aeron.Context#availableImageHandler(AvailableImageHandler)} and
* {@link Aeron.Context#unavailableImageHandler(UnavailableImageHandler)} from the {@link Aeron.Context}.
*
* @param channel for receiving the messages known to the media layer.
* @param streamId within the channel scope.
* @return the {@link Subscription} for the channel and streamId pair.
*/
public Subscription addSubscription(final String channel, final int streamId)
{
return conductor.addSubscription(channel, streamId);
}
/**
* Add a new {@link Subscription} for subscribing to messages from publishers.
*
* This method will override the default handlers from the {@link Aeron.Context}, i.e.
* {@link Aeron.Context#availableImageHandler(AvailableImageHandler)} and
* {@link Aeron.Context#unavailableImageHandler(UnavailableImageHandler)}. Null values are valid and will
* result in no action being taken.
*
* @param channel for receiving the messages known to the media layer.
* @param streamId within the channel scope.
* @param availableImageHandler called when {@link Image}s become available for consumption. Null is valid if no
* action is to be taken.
* @param unavailableImageHandler called when {@link Image}s go unavailable for consumption. Null is valid if no
* action is to be taken.
* @return the {@link Subscription} for the channel and streamId pair.
*/
public Subscription addSubscription(
final String channel,
final int streamId,
final AvailableImageHandler availableImageHandler,
final UnavailableImageHandler unavailableImageHandler)
{
return conductor.addSubscription(channel, streamId, availableImageHandler, unavailableImageHandler);
}
/**
* Generate the next correlation id that is unique for the connected Media Driver.
*
* This is useful generating correlation identifiers for pairing requests with responses in a clients own
* application protocol.
*
* This method is thread safe and will work across processes that all use the same media driver.
*
* @return next correlation id that is unique for the Media Driver.
*/
public long nextCorrelationId()
{
if (1 == isClosed)
{
throw new AeronException("client is closed");
}
return commandBuffer.nextCorrelationId();
}
/**
* Get the {@link CountersReader} for the Aeron media driver counters.
*
* @return new {@link CountersReader} for the Aeron media driver in use.
*/
public CountersReader countersReader()
{
if (1 == isClosed)
{
throw new AeronException("client is closed");
}
return conductor.countersReader();
}
/**
* Allocate a counter on the media driver and return a {@link Counter} for it.
*
* The counter should be freed by calling {@link Counter#close()}.
*
* @param typeId for the counter.
* @param keyBuffer containing the optional key for the counter.
* @param keyOffset within the keyBuffer at which the key begins.
* @param keyLength of the key in the keyBuffer.
* @param labelBuffer containing the mandatory label for the counter. The label should not be length prefixed.
* @param labelOffset within the labelBuffer at which the label begins.
* @param labelLength of the label in the labelBuffer.
* @return the newly allocated counter.
* @see org.agrona.concurrent.status.CountersManager#allocate(int, DirectBuffer, int, int, DirectBuffer, int, int)
*/
public Counter addCounter(
final int typeId,
final DirectBuffer keyBuffer,
final int keyOffset,
final int keyLength,
final DirectBuffer labelBuffer,
final int labelOffset,
final int labelLength)
{
return conductor.addCounter(typeId, keyBuffer, keyOffset, keyLength, labelBuffer, labelOffset, labelLength);
}
/**
* Allocate a counter on the media driver and return a {@link Counter} for it.
*
* The counter should be freed by calling {@link Counter#close()}.
*
* @param typeId for the counter.
* @param label for the counter. It should be US-ASCII.
* @return the newly allocated counter.
* @see org.agrona.concurrent.status.CountersManager#allocate(String, int)
*/
public Counter addCounter(final int typeId, final String label)
{
return conductor.addCounter(typeId, label);
}
/**
* Configuration options for the {@link Aeron} client.
*/
public static class Configuration
{
/**
* Duration in milliseconds for which the client conductor will sleep between duty cycles.
*/
static final long IDLE_SLEEP_MS = 16;
/**
* Duration in nanoseconds for which the client conductor will sleep between duty cycles.
*/
static final long IDLE_SLEEP_NS = TimeUnit.MILLISECONDS.toNanos(IDLE_SLEEP_MS);
/**
* Default interval between sending keepalive control messages to the driver.
*/
static final long KEEPALIVE_INTERVAL_NS = TimeUnit.MILLISECONDS.toNanos(500);
/**
* Duration to wait while lingering a entity such as an {@link Image} before deleting underlying resources
* such as memory mapped files.
*/
public static final String RESOURCE_LINGER_DURATION_PROP_NAME = "aeron.client.resource.linger.duration";
/**
* Default duration a resource should linger before deletion.
*/
public static final long RESOURCE_LINGER_DURATION_DEFAULT = TimeUnit.SECONDS.toNanos(3);
/**
* The Default handler for Aeron runtime exceptions.
* When a {@link DriverTimeoutException} is encountered, this handler will
* exit the program.
*
* The error handler can be overridden by supplying an {@link Context} with a custom handler.
*
* @see Context#errorHandler(ErrorHandler)
*/
public static final ErrorHandler DEFAULT_ERROR_HANDLER =
(throwable) ->
{
throwable.printStackTrace();
if (throwable instanceof DriverTimeoutException)
{
System.err.printf(
"%n***%n*** timeout from the MediaDriver - is it currently running? Exiting%n***%n");
System.exit(-1);
}
};
/**
* Duration to wait while lingering a entity such as an {@link Image} before deleting underlying resources
* such as memory mapped files.
*
* @return duration in nanoseconds to wait before deleting a expired resource.
* @see #RESOURCE_LINGER_DURATION_PROP_NAME
*/
public static long resourceLingerDurationNs()
{
return getDurationInNanos(RESOURCE_LINGER_DURATION_PROP_NAME, RESOURCE_LINGER_DURATION_DEFAULT);
}
}
/**
* This class provides configuration for the {@link Aeron} class via the {@link Aeron#connect(Aeron.Context)}
* method and its overloads. It gives applications some control over the interactions with the Aeron Media Driver.
* It can also set up error handling as well as application callbacks for image information from the
* Media Driver.
*
* A number of the properties are for testing and should not be set by end users.
*
* Note: Do not reuse instances of the context across different {@link Aeron} clients.
*
* The context will be owned by {@link ClientConductor} after a successful
* {@link Aeron#connect(Context)} and closed via {@link Aeron#close()}.
*/
public static class Context extends CommonContext
{
private long clientId;
private boolean useConductorAgentInvoker = false;
private AgentInvoker driverAgentInvoker;
private Lock clientLock;
private EpochClock epochClock;
private NanoClock nanoClock;
private IdleStrategy idleStrategy;
private CopyBroadcastReceiver toClientBuffer;
private RingBuffer toDriverBuffer;
private DriverProxy driverProxy;
private MappedByteBuffer cncByteBuffer;
private AtomicBuffer cncMetaDataBuffer;
private LogBuffersFactory logBuffersFactory;
private ErrorHandler errorHandler;
private AvailableImageHandler availableImageHandler;
private UnavailableImageHandler unavailableImageHandler;
private AvailableCounterHandler availableCounterHandler;
private UnavailableCounterHandler unavailableCounterHandler;
private long keepAliveInterval = Configuration.KEEPALIVE_INTERVAL_NS;
private long interServiceTimeout = 0;
private long resourceLingerDurationNs = Configuration.resourceLingerDurationNs();
private ThreadFactory threadFactory = Thread::new;
/**
* Perform a shallow copy of the object.
*
* @return a shallow copy of the object.
*/
public Context clone()
{
return (Context)super.clone();
}
/**
* This is called automatically by {@link Aeron#connect(Aeron.Context)} and its overloads.
* There is no need to call it from a client application. It is responsible for providing default
* values for options that are not individually changed through field setters.
*
* @return this for a fluent API.
*/
public Context conclude()
{
super.conclude();
if (null == clientLock)
{
clientLock = new ReentrantLock();
}
if (null == epochClock)
{
epochClock = new SystemEpochClock();
}
if (null == nanoClock)
{
nanoClock = new SystemNanoClock();
}
if (null == idleStrategy)
{
idleStrategy = new SleepingMillisIdleStrategy(Configuration.IDLE_SLEEP_MS);
}
if (cncFile() != null)
{
connectToDriver();
}
if (null == toDriverBuffer)
{
toDriverBuffer = new ManyToOneRingBuffer(
CncFileDescriptor.createToDriverBuffer(cncByteBuffer, cncMetaDataBuffer));
}
if (null == toClientBuffer)
{
toClientBuffer = new CopyBroadcastReceiver(new BroadcastReceiver(
CncFileDescriptor.createToClientsBuffer(cncByteBuffer, cncMetaDataBuffer)));
}
if (countersMetaDataBuffer() == null)
{
countersMetaDataBuffer(
CncFileDescriptor.createCountersMetaDataBuffer(cncByteBuffer, cncMetaDataBuffer));
}
if (countersValuesBuffer() == null)
{
countersValuesBuffer(CncFileDescriptor.createCountersValuesBuffer(cncByteBuffer, cncMetaDataBuffer));
}
interServiceTimeout = CncFileDescriptor.clientLivenessTimeout(cncMetaDataBuffer);
if (null == logBuffersFactory)
{
logBuffersFactory = new MappedLogBuffersFactory();
}
if (null == errorHandler)
{
errorHandler = Configuration.DEFAULT_ERROR_HANDLER;
}
if (null == driverProxy)
{
clientId = toDriverBuffer.nextCorrelationId();
driverProxy = new DriverProxy(toDriverBuffer, clientId);
}
return this;
}
/**
* Get the client identity that has been allocated for communicating with the media driver.
*
* @return the client identity that has been allocated for communicating with the media driver.
*/
public long clientId()
{
return clientId;
}
/**
* Should an {@link AgentInvoker} be used for running the {@link ClientConductor} rather than run it on
* a thread with a {@link AgentRunner}.
*
* @param useConductorAgentInvoker use {@link AgentInvoker} be used for running the {@link ClientConductor}?
* @return this for a fluent API.
*/
public Context useConductorAgentInvoker(final boolean useConductorAgentInvoker)
{
this.useConductorAgentInvoker = useConductorAgentInvoker;
return this;
}
/**
* Should an {@link AgentInvoker} be used for running the {@link ClientConductor} rather than run it on
* a thread with a {@link AgentRunner}.
*
* @return true if the {@link ClientConductor} will be run with an {@link AgentInvoker} otherwise false.
*/
public boolean useConductorAgentInvoker()
{
return useConductorAgentInvoker;
}
/**
* Set the {@link AgentInvoker} for the Media Driver to be used while awaiting a synchronous response.
*
* Useful for when running on a low thread count scenario.
*
* @param driverAgentInvoker to be invoked while awaiting a response in the client.
* @return this for a fluent API.
*/
public Context driverAgentInvoker(final AgentInvoker driverAgentInvoker)
{
this.driverAgentInvoker = driverAgentInvoker;
return this;
}
/**
* Get the {@link AgentInvoker} that is used to run the Media Driver while awaiting a synchronous response.
*
* @return the {@link AgentInvoker} that is used for running the Media Driver.
*/
public AgentInvoker driverAgentInvoker()
{
return driverAgentInvoker;
}
/**
* The {@link Lock} that is used to provide mutual exclusion in the Aeron client.
*
* If the {@link #useConductorAgentInvoker()} is set and only one thread accesses the client
* then the lock can be set to {@link NoOpLock} to elide the lock overhead.
*
* @param lock that is used to provide mutual exclusion in the Aeron client.
* @return this for a fluent API.
*/
public Context clientLock(final Lock lock)
{
clientLock = lock;
return this;
}
/**
* Get the {@link Lock} that is used to provide mutual exclusion in the Aeron client.
*
* @return the {@link Lock} that is used to provide mutual exclusion in the Aeron client.
*/
public Lock clientLock()
{
return clientLock;
}
/**
* Set the {@link EpochClock} to be used for tracking wall clock time when interacting with the driver.
*
* @param clock {@link EpochClock} to be used for tracking wall clock time when interacting with the driver.
* @return this for a fluent API.
*/
public Context epochClock(final EpochClock clock)
{
this.epochClock = clock;
return this;
}
/**
* Get the {@link EpochClock} used by the client for the epoch time in milliseconds.
*
* @return the {@link EpochClock} used by the client for the epoch time in milliseconds.
*/
public EpochClock epochClock()
{
return epochClock;
}
/**
* Set the {@link NanoClock} to be used for tracking high resolution time.
*
* @param clock {@link NanoClock} to be used for tracking high resolution time.
* @return this for a fluent API.
*/
public Context nanoClock(final NanoClock clock)
{
this.nanoClock = clock;
return this;
}
/**
* Get the {@link NanoClock} to be used for tracking high resolution time.
*
* @return the {@link NanoClock} to be used for tracking high resolution time.
*/
public NanoClock nanoClock()
{
return nanoClock;
}
/**
* Provides an IdleStrategy for the thread responsible for communicating with the Aeron Media Driver.
*
* @param idleStrategy Thread idle strategy for communication with the Media Driver.
* @return this for a fluent API.
*/
public Context idleStrategy(final IdleStrategy idleStrategy)
{
this.idleStrategy = idleStrategy;
return this;
}
/**
* Get the {@link IdleStrategy} employed by the client conductor thread.
*
* @return the {@link IdleStrategy} employed by the client conductor thread.
*/
public IdleStrategy idleStrategy()
{
return idleStrategy;
}
/**
* This method is used for testing and debugging.
*
* @param toClientBuffer Injected CopyBroadcastReceiver
* @return this for a fluent API.
*/
Context toClientBuffer(final CopyBroadcastReceiver toClientBuffer)
{
this.toClientBuffer = toClientBuffer;
return this;
}
/**
* The buffer used for communicating from the media driver to the Aeron client.
*
* @return the buffer used for communicating from the media driver to the Aeron client.
*/
public CopyBroadcastReceiver toClientBuffer()
{
return toClientBuffer;
}
/**
* This method is used for testing and debugging.
*
* @param toDriverBuffer Injected RingBuffer.
* @return this for a fluent API.
*/
Context toDriverBuffer(final RingBuffer toDriverBuffer)
{
this.toDriverBuffer = toDriverBuffer;
return this;
}
/**
* Get the {@link RingBuffer} used for sending commands to the media driver.
*
* @return the {@link RingBuffer} used for sending commands to the media driver.
*/
public RingBuffer toDriverBuffer()
{
return toDriverBuffer;
}
/**
* Set the proxy for communicating with the media driver.
*
* @param driverProxy for communicating with the media driver.
* @return this for a fluent API.
*/
Context driverProxy(final DriverProxy driverProxy)
{
this.driverProxy = driverProxy;
return this;
}
/**
* Get the proxy for communicating with the media driver.
*
* @return the proxy for communicating with the media driver.
*/
public DriverProxy driverProxy()
{
return driverProxy;
}
/**
* This method is used for testing and debugging.
*
* @param logBuffersFactory Injected LogBuffersFactory
* @return this for a fluent API.
*/
Context logBuffersFactory(final LogBuffersFactory logBuffersFactory)
{
this.logBuffersFactory = logBuffersFactory;
return this;
}
/**
* Get the factory for making log buffers.
*
* @return the factory for making log buffers.
*/
public LogBuffersFactory logBuffersFactory()
{
return logBuffersFactory;
}
/**
* Handle Aeron exceptions in a callback method. The default behavior is defined by
* {@link Configuration#DEFAULT_ERROR_HANDLER}. This is the error handler which will be used if an error occurs
* during the callback for poll operations such as {@link Subscription#poll(FragmentHandler, int)}.
*
* The error handler can be reset after {@link Aeron#connect()} and the latest version will always be used
* so that the boot strapping process can be performed such as replacing the default one with a
* {@link CountedErrorHandler}.
*
* @param errorHandler Method to handle objects of type Throwable.
* @return this for a fluent API.
* @see io.aeron.exceptions.DriverTimeoutException
* @see io.aeron.exceptions.RegistrationException
*/
public Context errorHandler(final ErrorHandler errorHandler)
{
this.errorHandler = errorHandler;
return this;
}
/**
* Get the error handler that will be called for errors reported back from the media driver.
*
* @return the error handler that will be called for errors reported back from the media driver.
*/
public ErrorHandler errorHandler()
{
return errorHandler;
}
/**
* Setup a default callback for when an {@link Image} is available.
*
* @param handler Callback method for handling available image notifications.
* @return this for a fluent API.
*/
public Context availableImageHandler(final AvailableImageHandler handler)
{
this.availableImageHandler = handler;
return this;
}
/**
* Get the default callback handler for notifying when {@link Image}s become available.
*
* @return the callback handler for notifying when {@link Image}s become available.
*/
public AvailableImageHandler availableImageHandler()
{
return availableImageHandler;
}
/**
* Setup a default callback for when an {@link Image} is unavailable.
*
* @param handler Callback method for handling unavailable image notifications.
* @return this for a fluent API.
*/
public Context unavailableImageHandler(final UnavailableImageHandler handler)
{
this.unavailableImageHandler = handler;
return this;
}
/**
* Get the callback handler for when an {@link Image} is unavailable.
*
* @return the callback handler for when an {@link Image} is unavailable.
*/
public UnavailableImageHandler unavailableImageHandler()
{
return unavailableImageHandler;
}
/**
* Setup a callback for when a counter is available.
*
* @param handler to be called for handling available counter notifications.
* @return this for a fluent API.
*/
public Context availableCounterHandler(final AvailableCounterHandler handler)
{
this.availableCounterHandler = handler;
return this;
}
/**
* Get the callback handler for when a counter is available.
*
* @return the callback handler for when a counter is available.
*/
public AvailableCounterHandler availableCounterHandler()
{
return availableCounterHandler;
}
/**
* Setup a callback for when a counter is unavailable.
*
* @param handler to be called for handling unavailable counter notifications.
* @return this for a fluent API.
*/
public Context unavailableCounterHandler(final UnavailableCounterHandler handler)
{
this.unavailableCounterHandler = handler;
return this;
}
/**
* Get the callback handler for when a counter is unavailable.
*
* @return the callback handler for when a counter is unavailable.
*/
public UnavailableCounterHandler unavailableCounterHandler()
{
return unavailableCounterHandler;
}
/**
* Set the interval in nanoseconds for which the client will perform keep-alive operations.
*
* @param value the interval in nanoseconds for which the client will perform keep-alive operations.
* @return this for a fluent API.
*/
public Context keepAliveInterval(final long value)
{
keepAliveInterval = value;
return this;
}
/**
* Get the interval in nanoseconds for which the client will perform keep-alive operations.
*
* @return the interval in nanoseconds for which the client will perform keep-alive operations.
*/
public long keepAliveInterval()
{
return keepAliveInterval;
}
/**
* Set the amount of time, in milliseconds, that this client will wait until it determines the
* Media Driver is unavailable. When this happens a
* {@link io.aeron.exceptions.DriverTimeoutException} will be generated for the error handler.
*
* @param value Number of milliseconds.
* @return this for a fluent API.
* @see #errorHandler(ErrorHandler)
*/
public Context driverTimeoutMs(final long value)
{
super.driverTimeoutMs(value);
return this;
}
/**
* Set the timeout between service calls the to {@link ClientConductor} duty cycles in nanoseconds.
*
* @param interServiceTimeout the timeout (ns) between service calls the to {@link ClientConductor} duty cycle.
* @return this for a fluent API.
*/
Context interServiceTimeout(final long interServiceTimeout)
{
this.interServiceTimeout = interServiceTimeout;
return this;
}
/**
* Return the timeout between service calls to the duty cycle for the client.
*
* When exceeded, {@link #errorHandler} will be called and the active {@link Publication}s and {@link Image}s
* closed.
*
* This value is controlled by the driver and included in the CnC file. It can be configured by adjusting
* the aeron.client.liveness.timeout property on the media driver.
*
* @return the timeout in nanoseconds between service calls.
*/
public long interServiceTimeout()
{
return interServiceTimeout;
}
/**
* Duration to wait while lingering a entity such as an {@link Image} before deleting underlying resources
* such as memory mapped files.
*
* @param resourceLingerDurationNs to wait before deleting a expired resource.
* @return this for a fluent API.
* @see Configuration#RESOURCE_LINGER_DURATION_PROP_NAME
*/
public Context resourceLingerDurationNs(final long resourceLingerDurationNs)
{
this.resourceLingerDurationNs = resourceLingerDurationNs;
return this;
}
/**
* Duration to wait while lingering a entity such as an {@link Image} before deleting underlying resources
* such as memory mapped files.
*
* @return duration in nanoseconds to wait before deleting a expired resource.
* @see Configuration#RESOURCE_LINGER_DURATION_PROP_NAME
*/
public long resourceLingerDurationNs()
{
return resourceLingerDurationNs;
}
/**
* @see CommonContext#aeronDirectoryName(String)
*/
public Context aeronDirectoryName(final String dirName)
{
super.aeronDirectoryName(dirName);
return this;
}
/**
* Specify the thread factory to use when starting the conductor thread.
*
* @param threadFactory thread factory to construct the thread.
* @return this for a fluent API.
*/
public Context threadFactory(final ThreadFactory threadFactory)
{
this.threadFactory = threadFactory;
return this;
}
/**
* The thread factory to be use to construct the conductor thread
*
* @return the specified thread factory or {@link Thread#Thread(Runnable)} if none is provided
*/
public ThreadFactory threadFactory()
{
return threadFactory;
}
/**
* Clean up all resources that the client uses to communicate with the Media Driver.
*/
public void close()
{
final MappedByteBuffer cncByteBuffer = this.cncByteBuffer;
this.cncByteBuffer = null;
IoUtil.unmap(cncByteBuffer);
super.close();
}
private void connectToDriver()
{
final long startTimeMs = epochClock.time();
final long deadlineMs = startTimeMs + driverTimeoutMs();
final File cncFile = cncFile();
while (null == toDriverBuffer)
{
while (!cncFile.exists() || cncFile.length() <= 0)
{
if (epochClock.time() > deadlineMs)
{
throw new DriverTimeoutException("CnC file not created: " + cncFile.getAbsolutePath());
}
sleep(Configuration.IDLE_SLEEP_MS);
}
cncByteBuffer = waitForFileMapping(cncFile, deadlineMs, epochClock);
cncMetaDataBuffer = CncFileDescriptor.createMetaDataBuffer(cncByteBuffer);
int cncVersion;
while (0 == (cncVersion = cncMetaDataBuffer.getIntVolatile(CncFileDescriptor.cncVersionOffset(0))))
{
if (epochClock.time() > deadlineMs)
{
throw new DriverTimeoutException("CnC file is created but not initialised");
}
sleep(1);
}
if (CncFileDescriptor.CNC_VERSION != cncVersion)
{
throw new AeronException("CnC file version not supported: version=" + cncVersion);
}
final ManyToOneRingBuffer ringBuffer = new ManyToOneRingBuffer(
CncFileDescriptor.createToDriverBuffer(cncByteBuffer, cncMetaDataBuffer));
while (0 == ringBuffer.consumerHeartbeatTime())
{
if (epochClock.time() > deadlineMs)
{
throw new DriverTimeoutException("no driver heartbeat detected");
}
sleep(1);
}
final long timeMs = epochClock.time();
if (ringBuffer.consumerHeartbeatTime() < (timeMs - driverTimeoutMs()))
{
if (timeMs > deadlineMs)
{
throw new DriverTimeoutException("no driver heartbeat detected");
}
IoUtil.unmap(cncByteBuffer);
cncByteBuffer = null;
cncMetaDataBuffer = null;
sleep(100);
continue;
}
toDriverBuffer = ringBuffer;
}
}
}
private static MappedByteBuffer waitForFileMapping(
final File cncFile, final long deadlineMs, final EpochClock epochClock)
{
try (FileChannel fileChannel = FileChannel.open(cncFile.toPath(), READ, WRITE))
{
while (fileChannel.size() < CncFileDescriptor.CNC_VERSION_FIELD_OFFSET + SIZE_OF_INT)
{
if (epochClock.time() > deadlineMs)
{
throw new AeronException("CnC file is created but not populated");
}
sleep(Configuration.IDLE_SLEEP_MS);
}
return fileChannel.map(READ_WRITE, 0, fileChannel.size());
}
catch (final IOException ex)
{
throw new AeronException("cannot open CnC file", ex);
}
}
static void sleep(final long durationMs)
{
try
{
Thread.sleep(durationMs);
}
catch (final InterruptedException ignore)
{
Thread.currentThread().interrupt();
}
}
}