/* * Copyright 2015 The Netty Project * * The Netty Project 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 io.netty.channel.pool; import io.netty.bootstrap.Bootstrap; import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.Channel; import io.netty.channel.ChannelInboundHandlerAdapter; import io.netty.channel.ChannelInitializer; import io.netty.channel.EventLoopGroup; import io.netty.channel.local.LocalAddress; import io.netty.channel.local.LocalChannel; import io.netty.channel.local.LocalEventLoopGroup; import io.netty.channel.local.LocalServerChannel; import io.netty.util.concurrent.Future; import org.hamcrest.CoreMatchers; import org.junit.Rule; import org.junit.Test; import org.junit.rules.ExpectedException; import java.util.Queue; import java.util.concurrent.LinkedBlockingQueue; import static org.junit.Assert.*; public class SimpleChannelPoolTest { private static final String LOCAL_ADDR_ID = "test.id"; @Rule public ExpectedException expectedException = ExpectedException.none(); @Test public void testAcquire() throws Exception { EventLoopGroup group = new LocalEventLoopGroup(); LocalAddress addr = new LocalAddress(LOCAL_ADDR_ID); Bootstrap cb = new Bootstrap(); cb.remoteAddress(addr); cb.group(group) .channel(LocalChannel.class); ServerBootstrap sb = new ServerBootstrap(); sb.group(group) .channel(LocalServerChannel.class) .childHandler(new ChannelInitializer<LocalChannel>() { @Override public void initChannel(LocalChannel ch) throws Exception { ch.pipeline().addLast(new ChannelInboundHandlerAdapter()); } }); // Start server Channel sc = sb.bind(addr).sync().channel(); CountingChannelPoolHandler handler = new CountingChannelPoolHandler(); ChannelPool pool = new SimpleChannelPool(cb, handler); Channel channel = pool.acquire().sync().getNow(); pool.release(channel).syncUninterruptibly(); Channel channel2 = pool.acquire().sync().getNow(); assertSame(channel, channel2); assertEquals(1, handler.channelCount()); pool.release(channel2).syncUninterruptibly(); // Should fail on multiple release calls. try { pool.release(channel2).syncUninterruptibly(); fail(); } catch (IllegalArgumentException e) { // expected assertFalse(channel.isActive()); } assertEquals(1, handler.acquiredCount()); assertEquals(2, handler.releasedCount()); sc.close().sync(); group.shutdownGracefully(); } @Test public void testBoundedChannelPoolSegment() throws Exception { EventLoopGroup group = new LocalEventLoopGroup(); LocalAddress addr = new LocalAddress(LOCAL_ADDR_ID); Bootstrap cb = new Bootstrap(); cb.remoteAddress(addr); cb.group(group) .channel(LocalChannel.class); ServerBootstrap sb = new ServerBootstrap(); sb.group(group) .channel(LocalServerChannel.class) .childHandler(new ChannelInitializer<LocalChannel>() { @Override public void initChannel(LocalChannel ch) throws Exception { ch.pipeline().addLast(new ChannelInboundHandlerAdapter()); } }); // Start server Channel sc = sb.bind(addr).sync().channel(); CountingChannelPoolHandler handler = new CountingChannelPoolHandler(); ChannelPool pool = new SimpleChannelPool(cb, handler, ChannelHealthChecker.ACTIVE) { private final Queue<Channel> queue = new LinkedBlockingQueue<Channel>(1); @Override protected Channel pollChannel() { return queue.poll(); } @Override protected boolean offerChannel(Channel ch) { return queue.offer(ch); } }; Channel channel = pool.acquire().sync().getNow(); Channel channel2 = pool.acquire().sync().getNow(); pool.release(channel).syncUninterruptibly().getNow(); try { pool.release(channel2).syncUninterruptibly(); fail(); } catch (IllegalStateException e) { // expected } channel2.close().sync(); assertEquals(2, handler.channelCount()); assertEquals(0, handler.acquiredCount()); assertEquals(1, handler.releasedCount()); sc.close().sync(); channel.close().sync(); channel2.close().sync(); group.shutdownGracefully(); } /** * Tests that if channel was unhealthy it is not offered back to the pool. * * @throws Exception */ @Test public void testUnhealthyChannelIsNotOffered() throws Exception { EventLoopGroup group = new LocalEventLoopGroup(); LocalAddress addr = new LocalAddress(LOCAL_ADDR_ID); Bootstrap cb = new Bootstrap(); cb.remoteAddress(addr); cb.group(group) .channel(LocalChannel.class); ServerBootstrap sb = new ServerBootstrap(); sb.group(group) .channel(LocalServerChannel.class) .childHandler(new ChannelInitializer<LocalChannel>() { @Override public void initChannel(LocalChannel ch) throws Exception { ch.pipeline().addLast(new ChannelInboundHandlerAdapter()); } }); // Start server Channel sc = sb.bind(addr).syncUninterruptibly().channel(); ChannelPoolHandler handler = new CountingChannelPoolHandler(); ChannelPool pool = new SimpleChannelPool(cb, handler); Channel channel1 = pool.acquire().syncUninterruptibly().getNow(); pool.release(channel1).syncUninterruptibly(); Channel channel2 = pool.acquire().syncUninterruptibly().getNow(); //first check that when returned healthy then it actually offered back to the pool. assertSame(channel1, channel2); expectedException.expect(IllegalStateException.class); channel1.close().syncUninterruptibly(); try { pool.release(channel1).syncUninterruptibly(); } finally { sc.close().syncUninterruptibly(); channel2.close().syncUninterruptibly(); group.shutdownGracefully(); } } /** * Tests that if channel was unhealthy it is was offered back to the pool because * it was requested not to validate channel health on release. * * @throws Exception */ @Test public void testUnhealthyChannelIsOfferedWhenNoHealthCheckRequested() throws Exception { EventLoopGroup group = new LocalEventLoopGroup(); LocalAddress addr = new LocalAddress(LOCAL_ADDR_ID); Bootstrap cb = new Bootstrap(); cb.remoteAddress(addr); cb.group(group) .channel(LocalChannel.class); ServerBootstrap sb = new ServerBootstrap(); sb.group(group) .channel(LocalServerChannel.class) .childHandler(new ChannelInitializer<LocalChannel>() { @Override public void initChannel(LocalChannel ch) throws Exception { ch.pipeline().addLast(new ChannelInboundHandlerAdapter()); } }); // Start server Channel sc = sb.bind(addr).syncUninterruptibly().channel(); ChannelPoolHandler handler = new CountingChannelPoolHandler(); ChannelPool pool = new SimpleChannelPool(cb, handler, ChannelHealthChecker.ACTIVE, false); Channel channel1 = pool.acquire().syncUninterruptibly().getNow(); channel1.close().syncUninterruptibly(); Future<Void> releaseFuture = pool.release(channel1, channel1.eventLoop().<Void>newPromise()).syncUninterruptibly(); assertThat(releaseFuture.isSuccess(), CoreMatchers.is(true)); Channel channel2 = pool.acquire().syncUninterruptibly().getNow(); //verifying that in fact the channel2 is different that means is not pulled from the pool assertNotSame(channel1, channel2); sc.close().syncUninterruptibly(); channel2.close().syncUninterruptibly(); group.shutdownGracefully(); } @Test public void testBootstrap() { final SimpleChannelPool pool = new SimpleChannelPool(new Bootstrap(), new CountingChannelPoolHandler()); try { // Checking for the actual bootstrap object doesn't make sense here, since the pool uses a copy with a // modified channel handler. assertNotNull(pool.bootstrap()); } finally { pool.close(); } } @Test public void testHandler() { final ChannelPoolHandler handler = new CountingChannelPoolHandler(); final SimpleChannelPool pool = new SimpleChannelPool(new Bootstrap(), handler); try { assertSame(handler, pool.handler()); } finally { pool.close(); } } @Test public void testHealthChecker() { final ChannelHealthChecker healthChecker = ChannelHealthChecker.ACTIVE; final SimpleChannelPool pool = new SimpleChannelPool( new Bootstrap(), new CountingChannelPoolHandler(), healthChecker); try { assertSame(healthChecker, pool.healthChecker()); } finally { pool.close(); } } @Test public void testReleaseHealthCheck() { final SimpleChannelPool healthCheckOnReleasePool = new SimpleChannelPool( new Bootstrap(), new CountingChannelPoolHandler(), ChannelHealthChecker.ACTIVE, true); try { assertTrue(healthCheckOnReleasePool.releaseHealthCheck()); } finally { healthCheckOnReleasePool.close(); } final SimpleChannelPool noHealthCheckOnReleasePool = new SimpleChannelPool( new Bootstrap(), new CountingChannelPoolHandler(), ChannelHealthChecker.ACTIVE, false); try { assertFalse(noHealthCheckOnReleasePool.releaseHealthCheck()); } finally { noHealthCheckOnReleasePool.close(); } } }