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

org.osgi.util.pushstream.SimplePushEventSourceImpl Maven / Gradle / Ivy

There is a newer version: 1.1.0
Show newest version
/*
 * Copyright (c) OSGi Alliance (2015, 2017). All Rights Reserved.
 * 
 * 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 org.osgi.util.pushstream;

import static java.util.Collections.emptyList;
import static java.util.concurrent.TimeUnit.NANOSECONDS;
import static java.util.stream.Collectors.toList;

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.Semaphore;

import org.osgi.util.promise.Deferred;
import org.osgi.util.promise.Promise;
import org.osgi.util.promise.PromiseFactory;

class SimplePushEventSourceImpl>>
		implements SimplePushEventSource {
	
	private final Object								lock		= new Object();

	private final PromiseFactory						promiseFactory;
	private final PromiseFactory						sameThread;

	private final QueuePolicy						queuePolicy;

	private final U										queue;

	private final int									parallelism;

	private final Semaphore								semaphore;

	private final List>	connected	= new ArrayList<>();

	private final Runnable								onClose;

	private boolean										closed;
	
	private Deferred								connectPromise;

	private boolean										waitForFinishes;


	public SimplePushEventSourceImpl(PromiseFactory promiseFactory,
			QueuePolicy queuePolicy,
			U queue, int parallelism, Runnable onClose) {
		this.promiseFactory = promiseFactory;
		this.sameThread = new PromiseFactory(
				PromiseFactory.inlineExecutor(),
				promiseFactory.scheduledExecutor());
		this.queuePolicy = queuePolicy;
		this.queue = queue;
		this.parallelism = parallelism;
		this.semaphore = new Semaphore(parallelism);
		this.onClose = onClose;
		this.closed = false;
		this.connectPromise = null;
	}

	@Override
	public AutoCloseable open(PushEventConsumer< ? super T> pec)
			throws Exception {
		Deferred toResolve = null;
		synchronized (lock) {
			if (closed) {
				throw new IllegalStateException(
						"This PushEventConsumer is closed");
			}

			toResolve = connectPromise;
			connectPromise = null;

			connected.add(pec);
		}

		if (toResolve != null) {
			toResolve.resolve(null);
		}

		return () -> {
			closeConsumer(pec, PushEvent.close());
		};
	}

	private void closeConsumer(PushEventConsumer< ? super T> pec,
			PushEvent event) {
		boolean sendClose;
		synchronized (lock) {
			sendClose = connected.remove(pec);
		}
		if (sendClose) {
			doSend(pec, event);
		}
	}

	private void doSend(PushEventConsumer< ? super T> pec, PushEvent event) {
		try {
			promiseFactory.executor().execute(() -> safePush(pec, event));
		} catch (RejectedExecutionException ree) {
			// TODO log?
			if (!event.isTerminal()) {
				close(PushEvent.error(ree));
			} else {
				safePush(pec, event);
			}
		}
	}

	private Promise doSendWithBackPressure(
			PushEventConsumer< ? super T> pec, PushEvent event) {
		Deferred d = sameThread.deferred();
		try {
			promiseFactory.executor().execute(
					() -> d.resolve(Long.valueOf(
							System.nanoTime() + safePush(pec, event))));
		} catch (RejectedExecutionException ree) {
			// TODO log?

			if (!event.isTerminal()) {
				close(PushEvent.error(ree));
				d.resolve(Long.valueOf(System.nanoTime()));
			} else {
				d.resolve(
						Long.valueOf(System.nanoTime() + safePush(pec, event)));
			}
		}
		return d.getPromise();
	}

	private long safePush(PushEventConsumer< ? super T> pec,
			PushEvent event) {
		try {
			long backpressure = pec.accept(event) * 1000000;
			if (backpressure < 0 && !event.isTerminal()) {
				closeConsumer(pec, PushEvent.close());
				return -1;
			}
			return event.isTerminal() ? -1 : backpressure;
		} catch (Exception e) {
			// TODO log?
			if (!event.isTerminal()) {
				closeConsumer(pec, PushEvent.error(e));
			}
			return -1;
		}
	}

	@Override
	public void close() {
		close(PushEvent.close());
	}

	private void close(PushEvent event) {
		List> toClose;
		Deferred toFail = null;
		synchronized (lock) {
			if(!closed) {
				closed = true;
				
				toClose = new ArrayList<>(connected);
				connected.clear();
				queue.clear();

				if(connectPromise != null) {
					toFail = connectPromise;
					connectPromise = null;
				}
			} else {
				toClose = emptyList();
			}
		}

		toClose.stream().forEach(pec -> doSend(pec, event));

		if (toFail != null) {
			toFail.resolveWith(closedConnectPromise());
		}

		onClose.run();
	}

	@Override
	public void publish(T t) {
		enqueueEvent(PushEvent.data(t));
	}

	@Override
	public void endOfStream() {
		enqueueEvent(PushEvent.close());
	}

	@Override
	public void error(Throwable t) {
		enqueueEvent(PushEvent.error(t));
	}

	private void enqueueEvent(PushEvent event) {
		synchronized (lock) {
			if (closed || connected.isEmpty()) {
				return;
			}
		}

		try {
			queuePolicy.doOffer(queue, event);
			boolean start;
			synchronized (lock) {
				start = !waitForFinishes && semaphore.tryAcquire();
			}
			if (start) {
				startWorker();
			}
		} catch (Exception e) {
			close(PushEvent.error(e));
			throw new IllegalStateException(
					"The queue policy threw an exception", e);
		}
	}

	@SuppressWarnings({
			"unchecked", "boxing"
	})
	private void startWorker() {
		promiseFactory.executor().execute(() -> {
			try {
				
				for(;;) {
					PushEvent event;
					List> toCall;
					boolean resetWait;
					synchronized (lock) {
						if(waitForFinishes) {
							semaphore.release();
							while(waitForFinishes) {
								lock.notifyAll();
								lock.wait();
							}
							semaphore.acquire();
						}

						event = (PushEvent) queue.poll();
						
						if(event == null) {
							break;
						}

						if (connected.isEmpty()) {
							queue.clear();
							break;
						}

						toCall = new ArrayList<>(connected);
						if (event.isTerminal()) {
							waitForFinishes = true;
							resetWait = true;
							connected.clear();
							while (!semaphore.tryAcquire(parallelism - 1)) {
								lock.wait();
							}
						} else {
							resetWait = false;
						}
					}
					
					Promise backPressure = deliver(toCall, event);
					
					if (backPressure.isDone()) {
						handleReset(resetWait);

						long toWait = backPressure.getValue()
								- System.nanoTime();

						if (toWait > 0) {
							promiseFactory.scheduledExecutor().schedule(
									this::startWorker, toWait,
									NANOSECONDS);
							return;
						}
					} else {
						backPressure.then(p -> {
							handleReset(resetWait);
							long toWait = p.getValue() - System.nanoTime();

							if (toWait > 0) {
								promiseFactory.scheduledExecutor().schedule(
										this::startWorker, toWait,
										NANOSECONDS);
							} else {
								startWorker();
							}
							return p;
						}, p -> close(
								PushEvent.error(p.getFailure())));
						return;
					}
				}

				semaphore.release();
			} catch (Exception e) {
				close(PushEvent.error(e));
			}
			if (queue.peek() != null && semaphore.tryAcquire()) {
				try {
					startWorker();
				} catch (Exception e) {
					close(PushEvent.error(e));
				}
			}
		});

	}

	private void handleReset(boolean resetWait) {
		if (resetWait == true) {
			synchronized (lock) {
				waitForFinishes = false;
				lock.notifyAll();
			}
		}
	}

	private Promise deliver(List> toCall,
			PushEvent event) {
		if (toCall.size() == 1) {
			return doCall(event, toCall.get(0));
		} else {
			List> calls = toCall.stream().map(pec -> {
				if (semaphore.tryAcquire()) {
					return doSendWithBackPressure(pec, event)
							.onResolve(() -> semaphore.release());
				} else {
					return doCall(event, pec);
				}
			}).collect(toList());
			return sameThread.all(calls)
					.map(l -> l.stream().max(Long::compareTo).orElseGet(
							() -> Long.valueOf(System.nanoTime())));
		}
	}

	private Promise doCall(PushEvent event,
			PushEventConsumer< ? super T> pec) {
		return sameThread.resolved(
				Long.valueOf(System.nanoTime() + safePush(pec, event)));
	}

	@Override
	public boolean isConnected() {
		synchronized (lock) {
			return !connected.isEmpty();
		}
	}

	@Override
	public Promise connectPromise() {
		synchronized (lock) {
			if (closed) {
				return closedConnectPromise();
			}

			if (connected.isEmpty()) {
				if (connectPromise == null) {
					connectPromise = promiseFactory.deferred();
				}
				return connectPromise.getPromise();
			} else {
				return promiseFactory.resolved(null);
			}
		}
	}

	private Promise closedConnectPromise() {
		return promiseFactory.failed(new IllegalStateException(
				"This SimplePushEventSource is closed"));
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy