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

io.helidon.security.jersey.OutputStreamPublisher Maven / Gradle / Ivy

/*
 * Copyright (c) 2018 Oracle and/or its affiliates. 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 io.helidon.security.jersey;

import java.io.IOException;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;

import io.helidon.common.reactive.Flow;

/**
 * Output stream that {@link Flow.Publisher publishes} any data written to it as {@link ByteBuffer} events.
 */
@SuppressWarnings("WeakerAccess")
class OutputStreamPublisher extends OutputStream implements Flow.Publisher {

    private final SingleSubscriberHolder subscriber = new SingleSubscriberHolder<>();
    private final Object invocationLock = new Object();

    private final RequestedCounter requested = new RequestedCounter();

    private final CompletableFuture completionResult = new CompletableFuture<>();

    @Override
    public void subscribe(Flow.Subscriber subscriberParam) {
        if (subscriber.register(subscriberParam)) {
            subscriberParam.onSubscribe(new Flow.Subscription() {
                @Override
                public void request(long n) {
                    requested.increment(n, t -> complete(t));
                }

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

    @Override
    public void write(byte[] b) throws IOException {

        publish(b, 0, b.length);
    }

    @Override
    public void write(byte[] b, int off, int len) throws IOException {
        publish(b, off, len);
    }

    @Override
    public void write(int b) throws IOException {
        byte bb = (byte) b;
        publish(new byte[] {bb}, 0, 1);
    }

    @Override
    public void close() throws IOException {
        complete();
        try {
            completionResult.get();
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
            throw new IOException(e);
        } catch (ExecutionException e) {
            throw new IOException(e.getCause());
        }
    }

    /**
     * Signals this publishing output stream that it can safely return from otherwise blocking invocation
     * to it's {@link #close()} method.
     * Subsequent multiple invocations of this method are allowed, but have no effect on this publishing output stream.
     * 

* When the {@link #close()} method on this output stream is invoked, it will block waiting for a signal to complete. * This is useful in cases, when the receiving side needs to synchronize it's completion with the publisher, e.g. to * ensure that any resources used by the subscribing party are not released prematurely due to a premature exit from * publishing output stream {@code close()} method. *

* Additionally, this mechanism can be used to propagate any downstream completion exceptions back to this publisher * and up it's call stack. When a non-null {@code throwable} parameter is passed into the method, it will be wrapped * in an {@link IOException} and thrown from the {@code close()} method when it is invoked. * * @param throwable represents a completion error condition that should be thrown when a {@code close()} method is invoked * on this publishing output stream. If set to {@code null}, the {@code close()} method will exit normally. */ public void signalCloseComplete(Throwable throwable) { if (throwable == null) { completionResult.complete(null); } else { completionResult.completeExceptionally(throwable); } } private void publish(byte[] buffer, int offset, int length) throws IOException { Objects.requireNonNull(buffer); try { final Flow.Subscriber sub = subscriber.get(); while (!subscriber.isClosed() && !requested.tryDecrement()) { //java9 //Thread.onSpinWait(); // wait until some data can be sent or the stream has been closed //java8 :( Thread.sleep(30); } synchronized (invocationLock) { if (subscriber.isClosed()) { throw new IOException("Output stream already closed."); } sub.onNext(ByteBuffer.wrap(buffer, offset, length)); } } catch (InterruptedException e) { Thread.currentThread().interrupt(); complete(e); throw new IOException(e); } catch (ExecutionException e) { complete(e.getCause()); throw new IOException(e.getCause()); } } private void complete() { subscriber.close(sub -> { synchronized (invocationLock) { sub.onComplete(); } }); } private void complete(Throwable t) { subscriber.close(sub -> { synchronized (invocationLock) { sub.onError(t); } }); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy