io.kroxylicious.proxy.internal.codec.CorrelationManager Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kroxylicious-runtime Show documentation
Show all versions of kroxylicious-runtime Show documentation
The proxy code which provides the runtime environment in which filters execute
The newest version!
/*
* 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.proxy.internal.codec;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import org.apache.kafka.common.protocol.ApiKeys;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import io.kroxylicious.proxy.filter.Filter;
import io.kroxylicious.proxy.tag.VisibleForTesting;
/**
* Manages correlation ids for a single connection (across the proxy) between a single client
* and a single broker.
*/
public class CorrelationManager {
private static final Logger LOGGER = LoggerFactory.getLogger(CorrelationManager.class);
// TODO use a specialized map
@VisibleForTesting
final Map brokerRequests = new HashMap<>();
/** The correlation id with the upstream broker */
private int upstreamId;
public CorrelationManager() {
this(0);
}
public CorrelationManager(int initialCorrelationId) {
upstreamId = initialCorrelationId;
}
/**
* Allocate and return a correlation id for an outgoing request to the broker.
*
* @param apiKey The API key.
* @param apiVersion The API version.
* @param downstreamCorrelationId The downstream client's correlation id.
* @param hasResponse Whether a response is expected.
* @param promise
* @param decodeResponse Whether the response should be decoded.
*/
public int putBrokerRequest(short apiKey,
short apiVersion,
int downstreamCorrelationId,
boolean hasResponse,
Filter recipient,
CompletableFuture> promise,
boolean decodeResponse) {
// need to allocate an id and put in a map for quick lookup, along with the "tag"
int upstreamCorrelationId = upstreamId++;
LOGGER.trace("Allocated upstream id {} for downstream id {}", upstreamCorrelationId, downstreamCorrelationId);
if (hasResponse) {
Correlation existing = this.brokerRequests.put(upstreamCorrelationId,
new Correlation(apiKey, apiVersion, downstreamCorrelationId, decodeResponse, recipient, promise));
if (existing != null) {
LOGGER.error("Duplicate upstream correlation id {}", upstreamCorrelationId);
}
}
return upstreamCorrelationId;
}
/**
* Find (and remove) the Correlation for an incoming response from the broker
* @param upstreamCorrelationId The (upstream) correlation id in the response.
*/
public Correlation getBrokerCorrelation(int upstreamCorrelationId) {
// Set the tag on the response object
return brokerRequests.remove(upstreamCorrelationId);
}
/**
* A record for which responses should be decoded, together with their
* API key and version.
*/
// TODO a perfect value type
public static class Correlation {
private final short apiKey;
private final short apiVersion;
private final int downstreamCorrelationId;
private final boolean decodeResponse;
private final Filter recipient;
private final CompletableFuture> promise;
private Correlation(short apiKey,
short apiVersion,
int downstreamCorrelationId,
boolean decodeResponse,
Filter recipient,
CompletableFuture> promise) {
this.apiKey = apiKey;
this.apiVersion = apiVersion;
this.downstreamCorrelationId = downstreamCorrelationId;
this.decodeResponse = decodeResponse;
this.recipient = recipient;
this.promise = promise;
}
public int downstreamCorrelationId() {
return downstreamCorrelationId;
}
@Override
public String toString() {
return "Correlation(" +
"apiKey=" + ApiKeys.forId(apiKey) +
", apiVersion=" + apiVersion +
", downstreamCorrelationId=" + downstreamCorrelationId +
", decodeResponse=" + decodeResponse +
", recipient=" + recipient +
", promise=" + promise +
')';
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
Correlation that = (Correlation) o;
return apiKey == that.apiKey && apiVersion == that.apiVersion && downstreamCorrelationId == that.downstreamCorrelationId
&& decodeResponse == that.decodeResponse;
}
@Override
public int hashCode() {
return Objects.hash(apiKey, apiVersion, downstreamCorrelationId, decodeResponse);
}
public short apiKey() {
return apiKey;
}
public short apiVersion() {
return apiVersion;
}
public boolean decodeResponse() {
return decodeResponse;
}
public Filter recipient() {
return recipient;
}
public CompletableFuture> promise() {
return promise;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy