io.jsync.net.impl.AsyncEventLoopGroup.java Source code

Java tutorial

Introduction

Here is the source code for io.jsync.net.impl.AsyncEventLoopGroup.java

Source

/*
 * Copyright (c) 2011-2013 The original author or authors
 * ------------------------------------------------------
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * and Apache License v2.0 which accompanies this distribution.
 *
 *     The Eclipse Public License is available at
 *     http://www.eclipse.org/legal/epl-v10.html
 *
 *     The Apache License v2.0 is available at
 *     http://www.opensource.org/licenses/apache2.0.php
 *
 * You may elect to redistribute this code under either of these licenses.
 */

package io.jsync.net.impl;

import io.netty.channel.*;
import io.netty.util.concurrent.*;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @author <a href="http://tfox.org">Tim Fox</a>
 */
@SuppressWarnings("deprecation")
public final class AsyncEventLoopGroup extends AbstractEventExecutorGroup implements EventLoopGroup {

    private final List<EventLoopHolder> workers = new ArrayList<>();
    private final CountDownLatch latch = new CountDownLatch(1);
    private final AtomicBoolean gracefulShutdown = new AtomicBoolean();
    private final Promise<?> terminationFuture = new DefaultPromise<Void>(GlobalEventExecutor.INSTANCE);
    private int pos;

    @Override
    public synchronized EventLoop next() {
        if (workers.isEmpty()) {
            throw new IllegalStateException();
        } else {
            EventLoop worker = workers.get(pos).worker;
            pos++;
            checkPos();
            return worker;
        }
    }

    @Override
    public Iterator<EventExecutor> iterator() {
        return new EventLoopIterator(workers.iterator());
    }

    @Override
    public ChannelFuture register(Channel channel) {
        return next().register(channel);
    }

    @Override
    public ChannelFuture register(ChannelPromise channelPromise) {
        return next().register(channelPromise);
    }

    @Deprecated
    @Override
    public ChannelFuture register(Channel channel, ChannelPromise promise) {
        return next().register(channel, promise);
    }

    @Override
    public boolean isShutdown() {
        return latch.getCount() == 0;
    }

    @Override
    public boolean isTerminated() {
        return isShutdown();
    }

    @Override
    public synchronized boolean awaitTermination(long timeout, TimeUnit unit) throws InterruptedException {
        return latch.await(timeout, unit);
    }

    public synchronized void addWorker(EventLoop worker) {
        EventLoopHolder holder = findHolder(worker);
        if (holder == null) {
            workers.add(new EventLoopHolder(worker));
        } else {
            holder.count++;
        }
    }

    public synchronized void shutdown() {
        for (EventLoopHolder holder : workers) {
            holder.worker.shutdown();
        }
        latch.countDown();
    }

    @Override
    public boolean isShuttingDown() {
        return gracefulShutdown.get();
    }

    @Override
    public Future<?> shutdownGracefully(long quietPeriod, long timeout, TimeUnit unit) {
        if (gracefulShutdown.compareAndSet(false, true)) {
            final AtomicInteger counter = new AtomicInteger(workers.size());
            for (EventLoopHolder holder : workers) {
                holder.worker.shutdownGracefully().addListener((GenericFutureListener) future -> {
                    if (counter.decrementAndGet() == 0) {
                        terminationFuture.setSuccess(null);
                    }
                });
            }
        }
        return terminationFuture;
    }

    @Override
    public Future<?> terminationFuture() {
        return terminationFuture;
    }

    private EventLoopHolder findHolder(EventLoop worker) {
        EventLoopHolder wh = new EventLoopHolder(worker);
        for (EventLoopHolder holder : workers) {
            if (holder.equals(wh)) {
                return holder;
            }
        }
        return null;
    }

    public synchronized void removeWorker(EventLoop worker) {
        //TODO can be optimised
        EventLoopHolder holder = findHolder(worker);
        if (holder != null) {
            holder.count--;
            if (holder.count == 0) {
                workers.remove(holder);
            }
            checkPos();
        } else {
            throw new IllegalStateException("Can't find worker to remove");
        }
    }

    public synchronized int workerCount() {
        return workers.size();
    }

    private void checkPos() {
        if (pos == workers.size()) {
            pos = 0;
        }
    }

    private static class EventLoopHolder {
        final EventLoop worker;
        int count = 1;

        EventLoopHolder(EventLoop worker) {
            this.worker = worker;
        }

        @Override
        public boolean equals(Object o) {
            if (this == o)
                return true;
            if (o == null || getClass() != o.getClass())
                return false;

            EventLoopHolder that = (EventLoopHolder) o;

            return worker != null ? worker.equals(that.worker) : that.worker == null;

        }

        @Override
        public int hashCode() {
            return worker != null ? worker.hashCode() : 0;
        }
    }

    private static final class EventLoopIterator implements Iterator<EventExecutor> {
        private final Iterator<EventLoopHolder> holderIt;

        public EventLoopIterator(Iterator<EventLoopHolder> holderIt) {
            this.holderIt = holderIt;
        }

        @Override
        public boolean hasNext() {
            return holderIt.hasNext();
        }

        @Override
        public EventExecutor next() {
            return holderIt.next().worker;
        }

        @Override
        public void remove() {
            throw new UnsupportedOperationException("read-only");
        }
    }
}