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

reactor.rx.action.combination.CombineLatestAction Maven / Gradle / Ivy

/*
 * Copyright (c) 2011-2015 Pivotal Software Inc, 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 reactor.rx.action.combination;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscription;
import reactor.core.Dispatcher;
import reactor.fn.Function;
import reactor.fn.tuple.Tuple;
import reactor.rx.subscription.PushSubscription;

import java.util.List;

/**
 * @author Stephane Maldini
 * @since 2.0
 */
public final class CombineLatestAction
		extends FanInAction, V, CombineLatestAction.InnerSubscriber> {

	private static final Object EMPTY_ZIPPED_DATA = new Object();

	final Function accumulator;

	int index = 0;

	Object[] toZip = new Object[1];

	public CombineLatestAction(Dispatcher dispatcher,
	                           Function accumulator, List>
			composables) {
		super(dispatcher, composables);
		this.accumulator = accumulator;
		this.toZip = new Object[composables != null ? composables.size() : 1];
	}

	@SuppressWarnings("unchecked")
	protected void broadcastTuple() {
		if (!checkAllFilled()) return;

		Object[] _toZip = toZip;

		V res = accumulator.apply((TUPLE) Tuple.of(_toZip));

		if (res != null) {
			broadcastNext(res);
		}
	}

	private boolean checkAllFilled() {
		for (int i = 0; i < toZip.length; i++) {
			if (toZip[i] == null) {
				return false;
			}
		}
		return true;
	}


	@Override
	protected FanInSubscription, V, InnerSubscriber> createFanInSubscription() {
		return new FanInSubscription<>(this);
	}

	@Override
	protected PushSubscription> createTrackingSubscription(Subscription subscription) {
		return innerSubscriptions;
	}

	@Override
	protected void doNext(Zippable ev) {
		toZip[ev.index] = ev.data == null ? EMPTY_ZIPPED_DATA : ev.data;

		broadcastTuple();
	}

	@Override
	protected void doComplete() {
			//can receive multiple queued complete signals
			cancel();
			broadcastComplete();
	}

	@Override
	protected InnerSubscriber createSubscriber() {
		return new CombineLatestAction.InnerSubscriber<>(this, index++);
	}

	@Override
	protected long initUpstreamPublisherAndCapacity() {
		long i = 0l;
		for (Publisher composable : publishers) {
			addPublisher(composable);
			i++;
		}
		return i;
	}

	@Override
	public String toString() {
		String formatted = super.toString();
		for (int i = 0; i < toZip.length; i++) {
			if (toZip[i] != null)
				formatted += "(" + (i) + "):" + toZip[i] + ",";
		}
		return formatted.substring(0, formatted.length());
	}

//Handling each new Publisher to zip

	public static final class InnerSubscriber extends FanInAction.InnerSubscriber, V> {
		final CombineLatestAction outerAction;
		final int                          index;

		InnerSubscriber(CombineLatestAction outerAction, int index) {
			super(outerAction);
			this.index = index;
			this.outerAction = outerAction;
		}

		@Override
		@SuppressWarnings("unchecked")
		public void onSubscribe(Subscription subscription) {
			setSubscription(new FanInSubscription.InnerSubscription, InnerSubscriber>(subscription,
					this));
			int newSize = outerAction.innerSubscriptions.runningComposables;
			outerAction.capacity(newSize);

			if (newSize > outerAction.toZip.length) {
				Object[] previousZip = outerAction.toZip;
				outerAction.toZip = new Object[newSize];
				System.arraycopy(previousZip, 0, outerAction.toZip, 0, newSize - 1);
			}

			long toRequest = pendingRequests;
			if (toRequest > 0) {
				pendingRequests = 0;
				request(toRequest);
			}
			if (outerAction.dynamicMergeAction != null) {
				outerAction.dynamicMergeAction.decrementWip();
			}
		}

		@Override
		public void onNext(O ev) {
			if (--pendingRequests > 0) pendingRequests = 0l;
			//emittedSignals++;
			outerAction.innerSubscriptions.serialNext(new Zippable(index, ev));
		}

		@Override
		public boolean isReactivePull(Dispatcher dispatcher, long producerCapacity) {
			return false;
		}

		@Override
		public String toString() {
			return "CombineLatest.InnerSubscriber{index=" + index + ", " +
					"pending=" + pendingRequests + ", emitted=" + emittedSignals + "}";
		}
	}

	public static final class Zippable {
		final int index;
		final O   data;

		public Zippable(int index, O data) {
			this.index = index;
			this.data = data;
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy