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

com.firefly.net.tcp.flex.client.FlexConnectionManager Maven / Gradle / Ivy

There is a newer version: 5.0.0-dev6
Show newest version
package com.firefly.net.tcp.flex.client;

import com.firefly.net.tcp.codec.flex.stream.FlexConnection;
import com.firefly.net.tcp.flex.exception.ConnectionException;
import com.firefly.utils.Assert;
import com.firefly.utils.CollectionUtils;
import com.firefly.utils.concurrent.Scheduler;
import com.firefly.utils.concurrent.Schedulers;
import com.firefly.utils.lang.AbstractLifeCycle;
import com.firefly.utils.lang.HostPort;
import com.firefly.utils.retry.RetryTaskBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.stream.Collectors;

import static com.firefly.utils.retry.RetryStrategies.ifException;
import static com.firefly.utils.retry.RetryStrategies.ifResult;
import static com.firefly.utils.retry.StopStrategies.afterExecute;
import static com.firefly.utils.retry.WaitStrategies.exponentialWait;

/**
 * @author Pengtao Qiu
 */
public class FlexConnectionManager extends AbstractLifeCycle {

    protected static final Logger log = LoggerFactory.getLogger("firefly-system");

    private final Map connectionMap = new HashMap<>();
    private final MultiplexingClient client;
    private final AtomicInteger index = new AtomicInteger(0);
    private final Scheduler scheduler = Schedulers.createScheduler();
    private final AddressProvider addressProvider;
    private volatile List activatedList;

    public FlexConnectionManager(MultiplexingClient client, AddressProvider addressProvider) {
        Assert.notNull(addressProvider);
        Assert.notNull(client);

        this.client = client;
        this.addressProvider = addressProvider;
        this.activatedList = convert(addressProvider.getAddressList());
        Assert.notEmpty(activatedList, "The address list is empty");
        start();
    }

    public List getActivatedList() {
        return activatedList;
    }

    public void updateActivatedList(Set activatedList) {
        this.activatedList = convert(activatedList);
    }

    public FlexConnection getConnection() {
        FlexConnection ret = RetryTaskBuilder.newTask()
                .retry(ifResult(Objects::isNull))
                .stop(afterExecute(activatedList.size()))
                .wait(exponentialWait(10, TimeUnit.MILLISECONDS))
                .task(() -> {
                    int i = Math.abs(index.getAndAdd(1)) % activatedList.size();
                    return getConnection(activatedList.get(i));
                })
                .call();
        if (ret == null) {
            throw new ConnectionException("Can not get connection");
        }
        return ret;
    }

    private synchronized FlexConnection getConnection(HostPort hostPort) {
        try {
            FlexConnection connection = connectionMap.get(hostPort);
            if (connection != null) {
                if (connection.isOpen()) {
                    return connection;
                } else {
                    FlexConnection newConnection = createConnection(hostPort);
                    if (newConnection != null) {
                        connectionMap.put(hostPort, newConnection);
                        return newConnection;
                    } else {
                        connectionMap.remove(hostPort);
                        updateActivatedList();
                        return null;
                    }
                }
            } else {
                FlexConnection newConnection = createConnection(hostPort);
                if (newConnection != null) {
                    connectionMap.put(hostPort, newConnection);
                    updateActivatedList();
                    return newConnection;
                } else {
                    connectionMap.remove(hostPort);
                    updateActivatedList();
                    return null;
                }
            }
        } catch (Exception e) {
            log.error("get connection exception", e);
            updateActivatedList();
            return null;
        }
    }

    private FlexConnection createConnection(HostPort hostPort) {
        return RetryTaskBuilder.newTask()
                .retry(ifException(ex -> ex != null && ex.getCause() instanceof TimeoutException))
                .stop(afterExecute(10))
                .wait(exponentialWait(10, TimeUnit.MILLISECONDS))
                .task(() -> this.connect(hostPort))
                .call();
    }

    private FlexConnection connect(HostPort hostPort) {
        try {
            return client.connect(hostPort.getHost(), hostPort.getPort()).get(5, TimeUnit.SECONDS);
        } catch (Exception e) {
            log.error("Connect " + hostPort + " exception", e);
            throw new ConnectionException("connection exception", e);
        }
    }

    private List convert(Set urls) {
        return Collections.unmodifiableList(
                urls.stream().map(HostPort::new)
                    .sorted(Comparator.comparing(HostPort::toString))
                    .collect(Collectors.toList()));
    }

    private void updateActivatedList() {
        activatedList = Collections.unmodifiableList(new ArrayList<>(connectionMap.keySet()));
    }

    @Override
    protected void init() {
        if (!CollectionUtils.isEmpty(activatedList)) {
            activatedList.forEach(hostPort -> {
                try {
                    FlexConnection connection = createConnection(hostPort);
                    if (connection != null) {
                        connectionMap.put(hostPort, connection);
                    }
                } catch (Exception e) {
                    log.error("Connect " + hostPort + " exception", e);
                }
            });
            updateActivatedList();
        }
        scheduler.scheduleWithFixedDelay(() -> {
            log.info("Client current activated address list: {}", activatedList);
            activatedList = convert(addressProvider.getAddressList());
        }, 5, 5, TimeUnit.SECONDS);
    }

    @Override
    protected void destroy() {
        scheduler.stop();
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy