com.navercorp.nbasearc.gcp.Gateway Maven / Gradle / Ivy
/*
* Copyright 2015 NAVER Corp.
*
* 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.navercorp.nbasearc.gcp;
import static com.navercorp.nbasearc.gcp.Gateway.State.*;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.atomic.AtomicInteger;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.MoreExecutors;
import com.google.common.util.concurrent.SettableFuture;
class Gateway {
/**
* USED --(deleted)--> UNUSED
*
* USED --unuse--> UNUSED
*/
enum State {
USED, UNUSED
}
private static final Logger log = LoggerFactory.getLogger(Gateway.class);
private final Integer id;
private final String ip;
private final int port;
private final int concnt;
private final PhysicalConnection[] cons;
private State state;
private AtomicInteger active; // active connection count
Gateway(Integer id, String ip, int port, int concnt) {
this.id = id;
this.ip = ip;
this.port = port;
this.concnt = concnt;
this.cons = new PhysicalConnection[concnt];
this.state = USED;
this.active = new AtomicInteger();
}
ListenableFuture> init(SingleThreadEventLoopTrunk eventLoopTrunk, int reconnectInterval) {
final SettableFuture> sf = SettableFuture.create();
for (int i = 0; i < concnt; i++) {
cons[i] = PhysicalConnection.create(ip, port, eventLoopTrunk.roundrobinEventLoop(), this,
reconnectInterval);
final ListenableFuture> conFuture = cons[i].connect();
conFuture.addListener(new Runnable() {
@Override
public void run() {
try {
conFuture.get();
sf.set(null);
} catch (InterruptedException e) {
log.error("Exception occured while connecting to {}", Gateway.this, e);
} catch (ExecutionException e) {
log.error("Exception occured while connecting to {}", Gateway.this, e);
}
}
}, MoreExecutors.directExecutor());
}
return sf;
}
ListenableFuture> unuse(Set vcConcurrentSet) {
state = UNUSED;
final SettableFuture> sf = SettableFuture.create();
final AtomicInteger closedConCnt = new AtomicInteger(cons.length);
for (PhysicalConnection con : cons) {
con.unuse(vcConcurrentSet).addListener(new Runnable() {
@Override
public void run() {
if (closedConCnt.decrementAndGet() == 0) {
sf.set(null);
}
}
}, MoreExecutors.directExecutor());
}
return sf;
}
boolean isSafeToAbandone() {
for (PhysicalConnection con : cons) {
if (con.getState() != PhysicalConnection.State.UNUSED) {
return false;
}
}
return true;
}
@Override
public String toString() {
return new StringBuilder()
.append("[")
.append("gwid: ").append(id)
.append(", ip: ").append(ip)
.append(", port: ").append(port)
.append(", concnt: ").append(concnt)
.append(", state: ").append(getState())
.append(", active: ").append(active)
.append("]").toString();
}
Integer getId() {
return id;
}
PhysicalConnection bestPc() {
PhysicalConnection bestPc = null;
long bestCost = Long.MAX_VALUE;
for (PhysicalConnection pc : cons) {
if (pc.getState() == PhysicalConnection.State.CONNECTED) {
long cost = pc.busyCost();
if (bestCost > cost) {
bestPc = pc;
bestCost = cost;
}
}
}
return bestPc;
}
void decreaseActive() {
active.getAndDecrement();
log.info("decrease active {}, {}", active, this);
assert active.get() >= 0 : this + " active is zero or larger, active: " + active;
}
void increaseActive() {
active.incrementAndGet();
log.info("increase active {}, {}", active, this);
assert active.get() <= concnt :
this + " active is smaller than concnt, concnt: " + concnt + ", active: " + active;
}
int getActive() {
return active.get();
}
State getState() {
return state;
}
private static byte[] PING = "ping\r\n".getBytes();
void healthCheck() {
for (PhysicalConnection pc : cons) {
if (pc.getState() == PhysicalConnection.State.CONNECTED) {
pc.execute(Request.systemRequest(PING, pc));
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy