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

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

/*
 * Copyright (c) OSGi Alliance (2015, 2018). 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.concurrent.TimeUnit.MILLISECONDS;
import static org.osgi.util.pushstream.AbstractPushStreamImpl.State.CLOSED;
import static org.osgi.util.pushstream.PushEvent.*;
import static org.osgi.util.pushstream.PushbackPolicyOption.LINEAR;
import static org.osgi.util.pushstream.QueuePolicyOption.FAIL;

import java.util.Iterator;
import java.util.Objects;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.stream.Stream;

import org.osgi.util.promise.PromiseFactory;

/**
 * A factory for {@link PushStream} instances, and utility methods for handling
 * {@link PushEventSource}s and {@link PushEventConsumer}s
 */
public final class PushStreamProvider {

	private final Lock					lock	= new ReentrantLock(true);

	private int							schedulerReferences;

	private ScheduledExecutorService	sharedScheduler;

	private ScheduledExecutorService acquireScheduler() {
		try {
			lock.lockInterruptibly();
			try {
				schedulerReferences += 1;

				if (schedulerReferences == 1) {
					sharedScheduler = Executors
							.newSingleThreadScheduledExecutor();
				}
				return sharedScheduler;
			} finally {
				lock.unlock();
			}
		} catch (InterruptedException e) {
			throw new IllegalStateException("Unable to acquire the Scheduler",
					e);
		}
	}

	private void releaseScheduler() {
		try {
			lock.lockInterruptibly();
			try {
				schedulerReferences -= 1;

				if (schedulerReferences == 0) {
					sharedScheduler.shutdown();
					sharedScheduler = null;
				}
			} finally {
				lock.unlock();
			}
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}

	/**
	 * Create a stream with the default configured buffer, executor size, queue,
	 * queue policy and pushback policy. This is equivalent to calling
	 * 
	 * 
	 * buildStream(source).create();
	 * 
*

* This stream will be buffered from the event producer, and will honor back * pressure even if the source does not. *

* Buffered streams are useful for "bursty" event sources which produce a * number of events close together, then none for some time. These bursts * can sometimes overwhelm downstream processors. Buffering will not, * however, protect downstream components from a source which produces * events faster (on average) than they can be consumed. *

* Event delivery will not begin until a terminal operation is reached on * the chain of PushStreams. Once a terminal operation is reached the stream * will be connected to the event source. * * @param eventSource * @return A {@link PushStream} with a default initial buffer */ public PushStream createStream(PushEventSource eventSource) { return createStream(eventSource, 1, null, null, new ArrayBlockingQueue<>(32), FAIL.getPolicy(), LINEAR.getPolicy(1000)); } /** * Builds a push stream with custom configuration. * *

* * The resulting {@link PushStream} may be buffered or unbuffered depending * on how it is configured. * * @param eventSource The source of the events * * @return A {@link PushStreamBuilder} for the stream */ public >> PushStreamBuilder buildStream( PushEventSource eventSource) { return new PushStreamBuilderImpl(this, null, null, eventSource); } @SuppressWarnings({ "rawtypes", "unchecked" }) >> PushStream createStream( PushEventSource eventSource, int parallelism, Executor executor, ScheduledExecutorService scheduler, U queue, QueuePolicy queuePolicy, PushbackPolicy pushbackPolicy) { if (eventSource == null) { throw new NullPointerException("There is no source of events"); } if (parallelism < 0) { throw new IllegalArgumentException( "The supplied parallelism cannot be less than zero. It was " + parallelism); } else if (parallelism == 0) { parallelism = 1; } boolean closeExecutorOnClose; Executor workerToUse; if (executor == null) { workerToUse = Executors.newFixedThreadPool(parallelism); closeExecutorOnClose = true; } else { workerToUse = Objects.requireNonNull(executor); closeExecutorOnClose = false; } boolean releaseSchedulerOnClose; ScheduledExecutorService timerToUse; if (scheduler == null) { timerToUse = acquireScheduler(); releaseSchedulerOnClose = true; } else { timerToUse = Objects.requireNonNull(scheduler); releaseSchedulerOnClose = false; } if (queue == null) { queue = (U) new ArrayBlockingQueue(32); } if (queuePolicy == null) { queuePolicy = FAIL.getPolicy(); } if (pushbackPolicy == null) { pushbackPolicy = LINEAR.getPolicy(1000); } PushStream stream = new BufferedPushStreamImpl<>(this, new PromiseFactory(workerToUse, timerToUse), queue, parallelism, queuePolicy, pushbackPolicy, aec -> { try { return eventSource.open(aec); } catch (Exception e) { throw new RuntimeException( "Unable to connect to event source", e); } }); return cleanupThreads(closeExecutorOnClose, workerToUse, releaseSchedulerOnClose, stream); } private PushStream cleanupThreads(boolean closeExecutorOnClose, Executor workerToUse, boolean releaseSchedulerOnClose, PushStream stream) { if (closeExecutorOnClose || releaseSchedulerOnClose) { stream = stream.onClose(() -> { if (closeExecutorOnClose) { ((ExecutorService) workerToUse).shutdown(); } if (releaseSchedulerOnClose) { releaseScheduler(); } }).map(x -> x); } return stream; } PushStream createUnbufferedStream(PushEventSource eventSource, Executor executor, ScheduledExecutorService scheduler) { boolean closeExecutorOnClose; Executor workerToUse; if (executor == null) { workerToUse = Executors.newFixedThreadPool(2); closeExecutorOnClose = true; } else { workerToUse = Objects.requireNonNull(executor); closeExecutorOnClose = false; } boolean releaseSchedulerOnClose; ScheduledExecutorService timerToUse; if (scheduler == null) { timerToUse = acquireScheduler(); releaseSchedulerOnClose = true; } else { timerToUse = Objects.requireNonNull(scheduler); releaseSchedulerOnClose = false; } PushStream stream = new UnbufferedPushStreamImpl<>(this, new PromiseFactory(workerToUse, timerToUse), aec -> { try { return eventSource.open(aec); } catch (Exception e) { throw new RuntimeException( "Unable to connect to event source", e); } }); return cleanupThreads(closeExecutorOnClose, workerToUse, releaseSchedulerOnClose, stream); } /** * Convert an {@link PushStream} into an {@link PushEventSource}. The first * call to {@link PushEventSource#open(PushEventConsumer)} will begin event * processing. The {@link PushEventSource} will remain active until the * backing stream is closed, and permits multiple consumers to * {@link PushEventSource#open(PushEventConsumer)} it. This is equivalent * to: * *

	 * buildEventSourceFromStream(stream).create();
	 * 
* * @param stream * @return a {@link PushEventSource} backed by the {@link PushStream} */ public PushEventSource createEventSourceFromStream( PushStream stream) { return buildEventSourceFromStream(stream).build(); } /** * Convert an {@link PushStream} into an {@link PushEventSource}. The first * call to {@link PushEventSource#open(PushEventConsumer)} will begin event * processing. * *

* The {@link PushEventSource} will remain active until the backing stream * is closed, and permits multiple consumers to * {@link PushEventSource#open(PushEventConsumer)} it. Note that this means * the caller of this method is responsible for closing the supplied * stream if it is not finite in length. * *

Late joining * consumers will not receive historical events, but will immediately * receive the terminal event which closed the stream if the stream is * already closed. * * @param stream * * @return a {@link PushEventSource} backed by the {@link PushStream} */ public >> BufferBuilder,T,U> buildEventSourceFromStream( PushStream stream) { BufferBuilder,T,U> builder = stream.buildBuffer(); return new BufferBuilder,T,U>() { @Override public BufferBuilder,T,U> withBuffer(U queue) { builder.withBuffer(queue); return this; } @Override public BufferBuilder,T,U> withQueuePolicy( QueuePolicy queuePolicy) { builder.withQueuePolicy(queuePolicy); return this; } @Override public BufferBuilder,T,U> withQueuePolicy( QueuePolicyOption queuePolicyOption) { builder.withQueuePolicy(queuePolicyOption); return this; } @Override public BufferBuilder,T,U> withPushbackPolicy( PushbackPolicy pushbackPolicy) { builder.withPushbackPolicy(pushbackPolicy); return this; } @Override public BufferBuilder,T,U> withPushbackPolicy( PushbackPolicyOption pushbackPolicyOption, long time) { builder.withPushbackPolicy(pushbackPolicyOption, time); return this; } @Override public BufferBuilder,T,U> withParallelism( int parallelism) { builder.withParallelism(parallelism); return this; } @Override public BufferBuilder,T,U> withExecutor( Executor executor) { builder.withExecutor(executor); return this; } @Override public BufferBuilder,T,U> withScheduler( ScheduledExecutorService scheduler) { builder.withScheduler(scheduler); return this; } @Override public PushEventSource build() { AtomicBoolean connect = new AtomicBoolean(); AtomicReference> terminalEvent = new AtomicReference<>(); CopyOnWriteArrayList> consumers = new CopyOnWriteArrayList<>(); return consumer -> { consumers.add(consumer); PushEvent terminal = terminalEvent.get(); if (terminal != null) { if (consumers.remove(consumer)) { // The stream is already done and we missed it consumer.accept(terminal); } return () -> { //Nothing to do, we have already sent the terminal event }; } if(!connect.getAndSet(true)) { // connect builder.build() .forEachEvent(new MultiplexingConsumer( terminalEvent, consumers)); } return () -> { if (consumers.remove(consumer)) { try { consumer.accept(PushEvent.close()); } catch (Exception ex) { // TODO Auto-generated catch block ex.printStackTrace(); } } }; }; } }; } private static class MultiplexingConsumer implements PushEventConsumer { private final AtomicReference> terminalEventStore; private final CopyOnWriteArrayList> consumers; public MultiplexingConsumer( AtomicReference> terminalEventStore, CopyOnWriteArrayList> consumers) { super(); this.terminalEventStore = terminalEventStore; this.consumers = consumers; } @Override public long accept(PushEvent< ? extends T> event) throws Exception { boolean isTerminal = event.isTerminal(); if(isTerminal) { if(!terminalEventStore.compareAndSet(null, event.nodata())) { // We got a duplicate terminal, silently ignore it return -1; } for (PushEventConsumer< ? super T> pushEventConsumer : consumers) { if(consumers.remove(pushEventConsumer)) { try { pushEventConsumer.accept(event); } catch (Exception ex) { // TODO Auto-generated catch block ex.printStackTrace(); } } } return -1; } else { long maxBP = 0; for (PushEventConsumer< ? super T> pushEventConsumer : consumers) { try { long tmpBP = pushEventConsumer.accept(event); if(tmpBP < 0 && consumers.remove(pushEventConsumer)) { try { pushEventConsumer.accept(PushEvent.close()); } catch (Exception ex) { // TODO Auto-generated catch block ex.printStackTrace(); } } else if (tmpBP > maxBP) { maxBP = tmpBP; } } catch (Exception ex) { if(consumers.remove(pushEventConsumer)) { try { pushEventConsumer.accept(PushEvent.error(ex)); } catch (Exception ex2) { // TODO Auto-generated catch block ex2.printStackTrace(); } } } } return maxBP; } } } /** * Create a {@link SimplePushEventSource} with the supplied type and default * buffering behaviors. The SimplePushEventSource will respond to back * pressure requests from the consumers connected to it. This is equivalent * to: * *

	 * buildSimpleEventSource(type).create();
	 * 
* * @param type * @return a {@link SimplePushEventSource} */ public SimplePushEventSource createSimpleEventSource(Class type) { return createSimplePushEventSource(1, null, new ArrayBlockingQueue<>(32), FAIL.getPolicy(), () -> { /* Nothing else to do */ }); } /** * Build a {@link SimplePushEventSource} with the supplied type and custom * buffering behaviors. The SimplePushEventSource will respond to back * pressure requests from the consumers connected to it. * * @param type * @return a {@link SimplePushEventSource} */ public >> BufferBuilder,T,U> buildSimpleEventSource( Class type) { return new AbstractBufferBuilder,T,U>() { @Override public SimplePushEventSource build() { return createSimplePushEventSource(concurrency, worker, buffer, bufferingPolicy, () -> { /* Nothing else to do */ }); } }; } @SuppressWarnings({ "unchecked", "rawtypes" }) >> SimplePushEventSource createSimplePushEventSource( int parallelism, Executor executor, U queue, QueuePolicy queuePolicy, Runnable onClose) { if (parallelism < 0) { throw new IllegalArgumentException( "The supplied parallelism cannot be less than zero. It was " + parallelism); } else if (parallelism == 0) { parallelism = 1; } boolean closeExecutorOnClose; Executor toUse; if (executor == null) { toUse = Executors.newFixedThreadPool(parallelism); closeExecutorOnClose = true; } else { toUse = Objects.requireNonNull(executor); closeExecutorOnClose = false; } if (queue == null) { queue = (U) new ArrayBlockingQueue(32); } if (queuePolicy == null) { queuePolicy = FAIL.getPolicy(); } SimplePushEventSourceImpl spes = new SimplePushEventSourceImpl( new PromiseFactory(toUse, acquireScheduler()), queuePolicy, queue, parallelism, () -> { try { onClose.run(); } catch (Exception e) { // TODO log this? } if (closeExecutorOnClose) { ((ExecutorService) toUse).shutdown(); } releaseScheduler(); }); return spes; } /** * Create a buffered {@link PushEventConsumer} with the default configured * buffer, executor size, queue, queue policy and pushback policy. This is * equivalent to calling * *
	 * buildBufferedConsumer(delegate).create();
	 * 
*

* The returned consumer will be buffered from the event source, and will * honor back pressure requests from its delegate even if the event source * does not. *

* Buffered consumers are useful for "bursty" event sources which produce a * number of events close together, then none for some time. These bursts * can sometimes overwhelm the consumer. Buffering will not, however, * protect downstream components from a source which produces events faster * than they can be consumed. * * @param delegate * @return a {@link PushEventConsumer} with a buffer directly before it */ public PushEventConsumer createBufferedConsumer( PushEventConsumer delegate) { return buildBufferedConsumer(delegate).build(); } /** * Build a buffered {@link PushEventConsumer} with custom configuration. *

* The returned consumer will be buffered from the event source, and will * honor back pressure requests from its delegate even if the event source * does not. *

* Buffered consumers are useful for "bursty" event sources which produce a * number of events close together, then none for some time. These bursts * can sometimes overwhelm the consumer. Buffering will not, however, * protect downstream components from a source which produces events faster * than they can be consumed. *

* Buffers are also useful as "circuit breakers". If a * {@link QueuePolicyOption#FAIL} is used then a full buffer will request * that the stream close, preventing an event storm from reaching the * client. *

* Note that this buffered consumer will close when it receives a terminal * event, or if the delegate returns negative backpressure. No further * events will be propagated after this time. * * @param delegate * @return a {@link PushEventConsumer} with a buffer directly before it */ public >> BufferBuilder,T,U> buildBufferedConsumer( PushEventConsumer delegate) { return new AbstractBufferBuilder,T,U>() { @Override public PushEventConsumer build() { PushEventPipe pipe = new PushEventPipe<>(); createStream(pipe, concurrency, worker, timer, buffer, bufferingPolicy, backPressure) .forEachEvent(delegate); return pipe; } }; } static final class PushEventPipe implements PushEventConsumer, PushEventSource { volatile PushEventConsumer< ? super T> delegate; @Override public AutoCloseable open(PushEventConsumer< ? super T> pec) throws Exception { return () -> { /* Nothing else to do */ }; } @Override public long accept(PushEvent< ? extends T> event) throws Exception { return delegate.accept(event); } } /** * Create an Unbuffered {@link PushStream} from a Java {@link Stream} The * data from the stream will be pushed into the PushStream synchronously as * it is opened. This may make terminal operations blocking unless a buffer * has been added to the {@link PushStream}. Care should be taken with * infinite {@link Stream}s to avoid blocking indefinitely. * * @param items The items to push into the PushStream * @return A PushStream containing the items from the Java Stream */ public PushStream streamOf(Stream items) { PushEventSource pes = aec -> { AtomicBoolean closed = new AtomicBoolean(false); items.mapToLong(i -> { try { long returnValue = closed.get() ? -1 : aec.accept(data(i)); if (returnValue < 0) { aec.accept(PushEvent. close()); } return returnValue; } catch (Exception e) { try { aec.accept(PushEvent. error(e)); } catch (Exception e2) {/* No further events needed */} return -1; } }).filter(i -> i < 0).findFirst().orElseGet(() -> { try { return aec.accept(PushEvent. close()); } catch (Exception e) { return -1; } }); return () -> closed.set(true); }; return this. createUnbufferedStream(pes, null, null); } /** * Create an Unbuffered {@link PushStream} from a Java {@link Stream} The * data from the stream will be pushed into the PushStream asynchronously * using the supplied Executor. * * @param executor The worker to use to push items from the Stream into the * PushStream * @param scheduler The scheduler to use to trigger timed events in the * PushStream * @param items The items to push into the PushStream * @return A PushStream containing the items from the Java Stream */ public PushStream streamOf(Executor executor, ScheduledExecutorService scheduler, Stream items) { boolean closeExecutorOnClose; Executor workerToUse; if (executor == null) { workerToUse = Executors.newFixedThreadPool(2); closeExecutorOnClose = true; } else { workerToUse = Objects.requireNonNull(executor); closeExecutorOnClose = false; } boolean releaseSchedulerOnClose; ScheduledExecutorService timerToUse; if (scheduler == null) { timerToUse = acquireScheduler(); releaseSchedulerOnClose = true; } else { timerToUse = Objects.requireNonNull(scheduler); releaseSchedulerOnClose = false; } PushStream stream = new UnbufferedPushStreamImpl>>( this, new PromiseFactory(workerToUse, timerToUse), aec -> { return () -> { /* No action to take */ }; }) { @Override protected boolean begin() { if (super.begin()) { Iterator it = items.iterator(); promiseFactory.executor().execute(() -> pushData(it)); return true; } return false; } private void pushData(Iterator it) { while (it.hasNext()) { try { long returnValue = closed.get() == CLOSED ? -1 : handleEvent(data(it.next())); if (returnValue != 0) { if (returnValue < 0) { close(); return; } else { promiseFactory.scheduledExecutor() .schedule( () -> promiseFactory.executor() .execute(() -> pushData(it)), returnValue, MILLISECONDS); return; } } } catch (Exception e) { close(error(e)); } } close(); } }; return cleanupThreads(closeExecutorOnClose, workerToUse, releaseSchedulerOnClose, stream); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy