io.kroxylicious.test.codec.DecodedFrame Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kroxylicious-integration-test-support Show documentation
Show all versions of kroxylicious-integration-test-support Show documentation
Support code to simplify writing integration tests for filters
/*
* Copyright Kroxylicious Authors.
*
* Licensed under the Apache Software License version 2.0, available at http://www.apache.org/licenses/LICENSE-2.0
*/
package io.kroxylicious.test.codec;
import org.apache.kafka.common.protocol.ApiKeys;
import org.apache.kafka.common.protocol.ApiMessage;
import org.apache.kafka.common.protocol.MessageSizeAccumulator;
import org.apache.kafka.common.protocol.ObjectSerializationCache;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* A frame that has been decoded.
*
* @param The header type
* @param The body type
*
*/
public abstract class DecodedFrame
implements Frame {
private static final Logger LOGGER = LoggerFactory.getLogger(DecodedFrame.class);
/**
* Number of bytes required for storing the frame length.
*/
private static final int FRAME_SIZE_LENGTH = Integer.BYTES;
/**
* apiVersion of the frame
*/
protected final short apiVersion;
/**
* correlationId of the frame
*/
protected final int correlationId;
/**
* header of the frame
*/
protected final H header;
/**
* body of the frame
*/
protected final B body;
private int headerAndBodyEncodedLength;
private ObjectSerializationCache serializationCache;
DecodedFrame(short apiVersion, int correlationId, H header, B body) {
this.apiVersion = apiVersion;
this.correlationId = correlationId;
this.header = header;
this.body = body;
this.headerAndBodyEncodedLength = -1;
}
@Override
public int correlationId() {
return correlationId;
}
/**
* header version of frame
* @return header version
*/
protected abstract short headerVersion();
/**
* Header of the frame
* @return header
*/
public H header() {
return header;
}
/**
* Body of the frame
* @return body
*/
public B body() {
return body;
}
/**
* Get apiKey of body
* @return apiKey
*/
public ApiKeys apiKey() {
return ApiKeys.forId(body.apiKey());
}
/**
* Get apiVersion of frame
* @return apiKey
*/
public short apiVersion() {
return apiVersion;
}
@Override
public final int estimateEncodedSize() {
if (headerAndBodyEncodedLength != -1) {
assert serializationCache != null;
return FRAME_SIZE_LENGTH + headerAndBodyEncodedLength;
}
var headerVersion = headerVersion();
MessageSizeAccumulator sizer = new MessageSizeAccumulator();
ObjectSerializationCache cache = new ObjectSerializationCache();
header().addSize(sizer, cache, headerVersion);
body().addSize(sizer, cache, apiVersion());
headerAndBodyEncodedLength = sizer.totalSize();
serializationCache = cache;
return FRAME_SIZE_LENGTH + headerAndBodyEncodedLength;
}
@Override
public final void encode(ByteBufAccessor out) {
if (headerAndBodyEncodedLength < 0) {
LOGGER.warn("Encoding estimation should happen before encoding, if possible");
}
final int encodedSize = estimateEncodedSize();
if (LOGGER.isTraceEnabled()) {
LOGGER.trace("Writing {} with 4 byte length ({}) plus bytes of header {}, and body {} to {}",
getClass().getSimpleName(), encodedSize, header, body, out);
}
out.ensureWritable(encodedSize);
final int initialIndex = out.writerIndex();
out.writeInt(headerAndBodyEncodedLength);
final ObjectSerializationCache cache = serializationCache;
header.write(out, cache, headerVersion());
body.write(out, cache, apiVersion());
assert (out.writerIndex() - initialIndex) == encodedSize;
}
@Override
public String toString() {
return getClass().getSimpleName() + "(" +
apiKey() + "(" + apiVersion + ")v" + apiVersion +
", header=" + header +
", body=" + body +
')';
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy