io.kroxylicious.test.ApiMessageSampleGenerator 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;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
import java.util.stream.Stream;
import org.apache.kafka.common.Uuid;
import org.apache.kafka.common.protocol.ApiKeys;
import org.apache.kafka.common.protocol.ApiMessage;
import org.apache.kafka.common.protocol.Message;
import org.apache.kafka.common.protocol.types.BoundField;
import org.apache.kafka.common.protocol.types.Schema;
import org.apache.kafka.common.record.BaseRecords;
import org.apache.kafka.common.record.CompressionType;
import org.apache.kafka.common.record.MemoryRecords;
import org.apache.kafka.common.record.MemoryRecordsBuilder;
import org.apache.kafka.common.record.RecordBatch;
import org.apache.kafka.common.record.SimpleRecord;
import org.apache.kafka.common.record.TimestampType;
import org.apache.kafka.common.utils.ImplicitLinkedHashCollection;
/**
* Generates an ApiMessage instance per ApiKey. The message is built
* with reflection and populated with randomised data. The data is built
* from the same random seed and the ApiKeys and methods are iterated over
* in a deterministic order so the output of `createRequestSamples` should
* be consistent.
*/
public class ApiMessageSampleGenerator {
public record ApiAndVersion(ApiKeys keys, short apiVersion) {
}
public static final int RANGE_MIN = 0;
public static final int RANGE_MAX = 1024;
private ApiMessageSampleGenerator() {
}
/**
* Generates a sample request ApiMessage for all ApiKeys
* @return ApiKeys to message
*/
static public Map createRequestSamples() {
Random random = new Random(0);
return instantiateAll(DataClasses.getRequestClasses(), random);
}
/**
* Generates a sample response ApiMessage for all ApiKeys
* @return ApiKeys to message
*/
static public Map createResponseSamples() {
Random random = new Random(0);
return instantiateAll(DataClasses.getResponseClasses(), random);
}
private static Map instantiateAll(Map> messages, Random random) {
return messages.entrySet().stream().sorted(Comparator.comparing(apiKeysClassEntry -> apiKeysClassEntry.getKey().name))
.flatMap(ApiMessageSampleGenerator::getSupportedVersions)
.flatMap(entry -> instantiateSample(entry, random)).collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
}
private static Stream>> getSupportedVersions(
Map.Entry> apiKeysClassEntry) {
short lowestSupportedVersion = apiKeysClassEntry.getKey().messageType.lowestSupportedVersion();
short highestSupportedVersion = apiKeysClassEntry.getKey().messageType.highestSupportedVersion(true);
return IntStream.range(lowestSupportedVersion, highestSupportedVersion + 1).mapToObj(version -> new AbstractMap.SimpleEntry<>(
new ApiAndVersion(apiKeysClassEntry.getKey(), (short) version), apiKeysClassEntry.getValue()));
}
private static Stream> instantiateSample(Map.Entry> entry, Random random) {
try {
Field schemasField = entry.getValue().getDeclaredField("SCHEMAS");
Schema[] schemas = (Schema[]) schemasField.get(null);
Field lowestSupportedVersion = entry.getValue().getDeclaredField("LOWEST_SUPPORTED_VERSION");
short lowestVersion = (short) lowestSupportedVersion.get(null);
Field highestSupportedVersion = entry.getValue().getDeclaredField("HIGHEST_SUPPORTED_VERSION");
short highestVersion = (short) highestSupportedVersion.get(null);
short apiVersion = entry.getKey().apiVersion;
if (apiVersion < lowestVersion || apiVersion > highestVersion) {
return Stream.of();
}
Schema schema = schemas[apiVersion];
ApiMessage message = (ApiMessage) instantiate(entry.getValue(), random, schema);
return Stream.of(new AbstractMap.SimpleEntry<>(entry.getKey(), message));
}
catch (Exception e) {
throw new RuntimeException(e);
}
}
static private Message instantiate(Class extends Message> clazz, Random random, Schema schema) {
try {
Message instance = clazz.getConstructor().newInstance();
Map fieldsForVersion = Arrays.stream(schema.fields())
.filter(boundField -> !boundField.def.name.startsWith("_"))
.collect(Collectors.toMap(ApiMessageSampleGenerator::toSetterName, boundField -> boundField.def));
Stream sortedMethods = Arrays.stream(clazz.getMethods())
.filter(method -> method.getName().startsWith("set"))
.filter(method -> fieldsForVersion.containsKey(method.getName()))
.sorted(Comparator.comparing(Method::getName));
sortedMethods.forEach(method -> {
try {
Object o = instantiateArg(method, random, fieldsForVersion.get(method.getName()).type);
method.invoke(instance, o);
}
catch (Exception e) {
throw new RuntimeException(e);
}
});
return instance;
}
catch (Exception e) {
throw new RuntimeException(e);
}
}
private static String toSetterName(BoundField boundField) {
return "set" + Arrays.stream(boundField.def.name.split("_")).map(s -> s.substring(0, 1).toUpperCase() + s.substring(1)).collect(Collectors.joining());
}
static private Object instantiateArg(Class> paramClass, Type paramGenericType, Random random, org.apache.kafka.common.protocol.types.Type type) {
if (paramClass == long.class || paramClass == Long.class) {
return random.nextLong(RANGE_MIN, RANGE_MAX);
}
else if (paramClass == int.class || paramClass == Integer.class) {
return random.nextInt(RANGE_MIN, RANGE_MAX);
}
else if (paramClass == short.class || paramClass == Short.class) {
return (short) random.nextInt(RANGE_MIN, RANGE_MAX);
}
else if (paramClass == double.class || paramClass == Double.class) {
return random.nextDouble(RANGE_MIN, RANGE_MAX);
}
else if (paramClass == String.class) {
return "random-string-" + random.nextInt(RANGE_MIN, RANGE_MAX);
}
else if (paramClass == byte.class || paramClass == Byte.class) {
return (byte) random.nextInt(RANGE_MIN, 127);
}
else if (paramClass == byte[].class) {
byte[] bytes = new byte[5];
random.nextBytes(bytes);
return bytes;
}
else if (paramClass == ByteBuffer.class) {
byte[] bytes = new byte[5];
random.nextBytes(bytes);
return ByteBuffer.wrap(bytes).flip();
}
else if (paramClass == boolean.class) {
return random.nextBoolean();
}
else if (paramClass == Uuid.class) {
return randomUuid(random);
}
else if (paramClass == BaseRecords.class) {
return randomMemoryRecords(random);
}
else if (Message.class.isAssignableFrom(paramClass)) {
return instantiate(paramClass.asSubclass(Message.class), random, (Schema) type);
}
else if (List.class.isAssignableFrom(paramClass)) {
return instantiateList(paramGenericType, random, type);
}
else if (ImplicitLinkedHashCollection.class.isAssignableFrom(paramClass)) {
return instantiateMessageCollection(paramClass.asSubclass(ImplicitLinkedHashCollection.class), random, type);
}
else {
throw new IllegalArgumentException("unexpected type " + paramClass.getSimpleName());
}
}
private static MemoryRecords randomMemoryRecords(Random random) {
String key = "key-" + random.nextInt(RANGE_MIN, RANGE_MAX);
String value = "value-" + random.nextInt(RANGE_MIN, RANGE_MAX);
long baseOffset = random.nextInt(RANGE_MIN, RANGE_MAX);
long logAppendTime = RecordBatch.NO_TIMESTAMP;
try (MemoryRecordsBuilder builder = MemoryRecords.builder(ByteBuffer.allocate(1), RecordBatch.CURRENT_MAGIC_VALUE, CompressionType.NONE,
TimestampType.CREATE_TIME, baseOffset,
logAppendTime,
RecordBatch.NO_PRODUCER_ID, RecordBatch.NO_PRODUCER_EPOCH, RecordBatch.NO_SEQUENCE, false,
RecordBatch.NO_PARTITION_LEADER_EPOCH)) {
builder.append(new SimpleRecord(random.nextLong(RANGE_MIN, RANGE_MAX), key.getBytes(StandardCharsets.UTF_8), value.getBytes(StandardCharsets.UTF_8)));
return builder.build();
}
}
/**
* Takes the guts of the JVM randomUUID and byte[] constructor
* so that we can create a random UUID using our random
* implementation. This is so that all random data used in generation
* comes from the same random seed to generate deterministic samples.
* @param random the random to use
* @return a random Uuid
*/
private static Uuid randomUuid(Random random) {
byte[] randomBytes = new byte[16];
random.nextBytes(randomBytes);
randomBytes[6] &= 0x0f; /* clear version */
randomBytes[6] |= 0x40; /* set to version 4 */
randomBytes[8] &= 0x3f; /* clear variant */
randomBytes[8] |= 0x80; /* set to IETF variant */
long mostSignificantBits = 0;
long leastSignificantBits = 0;
for (int i = 0; i < 8; i++) {
mostSignificantBits = (mostSignificantBits << 8) | (randomBytes[i] & 0xff);
}
for (int i = 8; i < 16; i++) {
leastSignificantBits = (leastSignificantBits << 8) | (randomBytes[i] & 0xff);
}
return new Uuid(mostSignificantBits, leastSignificantBits);
}
private static Object instantiateMessageCollection(Class extends ImplicitLinkedHashCollection> genericType, Random random,
org.apache.kafka.common.protocol.types.Type field) {
try {
ImplicitLinkedHashCollection collection = genericType.getConstructor().newInstance();
ParameterizedType paramClass1 = (ParameterizedType) genericType.getGenericSuperclass();
Type actualTypeArgument = paramClass1.getActualTypeArguments()[0];
Class> typeArgument = (Class>) actualTypeArgument;
Object o = instantiateArg(typeArgument, actualTypeArgument, random, field.arrayElementType().orElseThrow());
collection.add((ImplicitLinkedHashCollection.Element) o);
return collection;
}
catch (Exception e) {
throw new RuntimeException(e);
}
}
private static Object instantiateList(Type paramClass, Random random, org.apache.kafka.common.protocol.types.Type type) {
ArrayList
© 2015 - 2025 Weber Informatics LLC | Privacy Policy