com.github.milenkovicm.kafka.KafkaProducer.java Source code

Java tutorial

Introduction

Here is the source code for com.github.milenkovicm.kafka.KafkaProducer.java

Source

/*
 * Copyright 2015 Marko Milenkovic (http://github.com/milenkovicm)
 *
 * 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.github.milenkovicm.kafka;

import com.github.milenkovicm.kafka.connection.ControlKafkaBroker;
import com.github.milenkovicm.kafka.connection.DataKafkaBroker;
import com.github.milenkovicm.kafka.connection.KafkaPromise;
import com.github.milenkovicm.kafka.protocol.MetadataResponse;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.util.concurrent.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit;

public class KafkaProducer {

    private static final Logger LOGGER = LoggerFactory.getLogger(KafkaProducer.class);

    final EventExecutor eventExecutor;
    final Promise<Void> connectPromise;
    final Promise<Void> disconnectPromise;

    final String hostname;
    final int port;
    final String topicName;

    final KafkaTopic kafkaTopic;
    final ProducerProperties properties;
    final EventLoopGroup workerGroup;
    final Map<Integer, DataKafkaBroker> brokers = new ConcurrentHashMap<>();

    volatile ControlKafkaBroker control = null;
    volatile int numberOfPartitions;
    volatile boolean shutdown = false;

    public KafkaProducer(String hostname, int port, String topicName, ProducerProperties properties,
            EventLoopGroup workerGroup) {
        this.properties = properties;
        this.hostname = hostname;
        this.port = port;
        this.topicName = topicName;
        this.workerGroup = workerGroup;

        this.kafkaTopic = new KafkaTopic(properties.get(ProducerProperties.PARTITIONER), properties, topicName);

        this.eventExecutor = GlobalEventExecutor.INSTANCE;
        this.connectPromise = new DefaultPromise<>(eventExecutor);
        this.disconnectPromise = new DefaultPromise<>(eventExecutor);

    }

    public KafkaProducer(String hostname, int port, String topicName, ProducerProperties properties) {
        this(hostname, port, topicName, properties,
                new NioEventLoopGroup(properties.get(ProducerProperties.NETTY_THREAD_COUNT),
                        new DefaultThreadFactory("producer-" + topicName, Thread.MAX_PRIORITY)));
    }

    public KafkaProducer(String hostname, int port, String topicName) {
        this(hostname, port, topicName, new ProducerProperties());
    }

    public KafkaTopic topic() {
        return kafkaTopic;
    }

    public Future<Void> connect() {
        this.connectControl(hostname, port, connectPromise);
        return connectPromise;
    }

    public Future<Void> disconnect() {
        this.shutdown = true;

        List<Future<?>> futures = new ArrayList<>();
        if (control != null) {
            final ChannelFuture disconnectControl = control.channel().disconnect();
            futures.add(disconnectControl);
        }
        for (DataKafkaBroker data : brokers.values()) {
            final Future<?> disconnectData = data.disconnect();
            futures.add(disconnectData);
        }
        brokers.clear();
        control = null;

        if (futures.size() > 0) {
            final PromiseAggregator<Void, ChannelFuture> promiseAggregator = new PromiseAggregator<Void, ChannelFuture>(
                    disconnectPromise);
            promiseAggregator.add(futures.toArray(new Promise[1]));
        } else {
            disconnectPromise.setSuccess(null);
        }
        this.workerGroup.shutdownGracefully();
        this.eventExecutor.shutdownGracefully();
        return disconnectPromise;
    }

    // connect control channel which handles all metadata queries
    void connectControl(String hostname, int port, Promise<Void> promise) {
        ControlKafkaBroker control = new ControlKafkaBroker(hostname, port, topicName, workerGroup, properties);
        this.control = control;
        control.connect().addListener(new ControlConnectListener(promise));
        control.channel().closeFuture().addListener(new ControlDisconnectedListener(control));
    }

    void connectData(MetadataResponse metadataResponse, Promise<Void> promise) {
        LOGGER.debug("create data connections ...");

        MetadataResponse.TopicMetadata topic = metadataResponse.topics.get(0);
        this.numberOfPartitions = topic.partitions.size();
        this.kafkaTopic.initialize(this.numberOfPartitions);

        this.assignPartitions(metadataResponse, promise);
        LOGGER.debug("init done!");
    }

    void assignPartitions(MetadataResponse metadataResponse, Promise<Void> promise) {
        LOGGER.debug("assign partitions {}", metadataResponse);
        MetadataResponse.TopicMetadata topic = metadataResponse.topics.get(0);

        List<ChannelFuture> promises = new LinkedList<>();

        for (MetadataResponse.PartitionMetadata partition : topic.partitions) {
            LOGGER.debug("brokerId: [{}] assigned to partitionId: [{}]", partition.leader, partition.partitionId);
            DataKafkaBroker dataKafkaChannel = getDataChannel(metadataResponse, partition.leader, promises);
            this.kafkaTopic.set(dataKafkaChannel, partition.partitionId);
        }
        completePromise(promise, promises);
    }

    void completePromise(final Promise<Void> promise, List<ChannelFuture> promises) {
        if (promises.size() > 0) {
            final PromiseAggregator<Void, ChannelFuture> promiseAggregator = new PromiseAggregator<Void, ChannelFuture>(
                    promise) {
            };
            promiseAggregator.add(promises.toArray(new Promise[1]));
        } else {
            promise.setSuccess(null);
        }
    }

    DataKafkaBroker getDataChannel(MetadataResponse metadataResponse, Integer brokerId,
            List<ChannelFuture> promises) {
        DataKafkaBroker dataKafkaChannel = brokers.get(brokerId);

        if (dataKafkaChannel != null) {
            return dataKafkaChannel;
        } else {
            final MetadataResponse.Broker broker = findBroker(metadataResponse, brokerId);
            final DataKafkaBroker channel = connectDataChannel(promises, broker);
            brokers.put(broker.nodeId, channel);

            return channel;
        }
    }

    DataKafkaBroker connectDataChannel(List<ChannelFuture> promises, MetadataResponse.Broker broker) {
        LOGGER.debug("connecting data channel to broker:[{}] hostname: [{}] port:[{}]", broker.nodeId, broker.host,
                broker.port);
        final DataKafkaBroker channel = new DataKafkaBroker(broker.host, broker.port, broker.nodeId, topicName,
                workerGroup, properties);

        final ChannelFuture future = channel.connect();
        future.channel().closeFuture().addListener(new BrokerDisconnectedListener(channel));
        promises.add(future);
        return channel;
    }

    MetadataResponse.Broker findBroker(MetadataResponse metadataResponse, Integer brokerId) {
        for (MetadataResponse.Broker broker : metadataResponse.brokers) {
            if (broker.nodeId == brokerId) {
                return broker;
            }
        }
        throw new ProducerException("cant find broker with id: " + brokerId);
    }

    class ControlDisconnectedListener implements ChannelFutureListener {
        final ControlKafkaBroker controlKafkaChannel;

        ControlDisconnectedListener(ControlKafkaBroker controlKafkaChannel) {
            this.controlKafkaChannel = controlKafkaChannel;
        }

        @Override
        public void operationComplete(ChannelFuture future) throws Exception {
            if (shutdown) {
                return;
            }
            LOGGER.warn("control channel: [{}] disconnected! ", future.channel());
            control = null;
            controlKafkaChannel.disconnect();
            eventExecutor.schedule(new UpdateControlChannel(), properties.get(ProducerProperties.RETRY_BACKOFF),
                    TimeUnit.MILLISECONDS);
        }
    }

    class ControlConnectListener implements ChannelFutureListener {
        final Promise<Void> promise;

        ControlConnectListener(Promise<Void> promise) {
            this.promise = promise;
        }

        @Override
        public void operationComplete(ChannelFuture future) throws Exception {
            if (!future.isSuccess()) {
                //
                // couldn't connect control channel
                promise.setFailure(future.cause());
                LOGGER.debug("control connection failed! hostname:[{}] port:[{}]", hostname, port);
            } else {
                LOGGER.debug("control connection connected! hostname:[{}] port:[{}]", hostname, port);
                // control channel is connected
                // fetch metadata
                final KafkaPromise kafkaPromise = control.fetchMetadata(topicName);
                kafkaPromise.addListener(new FutureListener<MetadataResponse>() {
                    @Override
                    public void operationComplete(Future<MetadataResponse> future) throws Exception {
                        if (future.isSuccess()) {
                            // metadata fetched
                            // future will succeed when all data channels connect
                            connectData(future.get(), promise);
                        } else {
                            // error fetching metadata
                            promise.setFailure(future.cause());
                        }
                    }
                });
            }
        }
    }

    class BrokerDisconnectedListener implements ChannelFutureListener {

        final DataKafkaBroker dataKafkaChannel;

        BrokerDisconnectedListener(DataKafkaBroker dataKafkaChannel) {
            this.dataKafkaChannel = dataKafkaChannel;
        }

        @Override
        public void operationComplete(ChannelFuture future) throws Exception {
            if (shutdown) {
                return;
            }
            LOGGER.warn("brokerId: [{}] disconnected! ", dataKafkaChannel);
            kafkaTopic.remove(dataKafkaChannel);
            brokers.remove(dataKafkaChannel.brokerId);
            dataKafkaChannel.disconnect();
            eventExecutor.schedule(new UpdateDataChannels(), properties.get(ProducerProperties.RETRY_BACKOFF),
                    TimeUnit.MILLISECONDS);
        }
    }

    class UpdateDataChannels implements Runnable {

        @Override
        public void run() {
            ControlKafkaBroker control = KafkaProducer.this.control;
            if (control == null) {
                return;
            }
            control.fetchMetadata(topicName).addListener(new GenericFutureListener<Future<MetadataResponse>>() {
                @Override
                public void operationComplete(Future<MetadataResponse> future) throws Exception {
                    connectData(future.get(), new DefaultPromise<Void>(eventExecutor));
                }
            });
            LOGGER.debug("update data connection ... done");
        }
    }

    class UpdateControlChannel implements Runnable {

        @Override
        public void run() {
            final DataKafkaBroker dataKafkaChannel = kafkaTopic.get();
            final DefaultPromise<Void> promise = new DefaultPromise<Void>(eventExecutor);
            if (dataKafkaChannel != null) {
                connectControl(dataKafkaChannel.hostname, dataKafkaChannel.port, promise);
                LOGGER.debug("update control connection from data connection values... done");
            } else {
                connectControl(hostname, port, promise);
                LOGGER.debug("update control connection from default values ... done");
            }
        }
    }
}