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

io.datakernel.stream.StreamForwarder Maven / Gradle / Ivy

Go to download

Composable asynchronous/reactive streams with powerful data processing capabilities.

The newest version!
/*
 * Copyright (C) 2015 SoftIndex LLC.
 *
 * 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 io.datakernel.stream;

import io.datakernel.eventloop.Eventloop;

import java.util.ArrayList;

/**
 * Simple no-op forwarder of stream data, which supports wiring of itself to upstream producer or to downstream consumer at later time.
 * 

After both upstream producer and downstream consumer are connected, StreamForwarder switches to zero-overhead 'short-circuit' mode, which directly forwards StreamDataReceiver from downstream to upstream. * * @param type of data */ public class StreamForwarder extends AbstractStreamTransformer_1_1 implements StreamDataReceiver { private final ArrayList bufferedItems = new ArrayList<>(); private boolean bufferedEndOfStream; private boolean rewired; /** * Creates a new instance of this class * * @param eventloop eventloop in which forwarder will be running */ public StreamForwarder(Eventloop eventloop) { super(eventloop); } @Override public StreamDataReceiver getDataReceiver() { return rewired ? getDownstreamDataReceiver() : this; // short-circuit upstream producer and downstream consumer } @Override public void onData(T item) { if (rewired) { // Maybe downstreamDataReceiver is cached somewhere in upstream producer, even after rewire? // Anyway, let's just forward items to downstream. send(item); } else { bufferedItems.add(item); if (getUpstream() != null) { suspendUpstream(); } } } private void flushAndRewire() { if (error != null) { upstreamProducer.closeWithError(error); downstreamConsumer.onError(error); return; } for (T item : bufferedItems) { send(item); } bufferedItems.clear(); if (bufferedEndOfStream) { sendEndOfStream(); } rewired = true; getUpstream().bindDataReceiver(); if (status == CLOSED) { closeUpstream(); } else if (status != READY) { suspendUpstream(); } else { resumeUpstream(); } } @Override protected void onConsumerStarted() { if (getUpstream() != null && getDownstream() != null) { flushAndRewire(); } } @Override protected void onProducerStarted() { if (getUpstream() != null && getDownstream() != null) { flushAndRewire(); } } @Override public void onEndOfStream() { if (rewired) { sendEndOfStream(); } else { bufferedEndOfStream = true; } } @Override public void onSuspended() { if (rewired) { suspendUpstream(); } } @Override public void onResumed() { if (rewired) { resumeUpstream(); } } @Override public void onClosed() { if (rewired) { closeUpstream(); } } @Override public void onClosedWithError(Exception e) { if (rewired) { upstreamProducer.closeWithError(e); downstreamConsumer.onError(e); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy