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

com.sproutsocial.nsq.NsqdInstance Maven / Gradle / Ivy

There is a newer version: 1.4.8
Show newest version
package com.sproutsocial.nsq;

import org.slf4j.Logger;

import java.io.IOException;
import java.util.concurrent.TimeUnit;

import static org.slf4j.LoggerFactory.getLogger;
import static com.sproutsocial.nsq.Util.*;

class NsqdInstance {
    private enum State {
        CONNECTED,
        NOT_CONNECTED,
        FAILED
    }

    private static final Logger LOGGER = getLogger(NsqdInstance.class);
    private final Publisher parent;
    private final BasePubSub basePubSub;
    HostAndPort hostAndPort;
    private PubConnection con = null;
    long failoverStart = 0;
    private volatile int failoverDurationSecs;
    private State currentState = State.NOT_CONNECTED;

    public NsqdInstance(String hostAndPort, Publisher parent, int failoverDurationSecs, BasePubSub basePubSub) {
        checkNotNull(hostAndPort);
        checkNotNull(parent);
        checkNotNull(basePubSub);
        this.hostAndPort = HostAndPort.fromString(hostAndPort).withDefaultPort(4150);
        this.parent = parent;
        this.failoverDurationSecs = failoverDurationSecs;
        this.basePubSub = basePubSub;
    }

    /**
     * @return true if this host is ready to receive data
     */
    protected synchronized boolean makeReady() {
        if (currentState == State.NOT_CONNECTED) {
            if (parent.isStopping) {
                throw new NSQException("publisher stopped");
            }
            return connectAttempt();
        } else if (currentState == State.FAILED && canAttemptRecovery()) {
            return connectAttempt();
        }
        return currentState == State.CONNECTED;
    }

    private boolean canAttemptRecovery() {
        return Util.clock() - failoverStart > TimeUnit.SECONDS.toMillis(failoverDurationSecs);
    }

    private boolean connectAttempt() {
        if (con != null) {
            con.close();
        }
        con = new PubConnection(basePubSub.getClient(), this.hostAndPort, parent);
        try {
            con.connect(basePubSub.getConfig());
            currentState = State.CONNECTED;
        } catch (IOException e) {
            markFailure();
            return false;
        }
        LOGGER.info("publisher connected:{}", hostAndPort);
        return true;
    }

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

    @Override
    public int hashCode() {
        return hostAndPort.hashCode();
    }

    public synchronized void markFailure() {
        Util.closeQuietly(con);
        con = null;
        currentState = State.FAILED;
        failoverStart = Util.clock();
        LOGGER.warn("Marking the connection to host {} as failed , will retry after {} seconds", hostAndPort, this.failoverDurationSecs);
    }

    public void setFailoverDurationSecs(int failoverDurationSecs) {
        this.failoverDurationSecs = failoverDurationSecs;
    }


    public PubConnection getCon() {
        return con;
    }

    public void clearConnection() {
        this.con = null;
        currentState = State.NOT_CONNECTED;
    }

    @Override
    public String toString() {
        return "NsqdInstance{" + "parent=" + parent +
                ", hostAndPort=" + hostAndPort +
                ", con=" + con +
                ", failoverStart=" + failoverStart +
                ", failoverDurationSecs=" + failoverDurationSecs +
                ", currentState=" + currentState +
                '}';
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy