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

org.finos.tracdap.common.data.pipeline.ReactiveByteSink Maven / Gradle / Ivy

Go to download

TRAC D.A.P. data library, interfaces and core functionality for working with primary data

The newest version!
/*
 * Licensed to the Fintech Open Source Foundation (FINOS) under one or
 * more contributor license agreements. See the NOTICE file distributed
 * with this work for additional information regarding copyright ownership.
 * FINOS licenses this file to you 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.finos.tracdap.common.data.pipeline;

import org.finos.tracdap.common.data.DataPipeline;
import org.finos.tracdap.common.exception.ETracInternal;

import org.apache.arrow.memory.ArrowBuf;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.concurrent.Flow;


public class ReactiveByteSink
    extends
        BaseDataSink
    implements
        DataPipeline.StreamApi {

    private final Logger log = LoggerFactory.getLogger(getClass());

    private final Flow.Subscriber sink;
    private Flow.Subscription subscription;
    private int chunksRequested;
    private int chunksDelivered;

    ReactiveByteSink(DataPipelineImpl pipeline, Flow.Subscriber sink) {

        super(pipeline);

        this.sink = sink;
        this.chunksRequested = 0;
        this.chunksDelivered = 0;
    }

    @Override
    public DataPipeline.StreamApi dataInterface() {
        return this;
    }

    @Override
    public void connect() {

        if (subscription != null) {
            log.warn("Data stream started twice");
            return;
        }

        subscription = new Subscription();
        sink.onSubscribe(subscription);
    }

    private void doRequest(long n) {

        chunksRequested += n;
        pipeline.pumpData();
    }

    public void doCancel() {

        pipeline.requestCancel();
        close();
    }

    @Override
    public boolean isReady() {
        return chunksRequested > chunksDelivered && subscription != null;
    }

    @Override
    public void pump() {

        if (isReady())
            pipeline.pumpData();
    }

    @Override
    public void terminate(Throwable error) {

        if (isDone()) {
            log.warn("Requested termination, but stage is already down");
            return;
        }

        markAsDone();
        sink.onError(error);
    }

    @Override
    public void close() {

        // no-op
    }

    @Override
    public void onStart() {

        // No-op, already subscribed
    }

    @Override
    public void onNext(ArrowBuf chunk) {

        if (chunksRequested <= chunksDelivered) {
            var err = new ETracInternal("Data stream is out of sync");
            log.error(err.getMessage(), err);
            throw err;
        }

        chunksDelivered += 1;
        sink.onNext(chunk);
    }

    @Override
    public void onComplete() {

        markAsDone();
        sink.onComplete();

        reportComplete();
    }

    @Override
    public void onError(Throwable error) {

        markAsDone();
        sink.onError(error);

        reportRegularError(error);
    }

    private class Subscription implements Flow.Subscription {

        @Override
        public void request(long n) {
            doRequest(n);
        }

        @Override
        public void cancel() {
            doCancel();
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy