Package net.gleamynode.netty.handler.codec.replay

Source Code of net.gleamynode.netty.handler.codec.replay.ReplayingDecoder

/*
* Copyright (C) 2008  Trustin Heuiseung Lee
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, 5th Floor, Boston, MA 02110-1301 USA
*/
package net.gleamynode.netty.handler.codec.replay;

import net.gleamynode.netty.array.ByteArray;
import net.gleamynode.netty.array.ByteArrayBuffer;
import net.gleamynode.netty.channel.Channel;
import net.gleamynode.netty.channel.ChannelHandlerContext;
import net.gleamynode.netty.channel.ChannelPipelineCoverage;
import net.gleamynode.netty.channel.ChannelStateEvent;
import net.gleamynode.netty.channel.ChannelUtil;
import net.gleamynode.netty.channel.ExceptionEvent;
import net.gleamynode.netty.channel.MessageEvent;
import net.gleamynode.netty.channel.SimpleChannelHandler;

/**
* @author The Netty Project (netty@googlegroups.com)
* @author Trustin Lee (trustin@gmail.com)
*
* @version $Rev: 475 $, $Date: 2008-07-04 13:57:00 +0900 (Fri, 04 Jul 2008) $
*
*/
@ChannelPipelineCoverage("one")
public abstract class ReplayingDecoder extends SimpleChannelHandler {

    static final Error REWIND = new Rewind();

    private volatile ReplayableByteArrayBuffer cumulation = new ReplayableByteArrayBuffer();

    @Override
    public void messageReceived(
            ChannelHandlerContext ctx, MessageEvent e) throws Exception {

        Object m = e.getMessage();
        if (!(m instanceof ByteArray)) {
            ctx.sendUpstream(e);
            return;
        }

        ByteArray input = (ByteArray) m;
        if (input.empty()) {
            return;
        }

        ReplayableByteArrayBuffer cumulation = this.cumulation;

        // Avoid CompositeByteArray index overflow.
        if (Integer.MAX_VALUE - cumulation.endIndex() < input.length()) {
            ReplayableByteArrayBuffer newCumulation = new ReplayableByteArrayBuffer();
            for (ByteArray component: cumulation) {
                newCumulation.unwrap().write(component);
            }
            this.cumulation = cumulation = newCumulation;
        }

        cumulation.unwrap().write(input);
        callDecode(ctx, e.getChannel(), cumulation);
    }

    @Override
    public void channelDisconnected(
            ChannelHandlerContext ctx, ChannelStateEvent e) throws Exception {
        cleanup(ctx, e);
    }

    @Override
    public void exceptionCaught(
            ChannelHandlerContext ctx, ExceptionEvent e) throws Exception {
        ctx.sendUpstream(e);
    }

    private void callDecode(ChannelHandlerContext context,
            Channel channel, ReplayableByteArrayBuffer cumulation) throws Exception {
        while (!cumulation.empty()) {
            int oldFirstIndex = cumulation.unwrap().firstIndex();
            Object result = null;
            try {
                result = decode(context, channel, cumulation);
                // Successfully decoded a message; clean up recorded results.
                if (result != null) {
                    clear();
                } else if (oldFirstIndex == cumulation.unwrap().firstIndex()) {
                    throw new IllegalStateException(
                            "null cannot be returned if no data is consumed.");
                } else {
                    // Previous data has been discarded.
                    // Probably it's reading on.
                    clear();
                    continue;
                }
            } catch (Rewind rewind) {
                // Rewound
            }

            if (result == null) {
                // Seems like more data is required.
                // Let's wait for the next notification.
                break;
            }

            if (oldFirstIndex == cumulation.unwrap().firstIndex()) {
                throw new IllegalStateException(
                        "decode() method must consume at least one byte " +
                        "if it returned a decoded message.");
            }
            ChannelUtil.fireMessageReceived(context, channel, result);
        }
    }

    private void cleanup(ChannelHandlerContext ctx, ChannelStateEvent e)
            throws Exception {
        // Make sure all frames were read before notifying a closed channel.
        callDecode(ctx, e.getChannel(), cumulation);
        ctx.sendUpstream(e);
    }

    protected void rewind() {
        cumulation.rewind();
    }

    protected void clear() {
        cumulation.clear();
    }

    protected abstract Object decode(
            ChannelHandlerContext ctx, Channel channel, ByteArrayBuffer buffer) throws Exception;
}
TOP

Related Classes of net.gleamynode.netty.handler.codec.replay.ReplayingDecoder

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.