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

reactor.io.net.impl.zmq.ZeroMQChannelStream Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2011-2014 Pivotal Software, Inc.
 *
 *  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 reactor.io.net.impl.zmq;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import org.zeromq.ZFrame;
import org.zeromq.ZMQ;
import org.zeromq.ZMsg;
import reactor.Environment;
import reactor.core.Dispatcher;
import reactor.core.processor.CancelException;
import reactor.core.support.Exceptions;
import reactor.fn.Consumer;
import reactor.io.buffer.Buffer;
import reactor.io.codec.Codec;
import reactor.io.net.ChannelStream;
import reactor.rx.action.support.DefaultSubscriber;
import reactor.rx.broadcast.Broadcaster;
import reactor.rx.subscription.PushSubscription;

import java.net.InetSocketAddress;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;

/**
 * @author Jon Brisbin
 * @author Stephane Maldini
 */
public class ZeroMQChannelStream extends ChannelStream {

	private final ZeroMQConsumerSpec eventSpec = new ZeroMQConsumerSpec();
	private final InetSocketAddress remoteAddress;

	private volatile String     connectionId;
	private volatile ZMQ.Socket socket;

	private Subscriber inputSub;

	public ZeroMQChannelStream(Environment env,
	                           long prefetch,
	                           Dispatcher eventsDispatcher,
	                           InetSocketAddress remoteAddress,
	                           Codec codec) {
		super(env, codec, prefetch, eventsDispatcher);
		this.remoteAddress = remoteAddress;
	}

	@Override
	protected void doSubscribeWriter(Publisher writer, final Subscriber postWriter) {
		writer.subscribe(new DefaultSubscriber() {

			ZMsg currentMsg;

			@Override
			public void onSubscribe(final Subscription subscription) {
				eventSpec.close(new Consumer() {
					@Override
					public void accept(Void aVoid) {
						subscription.cancel();
					}
				});
				subscription.request(Long.MAX_VALUE);
				postWriter.onSubscribe(Broadcaster.HOT_SUBSCRIPTION);
			}

			@Override
			public void onNext(OUT out) {
				final ByteBuffer data;
				if (Buffer.class.isAssignableFrom(out.getClass())) {
					data = ((Buffer) out).byteBuffer();
				} else if (getEncoder() != null) {
					data = getEncoder().apply(out).byteBuffer();
				} else {
					postWriter.onError(
							Exceptions.addValueAsLastCause(new IllegalArgumentException("Data cannot be encoded"), out));
					return;
				}

				byte[] bytes = new byte[data.remaining()];
				data.get(bytes);
				boolean isNewMsg;
				ZMsg msg;
				msg = currentMsg;
				currentMsg = new ZMsg();
				if (msg == null) {
					msg = currentMsg;
					isNewMsg = true;
				} else {
					isNewMsg = false;
				}

				if (isNewMsg) {
					switch (socket.getType()) {
						case ZMQ.ROUTER:
							msg.add(new ZFrame(connectionId));
							break;
						default:
					}
				}
				msg.add(new ZFrame(bytes));
			}

			@Override
			public void onError(Throwable throwable) {
				doFlush(currentMsg, null);
				postWriter.onError(throwable);
			}

			@Override
			public void onComplete() {
				doFlush(currentMsg, postWriter);
			}
		});
	}

	@Override
	public void doDecoded(IN in) {
		try {
			if (inputSub != null) {
				inputSub.onNext(in);
			}
		} catch (CancelException ce) {
			//IGNORE
		}
	}

	@Override
	public void subscribe(Subscriber subscriber) {
		if (subscriber == null) {
			throw new IllegalStateException("Input Subscriber cannot be null");
		}
		synchronized (this) {
			if (inputSub != null) return;
			inputSub = subscriber;
		}

		inputSub.onSubscribe(new PushSubscription(null, inputSub));
	}

	public ZeroMQChannelStream setConnectionId(String connectionId) {
		this.connectionId = connectionId;
		return this;
	}

	public ZeroMQChannelStream setSocket(ZMQ.Socket socket) {
		this.socket = socket;
		return this;
	}

	@Override
	public InetSocketAddress remoteAddress() {
		return remoteAddress;
	}

	private void doFlush(ZMsg msg, final Subscriber onComplete) {
		if (null != msg) {
			boolean success = msg.send(socket);
			if (null != onComplete) {
				if (success) {
					onComplete.onComplete();
				} else {
					onComplete.onError(new RuntimeException("ZeroMQ Message could not be sent"));
				}
			}
		}
	}

	public void close() {
		getDispatcher().dispatch(null, new Consumer() {
			@Override
			public void accept(Void v) {
				try {
					final List> closeHandlers;
					synchronized (eventSpec.closeHandlers) {
						closeHandlers = new ArrayList>(eventSpec.closeHandlers);
					}

					for (Consumer r : closeHandlers) {
						r.accept(null);
					}
				} catch (Throwable t) {
					if (inputSub != null) {
						inputSub.onError(t);
					}
				}
			}
		}, null);
	}

	@Override
	public ConsumerSpec on() {
		return eventSpec;
	}


	@Override
	public ZMQ.Socket delegate() {
		return socket;
	}

	@Override
	public String toString() {
		return "ZeroMQNetChannel{" +
				"closeHandlers=" + eventSpec.closeHandlers +
				", connectionId='" + connectionId + '\'' +
				", socket=" + socket +
				'}';
	}

	private static class ZeroMQConsumerSpec implements ConsumerSpec {

		final List> closeHandlers = new ArrayList<>();

		@Override
		public ConsumerSpec close(Consumer onClose) {
			synchronized (closeHandlers) {
				closeHandlers.add(onClose);
			}
			return this;
		}

		@Override
		public ConsumerSpec readIdle(long idleTimeout, Consumer onReadIdle) {
			return this;
		}

		@Override
		public ConsumerSpec writeIdle(long idleTimeout, Consumer onWriteIdle) {
			return this;
		}
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy