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

com.opentable.kafka.builders.KafkaConsumerBaseBuilder Maven / Gradle / Ivy

The newest version!
/*
 * 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 com.opentable.kafka.builders;

import java.time.Duration;
import java.util.Arrays;
import java.util.Map;
import java.util.Optional;
import java.util.OptionalInt;
import java.util.OptionalLong;

import org.apache.kafka.clients.consumer.ConsumerConfig;
import org.apache.kafka.clients.consumer.ConsumerInterceptor;
import org.apache.kafka.clients.consumer.ConsumerPartitionAssignor;
import org.apache.kafka.clients.consumer.RangeAssignor;
import org.apache.kafka.common.serialization.Deserializer;

import com.opentable.kafka.logging.LoggingConsumerInterceptor;

@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
public abstract class KafkaConsumerBaseBuilder, K, V> extends KafkaBaseBuilder {

    private Optional groupId = Optional.empty();
    private Optional enableAutoCommit = Optional.empty();
    private OptionalInt maxPollRecords = OptionalInt.empty();
    private OptionalInt maxPartitionFetch = OptionalInt.empty();
    private OptionalLong sessionTimeoutMs = OptionalLong.empty();
    private OptionalLong maxPollIntervalMs = OptionalLong.empty();
    private Optional autoOffsetResetType = Optional.empty();
    private Class partitionStrategy = RangeAssignor.class;
    // Kafka is really stupid. In the properties you can only configure a no-args
    // and then they hack around it if you have one supplied
    private Class> keyDe;
    private Class> valueDe;
    protected Deserializer keyDeserializerInstance;
    protected Deserializer valueDeserializerInstance;

    protected KafkaConsumerBaseBuilder(Map props, EnvironmentProvider environmentProvider) {
        super(props, environmentProvider);
    }

    public SELF withGroupId(String val) {
        groupId = Optional.ofNullable(val);
        return self();
    }

    public SELF withAutoOffsetReset(KafkaConsumerBuilder.AutoOffsetResetType val) {
        autoOffsetResetType = Optional.ofNullable(val);
        return self();
    }

    public SELF withMaxPollRecords(int val) {
        maxPollRecords = OptionalInt.of(val);
        return self();
    }

    /**
     * Provide an class. If you have a no-args constructor use this
     * @param keyDeSer key deserializer
     * @param valDeSer value deserializer
     * @param  The type of the key returned by de-serializer
     * @param  The type of the value returned by de-serializer
     * @return this
     */
    @SuppressWarnings("unchecked")
    protected  KafkaConsumerBaseBuilder withDeserializers(Class> keyDeSer, Class> valDeSer) {
        KafkaConsumerBaseBuilder res = (KafkaConsumerBaseBuilder) self();
        res.keyDe = keyDeSer;
        res.valueDe = valDeSer;
        res.keyDeserializerInstance = null;
        res.valueDeserializerInstance = null;
        return res;
    }

    /**
     * Provide an instance. If you don't have a no-args constructor use this
     * @param keyDeSer key deserializer
     * @param valDeSer value deserializer
     * @param  The type of the key returned by de-serializer
     * @param  The type of the value returned by de-serializer
     * @return this
     */
    @SuppressWarnings("unchecked")
    protected  KafkaConsumerBaseBuilder withDeserializers(Deserializer keyDeSer, Deserializer valDeSer) {
        KafkaConsumerBaseBuilder res = (KafkaConsumerBaseBuilder) self();
        res.keyDeserializerInstance = keyDeSer;
        res.valueDeserializerInstance = valDeSer;
        this.keyDe = null;
        this.valueDe = null;
        return res;
    }

    public SELF withPartitionAssignmentStrategy(Class partitionAssignmentStrategy) {
        partitionStrategy = partitionAssignmentStrategy;
        return self();
    }


    public SELF withAutoCommit(boolean val) {
        enableAutoCommit = Optional.of(val);
        return self();
    }


    public SELF withMaxPartitionFetchBytes(int bytes) {
        this.maxPartitionFetch = OptionalInt.of(bytes);
        return self();
    }


    public SELF withPollInterval(Duration duration) {
        if (duration != null) {
            maxPollIntervalMs = OptionalLong.of(duration.toMillis());
        }
        return self();
    }

    public SELF withSessionTimeoutMs(Duration duration) {
        if (duration != null) {
            sessionTimeoutMs = OptionalLong.of(duration.toMillis());
        }
        return self();
    }

    public SELF withInterceptor(Class> clazz) {
        this.addInterceptor(clazz.getName());
        return self();
    }

    public Map buildProperties() {
        internalBuild();
        return getFinalProperties();
    }

    protected void internalBuild() {
        if (partitionStrategy != null) {
            addProperty(ConsumerConfig.PARTITION_ASSIGNMENT_STRATEGY_CONFIG, partitionStrategy.getName());
        }
        maxPartitionFetch.ifPresent(m -> this.addProperty(ConsumerConfig.MAX_PARTITION_FETCH_BYTES_CONFIG, String.valueOf(m)));
        maxPollIntervalMs.ifPresent(m -> this.addProperty(ConsumerConfig.MAX_POLL_INTERVAL_MS_CONFIG, String.valueOf(m)));
        sessionTimeoutMs.ifPresent(s -> this.addProperty(ConsumerConfig.SESSION_TIMEOUT_MS_CONFIG, String.valueOf(s)));
        enableAutoCommit.ifPresent(e -> this.addProperty(ConsumerConfig.ENABLE_AUTO_COMMIT_CONFIG, String.valueOf(e)));
        this.setupInterceptors(ConsumerConfig.INTERCEPTOR_CLASSES_CONFIG, LoggingConsumerInterceptor.class.getName());
        groupId.ifPresent(gid -> this.addProperty(ConsumerConfig.GROUP_ID_CONFIG, gid));
        autoOffsetResetType.ifPresent(a -> this.addProperty(ConsumerConfig.AUTO_OFFSET_RESET_CONFIG, a.value));
        maxPollRecords.ifPresent(mpr -> this.addProperty(ConsumerConfig.MAX_POLL_RECORDS_CONFIG, mpr));
        if (keyDe != null) {
            this.addProperty(ConsumerConfig.KEY_DESERIALIZER_CLASS_CONFIG, keyDe.getName());
        }
        if (valueDe != null) {
            this.addProperty(ConsumerConfig.VALUE_DESERIALIZER_CLASS_CONFIG, valueDe.getName());
        }
        // Merge in common and user supplied properties.
        this.finishBuild();
        this.cantBeNull(ConsumerConfig.PARTITION_ASSIGNMENT_STRATEGY_CONFIG, "Partition assignment strategy can't be null");
    }

    public enum AutoOffsetResetType {
        Latest("latest"), Earliest("earliest"), None("none");
        final String value;
        AutoOffsetResetType(String value) {
            this.value = value;
        }

        public static AutoOffsetResetType fromString(String c) {
            return Arrays.stream(values()).filter(t -> t.value.equalsIgnoreCase(c))
                .findFirst().orElseThrow(() -> new IllegalArgumentException("Can't convert " + c));
        }
    }


}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy