com.l2jmobius.commons.network.codecs.PacketDecoder.java Source code

Java tutorial

Introduction

Here is the source code for com.l2jmobius.commons.network.codecs.PacketDecoder.java

Source

/*
 * This file is part of the L2J Mobius project.
 * 
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 * 
 * This program 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
 * General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */
package com.l2jmobius.commons.network.codecs;

import java.nio.ByteOrder;
import java.util.List;
import java.util.logging.Logger;

import com.l2jmobius.commons.network.IConnectionState;
import com.l2jmobius.commons.network.IIncomingPacket;
import com.l2jmobius.commons.network.IIncomingPackets;
import com.l2jmobius.commons.network.PacketReader;

import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.ByteToMessageDecoder;

/**
 * @author Nos
 * @param <T>
 */
public class PacketDecoder<T> extends ByteToMessageDecoder {
    private static final Logger LOGGER = Logger.getLogger(PacketDecoder.class.getName());

    private final ByteOrder _byteOrder;
    private final IIncomingPackets<T>[] _incomingPackets;
    private final T _client;

    public PacketDecoder(ByteOrder byteOrder, IIncomingPackets<T>[] incomingPackets, T client) {
        _byteOrder = byteOrder;
        _incomingPackets = incomingPackets;
        _client = client;
    }

    @Override
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) {
        if ((in == null) || !in.isReadable()) {
            return;
        }

        if (in.order() != _byteOrder) {
            in = in.order(_byteOrder);
        }

        try {
            final short packetId = in.readUnsignedByte();
            if (packetId >= _incomingPackets.length) {
                LOGGER.finer("Unknown packet: " + Integer.toHexString(packetId));
                return;
            }

            final IIncomingPackets<T> incomingPacket = _incomingPackets[packetId];
            if (incomingPacket == null) {
                LOGGER.finer("Unknown packet: " + Integer.toHexString(packetId));
                return;
            }

            final IConnectionState connectionState = ctx.channel().attr(IConnectionState.ATTRIBUTE_KEY).get();
            if ((connectionState == null) || !incomingPacket.getConnectionStates().contains(connectionState)) {
                // LOGGER.warning(incomingPacket + ": Connection at invalid state: " + connectionState + " Required States: " + incomingPacket.getConnectionStates());
                return;
            }

            final IIncomingPacket<T> packet = incomingPacket.newIncomingPacket();
            if ((packet != null) && packet.read(_client, new PacketReader(in))) {
                out.add(packet);
            }
        } finally {
            // We always consider that we read whole packet.
            in.readerIndex(in.writerIndex());
        }
    }
}