net.openhft.chronicle.wire.AbstractGeneratedMethodReader Maven / Gradle / Ivy
/*
* Copyright 2016-2020 chronicle.software
*
* https://chronicle.software
*
* 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 net.openhft.chronicle.wire;
import net.openhft.chronicle.bytes.Bytes;
import net.openhft.chronicle.bytes.MethodReader;
import net.openhft.chronicle.bytes.MethodReaderInterceptorReturns;
import net.openhft.chronicle.core.Jvm;
import net.openhft.chronicle.core.io.ClosedIllegalStateException;
import net.openhft.chronicle.core.util.Mocker;
import net.openhft.chronicle.wire.utils.MethodReaderStatus;
import org.jetbrains.annotations.NotNull;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Consumer;
import java.util.function.Predicate;
import static java.lang.ThreadLocal.withInitial;
import static net.openhft.chronicle.core.io.Closeable.closeQuietly;
/**
* This is the AbstractGeneratedMethodReader class implementing the MethodReader interface.
* It serves as a base class for generated method readers, providing foundational functionality
* and utility methods to facilitate method reading.
*/
public abstract class AbstractGeneratedMethodReader implements MethodReader {
// A no-operation message history consumer.
private static final Consumer NO_OP_MH_CONSUMER = Jvm.uncheckedCast(Mocker.ignored(Consumer.class));
public final static ThreadLocal SERVICE_NAME = new ThreadLocal<>();
private final static ConcurrentHashMap TEMP_MESSAGE_HISTORY_BY_SERVICE_NAME = new ConcurrentHashMap<>();
protected final WireParselet debugLoggingParselet;
private final MarshallableIn in;
private final MessageHistoryThreadLocal tempMessageHistory;
protected MessageHistory messageHistory;
protected boolean dataEventProcessed;
// Flag to determine if the input should be closed.
private boolean closeIn = false;
// Flag to determine if the reader is closed.
private boolean closed;
// Consumer for processing message history.
private Consumer historyConsumer = NO_OP_MH_CONSUMER;
private Predicate predicate;
private boolean scanning;
/**
* Constructs a new AbstractGeneratedMethodReader with the provided input interface
* and a debug logging parselet.
*
* @param in The input interface for marshallable data
* @param debugLoggingParselet The parselet used for debugging and logging
*/
protected AbstractGeneratedMethodReader(MarshallableIn in,
WireParselet debugLoggingParselet) {
this.in = in;
this.debugLoggingParselet = debugLoggingParselet;
// gets the name of the service so we can offer history message caching
// the services name is set by the chronicle services framework
String serviceName = SERVICE_NAME.get();
if (serviceName == null)
serviceName = "";
// this was handled to support when multiple services are using the same thread.
this.tempMessageHistory = TEMP_MESSAGE_HISTORY_BY_SERVICE_NAME.computeIfAbsent(serviceName, x -> new MessageHistoryThreadLocal());
}
/**
* Sets a predicate to be used by the method reader.
*
* @param predicate The predicate for filtering
* @return The current instance of the AbstractGeneratedMethodReader class
*/
public AbstractGeneratedMethodReader predicate(Predicate predicate) {
this.predicate = predicate;
return this;
}
/**
* A utility method that assists implementations in retrieving a method from a class.
* It looks up the method by its name and parameter types, making it accessible if it's private or protected.
*
* @param clazz The class containing the method
* @param name The name of the method
* @param parameterTypes The parameter types of the method
* @return The method if found, otherwise throws an AssertionError
*/
protected static Method lookupMethod(Class> clazz, String name, Class>... parameterTypes) {
try {
final Method method = clazz.getMethod(name, parameterTypes);
Jvm.setAccessible(method);
return method;
} catch (NoSuchMethodException e) {
throw new AssertionError(e);
}
}
/**
* Sets a consumer for processing message history.
* This consumer is invoked if the next message is for a different queue and the history
* message isn't written to the output queue. It ensures that the LAST_WRITTEN mechanism
* works even when no output is present for a given message.
*
* @param historyConsumer The consumer for processing message history
*/
public void historyConsumer(Consumer historyConsumer) {
this.historyConsumer = historyConsumer;
}
/**
* Reads call name and arguments from the wire and performs invocation on a target object instance.
* The implementation of this method is generated at runtime, see {@link GenerateMethodReader}.
*
* @param wireIn Data input.
* @return MethodReaderStatus.
*/
protected abstract MethodReaderStatus readOneGenerated(WireIn wireIn);
protected abstract MethodReaderStatus readOneMetaGenerated(WireIn wireIn);
/**
* Reads the content based on the provided document context.
*
* @param context Reading document context.
* @return KNOWN if the read event is known, UNKNOWN if the event is not recognized, or EMPTY if no content is present.
*/
public MethodReaderStatus readOne0(DocumentContext context) {
WireIn wireIn = context.wire();
// Return EMPTY status if no content.
if (wireIn == null)
return MethodReaderStatus.EMPTY;
// Check if we need to write the unwritten message history.
if (historyConsumer != NO_OP_MH_CONSUMER) {
writeUnwrittenMessageHistory(context);
// Another reader may have swapped MessageHistory.get() and TEMP_MESSAGE_HISTORY
// Clearing local reference to recover link to the proper thread-local, which is MessageHistory.get()
messageHistory = null;
}
messageHistory().reset(context.sourceId(), context.index());
try {
wireIn.startEvent();
wireIn.consumePadding();
Bytes> bytes = wireIn.bytes();
dataEventProcessed = false;
MethodReaderStatus decoded = MethodReaderStatus.EMPTY; // Initialize status as no message.
// Read and process all remaining bytes.
while (bytes.readRemaining() > 0) {
if (wireIn.isEndEvent())
break;
long start = bytes.readPosition();
// Read the wire based on whether it's data or metadata.
MethodReaderStatus mrs = context.isData()
? readOneGenerated(wireIn)
: readOneMetaGenerated(wireIn);
// Update the decoding status based on the current read status.
switch (mrs) {
case HISTORY:
// Status remains unchanged.
break;
case KNOWN:
decoded = MethodReaderStatus.KNOWN;
break;
case UNKNOWN:
if (decoded == MethodReaderStatus.EMPTY)
decoded = MethodReaderStatus.UNKNOWN;
break;
default:
throw new AssertionError(mrs);
}
// If any bytes are ignored, return the decoding status.
if (restIgnored())
return decoded;
wireIn.consumePadding();
if (bytes.readPosition() == start) {
logNonProgressWarning(bytes.readRemaining());
return decoded;
}
}
wireIn.endEvent();
return decoded;
} finally {
// Don't save message history if we are reading non-data event (e.g. another "message history only" message)
// Infinite loop between services is possible otherwise
if (historyConsumer != NO_OP_MH_CONSUMER && dataEventProcessed)
swapMessageHistoryIfDirty();
messageHistory.reset();
}
}
/**
* Logs a warning message indicating that there has been no progress in reading the wire data.
* The warning provides information about the number of bytes that are left unread.
*
* @param bytes The number of bytes that are left unread.
*/
private void logNonProgressWarning(long bytes) {
Jvm.warn().on(getClass(), "Failed to progress reading " + bytes + " bytes left.");
}
/**
* Determines if the rest of the wire data should be ignored.
*
* @return {@code true} if the rest should be ignored, {@code false} otherwise. Default implementation returns false.
*/
protected boolean restIgnored() {
return false;
}
/**
* Swaps the current message history with a temporary message history using a double buffer technique.
* The method ensures that if an input event did not generate an output event, its message history
* will be saved, potentially to be written by another method reader in the future.
* If an input event did generate an output, stale information from a previous input event is cleared.
*/
private void swapMessageHistoryIfDirty() {
if (messageHistory.isDirty()) {
// This input event didn't generate an output event.
// Saving message history - in case next input event will be processed by another method reader,
// that method reader will cooperatively write saved history.
messageHistory = tempMessageHistory.getAndSet(messageHistory);
MessageHistory.set(messageHistory);
assert (messageHistory != tempMessageHistory.get());
} else {
// This input event generated an output event.
// In case previous input event was processed by this method reader, TEMP_MESSAGE_HISTORY may contain
// stale info on event's message history, which is superseded by the message history written now.
tempMessageHistory.get().reset();
}
}
/**
* Writes the message history of the last message to the output queue, but only if required.
* This is determined by checking if the message history has not yet been written to the output queue.
*
* @param context The {@link DocumentContext} of the output queue where the message history might be written.
*/
private void writeUnwrittenMessageHistory(DocumentContext context) {
final MessageHistory mh = tempMessageHistory.get();
if (mh.sources() != 0 && context.sourceId() != mh.lastSourceId() && mh.isDirty())
historyConsumer.accept(mh);
}
@Override
public boolean readOne() {
if (!predicate.test(this))
return false;
do {
throwExceptionIfClosed();
try (DocumentContext context = in.readingDocument()) {
if (!context.isPresent()) {
break;
}
MethodReaderStatus mrs = readOne0(context);
switch (mrs) {
case KNOWN:
if (scanning && context.isMetaData())
break; // continue
return true;
case EMPTY:
case UNKNOWN:
if (scanning)
break; // continue looking
return true;
default:
throw new AssertionError(mrs);
}
// retry on a data message unless a known message is found.
}
} while (scanning);
return false;
}
/**
* Checks if the method reader instance has been closed and throws a {@link ClosedIllegalStateException} if it is.
* This method is intended to guard against operations on a closed instance, ensuring that no further
* operations can be performed once the instance has been closed. This is particularly useful for
* methods that modify the state of the instance or rely on its open state to function correctly.
*
* @throws ClosedIllegalStateException if this instance has already been closed.
*/
public void throwExceptionIfClosed() throws ClosedIllegalStateException {
if (isClosed())
throw new ClosedIllegalStateException("Closed");
}
@Override
public MethodReaderInterceptorReturns methodReaderInterceptorReturns() {
return null;
}
@Override
public void close() {
if (closeIn)
closeQuietly(in);
closed = true;
}
@Override
public boolean isClosed() {
return closed;
}
@Override
public MethodReader closeIn(boolean closeIn) {
throwExceptionIfClosed();
this.closeIn = closeIn;
return this;
}
/**
* Offers a workaround to selectively disable "object recycling read" provided by {@link ValueIn#object(Object, Class)}
* for specific object types. This ensures that certain objects, such as arrays and collections, are
* not unintentionally reused or recycled during the reading process.
*
* @param The generic type of the object to check.
* @param o The object instance to verify and possibly recycle.
* @return The object itself if recycling is not applied, or {@code null} if the object is either
* {@code null} or an array. If the object is a collection or map, the method will clear its
* content and return the object.
*/
protected T checkRecycle(T o) {
if (o == null || o.getClass().isArray()) // If the object is null or an array, return null to prevent recycling.
return null;
if (o instanceof Collection) { // If the object is a collection, clear its content.
((Collection>) o).clear();
}
if (o instanceof Map) { // If the object is a map, clear its content.
((Map, ?>) o).clear();
}
// For objects of type AbstractMarshallableCfg, reset them to their default state.
if (o instanceof AbstractMarshallableCfg) {
((AbstractMarshallableCfg) o).reset();
}
// Return the potentially modified object.
return o;
}
/**
* Invokes a given method on the provided object with the specified arguments.
*
* @param method The method to be invoked.
* @param o The target object on which the method is to be invoked.
* @param objects The arguments to the method.
* @return Returns the result of the method invocation.
* @throws RuntimeException if the method invocation throws an exception.
*/
protected Object actualInvoke(Method method, Object o, Object[] objects) {
try {
return method.invoke(o, objects);
} catch (Exception e) {
throw Jvm.rethrow(e);
}
}
/**
* Retrieves the current message history.
* If the message history is not initialized, fetches it from the MessageHistory class.
*
* @return The current message history.
*/
private MessageHistory messageHistory() {
if (messageHistory == null)
messageHistory = MessageHistory.get();
return messageHistory;
}
/**
* Sets the scanning state.
*
* @param scanning The desired state of scanning. True to indicate scanning, false otherwise.
*/
public void scanning(boolean scanning) {
this.scanning = scanning;
}
/**
* This is a helper class that manages thread-local instances of MessageHistory.
* It provides methods to get and set the current thread's message history using a {@link ThreadLocal}.
*/
private static final class MessageHistoryThreadLocal {
private final ThreadLocal messageHistoryTL = withInitial(() -> {
@NotNull VanillaMessageHistory veh = new VanillaMessageHistory();
veh.addSourceDetails(true);
return veh;
});
/**
* Replaces the current thread's message history with the provided one
* and returns the replaced message history.
*
* @param mh The new message history to set for the current thread.
* @return The replaced message history.
*/
private MessageHistory getAndSet(MessageHistory mh) {
final MessageHistory result = messageHistoryTL.get();
messageHistoryTL.set(mh);
return result;
}
/**
* Retrieves the current thread's message history.
*
* @return The current thread's message history.
*/
public MessageHistory get() {
return messageHistoryTL.get();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy