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

io.aeron.CommonContext Maven / Gradle / Ivy

/*
 * Copyright 2014-2021 Real Logic Limited.
 *
 * 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 io.aeron;

import io.aeron.exceptions.ConcurrentConcludeException;
import io.aeron.exceptions.DriverTimeoutException;
import org.agrona.*;
import org.agrona.concurrent.AtomicBuffer;
import org.agrona.concurrent.UnsafeBuffer;
import org.agrona.concurrent.errors.DistinctErrorLog;
import org.agrona.concurrent.errors.ErrorConsumer;
import org.agrona.concurrent.errors.ErrorLogReader;
import org.agrona.concurrent.errors.LoggingErrorHandler;
import org.agrona.concurrent.ringbuffer.ManyToOneRingBuffer;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.PrintStream;
import java.nio.ByteBuffer;
import java.nio.MappedByteBuffer;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.function.Consumer;

import static io.aeron.Aeron.sleep;
import static io.aeron.CncFileDescriptor.cncVersionOffset;
import static java.lang.Long.getLong;
import static java.lang.System.getProperty;
import static java.util.concurrent.atomic.AtomicIntegerFieldUpdater.newUpdater;

/**
 * This class provides the Media Driver and client with common configuration for the Aeron directory.
 * 

* This class should have {@link #conclude()} called before the methods are used or at least * {@link #concludeAeronDirectory()} to avoid NPEs. *

* Properties *

    *
  • aeron.dir: Use value as directory name for Aeron buffers and status.
  • *
*/ public class CommonContext implements Cloneable { /** * Condition to specify a triple state conditional of always override to be true, always override to be false, * or infer value. */ public enum InferableBoolean { /** * Force the conditional to be false. */ FORCE_FALSE, /** * Force the conditional to be true. */ FORCE_TRUE, /** * Try to infer if true or false is most appropriate. */ INFER; /** * Parse the string looking for {@code true}, {@code false}, or {@code infer}. * * @param value to be parsed. * @return {@link InferableBoolean} which matches the string. */ public static InferableBoolean parse(final String value) { if (null == value || "infer".equals(value)) { return INFER; } return "true".equals(value) ? FORCE_TRUE : FORCE_FALSE; } } /** * Property name for driver timeout after which the driver is considered inactive. */ public static final String DRIVER_TIMEOUT_PROP_NAME = "aeron.driver.timeout"; /** * Property name for the timeout to use in debug mode. By default, this is not set and the configured * timeouts will be used. Setting this value adjusts timeouts to make debugging easier. */ public static final String DEBUG_TIMEOUT_PROP_NAME = "aeron.debug.timeout"; /** * Default timeout in which the driver is expected to respond or heartbeat. */ public static final long DEFAULT_DRIVER_TIMEOUT_MS = 10_000; /** * Timeout in which the driver is expected to respond or heartbeat. */ public static final long DRIVER_TIMEOUT_MS = getLong(DRIVER_TIMEOUT_PROP_NAME, DEFAULT_DRIVER_TIMEOUT_MS); /** * Value to represent a sessionId that is not to be used. */ public static final int NULL_SESSION_ID = Aeron.NULL_VALUE; /** * The top level Aeron directory used for communication between a Media Driver and client. */ public static final String AERON_DIR_PROP_NAME = "aeron.dir"; /** * The value of the top level Aeron directory unless overridden by {@link #aeronDirectoryName(String)} */ public static final String AERON_DIR_PROP_DEFAULT; /** * Media type used for IPC shared memory from {@link Publication} to {@link Subscription} channels. */ public static final String IPC_MEDIA = "ipc"; /** * Media type used for UDP sockets from {@link Publication} to {@link Subscription} channels. */ public static final String UDP_MEDIA = "udp"; /** * URI base used for IPC channels for {@link Publication}s and {@link Subscription}s */ public static final String IPC_CHANNEL = "aeron:ipc"; /** * URI base used for UDP channels for {@link Publication}s and {@link Subscription}s */ public static final String UDP_CHANNEL = "aeron:udp"; /** * URI used for Spy {@link Subscription}s whereby an outgoing unicast or multicast publication can be spied on * by IPC without receiving it again via the network. */ public static final String SPY_PREFIX = "aeron-spy:"; /** * The address and port used for a UDP channel. For the publisher it is the socket to send to, * for the subscriber it is the socket to receive from. */ public static final String ENDPOINT_PARAM_NAME = "endpoint"; /** * The network interface via which the socket will be routed. */ public static final String INTERFACE_PARAM_NAME = "interface"; /** * Initial term id to be used when creating an {@link ExclusivePublication}. */ public static final String INITIAL_TERM_ID_PARAM_NAME = "init-term-id"; /** * Current term id to be used when creating an {@link ExclusivePublication}. */ public static final String TERM_ID_PARAM_NAME = "term-id"; /** * Current term offset to be used when creating an {@link ExclusivePublication}. */ public static final String TERM_OFFSET_PARAM_NAME = "term-offset"; /** * The param name to be used for the term length as a channel URI param. */ public static final String TERM_LENGTH_PARAM_NAME = "term-length"; /** * MTU length parameter name for using as a channel URI param. If this is greater than the network MTU for UDP * then the packet will be fragmented and can amplify the impact of loss. */ public static final String MTU_LENGTH_PARAM_NAME = "mtu"; /** * Time To Live param for a multicast datagram. */ public static final String TTL_PARAM_NAME = "ttl"; /** * The param for the control channel IP address and port for multi-destination-cast semantics. */ public static final String MDC_CONTROL_PARAM_NAME = "control"; /** * Key for the mode of control that such be used for multi-destination-cast semantics. */ public static final String MDC_CONTROL_MODE_PARAM_NAME = "control-mode"; /** * Valid value for {@link #MDC_CONTROL_MODE_PARAM_NAME} when manual control is desired. */ public static final String MDC_CONTROL_MODE_MANUAL = "manual"; /** * Valid value for {@link #MDC_CONTROL_MODE_PARAM_NAME} when dynamic control is desired. Default value. */ public static final String MDC_CONTROL_MODE_DYNAMIC = "dynamic"; /** * Key for the session id for a publication or restricted subscription. */ public static final String SESSION_ID_PARAM_NAME = "session-id"; /** * Key for timeout a publication to linger after draining in nanoseconds. */ public static final String LINGER_PARAM_NAME = "linger"; /** * Parameter name for channel URI param to indicate if a subscribed stream must be reliable or not. * Value is boolean with true to recover loss and false to gap fill. */ public static final String RELIABLE_STREAM_PARAM_NAME = "reliable"; /** * Key for the tags for a channel */ public static final String TAGS_PARAM_NAME = "tags"; /** * Qualifier for a value which is a tag for reference. This prefix is use in the param value. */ public static final String TAG_PREFIX = "tag:"; /** * Parameter name for channel URI param to indicate if term buffers should be sparse. Value is boolean. */ public static final String SPARSE_PARAM_NAME = "sparse"; /** * Parameter name for channel URI param to indicate an alias for the given URI. Value not interpreted by Aeron. *

* This is a reserved application level param used to identify a particular channel for application purposes. */ public static final String ALIAS_PARAM_NAME = "alias"; /** * Parameter name for channel URI param to indicate if End of Stream (EOS) should be sent or not. Value is boolean. */ public static final String EOS_PARAM_NAME = "eos"; /** * Parameter name for channel URI param to indicate if a subscription should tether for local flow control. * Value is boolean. A tether only applies when there is more than one matching active subscription. If tether is * true then that subscription is included in flow control. If only one subscription then it tethers pace. */ public static final String TETHER_PARAM_NAME = "tether"; /** * Parameter name for channel URI param to indicate if a Subscription represents a group member or individual * from the perspective of message reception. This can inform loss handling and similar semantics. *

* When configuring a subscription for an MDC publication then should be added as this is effective multicast. * * @see CommonContext#MDC_CONTROL_MODE_PARAM_NAME * @see CommonContext#MDC_CONTROL_PARAM_NAME */ public static final String GROUP_PARAM_NAME = "group"; /** * Parameter name for Subscription URI param to indicate if Images that go unavailable should be allowed to * rejoin after a short cooldown or not. */ public static final String REJOIN_PARAM_NAME = "rejoin"; /** * Parameter name for Subscription URI param to indicate the congestion control algorithm to be used. * Options include {@code static} and {@code cubic}. */ public static final String CONGESTION_CONTROL_PARAM_NAME = "cc"; /** * Parameter name for Publication URI param to indicate the flow control strategy to be used. * Options include {@code min}, {@code max}, and {@code pref}. */ public static final String FLOW_CONTROL_PARAM_NAME = "fc"; /** * Parameter name for Subscription URI param to indicate the receiver tag to be sent in SMs. */ public static final String GROUP_TAG_PARAM_NAME = "gtag"; /** * Parameter name for Publication URI param to indicate whether spy subscriptions should simulate a connection. */ public static final String SPIES_SIMULATE_CONNECTION_PARAM_NAME = "ssc"; /** * Parameter name for the underlying OS socket send buffer length. */ public static final String SOCKET_SNDBUF_PARAM_NAME = "so-sndbuf"; /** * Parameter name for the underlying OS socket receive buffer length. */ public static final String SOCKET_RCVBUF_PARAM_NAME = "so-rcvbuf"; /** * Parameter name for the congestion control's initial receiver window length. */ public static final String RECEIVER_WINDOW_LENGTH_PARAM_NAME = "rcv-wnd"; /** * Parameter name of the offset for the media receive timestamp to be inserted into the incoming message on a * subscription. The special value of 'reserved' can be used to insert into the reserved value field. Media * receive timestamp is taken as the earliest possible point after the packet is received from the network. This * is only supported in the C media driver, the Java Media Driver will generate an error if used. */ public static final String MEDIA_RCV_TIMESTAMP_OFFSET_PARAM_NAME = "media-rcv-ts-offset"; /** * Parameter name of the offset for the channel receive timestamp to be inserted into the incoming message on a * subscription. The special value of 'reserved' can be used to insert into the reserved value field. Channel * receive timestamp is taken as soon a possible after the packet is received by Aeron receiver from the transport * bindings. */ public static final String CHANNEL_RECEIVE_TIMESTAMP_OFFSET_PARAM_NAME = "channel-rcv-ts-offset"; /** * Parameter name of the offset for the channel send timestamp to be inserted into the outgoing message * on a publication. The special value of 'reserved' can be used to insert into the reserved value * field. Channel send timestamp is taken shortly before passing the message over to the configured transport * bindings. */ public static final String CHANNEL_SEND_TIMESTAMP_OFFSET_PARAM_NAME = "channel-snd-ts-offset"; /** * Placeholder value to use in URIs to specify that a timestamp should be stored in the reserved value field. */ public static final String RESERVED_OFFSET = "reserved"; /** * Using an integer because there is no support for boolean. 1 is concluded, 0 is not concluded. */ private static final AtomicIntegerFieldUpdater IS_CONCLUDED_UPDATER = newUpdater( CommonContext.class, "isConcluded"); private static final Map DEBUG_FIELDS_SEEN = new ConcurrentHashMap<>(); private volatile int isConcluded; private long driverTimeoutMs = DRIVER_TIMEOUT_MS; private String aeronDirectoryName = getAeronDirectoryName(); private File aeronDirectory; private File cncFile; private UnsafeBuffer countersMetaDataBuffer; private UnsafeBuffer countersValuesBuffer; static { String baseDirName = null; if (SystemUtil.isLinux()) { final File devShmDir = new File("/dev/shm"); if (devShmDir.exists()) { baseDirName = "/dev/shm/aeron"; } } if (null == baseDirName) { baseDirName = SystemUtil.tmpDirName() + "aeron"; } AERON_DIR_PROP_DEFAULT = baseDirName + '-' + System.getProperty("user.name", "default"); } /** * Perform a shallow copy of the object. * * @return a shallow copy of the object. */ public CommonContext clone() { try { return (CommonContext)super.clone(); } catch (final CloneNotSupportedException ex) { throw new RuntimeException(ex); } } /** * Get the default directory name to be used if {@link #aeronDirectoryName(String)} is not set. This will take * the {@link #AERON_DIR_PROP_NAME} if set and if not then {@link #AERON_DIR_PROP_DEFAULT}. * * @return the default directory name to be used if {@link #aeronDirectoryName(String)} is not set. */ public static String getAeronDirectoryName() { return getProperty(AERON_DIR_PROP_NAME, AERON_DIR_PROP_DEFAULT); } /** * Convert the default Aeron directory name to be a random name for use with embedded drivers. * * @return random directory name with default directory name as base */ public static String generateRandomDirName() { return AERON_DIR_PROP_DEFAULT + "-" + UUID.randomUUID(); } /** * This completes initialization of the CommonContext object. It is automatically called by subclasses. * * @return this Object for method chaining. */ public CommonContext conclude() { if (0 != IS_CONCLUDED_UPDATER.getAndSet(this, 1)) { throw new ConcurrentConcludeException(); } concludeAeronDirectory(); cncFile = new File(aeronDirectory, CncFileDescriptor.CNC_FILE); return this; } /** * Has the context had the {@link #conclude()} method called. * * @return true of the {@link #conclude()} method has been called. */ public boolean isConcluded() { return 1 == isConcluded; } /** * Conclude the {@link #aeronDirectory()} so it does not need to keep being recreated. * * @return this for a fluent API. */ public CommonContext concludeAeronDirectory() { if (null == aeronDirectory) { aeronDirectory = new File(aeronDirectoryName); } return this; } /** * Get the top level Aeron directory used for communication between the client and Media Driver, and * the location of the data buffers. * * @return The top level Aeron directory. */ public String aeronDirectoryName() { return aeronDirectoryName; } /** * Get the directory in which the aeron config files are stored. *

* This is valid after a call to {@link #conclude()} or {@link #concludeAeronDirectory()}. * * @return the directory in which the aeron config files are stored. * @see #aeronDirectoryName() */ public File aeronDirectory() { return aeronDirectory; } /** * Set the top level Aeron directory used for communication between the client and Media Driver, and the location * of the data buffers. * * @param dirName New top level Aeron directory. * @return this for a fluent API. */ public CommonContext aeronDirectoryName(final String dirName) { this.aeronDirectoryName = dirName; return this; } /** * Create a new command and control file in the administration directory. * * @return The newly created File. */ public static File newDefaultCncFile() { return new File(getProperty(AERON_DIR_PROP_NAME, AERON_DIR_PROP_DEFAULT), CncFileDescriptor.CNC_FILE); } /** * Create a new command and control file in the administration directory. * * @param aeronDirectoryName name of the aeronDirectory that containing the cnc file. * @return The newly created File. */ public static File newCncFile(final String aeronDirectoryName) { return new File(aeronDirectoryName, CncFileDescriptor.CNC_FILE); } /** * Get the buffer containing the counter metadata. These counters are R/W for the driver, read only for all * other users. * * @return The buffer storing the counter metadata. */ public UnsafeBuffer countersMetaDataBuffer() { return countersMetaDataBuffer; } /** * Set the buffer containing the counter metadata. Testing/internal purposes only. * * @param countersMetaDataBuffer The new counter metadata buffer. * @return this for a fluent API. */ public CommonContext countersMetaDataBuffer(final UnsafeBuffer countersMetaDataBuffer) { this.countersMetaDataBuffer = countersMetaDataBuffer; return this; } /** * Get the buffer containing the counters. These counters are R/W for the driver, read only for all other users. * * @return The buffer storing the counters. */ public UnsafeBuffer countersValuesBuffer() { return countersValuesBuffer; } /** * Set the buffer containing the counters. Testing/internal purposes only. * * @param countersValuesBuffer The new counters buffer. * @return this for a fluent API. */ public CommonContext countersValuesBuffer(final UnsafeBuffer countersValuesBuffer) { this.countersValuesBuffer = countersValuesBuffer; return this; } /** * Get the command and control file. * * @return The command and control file. */ public File cncFile() { return cncFile; } /** * Set the driver timeout in milliseconds * * @param driverTimeoutMs to indicate liveness of driver * @return this for a fluent API. */ public CommonContext driverTimeoutMs(final long driverTimeoutMs) { this.driverTimeoutMs = driverTimeoutMs; return this; } /** * Get the driver timeout in milliseconds. * * @return driver timeout in milliseconds. */ public long driverTimeoutMs() { return checkDebugTimeout(driverTimeoutMs, TimeUnit.MILLISECONDS); } /** * Override the supplied timeout with the debug value if it has been set, and we are in debug mode. * * @param timeout The timeout value currently in use. * @param timeUnit The units of the timeout value. Debug timeout is specified in ns, so will be converted to this * unit. * @return The debug timeout if specified, and we are being debugged or the supplied value if not. Will be in * timeUnit units. */ public static long checkDebugTimeout(final long timeout, final TimeUnit timeUnit) { final String debugTimeoutString = getProperty(DEBUG_TIMEOUT_PROP_NAME); if (null == debugTimeoutString || !SystemUtil.isDebuggerAttached()) { return timeout; } try { final long debugTimeoutNs = SystemUtil.parseDuration(DEBUG_TIMEOUT_PROP_NAME, debugTimeoutString); final long debugTimeout = timeUnit.convert(debugTimeoutNs, TimeUnit.NANOSECONDS); final StackTraceElement[] stackTrace = Thread.currentThread().getStackTrace(); final String methodName = stackTrace[2].getMethodName(); final String className = stackTrace[2].getClassName(); final String debugFieldName = className + "." + methodName; if (null == DEBUG_FIELDS_SEEN.putIfAbsent(debugFieldName, true)) { final String message = "Using debug timeout [" + debugTimeout + "] for " + debugFieldName + " replacing [" + timeout + "]"; System.out.println(message); } return debugTimeout; } catch (final NumberFormatException ignore) { return timeout; } } /** * Delete the current Aeron directory, throwing errors if not possible. */ public void deleteAeronDirectory() { IoUtil.delete(aeronDirectory, false); } /** * Map the CnC file if it exists. * * @param logger for feedback * @return a new mapping for the file if it exists otherwise null; */ public MappedByteBuffer mapExistingCncFile(final Consumer logger) { final File cncFile = new File(aeronDirectory, CncFileDescriptor.CNC_FILE); if (cncFile.exists() && cncFile.length() > 0) { if (null != logger) { logger.accept("INFO: Aeron CnC file exists: " + cncFile); } return IoUtil.mapExistingFile(cncFile, CncFileDescriptor.CNC_FILE); } return null; } /** * Is a media driver active in the given directory? * * @param directory to check * @param driverTimeoutMs for the driver liveness check. * @param logger for feedback as liveness checked. * @return true if a driver is active or false if not. */ public static boolean isDriverActive( final File directory, final long driverTimeoutMs, final Consumer logger) { final File cncFile = new File(directory, CncFileDescriptor.CNC_FILE); if (cncFile.exists() && cncFile.length() > 0) { logger.accept("INFO: Aeron CnC file exists: " + cncFile); final MappedByteBuffer cncByteBuffer = IoUtil.mapExistingFile(cncFile, "CnC file"); try { return isDriverActive(driverTimeoutMs, logger, cncByteBuffer); } finally { BufferUtil.free(cncByteBuffer); } } return false; } /** * Is a media driver active in the current Aeron directory? * * @param driverTimeoutMs for the driver liveness check. * @param logger for feedback as liveness checked. * @return true if a driver is active or false if not. */ public boolean isDriverActive(final long driverTimeoutMs, final Consumer logger) { final MappedByteBuffer cncByteBuffer = mapExistingCncFile(logger); try { return isDriverActive(driverTimeoutMs, logger, cncByteBuffer); } finally { BufferUtil.free(cncByteBuffer); } } /** * Is a media driver active in the current mapped CnC buffer? If the driver is starting then it will wait for * up to the driverTimeoutMs by checking for the cncVersion being set. * * @param driverTimeoutMs for the driver liveness check. * @param logger for feedback as liveness checked. * @param cncByteBuffer for the existing CnC file. * @return true if a driver is active or false if not. */ public static boolean isDriverActive( final long driverTimeoutMs, final Consumer logger, final ByteBuffer cncByteBuffer) { if (null == cncByteBuffer) { return false; } final UnsafeBuffer cncMetaDataBuffer = CncFileDescriptor.createMetaDataBuffer(cncByteBuffer); final long startTimeMs = System.currentTimeMillis(); int cncVersion; while (0 == (cncVersion = cncMetaDataBuffer.getIntVolatile(CncFileDescriptor.cncVersionOffset(0)))) { if (System.currentTimeMillis() > (startTimeMs + driverTimeoutMs)) { throw new DriverTimeoutException("CnC file is created but not initialised."); } sleep(1); } CncFileDescriptor.checkVersion(cncVersion); final ManyToOneRingBuffer toDriverBuffer = new ManyToOneRingBuffer( CncFileDescriptor.createToDriverBuffer(cncByteBuffer, cncMetaDataBuffer)); final long timestampMs = toDriverBuffer.consumerHeartbeatTime(); final long nowMs = System.currentTimeMillis(); final long timestampAgeMs = nowMs - timestampMs; logger.accept("INFO: Aeron toDriver consumer heartbeat is (ms): " + timestampAgeMs); return timestampAgeMs <= driverTimeoutMs; } /** * Request a driver to run its termination hook. * * @param directory for the driver. * @param tokenBuffer containing the optional token for the request. * @param tokenOffset within the tokenBuffer at which the token begins. * @param tokenLength of the token in the tokenBuffer. * @return true if request was sent or false if request could not be sent. */ public static boolean requestDriverTermination( final File directory, final DirectBuffer tokenBuffer, final int tokenOffset, final int tokenLength) { final File cncFile = new File(directory, CncFileDescriptor.CNC_FILE); if (cncFile.exists() && cncFile.length() > 0) { final MappedByteBuffer cncByteBuffer = IoUtil.mapExistingFile(cncFile, "CnC file"); try { final UnsafeBuffer cncMetaDataBuffer = CncFileDescriptor.createMetaDataBuffer(cncByteBuffer); final int cncVersion = cncMetaDataBuffer.getIntVolatile(cncVersionOffset(0)); CncFileDescriptor.checkVersion(cncVersion); final ManyToOneRingBuffer toDriverBuffer = new ManyToOneRingBuffer( CncFileDescriptor.createToDriverBuffer(cncByteBuffer, cncMetaDataBuffer)); final long clientId = toDriverBuffer.nextCorrelationId(); final DriverProxy driverProxy = new DriverProxy(toDriverBuffer, clientId); return driverProxy.terminateDriver(tokenBuffer, tokenOffset, tokenLength); } finally { BufferUtil.free(cncByteBuffer); } } return false; } /** * Read the error log to a given {@link PrintStream} * * @param out to write the error log contents to. * @return the number of observations from the error log. */ public int saveErrorLog(final PrintStream out) { final MappedByteBuffer cncByteBuffer = mapExistingCncFile(null); try { return saveErrorLog(out, cncByteBuffer); } finally { BufferUtil.free(cncByteBuffer); } } /** * Read the error log to a given {@link PrintStream} * * @param out to write the error log contents to. * @param cncByteBuffer containing the error log. * @return the number of observations from the error log. */ public int saveErrorLog(final PrintStream out, final ByteBuffer cncByteBuffer) { if (null == cncByteBuffer) { return 0; } return printErrorLog(errorLogBuffer(cncByteBuffer), out); } /** * Release resources used by the CommonContext. */ public void close() { } /** * Print the contents of an error log to a {@link PrintStream} in human-readable format. * * @param errorBuffer to read errors from. * @param out print the errors to. * @return number of distinct errors observed. */ public static int printErrorLog(final AtomicBuffer errorBuffer, final PrintStream out) { int distinctErrorCount = 0; if (errorBuffer.capacity() > 0 && ErrorLogReader.hasErrors(errorBuffer)) { final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSSZ"); final ErrorConsumer errorConsumer = (count, firstTimestamp, lastTimestamp, ex) -> out.format( "***%n%d observations from %s to %s for:%n %s%n", count, dateFormat.format(new Date(firstTimestamp)), dateFormat.format(new Date(lastTimestamp)), ex); distinctErrorCount = ErrorLogReader.read(errorBuffer, errorConsumer); out.format("%n%d distinct errors observed.%n", distinctErrorCount); } return distinctErrorCount; } /** * Save the existing errors from a {@link MarkFile} to a file in the same directory as the original {@link MarkFile} * and optionally print location of such file to the supplied {@link PrintStream}. * * @param markFile which contains the error buffer. * @param errorBuffer which wraps the error log. * @param logger to which the existing errors will be printed. * @param errorFilePrefix to add to the generated error file. */ public static void saveExistingErrors( final File markFile, final AtomicBuffer errorBuffer, final PrintStream logger, final String errorFilePrefix) { try { final ByteArrayOutputStream baos = new ByteArrayOutputStream(); final int observations = printErrorLog(errorBuffer, new PrintStream(baos, false, "US-ASCII")); if (observations > 0) { final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss-SSSZ"); final File errorLogFile = new File( markFile.getParentFile(), errorFilePrefix + '-' + dateFormat.format(new Date()) + "-error.log"); if (null != logger) { logger.println("WARNING: existing errors saved to: " + errorLogFile); } try (FileOutputStream out = new FileOutputStream(errorLogFile)) { baos.writeTo(out); } } } catch (final Exception ex) { LangUtil.rethrowUnchecked(ex); } } /** * Get an {@link AtomicBuffer} which wraps the error log in the CnC file. * * @param cncByteBuffer which contains the error log. * @return an {@link AtomicBuffer} which wraps the error log in the CnC file. */ public static AtomicBuffer errorLogBuffer(final ByteBuffer cncByteBuffer) { final DirectBuffer cncMetaDataBuffer = CncFileDescriptor.createMetaDataBuffer(cncByteBuffer); final int cncVersion = cncMetaDataBuffer.getInt(cncVersionOffset(0)); CncFileDescriptor.checkVersion(cncVersion); return CncFileDescriptor.createErrorLogBuffer(cncByteBuffer, cncMetaDataBuffer); } /** * Wrap a user ErrorHandler so that error will continue to write to the errorLog. * * @param userErrorHandler the user specified ErrorHandler, can be null. * @param errorLog the configured errorLog, either the default or user supplied. * @return an error handler that will delegate to both the userErrorHandler and the errorLog. */ public static ErrorHandler setupErrorHandler(final ErrorHandler userErrorHandler, final DistinctErrorLog errorLog) { final LoggingErrorHandler loggingErrorHandler = new LoggingErrorHandler(errorLog); if (null == userErrorHandler) { return loggingErrorHandler; } else { return (throwable) -> { loggingErrorHandler.onError(throwable); userErrorHandler.onError(throwable); }; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy