org.jfxvnc.net.rfb.codec.handshaker.RfbClient33Decoder.java Source code

Java tutorial

Introduction

Here is the source code for org.jfxvnc.net.rfb.codec.handshaker.RfbClient33Decoder.java

Source

/*******************************************************************************
 * Copyright (c) 2016 comtel inc.
 *
 * 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 org.jfxvnc.net.rfb.codec.handshaker;

import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
import java.util.List;

import org.jfxvnc.net.rfb.codec.PixelFormat;
import org.jfxvnc.net.rfb.codec.handshaker.event.SecurityResultEvent;
import org.jfxvnc.net.rfb.codec.handshaker.event.SecurityTypesEvent;
import org.jfxvnc.net.rfb.codec.handshaker.event.ServerInitEvent;
import org.jfxvnc.net.rfb.codec.security.SecurityType;
import org.jfxvnc.net.rfb.codec.security.vncauth.VncAuthSecurityMessage;
import org.jfxvnc.net.rfb.exception.ProtocolException;
import org.jfxvnc.net.rfb.exception.SecurityException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

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

class RfbClient33Decoder extends ReplayingDecoder<RfbClient33Decoder.State> implements RfbClientDecoder {

    private static Logger logger = LoggerFactory.getLogger(RfbClient33Decoder.class);

    protected final Charset ASCII = StandardCharsets.US_ASCII;

    public RfbClient33Decoder() {
        super(State.SEC_TYPES);
    }

    enum State {
        SEC_TYPES, VNC_AUTH, SEC_RESULT, SERVER_INIT
    }

    @Override
    protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {
        switch (state()) {
        case SEC_TYPES:
            int type = in.readInt();
            SecurityType securityType = SecurityType.valueOf(type);
            logger.debug("supported security type: {}", securityType);
            checkpoint(securityType == SecurityType.VNC_Auth ? State.VNC_AUTH : State.SERVER_INIT);
            out.add(new SecurityTypesEvent(false, securityType));
            break;
        case VNC_AUTH:
            byte[] challenge = new byte[16];
            in.readBytes(challenge);
            out.add(new VncAuthSecurityMessage(challenge));
            checkpoint(State.SEC_RESULT);
            break;
        case SEC_RESULT:
            int secResult = in.readInt();
            logger.debug("server login {}", secResult == 0 ? "succeed" : "failed");
            if (secResult == 1) {
                int length = in.readInt();
                if (length == 0) {
                    out.add(new SecurityResultEvent(false, new ProtocolException("decode error message failed")));
                    return;
                }
                byte[] text = new byte[length];
                in.readBytes(text);
                out.add(new SecurityResultEvent(false, new SecurityException(new String(text, ASCII))));
                return;
            }
            out.add(new SecurityResultEvent(true));
            checkpoint(State.SERVER_INIT);
            break;
        case SERVER_INIT:
            ServerInitEvent initMsg = new ServerInitEvent();

            initMsg.setFrameBufferWidth(in.readUnsignedShort());
            initMsg.setFrameBufferHeight(in.readUnsignedShort());

            initMsg.setPixelFormat(parsePixelFormat(in));

            byte[] name = new byte[in.readInt()];
            in.readBytes(name);

            initMsg.setServerName(new String(name, ASCII));
            out.add(initMsg);
            break;
        default:
            break;
        }
    }

    protected void decodeErrorMessage(ChannelHandlerContext ctx, ByteBuf in) {
        if (!in.isReadable()) {
            ctx.fireExceptionCaught(new ProtocolException("decode error message failed"));
            return;
        }

        byte[] reason = new byte[actualReadableBytes()];
        in.readBytes(reason);
        String error = new String(reason, ASCII);
        ctx.fireExceptionCaught(new ProtocolException(error.trim()));
    }

    protected PixelFormat parsePixelFormat(ByteBuf m) {

        PixelFormat pf = new PixelFormat();

        pf.setBitPerPixel(m.readUnsignedByte());
        pf.setDepth(m.readUnsignedByte());
        pf.setBigEndian(m.readUnsignedByte() == 1);
        pf.setTrueColor(m.readUnsignedByte() == 1);
        pf.setRedMax(m.readUnsignedShort());
        pf.setGreenMax(m.readUnsignedShort());
        pf.setBlueMax(m.readUnsignedShort());

        pf.setRedShift(m.readUnsignedByte());
        pf.setGreenShift(m.readUnsignedByte());
        pf.setBlueShift(m.readUnsignedByte());
        m.skipBytes(3);

        return pf;
    }

}