In the end, this is what I did. I created a FakeChannel extending AbstractChannel, with a FakeChannelSink that simply captured all events sent downstream in a queue. At the other end of the pipeline I created a simple upstream handler that captured all upstream events to a queue. This then allowed me to test channel handlers which selectively forwarded events up and down the stack, and the order in which events were forwarded, without getting tied up in mocking multiple objects which returned other mocks (i.e. ChannelContextHandler returning Channel, etc).<div>
<br></div><div>Iain<br><div><br><div class="gmail_quote">On Thu, Aug 6, 2009 at 7:12 AM, &quot;이희승 (Trustin Lee)&quot; <span dir="ltr">&lt;<a href="mailto:trustin@gmail.com">trustin@gmail.com</a>&gt;</span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex;">
Hi Iain,<br>
<br>
Because Channel or other interfaces have many operations and hold<br>
somewhat complicated state, it will not be easy to implement a mock with<br>
jMock or EasyMock.  Instead, I&#39;d recommend to extending an &#39;Abstract*&#39;<br>
classes in Netty.  You might want to refer to the codec embedder source<br>
code, which simplifies the unit testing of codec handlers.<br>
<br>
HTH,<br>
Trustin<br>
<div><div></div><div class="h5"><br>
On 07/28/2009 05:42 AM, Iain McGinniss wrote:<br>
&gt; Hi guys,<br>
&gt;<br>
&gt; I&#39;ve been trying to implement the binary protocol used in JXTA using<br>
&gt; Netty for the past week or so (mixed in with trying to implement an HTTP<br>
&gt; tunnel that doesn&#39;t use servlets, but that&#39;s parked for the moment). In<br>
&gt; doing this I&#39;ve been trying to unit test my channel handlers, which I<br>
&gt; have found to be quite challenging. I&#39;m starting to wonder whether this<br>
&gt; is because I am taking the wrong approach. First, I&#39;ll give a brief<br>
&gt; overview of the protocol:<br>
&gt;<br>
&gt; 1. When a connection is opened, I must send a &quot;welcome&quot; message. This<br>
&gt; contains some dynamic data used to identify myself, such as a unique ID,<br>
&gt; the version of the protocol I am using and some meta-data about how the<br>
&gt; connection should be used.<br>
&gt; 2. While this is happening, the peer on the other end will be sending<br>
&gt; their welcome message. This message ends with CRLF.<br>
&gt; 3. If I don&#39;t receive a full welcome message from the other peer within<br>
&gt; 5 seconds, or if it is invalid in some way (wrong protocol version,<br>
&gt; unsupported modes specified in the meta-data, etc), then I must close<br>
&gt; the connection.<br>
&gt; 4. I cannot send any messages to the other peer until I have received a<br>
&gt; full, valid welcome message from them.<br>
&gt; 5. Once this welcome message handshake is complete, I can send messages<br>
&gt; to the peer. These messages are structured somewhat like HTTP requests,<br>
&gt; except in binary with known binary keys mapping to known binary values.<br>
&gt; One of these &quot;headers&quot; specifies the length of the rest of the message,<br>
&gt; so as a receiver I can predict how long the message is, preallocate a<br>
&gt; buffer of the correct size, read the message and then process it.<br>
&gt; 6. Either peer can close the connection at the end of any message,<br>
&gt; without any kind of &quot;goodbye&quot; message required.<br>
&gt;<br>
&gt; In order to deal with this protocol I&#39;ve created a ChannelHandler,<br>
&gt; extending SimpleChannelHandler.<br>
&gt;<br>
&gt; 1. I override the channelConnected() method in order to send my welcome<br>
&gt; message, and I do not allow the connected event to bubble up the<br>
&gt; pipeline at this point.<br>
&gt; 2. I override the messageReceived() method to process the welcome<br>
&gt; message, and call channel.close() within this if the welcome message is<br>
&gt; invalid or not what I expect. Once I receive a full welcome message, I<br>
&gt; fire my own UpstreamChannelStateEvent to state we are connected<br>
&gt; (properly). I actually don&#39;t know if this is &quot;legal&quot; or best practice.<br>
&gt; 3. I override the writeRequested() method to perform the framing of the<br>
&gt; ChannelBuffers containing full messages passed down from the previous<br>
&gt; handler which has done the serialization.<br>
&gt;<br>
&gt; Now, unit testing this has proved challenging. For instance, take the<br>
&gt; simple unit test of &quot;when channelConnected() is called, write a welcome<br>
&gt; message to the channel&quot;. The implementation of channelConnected is:<br>
&gt;<br>
&gt; public void channelConnected(ChannelHandlerContext ctx,<br>
&gt; ChannelStateEvent e) throws Exception {<br>
&gt;     SocketAddress remoteAddress = ctx.getChannel().getRemoteAddress();<br>
&gt;     EndpointAddress destAddr = new<br>
&gt; EndpointAddress(endpointAddr.getProtocolName(),<br>
&gt; remoteAddress.toString(), null, null);<br>
&gt;     WelcomeMessage welcome = new WelcomeMessage(destAddr, endpointAddr,<br>
&gt; peerId, false);<br>
&gt;     ChannelBuffer welcomeBytes =<br>
&gt; ChannelBuffers.copiedBuffer(welcome.getByteBuffer());<br>
&gt;     Channels.write(ctx, Channels.future(ctx.getChannel()), welcomeBytes);<br>
&gt; }<br>
&gt;<br>
&gt; My approach so far to unit testing this has been to use jMock 2,<br>
&gt; producing tests blocks like this:<br>
&gt;<br>
&gt; @Test<br>
&gt; public void testSendsWelcomeMessageImmediately() throws Exception {<br>
&gt;     final UpstreamChannelStateEvent ev = new<br>
&gt; UpstreamChannelStateEvent(channel, ChannelState.CONNECTED, true);<br>
&gt;     final WelcomeMessage expectedWelcomeMessage = new<br>
&gt; WelcomeMessage(REMOTE_ENDPOINT_ADDR, LOCAL_ENDPOINT_ADDR, LOCAL_PEER_ID,<br>
&gt; false);<br>
&gt;<br>
&gt;     mockContext.checking(new Expectations() {{<br>
&gt;         ignoring(ctx).getChannel(); will(returnValue(channel));<br>
&gt;         atLeast(1).of(channel).getRemoteAddress();<br>
&gt; will(returnValue(REMOTE_SOCK_ADDR));<br>
&gt;<br>
&gt;  one(ctx).sendDownstream(with(aDownstreamMessageEvent(aWelcomeMessage(expectedWelcomeMessage))));<br>
&gt;     }});<br>
&gt;<br>
&gt;     handler.channelConnected(ctx, ev);<br>
&gt; }<br>
&gt;<br>
&gt; The real sticking point was the &quot;aDownstreamMessageEvent&quot; and<br>
&gt; &quot;aWelcomeMessage&quot; Matchers I had to write by hand. The various event<br>
&gt; types in Netty do not implement equals() so I cannot do something like:<br>
&gt;<br>
&gt; DownstreamMessageEvent expectedEvent = new DownstreamMessageEvent(...);<br>
&gt; one(ctx).sendDownstream(with(equals(expectedDownStreamMessage));<br>
&gt;<br>
&gt; Additionally, in other test cases where I want to assert that my code<br>
&gt; fires events like an UpstreamChannelStateEvent with State=CONNECTED,<br>
&gt; Value=TRUE, I can&#39;t do this using equals() either. Instead, I must write<br>
&gt; custom matchers to compensate for this. All this results in, for a<br>
&gt; simple method like channelConnected(), roughly 5-6 times the amount of<br>
&gt; code required to write a test compared to the code itself (aggravated by<br>
&gt; most aspects of JXTA being difficult to test too, and using NIO<br>
&gt; ByteBuffers rather than ChannelBuffers... sigh). This will get easier<br>
&gt; over time as I start building more matchers for jmock specifically for<br>
&gt; Netty, but I still can&#39;t help feeling that there must be an easier way<br>
&gt; to test ChannelHandlers in isolation.<br>
&gt;<br>
&gt; Do you guys have any advice on how best to test Netty code?<br>
&gt; Iain<br>
&gt;<br>
&gt;<br>
</div></div>&gt; ------------------------------------------------------------------------<br>
&gt;<br>
&gt; _______________________________________________<br>
&gt; netty-users mailing list<br>
&gt; <a href="mailto:netty-users@lists.jboss.org">netty-users@lists.jboss.org</a><br>
&gt; <a href="https://lists.jboss.org/mailman/listinfo/netty-users" target="_blank">https://lists.jboss.org/mailman/listinfo/netty-users</a><br>
<br>
_______________________________________________<br>
netty-users mailing list<br>
<a href="mailto:netty-users@lists.jboss.org">netty-users@lists.jboss.org</a><br>
<a href="https://lists.jboss.org/mailman/listinfo/netty-users" target="_blank">https://lists.jboss.org/mailman/listinfo/netty-users</a><br>
</blockquote></div><br></div></div>