snapchatproto.servers.queue.OutboundAppWorker.java Source code

Java tutorial

Introduction

Here is the source code for snapchatproto.servers.queue.OutboundAppWorker.java

Source

/*
 * copyright 2015, gash
 * 
 * Gash licenses this file to you 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 snapchatproto.servers.queue;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import snapchatproto.servers.queue.DiscreteQueue.OneQueueEntry;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;

import com.google.protobuf.GeneratedMessage;

public class OutboundAppWorker extends Thread {
    protected static Logger logger = LoggerFactory.getLogger("server");

    int workerId;
    boolean forever = true;

    public OutboundAppWorker(ThreadGroup tgrp, int workerId) {
        super(tgrp, "outbound-" + workerId);
        this.workerId = workerId;

        if (DiscreteQueue.outbound == null)
            throw new RuntimeException("connection worker detected no outbound queue");
    }

    @Override
    public void run() {

        while (true) {
            if (!forever && DiscreteQueue.outbound.size() == 0)
                break;

            try {
                // block until a message is enqueued
                OneQueueEntry msg = DiscreteQueue.outbound.take();
                if (msg.channel.isWritable()) {
                    boolean rtn = false;
                    if (msg.channel != null && msg.channel.isOpen() && msg.channel.isWritable()) {
                        ChannelFuture cf = msg.channel.writeAndFlush(msg);

                        // blocks on write - use listener to be async
                        cf.awaitUninterruptibly();
                        rtn = cf.isSuccess();
                        if (!rtn)
                            DiscreteQueue.outbound.putFirst(msg);
                    }

                } else
                    DiscreteQueue.outbound.putFirst(msg);
            } catch (InterruptedException ie) {
                break;
            } catch (Exception e) {
                logger.error("Unexpected communcation failure", e);
                break;
            }
        }

        if (!forever) {
            logger.info("connection queue closing");
        }
    }
}