io.nats.streaming.StreamingConnectionImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2015-2016 Apcera Inc. All rights reserved. This program and the accompanying
* materials are made available under the terms of the MIT License (MIT) which accompanies this
* distribution, and is available at http://opensource.org/licenses/MIT
*/
package io.nats.streaming;
import static io.nats.streaming.NatsStreaming.ERR_CONNECTION_REQ_TIMEOUT;
import static io.nats.streaming.NatsStreaming.ERR_SUB_REQ_TIMEOUT;
import com.google.protobuf.ByteString;
import com.google.protobuf.InvalidProtocolBufferException;
import io.nats.client.Message;
import io.nats.client.MessageHandler;
import io.nats.client.NUID;
import io.nats.client.Nats;
import io.nats.streaming.protobuf.Ack;
import io.nats.streaming.protobuf.CloseRequest;
import io.nats.streaming.protobuf.CloseResponse;
import io.nats.streaming.protobuf.ConnectRequest;
import io.nats.streaming.protobuf.ConnectResponse;
import io.nats.streaming.protobuf.MsgProto;
import io.nats.streaming.protobuf.PubAck;
import io.nats.streaming.protobuf.PubMsg;
import io.nats.streaming.protobuf.SubscriptionRequest;
import io.nats.streaming.protobuf.SubscriptionResponse;
import java.io.IOException;
import java.time.Duration;
import java.time.Instant;
import java.time.temporal.ChronoUnit;
import java.util.HashMap;
import java.util.Map;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
class StreamingConnectionImpl implements StreamingConnection, io.nats.client.MessageHandler {
static final String ERR_MANUAL_ACK = NatsStreaming.PFX + "cannot manually ack in auto-ack mode";
private final ReadWriteLock mu = new ReentrantReadWriteLock();
private String clientId;
private String clusterId;
String pubPrefix; // Publish prefix set by streaming, append our subject.
String subRequests; // Subject to send subscription requests.
String unsubRequests; // Subject to send unsubscribe requests.
String subCloseRequests; // Subject to send subscription close requests.
String closeRequests; // Subject to send close requests.
String ackSubject; // publish acks
private io.nats.client.Subscription ackSubscription;
io.nats.client.Subscription hbSubscription;
io.nats.client.MessageHandler hbCallback;
Map subMap;
Map pubAckMap;
private BlockingQueue pubAckChan;
Options opts;
io.nats.client.Connection nc;
final Timer ackTimer = new Timer(true);
boolean ncOwned = false;
ExecutorService exec = Executors.newCachedThreadPool();
StreamingConnectionImpl() {
}
StreamingConnectionImpl(String stanClusterId, String clientId) {
this(stanClusterId, clientId, null);
}
StreamingConnectionImpl(String stanClusterId, String clientId, Options opts) {
this.clusterId = stanClusterId;
this.clientId = clientId;
if (opts == null) {
this.opts = new Options.Builder().build();
} else {
this.opts = opts;
// Check if the user has provided a connection as an option
if (this.opts.getNatsConn() != null) {
setNatsConnection(this.opts.getNatsConn());
}
}
}
// Connect will form a connection to the STAN subsystem.
StreamingConnectionImpl connect() throws IOException, InterruptedException {
boolean exThrown = false;
io.nats.client.Connection nc = getNatsConnection();
// Create a NATS connection if it doesn't exist
if (nc == null) {
nc = createNatsConnection();
setNatsConnection(nc);
ncOwned = true;
} else if (!nc.isConnected()) {
// Bail if the custom NATS connection is disconnected
throw new IOException(NatsStreaming.ERR_BAD_CONNECTION);
}
try {
// Create a heartbeat inbox
String hbInbox = nc.newInbox();
hbCallback = msg -> processHeartBeat(msg);
hbSubscription = nc.subscribe(hbInbox, hbCallback);
// Send Request to discover the cluster
String discoverSubject = String.format("%s.%s", opts.getDiscoverPrefix(), clusterId);
ConnectRequest req = ConnectRequest.newBuilder().setClientID(clientId)
.setHeartbeatInbox(hbInbox).build();
byte[] bytes = req.toByteArray();
Message reply;
reply = nc.request(discoverSubject, bytes, opts.getConnectTimeout().toMillis());
if (reply == null) {
throw new IOException(ERR_CONNECTION_REQ_TIMEOUT);
}
ConnectResponse cr = ConnectResponse.parseFrom(reply.getData());
if (!cr.getError().isEmpty()) {
// This is already a properly formatted streaming error message
// (StreamingConnectionImpl.SERVER_ERR_INVALID_CLIENT)
throw new IOException(cr.getError());
}
// Capture cluster configuration endpoints to publish and
// subscribe/unsubscribe.
pubPrefix = cr.getPubPrefix();
subRequests = cr.getSubRequests();
unsubRequests = cr.getUnsubRequests();
subCloseRequests = cr.getSubCloseRequests();
closeRequests = cr.getCloseRequests();
// Setup the ACK subscription
ackSubject = String.format("%s.%s", NatsStreaming.DEFAULT_ACK_PREFIX, NUID.nextGlobal()
);
ackSubscription = nc.subscribe(ackSubject, msg -> processAck(msg));
ackSubscription.setPendingLimits(1024 ^ 2, 32 * 1024 ^ 2);
pubAckMap = new HashMap<>();
// Create Subscription map
subMap = new HashMap<>();
pubAckChan = new LinkedBlockingQueue<>(opts.getMaxPubAcksInFlight());
} catch (IOException e) {
exThrown = true;
if (io.nats.client.Nats.ERR_TIMEOUT.equals(e.getMessage())) {
throw new IOException(ERR_CONNECTION_REQ_TIMEOUT, e);
} else {
throw e;
}
} finally {
if (exThrown) {
try {
close();
} catch (Exception e) {
/* NOOP -- can't do anything if close fails */
}
}
}
return this;
}
io.nats.client.Connection createNatsConnection() throws IOException {
// Create a NATS connection if it doesn't exist
io.nats.client.Connection nc = null;
if (getNatsConnection() == null) {
if (opts.getNatsUrl() != null) {
io.nats.client.Options natsOpts = new io.nats.client.Options.Builder()
.name(clientId).build();
nc = Nats.connect(opts.getNatsUrl(), natsOpts);
} else {
nc = Nats.connect();
}
ncOwned = true;
}
return nc;
}
@Override
public void close() throws IOException, InterruptedException {
io.nats.client.Connection nc;
this.lock();
try {
// Capture for NATS calls below
if (getNatsConnection() == null) {
// We are already closed
return;
}
// Capture for NATS calls below.
nc = getNatsConnection();
// if ncOwned, we close it at the end
try {
// Signals we are closed.
setNatsConnection(null);
// Now close ourselves.
if (getAckSubscription() != null) {
try {
getAckSubscription().unsubscribe();
} catch (Exception e) {
// ignore
}
}
if (getHbSubscription() != null) {
try {
getHbSubscription().unsubscribe();
} catch (Exception e) {
// ignore
}
}
CloseRequest req = CloseRequest.newBuilder().setClientID(clientId).build();
byte[] bytes = req.toByteArray();
Message reply;
reply = nc.request(closeRequests, bytes, opts.getConnectTimeout().toMillis());
if (reply == null) {
throw new IOException(NatsStreaming.ERR_CLOSE_REQ_TIMEOUT);
}
if (reply.getData() != null) {
CloseResponse cr = CloseResponse.parseFrom(reply.getData());
if (!cr.getError().isEmpty()) {
throw new IOException(cr.getError());
}
}
} finally {
if (ncOwned) {
try {
nc.close();
} catch (Exception ignore) {
// ignore
}
}
} // first finally
} finally {
this.unlock();
}
}
AckClosure createAckClosure(AckHandler ah, BlockingQueue ch) {
return new AckClosure(ah, ch);
}
private SubscriptionImpl createSubscription(String subject, String qgroup,
io.nats.streaming.MessageHandler cb,
StreamingConnectionImpl conn,
SubscriptionOptions opts) {
return new SubscriptionImpl(subject, qgroup, cb, conn, opts);
}
void processHeartBeat(Message msg) {
// No payload assumed, just reply.
io.nats.client.Connection nc;
this.rLock();
nc = this.nc;
this.rUnlock();
if (nc != null) {
try {
nc.publish(msg.getReplyTo(), null);
} catch (IOException e) {
// ignore exception; nothing we can do.
}
}
}
BlockingQueue createErrorChannel() {
return new LinkedBlockingQueue<>();
}
// Publish will publish to the cluster and wait for an ACK.
@Override
public void publish(String subject, byte[] data) throws IOException, InterruptedException {
final BlockingQueue ch = createErrorChannel();
publish(subject, data, null, ch);
String err;
try {
err = ch.take();
if (!err.isEmpty()) {
throw new IOException(err);
}
} catch (InterruptedException e) {
// TODO: ignore for now, but re-evaluatate this
}
}
/*
* PublishAsync will publish to the cluster on pubPrefix+subject and asynchronously process the
* ACK or error state. It will return the GUID for the message being sent.
*/
@Override
public String publish(String subject, byte[] data, AckHandler ah) throws IOException,
InterruptedException {
return publish(subject, data, ah, null);
}
private String publish(String subject, byte[] data, AckHandler ah, BlockingQueue ch)
throws IOException, InterruptedException {
String subj;
String ackSubject;
Duration ackTimeout;
BlockingQueue pac;
final AckClosure a;
final PubMsg pe;
String guid;
byte[] bytes;
a = createAckClosure(ah, ch);
this.lock();
try {
if (getNatsConnection() == null) {
throw new IllegalStateException(NatsStreaming.ERR_CONNECTION_CLOSED);
}
subj = pubPrefix + "." + subject;
guid = NUID.nextGlobal();
PubMsg.Builder pb =
PubMsg.newBuilder().setClientID(clientId).setGuid(guid).setSubject(subject);
if (data != null) {
pb = pb.setData(ByteString.copyFrom(data));
}
pe = pb.build();
bytes = pe.toByteArray();
// Map ack to guid
pubAckMap.put(guid, a);
// snapshot
ackSubject = this.ackSubject;
ackTimeout = opts.getAckTimeout();
pac = pubAckChan;
} finally {
this.unlock();
}
// Use the buffered channel to control the number of outstanding acks.
try {
pac.put(PubAck.getDefaultInstance());
} catch (InterruptedException e) {
// TODO: Reevaluate this.
// Eat this because you can't really do anything with it
}
try {
nc.publish(subj, ackSubject, bytes, true);
} catch (IOException e) {
removeAck(guid);
throw (e);
}
// Setup the timer for expiration.
this.lock();
try {
a.ackTask = createAckTimerTask(guid);
ackTimer.schedule(a.ackTask, ackTimeout.toMillis());
} finally {
this.unlock();
}
return guid;
}
@Override
public Subscription subscribe(String subject, io.nats.streaming.MessageHandler cb)
throws IOException, InterruptedException {
return subscribe(subject, cb, null);
}
@Override
public Subscription subscribe(String subject, io.nats.streaming.MessageHandler cb,
SubscriptionOptions opts) throws IOException,
InterruptedException {
return subscribe(subject, null, cb, opts);
}
@Override
public Subscription subscribe(String subject, String queue, io.nats.streaming.MessageHandler cb)
throws IOException, InterruptedException {
return subscribe(subject, queue, cb, null);
}
@Override
public Subscription subscribe(String subject, String queue, io.nats.streaming.MessageHandler cb,
SubscriptionOptions opts) throws IOException,
InterruptedException {
SubscriptionImpl sub;
io.nats.client.Connection nc;
this.lock();
try {
if (getNatsConnection() == null) {
throw new IllegalStateException(NatsStreaming.ERR_CONNECTION_CLOSED);
}
sub = createSubscription(subject, queue, cb, this, opts);
// Register subscription.
subMap.put(sub.inbox, sub);
nc = getNatsConnection();
} finally {
this.unlock();
}
// Hold lock throughout.
sub.wLock();
try {
// Listen for actual messages
sub.inboxSub = nc.subscribe(sub.inbox, this);
// Create a subscription request
// FIXME(dlc) add others.
SubscriptionRequest sr = createSubscriptionRequest(sub);
Message reply;
reply = nc.request(subRequests, sr.toByteArray(), 2L, TimeUnit.SECONDS);
if (reply == null) {
sub.inboxSub.unsubscribe();
throw new IOException(ERR_SUB_REQ_TIMEOUT);
}
SubscriptionResponse response;
try {
response = SubscriptionResponse.parseFrom(reply.getData());
} catch (InvalidProtocolBufferException e) {
sub.inboxSub.unsubscribe();
throw e;
}
if (!response.getError().isEmpty()) {
sub.inboxSub.unsubscribe();
throw new IOException(response.getError());
}
sub.setAckInbox(response.getAckInbox());
} finally {
sub.wUnlock();
}
return sub;
}
SubscriptionRequest createSubscriptionRequest(SubscriptionImpl sub) {
SubscriptionOptions subOpts = sub.getOptions();
SubscriptionRequest.Builder srb = SubscriptionRequest.newBuilder();
String clientId = sub.getConnection().getClientId();
String queue = sub.getQueue();
String subject = sub.getSubject();
srb.setClientID(clientId).setSubject(subject).setQGroup(queue == null ? "" : queue)
.setInbox(sub.getInbox()).setMaxInFlight(subOpts.getMaxInFlight())
.setAckWaitInSecs((int) subOpts.getAckWait().getSeconds());
switch (subOpts.getStartAt()) {
case First:
break;
case LastReceived:
break;
case NewOnly:
break;
case SequenceStart:
srb.setStartSequence(subOpts.getStartSequence());
break;
case TimeDeltaStart:
long delta = ChronoUnit.NANOS.between(subOpts.getStartTime(), Instant.now());
srb.setStartTimeDelta(delta);
break;
case UNRECOGNIZED:
default:
break;
}
srb.setStartPosition(subOpts.getStartAt());
if (subOpts.getDurableName() != null) {
srb.setDurableName(subOpts.getDurableName());
}
return srb.build();
}
// Process an ack from the STAN cluster
void processAck(Message msg) {
PubAck pa;
Exception ex = null;
try {
pa = PubAck.parseFrom(msg.getData());
} catch (InvalidProtocolBufferException e) {
// If we are speaking to a server we don't understand, let the
// user know.
System.err.println("Protocol error: " + e.getStackTrace());
return;
}
// Remove
AckClosure ackClosure = removeAck(pa.getGuid());
if (ackClosure != null) {
// Capture error if it exists.
String ackError = pa.getError();
if (ackClosure.ah != null) {
if (!ackError.isEmpty()) {
ex = new IOException(ackError);
}
// Perform the ackHandler callback
ackClosure.ah.onAck(pa.getGuid(), ex);
} else if (ackClosure.ch != null) {
try {
ackClosure.ch.put(ackError);
} catch (InterruptedException e) {
// ignore
}
}
}
}
TimerTask createAckTimerTask(String guid) {
return new TimerTask() {
public void run() {
try {
processAckTimeout(guid);
} catch (Exception e) {
// catch exception to prevent the timer to be closed, but cancel this task
cancel();
}
}
};
}
void processAckTimeout(String guid) {
AckClosure ackClosure = removeAck(guid);
if (ackClosure == null) {
return;
}
if (ackClosure.ah != null) {
ackClosure.ah.onAck(guid, new TimeoutException(NatsStreaming.ERR_TIMEOUT));
}
}
AckClosure removeAck(String guid) {
AckClosure ackClosure;
BlockingQueue pac;
TimerTask timerTask = null;
this.lock();
try {
ackClosure = pubAckMap.get(guid);
if (ackClosure != null) {
timerTask = ackClosure.ackTask;
pubAckMap.remove(guid);
}
pac = pubAckChan;
} finally {
this.unlock();
}
// Cancel timer if needed
if (timerTask != null) {
timerTask.cancel();
}
// Remove from channel to unblock async publish
if (ackClosure != null && pac.size() > 0) {
try {
// remove from queue to unblock publish
pac.take();
} catch (InterruptedException e) {
// TODO: Ignore, but re-evaluate this
}
}
return ackClosure;
}
@Override
public void onMessage(io.nats.client.Message msg) {
// For handling inbound NATS messages
processMsg(msg);
}
io.nats.streaming.Message createStanMessage(MsgProto msgp) {
return new io.nats.streaming.Message(msgp);
}
void processMsg(io.nats.client.Message raw) {
io.nats.streaming.Message stanMsg = null;
boolean isClosed;
SubscriptionImpl sub;
io.nats.client.Connection nc;
try {
MsgProto msgp = MsgProto.parseFrom(raw.getData());
stanMsg = createStanMessage(msgp);
} catch (InvalidProtocolBufferException e) {
// TODO: Ignore, but re-evaluate this
}
// Lookup the subscription
lock();
try {
nc = getNatsConnection();
isClosed = (nc == null);
sub = (SubscriptionImpl) subMap.get(raw.getSubject());
} finally {
unlock();
}
// Check if sub is no longer valid or connection has been closed.
if (sub == null || isClosed) {
return;
}
// Store in msg for backlink
stanMsg.setSubscription(sub);
io.nats.streaming.MessageHandler cb;
String ackSubject;
boolean isManualAck;
StreamingConnectionImpl subsc;
sub.rLock();
try {
cb = sub.getMessageHandler();
ackSubject = sub.getAckInbox();
isManualAck = sub.getOptions().isManualAcks();
subsc = sub.getConnection(); // Can be nil if sub has been unsubscribed
} finally {
sub.rUnlock();
}
// Perform the callback
if (cb != null && subsc != null) {
cb.onMessage(stanMsg);
}
// Process auto-ack
if (!isManualAck) {
Ack ack = Ack.newBuilder().setSubject(stanMsg.getSubject())
.setSequence(stanMsg.getSequence()).build();
try {
nc.publish(ackSubject, ack.toByteArray());
} catch (IOException e) {
// FIXME(dlc) - Async error handler? Retry?
// This really won't happen since the publish is executing in the NATS thread.
}
}
}
public String getClientId() {
return this.clientId;
}
@Override
public io.nats.client.Connection getNatsConnection() {
return this.nc;
}
private void setNatsConnection(io.nats.client.Connection nc) {
this.nc = nc;
}
public String newInbox() {
return nc.newInbox();
}
void lock() {
mu.writeLock().lock();
}
void unlock() {
mu.writeLock().unlock();
}
private void rLock() {
mu.readLock().lock();
}
private void rUnlock() {
mu.readLock().unlock();
}
io.nats.client.Subscription getAckSubscription() {
return this.ackSubscription;
}
io.nats.client.Subscription getHbSubscription() {
return this.hbSubscription;
}
// test injection setter/getters
void setPubAckChan(BlockingQueue ch) {
this.pubAckChan = ch;
}
BlockingQueue getPubAckChan() {
return pubAckChan;
}
void setPubAckMap(Map map) {
this.pubAckMap = map;
}
Map getPubAckMap() {
return pubAckMap;
}
void setSubMap(Map map) {
this.subMap = map;
}
Map getSubMap() {
return subMap;
}
class AckClosure {
TimerTask ackTask;
AckHandler ah;
BlockingQueue ch;
AckClosure(final AckHandler ah, final BlockingQueue ch) {
this.ah = ah;
this.ch = ch;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy