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

io.servicetalk.http.netty.SniCompleteChannelSingle Maven / Gradle / Ivy

The newest version!
/*
 * Copyright © 2021 Apple Inc. and the ServiceTalk project authors
 *
 * 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.servicetalk.http.netty;

import io.servicetalk.concurrent.SingleSource;
import io.servicetalk.transport.netty.internal.ChannelCloseUtils;
import io.servicetalk.transport.netty.internal.ChannelInitializer;
import io.servicetalk.transport.netty.internal.StacklessClosedChannelException;

import io.netty.channel.Channel;
import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.handler.ssl.SniCompletionEvent;

import javax.annotation.Nullable;

import static io.servicetalk.transport.netty.internal.ChannelCloseUtils.assignConnectionError;

final class SniCompleteChannelSingle extends ChannelInitSingle {
    SniCompleteChannelSingle(final Channel channel, final ChannelInitializer channelInitializer) {
        super(channel, channelInitializer);
    }

    @Override
    protected ChannelHandler newChannelHandler(final Subscriber subscriber) {
        return new SniCompleteChannelHandler(subscriber);
    }

    private static final class SniCompleteChannelHandler extends ChannelInboundHandlerAdapter {
        @Nullable
        private Subscriber subscriber;

        SniCompleteChannelHandler(Subscriber subscriber) {
            this.subscriber = subscriber;
        }

        @Override
        public void handlerAdded(final ChannelHandlerContext ctx) throws Exception {
            super.handlerAdded(ctx);
            assert ctx.channel().isActive();
            // Force a read to get the SSL handshake started. We initialize pipeline before
            // SslHandshakeCompletionEvent will complete, therefore, no data will be propagated before we finish
            // initialization.
            ctx.read();
        }

        @Override
        public void userEventTriggered(ChannelHandlerContext ctx, Object evt) {
            if (evt instanceof SniCompletionEvent && subscriber != null) {
                ctx.pipeline().remove(this);
                Subscriber subscriberCopy = subscriber;
                subscriber = null;
                subscriberCopy.onSuccess((SniCompletionEvent) evt);
            }
            ctx.fireUserEventTriggered(evt);
        }

        @Override
        public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) {
            if (subscriber != null) {
                propagateError(ctx.channel(), StacklessClosedChannelException.newInstance(
                        SniCompleteChannelHandler.class, "exceptionCaught(...)").initCause(cause));
            } else {
                // Propagate exception in the pipeline if subscriber is already complete
                ctx.fireExceptionCaught(cause);
                ChannelCloseUtils.close(ctx, cause);
            }
        }

        @Override
        public void channelInactive(ChannelHandlerContext ctx) {
            if (subscriber != null) {
                propagateError(ctx.channel(), StacklessClosedChannelException.newInstance(
                        SniCompleteChannelHandler.class, "channelInactive(...)"));
            } else {
                ctx.fireChannelInactive();
            }
        }

        private void propagateError(Channel channel, Throwable cause) {
            assert subscriber != null;
            assignConnectionError(channel, cause);
            final SingleSource.Subscriber subscriberCopy = subscriber;
            subscriber = null;
            subscriberCopy.onError(cause);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy