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

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

/*
 * 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.rx.action.combination;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
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.Arrays;
import java.util.List;

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

    private static final Object EMPTY_ZIPPED_DATA = new Object();

    final Function accumulator;

    int index = 0;
    int count = 0;

    Object[] toZip;

    @SuppressWarnings("unchecked")
    public static  Function> joinZipper() {
        return new Function>() {
            @Override
            public List apply(TUPLE ts) {
                return Arrays.asList((V[]) ts.toArray());
            }
        };
    }

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

    @Override
    protected void doOnSubscribe(Subscription subscription) {
        if (status.compareAndSet(NOT_STARTED, RUNNING)) {
            if (publishers != null) {
                for (Publisher publisher : publishers) {
                    addPublisher(publisher);
                }
            }
        }
    }

    @SuppressWarnings("unchecked")
    protected void broadcastTuple(boolean isFinishing) {
        long capacity = this.capacity;
        if (count >= capacity) {

            if (!checkAllFilled()) return;

            count = 0;
            Object[] _toZip = toZip;
            toZip = new Object[toZip.length];

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

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

                if (!isFinishing && upstreamSubscription.pendingRequestSignals() > 0) {
                    dispatcher.dispatch(capacity, upstreamSubscription, null);
                }
            }
        }
        if (isFinishing) {
            broadcastComplete();
        }
    }

    @Override
    protected void requestUpstream(long capacity, boolean terminated, long elements) {
        long upstream = innerSubscriptions.runningComposables;
        upstream = upstream == 0 ? elements : (upstream * elements < 0 ? Long.MAX_VALUE : upstream * elements);
        if (publishers != null && innerSubscriptions.pendingRequestSignals() > 0) {
            innerSubscriptions.updatePendingRequests(upstream);
        } else {
            requestMore(upstream);
            if (dynamicMergeAction != null) {
                dynamicMergeAction.requestUpstream(capacity, terminated, elements);
            }
        }
    }

    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 ZipSubscription(this);
    }

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

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

        broadcastTuple(false);
    }

    @Override
    protected void doComplete() {
        broadcastTuple(true);
    }

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

    @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, count > 0 ? formatted.length() - 1 : formatted.length());
    }

//Handling each new Publisher to zip

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

        InnerSubscriber(ZipAction 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;
            newSize = newSize == 0 ? 1 : newSize;
            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);
            }

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

        @Override
        public void onComplete() {
            if (TERMINATE_UPDATER.compareAndSet(this, 0, 1)) {
                outerAction.innerSubscriptions.remove(sequenceId);
                outerAction.status.compareAndSet(RUNNING, COMPLETING);
                outerAction.capacity(outerAction.innerSubscriptions.runningComposables);

                if(FanInSubscription.RUNNING_COMPOSABLE_UPDATER.decrementAndGet(outerAction.innerSubscriptions) == 0){
                    outerAction.innerSubscriptions.serialComplete();
                }else{
                    outerAction.innerSubscriptions.serialNext(new Zippable(index, null));
                }
            }
        }

        @Override
        public void request(long n) {
            super.request(1);
        }

        @Override
        public void onNext(O ev) {
            if (--pendingRequests < 0) pendingRequests = 0l;
            emittedSignals = 1;
            if (outerAction.status.get() == COMPLETING) {
                if (TERMINATE_UPDATER.compareAndSet(this, 0, 1)) {
                    outerAction.innerSubscriptions.remove(sequenceId);
                    long left = FanInSubscription.RUNNING_COMPOSABLE_UPDATER.decrementAndGet(outerAction
                      .innerSubscriptions);
                    if (0 == left) {
                        outerAction.innerSubscriptions.serialNext(new Zippable(index, ev));
                        outerAction.innerSubscriptions.serialComplete();
                        return;
                    }
                }
            }
            outerAction.innerSubscriptions.serialNext(new Zippable(index, ev));
        }


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

        @Override
        public long getCapacity() {
            return 1;
        }

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

    private final class ZipSubscription extends FanInSubscription, V, ZipAction.InnerSubscriber> {

        public ZipSubscription(Subscriber> subscriber) {
            super(subscriber);
        }

        @Override
        public boolean shouldRequestPendingSignals() {
            synchronized (this) {
                return pendingRequestSignals > 0 && pendingRequestSignals != Long.MAX_VALUE && count == maxCapacity;
            }
        }

        @Override
        public void request(long elements) {
            if (pendingRequestSignals == Long.MAX_VALUE) {
                super.parallelRequest(1);
            } else {
                super.request(elements);
            }
        }

        @Override
        public void onNext(Zippable ev) {
            if (ev.data != null) {
                super.onNext(ev);
            } else if(ZipAction.this.toZip.length > ev.index && ZipAction.this.toZip[ev.index] == null) {
				super.onComplete();
            }
        }
    }

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

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

        @Override
        public String toString() {
            return "Zippable{" +
              "index=" + index +
              ", data=" + data +
              '}';
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy