io.grpc.netty.UtilsTest.java Source code

Java tutorial

Introduction

Here is the source code for io.grpc.netty.UtilsTest.java

Source

/*
 * Copyright 2015 The gRPC 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.grpc.netty;

import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.TruthJUnit.assume;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertSame;

import com.google.common.base.MoreObjects;
import io.grpc.InternalChannelz;
import io.grpc.InternalChannelz.SocketOptions;
import io.grpc.Metadata;
import io.grpc.Status;
import io.grpc.internal.GrpcUtil;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFactory;
import io.netty.channel.ChannelOption;
import io.netty.channel.ConnectTimeoutException;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.ServerChannel;
import io.netty.channel.WriteBufferWaterMark;
import io.netty.channel.embedded.EmbeddedChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.http2.DefaultHttp2Headers;
import io.netty.handler.codec.http2.Http2Error;
import io.netty.handler.codec.http2.Http2Exception;
import io.netty.handler.codec.http2.Http2Headers;
import io.netty.util.AsciiString;
import java.util.Map;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;

/** Unit tests for {@link Utils}. */
@RunWith(JUnit4.class)
public class UtilsTest {
    private final Metadata.Key<String> userKey = Metadata.Key.of("user-key", Metadata.ASCII_STRING_MARSHALLER);
    private final String userValue = "user-value";

    @Test
    public void testStatusFromThrowable() {
        Status s = Status.CANCELLED.withDescription("msg");
        assertSame(s, Utils.statusFromThrowable(new Exception(s.asException())));
        Throwable t;
        t = new ConnectTimeoutException("msg");
        assertStatusEquals(Status.UNAVAILABLE.withCause(t), Utils.statusFromThrowable(t));
        t = new Http2Exception(Http2Error.INTERNAL_ERROR, "msg");
        assertStatusEquals(Status.INTERNAL.withCause(t), Utils.statusFromThrowable(t));
        t = new Exception("msg");
        assertStatusEquals(Status.UNKNOWN.withCause(t), Utils.statusFromThrowable(t));
    }

    @Test
    public void convertClientHeaders_sanitizes() {
        Metadata metaData = new Metadata();

        // Intentionally being explicit here rather than relying on any pre-defined lists of headers,
        // since the goal of this test is to validate the correctness of such lists in the first place.
        metaData.put(GrpcUtil.CONTENT_TYPE_KEY, "to-be-removed");
        metaData.put(GrpcUtil.USER_AGENT_KEY, "to-be-removed");
        metaData.put(GrpcUtil.TE_HEADER, "to-be-removed");
        metaData.put(userKey, userValue);

        String scheme = "https";
        String userAgent = "user-agent";
        String method = "POST";
        String authority = "authority";
        String path = "//testService/test";

        Http2Headers output = Utils.convertClientHeaders(metaData, new AsciiString(scheme), new AsciiString(path),
                new AsciiString(authority), new AsciiString(method), new AsciiString(userAgent));
        DefaultHttp2Headers headers = new DefaultHttp2Headers();
        for (Map.Entry<CharSequence, CharSequence> entry : output) {
            headers.add(entry.getKey(), entry.getValue());
        }

        // 7 reserved headers, 1 user header
        assertEquals(7 + 1, headers.size());
        // Check the 3 reserved headers that are non pseudo
        // Users can not create pseudo headers keys so no need to check for them here
        assertEquals(GrpcUtil.CONTENT_TYPE_GRPC, headers.get(GrpcUtil.CONTENT_TYPE_KEY.name()).toString());
        assertEquals(userAgent, headers.get(GrpcUtil.USER_AGENT_KEY.name()).toString());
        assertEquals(GrpcUtil.TE_TRAILERS, headers.get(GrpcUtil.TE_HEADER.name()).toString());
        // Check the user header is in tact
        assertEquals(userValue, headers.get(userKey.name()).toString());
    }

    @Test
    @SuppressWarnings("UndefinedEquals") // AsciiString.equals
    public void convertServerHeaders_sanitizes() {
        Metadata metaData = new Metadata();

        // Intentionally being explicit here rather than relying on any pre-defined lists of headers,
        // since the goal of this test is to validate the correctness of such lists in the first place.
        metaData.put(GrpcUtil.CONTENT_TYPE_KEY, "to-be-removed");
        metaData.put(GrpcUtil.TE_HEADER, "to-be-removed");
        metaData.put(GrpcUtil.USER_AGENT_KEY, "to-be-removed");
        metaData.put(userKey, userValue);

        Http2Headers output = Utils.convertServerHeaders(metaData);
        DefaultHttp2Headers headers = new DefaultHttp2Headers();
        for (Map.Entry<CharSequence, CharSequence> entry : output) {
            headers.add(entry.getKey(), entry.getValue());
        }
        // 2 reserved headers, 1 user header
        assertEquals(2 + 1, headers.size());
        assertEquals(Utils.CONTENT_TYPE_GRPC, headers.get(GrpcUtil.CONTENT_TYPE_KEY.name()));
    }

    @Test
    public void channelOptionsTest_noLinger() {
        Channel channel = new EmbeddedChannel();
        assertNull(channel.config().getOption(ChannelOption.SO_LINGER));
        InternalChannelz.SocketOptions socketOptions = Utils.getSocketOptions(channel);
        assertNull(socketOptions.lingerSeconds);
    }

    @Test
    @SuppressWarnings("deprecation")
    public void channelOptionsTest_oio() {
        Channel channel = new io.netty.channel.socket.oio.OioSocketChannel();
        SocketOptions socketOptions = setAndValidateGeneric(channel);
        assertEquals(250, (int) socketOptions.soTimeoutMillis);
    }

    @Test
    public void channelOptionsTest_nio() {
        Channel channel = new NioSocketChannel();
        SocketOptions socketOptions = setAndValidateGeneric(channel);
        assertNull(socketOptions.soTimeoutMillis);
    }

    private static InternalChannelz.SocketOptions setAndValidateGeneric(Channel channel) {
        channel.config().setOption(ChannelOption.SO_LINGER, 3);
        // only applicable for OIO channels:
        channel.config().setOption(ChannelOption.SO_TIMEOUT, 250);
        // Test some arbitrarily chosen options with a non numeric values
        channel.config().setOption(ChannelOption.SO_KEEPALIVE, true);
        WriteBufferWaterMark writeBufWaterMark = new WriteBufferWaterMark(10, 20);
        channel.config().setOption(ChannelOption.WRITE_BUFFER_WATER_MARK, writeBufWaterMark);

        InternalChannelz.SocketOptions socketOptions = Utils.getSocketOptions(channel);
        assertEquals(3, (int) socketOptions.lingerSeconds);
        assertEquals("true", socketOptions.others.get("SO_KEEPALIVE"));
        assertEquals(writeBufWaterMark.toString(),
                socketOptions.others.get(ChannelOption.WRITE_BUFFER_WATER_MARK.toString()));
        return socketOptions;
    }

    private static void assertStatusEquals(Status expected, Status actual) {
        assertEquals(expected.getCode(), actual.getCode());
        assertThat(MoreObjects.firstNonNull(actual.getDescription(), ""))
                .contains(MoreObjects.firstNonNull(expected.getDescription(), ""));
        assertEquals(expected.getCause(), actual.getCause());
    }

    @Test
    public void defaultEventLoopGroup_whenEpollIsAvailable() {
        assume().that(Utils.isEpollAvailable()).isTrue();

        EventLoopGroup defaultBossGroup = Utils.DEFAULT_BOSS_EVENT_LOOP_GROUP.create();
        EventLoopGroup defaultWorkerGroup = Utils.DEFAULT_WORKER_EVENT_LOOP_GROUP.create();

        assertThat(defaultBossGroup.getClass().getName()).isEqualTo("io.netty.channel.epoll.EpollEventLoopGroup");
        assertThat(defaultWorkerGroup.getClass().getName()).isEqualTo("io.netty.channel.epoll.EpollEventLoopGroup");

        defaultBossGroup.shutdownGracefully();
        defaultWorkerGroup.shutdownGracefully();
    }

    @Test
    public void defaultClientChannelType_whenEpollIsAvailable() {
        assume().that(Utils.isEpollAvailable()).isTrue();

        Class<? extends Channel> clientChannelType = Utils.DEFAULT_CLIENT_CHANNEL_TYPE;

        assertThat(clientChannelType.getName()).isEqualTo("io.netty.channel.epoll.EpollSocketChannel");
    }

    @Test
    public void defaultServerChannelFactory_whenEpollIsAvailable() {
        assume().that(Utils.isEpollAvailable()).isTrue();

        ChannelFactory<? extends ServerChannel> channelFactory = Utils.DEFAULT_SERVER_CHANNEL_FACTORY;

        assertThat(channelFactory.toString()).isEqualTo("ReflectiveChannelFactory(EpollServerSocketChannel.class)");
    }

    @Test
    public void maybeGetTcpUserTimeoutOption() {
        assume().that(Utils.isEpollAvailable()).isTrue();

        assertThat(Utils.maybeGetTcpUserTimeoutOption()).isNotNull();
    }
}