1、本代码需要一定java基础

Netty原理

我们都知道 Netty 是一个高性能、异步事件驱动的 NIO 框架,基于 JAVA NIO 提供的 API 实现

2、代码文件目录

java netty转发 netty 转发服务器_java netty转发

直接上代码

1、往BBWConfig.properties写入服务信息

package com.com.test.bbw;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.PropertySource;
import org.springframework.stereotype.Component;

import java.io.*;
import java.util.Properties;

@Component
@ConfigurationProperties
@PropertySource("classpath:BBWConfig.properties")
public class BBWConfig {

    private static final Log logger = LogFactory.getLog(BBWConfig.class);


    private String config_path = "src/main/resources/";   //本地路径
    private String config_filename = "BBWConfig.properties";
    private static Properties properties;

    private static String bbwServerIp;   //BBW服务器地址
    private static String bbwServerPort; //BBW服务器端口
    private static String paySendPort;   //支付发出端口
    private static String payRcvServerIp; //支付接收地址
    private static String payTcpRcvPort; //tcp支付接收端口
    private static String payHttpRcvPort; //http支付接收端口
    private static String warningNum; //AGENt重连次数警告consumer
    private static String encoding;      //报文编码格式
    private static String reStartTime;   //AGNET异常重连间隔(单位:毫秒)


    // Netty服务连接状态:连接中
    public static final int CONNECT_STATE_CONNECTING = 0;
    //  Netty服务连接状态:连接成功
    public static final int CONNECT_STATE_SUCC = 1;
    //  Netty服务连接状态:连接失败
    public static final int CONNECT_STATE_FAIL = -1;

    public BBWConfig() {
        String home =System.getProperty("user.dir");
        // set config file path
        if (null != home && !"".equals(home.trim())){
            if (!home.endsWith("/")){
                home = home + "/";
            }
            config_filename = home + config_path + config_filename;
        }
        else{
            config_filename = config_path + config_filename;
        }
        // check config file
        creatConfigFile();
        this.properties = getConfigFile();
        init();
    }

    private Properties getConfigFile() {
        properties = new Properties();
        try {
            FileReader fileReader = new FileReader(config_filename);
            properties.load(fileReader);
            fileReader.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return properties;
    }

    public static Properties getProperties() {
        return properties;
    }

    public static void setProperties(Properties properties) {
        BBWConfig.properties = properties;
    }

    public static String getBbwServerIp() {
        return bbwServerIp;
    }

    public static void setBbwServerIp(String bbwServerIp) {
        BBWConfig.bbwServerIp = bbwServerIp;
    }

    public static String getBbwServerPort() {
        return bbwServerPort;
    }

    public static void setBbwServerPort(String bbwServerPort) {
        BBWConfig.bbwServerPort = bbwServerPort;
    }

    public static String getPaySendPort() {
        return paySendPort;
    }

    public static void setPaySendPort(String paySendPort) {
        BBWConfig.paySendPort = paySendPort;
    }

    public static String getPayRcvServerIp() {
        return payRcvServerIp;
    }

    public static void setPayRcvServerIp(String payRcvServerIp) {
        BBWConfig.payRcvServerIp = payRcvServerIp;
    }

    public static String getPayTcpRcvPort() {
        return payTcpRcvPort;
    }

    public static void setPayTcpRcvPort(String payTcpRcvPort) {
        BBWConfig.payTcpRcvPort = payTcpRcvPort;
    }

    public static String getPayHttpRcvPort() {
        return payHttpRcvPort;
    }

    public static void setPayHttpRcvPort(String payHttpRcvPort) {
        BBWConfig.payHttpRcvPort = payHttpRcvPort;
    }

    public static String getWarningNum() {
        return warningNum;
    }

    public static void setWarningNum(String warningNum) {
        BBWConfig.warningNum = warningNum;
    }

    public static String getEncoding() {
        return encoding;
    }

    public static void setEncoding(String encoding) {
        BBWConfig.encoding = encoding;
    }

    public static String getReStartTime() {
        return reStartTime;
    }

    public static void setReStartTime(String reStartTime) {
        BBWConfig.reStartTime = reStartTime;
    }

    /**
     *
     * @说明 创建默认的配置文件
     */
    private File creatConfigFile() {
        File file = new File(config_filename);

        if (logger.isInfoEnabled()) {
            logger.info("BBWConfig config_filename = " + config_filename);
        }

        if (!file.exists()) {
            if (logger.isInfoEnabled()) {
                logger.info("BBWConfig start creatConfigFile  ..............");
            }
            try {
                file.createNewFile();
                OutputStreamWriter osw = new OutputStreamWriter(
                        new FileOutputStream(file));
                osw.write("#BBW服务器地址" + "\n");
                osw.write("bbwServerIp=127.0.0.1" + "\n");
                osw.write("#BBW服务器端口" + "\n");
                osw.write("bbwServerPort=4444" + "\n");
                osw.write("#支付接出端口Tcp" + "\n");
                osw.write("paySendPort=50039" + "\n");
                osw.write("#支付接入平台地址" + "\n");
                osw.write("payRcvServerIp=127.0.0.1" + "\n");
                osw.write("#支付平台接入端口tcp" + "\n");
                osw.write("payTcpRcvPort=50040" + "\n");
                osw.write("#支付平台接入http端口" + "\n");
                osw.write("payHttpRcvPort=9004" + "\n");
                osw.write("#AGENt重连次数警告consumer" + "\n");
                osw.write("warningNum=10" + "\n");
                osw.write("#报文编码格式" + "\n");
                osw.write("encoding=UTF8" + "\n");
                osw.write("#AGNET异常重连间隔(单位:毫秒)" + "\n");
                osw.write("reStartTime=10000" + "\n");
                osw.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
        return file;
    }
    private void init(){
        BBWConfig.bbwServerIp=BBWConfig.properties.getProperty("bbwServerIp");
        BBWConfig.bbwServerPort=BBWConfig.properties.getProperty("bbwServerPort");
        BBWConfig.paySendPort=BBWConfig.properties.getProperty("paySendPort");
        BBWConfig.payRcvServerIp=BBWConfig.properties.getProperty("payRcvServerIp");
        BBWConfig.payTcpRcvPort=BBWConfig.properties.getProperty("payTcpRcvPort");
        BBWConfig.payHttpRcvPort=BBWConfig.properties.getProperty("payHttpRcvPort");
        BBWConfig.encoding=BBWConfig.properties.getProperty("encoding");
        BBWConfig.warningNum=BBWConfig.properties.getProperty("warningNum");
    }
    public static void main(String[] args) {
        BBWConfig nc = new BBWConfig();
        if (logger.isDebugEnabled()) {
            logger.debug("BBWConfig toString \n" + nc.getConfigFile().toString());
        }
    }

}

 2、读取数据工具类

package com.com.test.bbw;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;

public class ConverUtils {

    private static final Logger log = LoggerFactory.getLogger(ConverUtils.class);

    private static final Integer msgLeng = 5;

    /**
     * 循环去写数据
     *
     * @param bos
     * @param buff
     * @param bufferCapacity
     * @throws IOException
     */
    public static void cycleWriteMessage(BufferedOutputStream bos, byte[] buff, int bufferCapacity) throws IOException {
        int len = buff.length;

        int count = 0;
        int tmpLen = len;
        while (true) {
            if (tmpLen <= bufferCapacity) {
                bos.write(buff, count, tmpLen);
                bos.flush();
                count += tmpLen;
                tmpLen = 0;
            } else {
                bos.write(buff, count, bufferCapacity);
                bos.flush();
                count += bufferCapacity;
                tmpLen -= bufferCapacity;
            }
            if (tmpLen == 0)
                break;
        }
    }

    /**
     * 循环去读数据
     *
     * @param bis
     * @param baos
     * @param bufferCapacity
     * @param len
     * @param socketReaderCircleTimes
     * @throws IOException
     */
    public static String cycleReadMessage(BufferedInputStream bis, StringBuffer msgInfo, int bufferCapacity, int len,
                                          int socketReaderCircleTimes) throws IOException {
        int index = 0;
        // 读取的明文长度
        int numberRead = 0;
        // 读取的字节长度
        int numberReadByte = 0;
        // 明文的长度变量
        int tmpLen = len;
        byte[] buff = null;
        // 定义可变字符串用于存储取到的明文数据
        StringBuffer msgTemp = new StringBuffer();
        for (int i = 0; i < socketReaderCircleTimes; i++) {
            buff = new byte[tmpLen];
            numberReadByte = bis.read(buff, 0, tmpLen);
            if (numberReadByte <= 0) {
                break;
            }
            // 拿到的字节数据转为16进制数据
            String asciiBytesToMsg = byteArr2HexStr(buff);
            String msg = convertHexToString(asciiBytesToMsg);
            msgTemp.append(msg);
            // 判断长度
            numberRead = asciiBytesToMsg.length();
            // 如果明文长度小于剩余报文头长度
            if (numberRead < tmpLen) {
                index += numberRead;
                tmpLen -= numberRead;
            } else {
                if (numberRead != tmpLen)
                    break;
                index += numberRead;
                tmpLen -= numberRead;
            }

            if (tmpLen <= 0)
                break;
        }
        // 判断长度
        if (msgTemp.length() == len) {
            msgInfo.append(msgTemp);
            return msgInfo.toString();
        }
        return null;
    }



    /**
     * 将两个字节数组合并
     *
     * @param data1
     * @param data2
     * @return
     */
    public static byte[] addBytes(byte[] headData, byte[] bodyData) {
        byte[] data3 = null;
        if(null != bodyData && 0 < bodyData.length){
            data3 = new byte[headData.length + bodyData.length];
            System.arraycopy(headData, 0, data3, 0, headData.length);
            System.arraycopy(bodyData, 0, data3, headData.length, bodyData.length);
        }else {
            data3 = "00000".getBytes();
        }
        return data3;
    }

    /**
     * 循环去读报文的长度
     *
     * @param bis
     * @param baos
     * @param bufferCapacity
     * @param len
     * @param socketReaderCircleTimes
     * @throws IOException
     */
    public static void cycleReadMessageHeadLeng(BufferedInputStream bis, ByteArrayOutputStream baos, int bufferCapacity,
                                                int len, int socketReaderCircleTimes) throws IOException {

        int numberRead = 0;
        int tmpLen = len;
        int index = 0;
        byte[] buff = null;
        for (int i = 0; i < socketReaderCircleTimes; i++) {
            if (tmpLen < bufferCapacity) {
                buff = new byte[tmpLen];
                int available = bis.available();
                if(available >0){
                    numberRead = bis.read(buff, 0, tmpLen);
                }else{
                    continue;
                }
            } else {
                buff = new byte[bufferCapacity];
                numberRead = bis.read(buff, 0, bufferCapacity);
            }

            if (numberRead < 0) {
                return;
            }
            if (numberRead < tmpLen) {
                index += numberRead;
                tmpLen -= numberRead;

                byte[] tmpByte = new byte[numberRead];
                System.arraycopy(buff, 0, tmpByte, 0, numberRead);
                baos.write(tmpByte);
            } else {
                if (numberRead != tmpLen)
                    break;
                index += numberRead;
                tmpLen -= numberRead;
                baos.write(buff);
            }

            if (tmpLen <= 0)
                break;
        }
    }

    /**
     * 将报文的2位ASCII码字符码转为10进制
     *
     * @param bytes
     * @return
     */
    public static int decodeMsgLength(byte[] bytes) {
        if (bytes != null && bytes.length == 2) {

            return bytes2Short(bytes, 0);
        } else {
            log.error("bytes of length cannot decode to int value.");
        }
        return 0;
    }

    public static short bytes2Short(byte[] b, int offset) {
        short n = (short) (((b[offset] < 0 ? b[offset] + 256 : b[offset]) << 8)
                + (b[offset + 1] < 0 ? b[offset + 1] + 256 : b[offset + 1]));
        return n;
    }

    public static byte[] encode(int lengthValue, int length) {
        if (lengthValue > 0 && length > 0) {
            byte[] bytes;
            try {
                String hexStr = Integer.toHexString(lengthValue);
                if (hexStr.length() % 2 != 0) {
                    hexStr = "0" + hexStr;
                }
                bytes = hexstr2ByteArr(hexStr);
                byte[] ret = new byte[length];
                for (int i = 0; i < length; i++) {
                    ret[i] = 0x00;
                }
                System.arraycopy(bytes, 0, ret, length - bytes.length, bytes.length);
                return ret;
            } catch (Exception e) {
                log.error("报文长度转化出错");
            }
        } else {
            if(lengthValue == 0){
                return "00000".getBytes();
            }
            log.error("无效的报文长度:"+lengthValue);
        }
        return new byte[0];
    }

    public static byte[] hexstr2ByteArr(String strIn) {
        byte[] arrB = strIn.getBytes();
        int iLen = arrB.length;
        byte[] arrOut = new byte[iLen / 2];

        for (int i = 0; i < iLen; i += 2) {
            String strTmp = new String(arrB, i, 2);
            arrOut[i / 2] = (byte) Integer.parseInt(strTmp, 16);
        }

        return arrOut;
    }

    /**
     * hex字符串转byte数组
     *
     * @param inHex
     *            待转换的Hex字符串
     * @return 转换后的byte数组结果
     */
    public static byte[] hexToByteArray(String inHex) {
        if (inHex != null && !inHex.isEmpty()) {
            int hexlen = inHex.trim().length();
            byte[] result;
            if (hexlen % 2 == 1) {
                // 奇数

            } else {
                // 偶数
                result = new byte[(hexlen / 2)];
                int j = 0;
                for (int i = 0; i < hexlen; i += 2) {
                    result[j] = hexToByte(inHex.substring(i, i + 2));
                    j++;
                }
                return result;
            }
        }
        return new byte[0];

    }

    /**
     * Hex字符串转byte
     *
     * @param inHex
     *            待转换的Hex字符串
     * @return 转换后的byte
     */
    public static byte hexToByte(String inHex) {
        return (byte) Integer.parseInt(inHex, 16);
    }

    /**
     * ASCII码转换为16进制
     *
     * @param str
     * @return
     */
    public static String convertStringToHex(String str) {

        char[] chars = str.toCharArray();

        StringBuffer hex = new StringBuffer();
        for (int i = 0; i < chars.length; i++) {
            hex.append(Integer.toHexString((int) chars[i]));
        }

        return hex.toString();
    }

    /**
     * 将字节数组转为16进制数据
     *
     * @param arrB
     * @return
     * @throws IOException
     */
    public static String byteArr2HexStr(byte[] arrB) throws IOException {
        int iLen = arrB.length;
        StringBuffer sb = new StringBuffer(iLen * 2);

        for (int i = 0; i < iLen; ++i) {
            int intTmp;
            for (intTmp = arrB[i]; intTmp < 0; intTmp += 256) {
            }

            if (intTmp < 16) {
                sb.append("0");
            }

            sb.append(Integer.toString(intTmp, 16));
        }

        return sb.toString();
    }

    /**
     *
     * 16进制数据转为明文
     * @param arrB
     * @return
     * @throws IOException
     */
    public static String convertHexToString(String hex) {

        StringBuilder sb = new StringBuilder();
        StringBuilder temp = new StringBuilder();
        for (int i = 0; i < hex.length() - 1; i += 2) {
            String output = hex.substring(i, (i + 2));
            int decimal = Integer.parseInt(output, 16);
            sb.append((char) decimal);
            temp.append(decimal);
        }

        return sb.toString();
    }
    /**
     * 左补齐
     *
     * @param 原字符串
     * @param 左补字符
     * @param 补齐长度
     * @return 如果原字符串超长,返回原字符串
     */
    public static String lpad(String str, char ch, int len) {
        if (str.length() >= len) {
            return str;
        }

        char[] origChs = str.toCharArray();
        char[] chs = new char[len];
        int pos = len - origChs.length;

        for (int i = 0; i < len; i++) {
            if (i < pos) {
                chs[i] = ch;
            } else {
                chs[i] = origChs[i - pos];
            }
        }
        return new String(chs);
    }

}

 3、解码

package com.com.test.bbw;

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

import java.util.List;

public class MsgPckDecode extends MessageToMessageDecoder<ByteBuf> {
	//明文长度
	private static final Integer msgLeng = 5;
	@Override
	protected void decode(ChannelHandlerContext ctx, ByteBuf msg, List<Object> out) throws Exception {

		final byte[] headMessage;
		final byte[] contentMessage;
		headMessage = new byte[2];
		// 定义可变字符串用于存储取到的明文数据
		StringBuffer msgTemp = new StringBuffer();
		//获取报文长度
		msg.getBytes(msg.readerIndex(), headMessage, 0, 2);
		// 将两位报文ASII码数据转为10进制长度
		int msgLength = ConverUtils.decodeMsgLength(headMessage);
		String msgLengthStr = msgLength+"";
		msgLengthStr = ConverUtils.lpad(msgLengthStr, '0', msgLeng);
		msgTemp.append(msgLengthStr);
        //根据报文长度读取内容
		contentMessage = new byte[msgLength];
		msg.getBytes(2, contentMessage, 0, msgLength);
		// 拿到的字节数据转为16进制数据
		String asciiBytesToMsg = ConverUtils.byteArr2HexStr(contentMessage);
		String message = ConverUtils.convertHexToString(asciiBytesToMsg);
		msgTemp.append(message);
		out.add(msgTemp.toString());

	}

}
package com.com.test.bbw;

import io.netty.buffer.ByteBuf;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.MessageToByteEncoder;
import org.msgpack.MessagePack;

public class MsgPckEncode extends MessageToByteEncoder<Object> {

	private static final Integer msgLeng = 5;
	@Override
	protected void encode(ChannelHandlerContext ctx, Object msg, ByteBuf buf) throws Exception {
		MessagePack pack = new MessagePack();
		String dataStr = (String) msg;
		// 第一步:获取报文长度
		int msgLength = Integer.parseInt(dataStr.substring(0, msgLeng));
		String msgInfo = dataStr.substring(msgLeng);
		//将报文 长度转为16进制 ,内容转为ASCII码
		String convertStringToHex  = ConverUtils.convertStringToHex(msgInfo);
		byte[] message = ConverUtils.addBytes(ConverUtils.encode(msgLength, 2), ConverUtils.hexToByteArray(convertStringToHex));
		buf.writeBytes(message);

	}
}

4、前置与服务B之间长链接,支持断线重连机制 

package com.com.test.bbw;

import io.netty.bootstrap.Bootstrap;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.timeout.IdleStateHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.concurrent.TimeUnit;

public class TCPBBWAdaptor {
    private static final Logger log = LoggerFactory.getLogger(TCPBBWAdaptor.class);
    private String bbwServerIp;
    private int bbwServerPort;/* 端口 */
    private String encoding;/* 编码方式 */
    private int timeout=10000;/* 超时时间:秒 */
    private EventLoopGroup group;
    private Bootstrap client;
    private Channel channel;

    public static Logger getLog() {
        return log;
    }

    private ChannelFuture future;

    public TCPBBWAdaptor() {
        init();
        group = new NioEventLoopGroup();
        client = new Bootstrap();
        client.group(group);
        client.channel(NioSocketChannel.class);
        client.option(ChannelOption.SO_KEEPALIVE, true);
        client.handler(new ChannelInitializer<SocketChannel>() {

            @Override
            protected void initChannel(SocketChannel ch) throws Exception {
                // 按照\r\n进行解码
                //ch.pipeline().addLast(new DelimiterBasedFrameDecoder(Integer.MAX_VALUE, Delimiters.lineDelimiter()[0]));
                ch.pipeline().addLast(new IdleStateHandler(0, 0, 5))
                        .addLast(new MsgPckDecode())
                        .addLast(new MsgPckEncode())
                        .addLast(new TcpBBWClientHandler(TCPBBWAdaptor.this));
            }

        });
    }

    public Object send(Object msg) {
        try {
            if (channel != null && channel.isActive()) {
                doSend(msg);
            } else {
                restConnOrConn();
                send(msg);
            }
        } catch (Exception e) {
            log.info("连接异常!");
            group.shutdownGracefully();
        }

        return msg;
    }

    private void doSend(Object msg) {
        group.execute(new Runnable() {
            @Override
            public void run() {
                channel.writeAndFlush(msg);
            }
        });
    }

    public void restConnOrConn() throws InterruptedException {
        future = client.connect(bbwServerIp, bbwServerPort).sync();
        future.addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) throws Exception {
                if (future.isSuccess()) {
                    channel = future.channel();
                } else {
                    future.channel().eventLoop().schedule(new Runnable() {

                        @Override
                        public void run() {
                            try {
                                restConnOrConn();
                            } catch (InterruptedException e) {
                                // TODO Auto-generated catch block
                                e.printStackTrace();
                            }

                        }
                    }, 2, TimeUnit.SECONDS);
                }
            }
        });

    }
    public void init(){
        this.bbwServerIp=BBWConfig.getBbwServerIp();
        this.bbwServerPort= Integer.parseInt(BBWConfig.getBbwServerPort());
        this.encoding=BBWConfig.getEncoding();
    }
}
package com.com.test.bbw;

import com.com.test.client.TCPPayAdaptor;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.handler.timeout.IdleStateEvent;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;

public class TcpBBWClientHandler extends ChannelInboundHandlerAdapter {
	private static final Logger log = LoggerFactory.getLogger(TcpBBWClientHandler.class);
	// 明文报文长度
	private static final Integer msgLeng = 5;
	private TCPBBWAdaptor clinet;
	public TcpBBWClientHandler(TCPBBWAdaptor clinet) {
        this.clinet = clinet;
	}
	/**
	 * 客户端与服务端创建连接的时候调用
	 */
	@Override
	public void channelActive(ChannelHandlerContext ctx) throws Exception {
		   ctx.fireChannelActive();
	}

	/**
	 * 客户端与服务端断开连接时调用
	 */
	@Override
	public void channelInactive(ChannelHandlerContext ctx) throws Exception {
		clinet.restConnOrConn();
	}

	/**
	 * 服务端接收客户端发送过来的数据结束之后调用
	 */
	@Override
	public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
		ctx.flush();
	}

	/**
	 * 工程出现异常的时候调用
	 */
	@Override
	public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
		cause.printStackTrace();
		ctx.close();
		//重新连接
		clinet.restConnOrConn();
	}
	/**
	 * 心跳处理方法
	 */
	@Override
	public void userEventTriggered(ChannelHandlerContext ctx, Object evt) throws Exception {
		// 如果属于心跳处理
		if (evt instanceof IdleStateEvent) {
			IdleStateEvent event = (IdleStateEvent) evt;
			switch (event.state()) {
			case READER_IDLE:
				break;
			case WRITER_IDLE:
				break;
			case ALL_IDLE:
				//发送心跳
				sendMsg(ctx);
				break;
			}
		}
		super.userEventTriggered(ctx, evt);
	}
	private void sendMsg(ChannelHandlerContext ctx) {
		ctx.channel().writeAndFlush("");
	}
	@Override
	public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
		ctx.channel().eventLoop().execute(new Runnable() {
			@Override
			public void run() {
				final String msgL=(String)msg;
				if (!StringUtils.isEmpty(msgL)) {
					log.debug("接收到报文为:" +msg);
					TCPPayAdaptor payAdaptor=new TCPPayAdaptor();
					payAdaptor.send(msg);
				}
			}
		});
	}

}

 5、前置与服务A之间短连接,异步调用

package com.com.test.client;

import com.com.test.bbw.BBWConfig;
import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.*;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;
import io.netty.handler.timeout.IdleStateHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.StringUtils;

import java.io.UnsupportedEncodingException;

public class TCPPayAdaptor {
	private static final Logger log = LoggerFactory.getLogger(TCPPayAdaptor.class);
	private String payRcvServerIp;
	private int payTcpRcvPort;/* 端口 */
	private String encoding;/* 编码方式 */
	private int timeout=20000;/* 超时时间:秒 */
	private EventLoopGroup group;
	private Bootstrap client;
	private Channel channel;

	public static Logger getLog() {
		return log;
	}

	private ChannelFuture future;

	public TCPPayAdaptor() {
		init();
		group = new NioEventLoopGroup();
		client = new Bootstrap();
		client.group(group);
		client.channel(NioSocketChannel.class);
		client.option(ChannelOption.SO_KEEPALIVE, true);
		client.handler(new ChannelInitializer<SocketChannel>() {

			@Override
			protected void initChannel(SocketChannel ch) throws Exception {
				// 按照\r\n进行解码
				//ch.pipeline().addLast(new DelimiterBasedFrameDecoder(Integer.MAX_VALUE, Delimiters.lineDelimiter()[0]));
				ch.pipeline().addLast(new IdleStateHandler(0, 0, 5))
						.addLast(new StringDecoder())
						.addLast(new StringEncoder())
						.addLast(new TcpPayClientHandler());
			}

		});
		try {
			future = client.connect(payRcvServerIp, payTcpRcvPort).sync();
			if(future!=null&&future.isSuccess()){
				channel=future.channel();
			}
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
	}

	public Object send(Object msg) {
		try {
			if (channel != null && channel.isActive()) {
				doSend(msg);
			}
		} catch (Exception e) {
			log.info("连接异常!");
			group.shutdownGracefully();
		}

		return msg;
	}

	private void doSend(Object msg) {
		try {
			final String msgL=(String)msg;
			byte[] bytes = msgL.getBytes(encoding);
			ByteBuf buf = Unpooled.wrappedBuffer(bytes);
			byte[] req = new byte[buf.readableBytes()];
			buf.readBytes(req);
			String smsg = new String(req, encoding);
			if (!StringUtils.isEmpty(smsg)) {
				log.debug("接收到报文为:" + smsg);
				channel.writeAndFlush(smsg);
			}
			ByteBuf buffer = Unpooled.buffer();
			buffer.writeBytes(req);
		} catch (UnsupportedEncodingException e) {
			log.info("bbw返回发送异常");
		}
	}

	public void init(){
		this.payTcpRcvPort= Integer.parseInt(BBWConfig.getPayTcpRcvPort());
		this.payRcvServerIp= BBWConfig.getPayRcvServerIp();
		this.encoding= BBWConfig.getEncoding();
	}
}
package com.com.test.client;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.handler.timeout.IdleStateEvent;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.lang.reflect.Method;

public class TcpPayClientHandler extends ChannelInboundHandlerAdapter {
	private static final Logger log = LoggerFactory.getLogger(TcpPayClientHandler.class);
	// 明文报文长度
	private static final Integer msgLeng = 5;
	/**
	 * 客户端与服务端创建连接的时候调用
	 */
	@Override
	public void channelActive(ChannelHandlerContext ctx) throws Exception {
		ctx.fireChannelActive();
	}

	/**
	 * 客户端与服务端断开连接时调用
	 */
	@Override
	public void channelInactive(ChannelHandlerContext ctx) throws Exception {
		ctx.channel().writeAndFlush("");
	}

	/**
	 * 服务端接收客户端发送过来的数据结束之后调用
	 */
	@Override
	public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
		ctx.flush();
	}

	/**
	 * 工程出现异常的时候调用
	 */
	@Override
	public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
		cause.printStackTrace();
		ctx.close();
	}
}

6、 服务B监听器,用来监听B系统发送给前置消息,用来转发到服务A

package com.com.test.listener;

import com.com.test.server.Server;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.stereotype.Component;



@Component
public class BBWListener implements ApplicationRunner{
	private static final Log logger = LogFactory.getLog(BBWListener.class);

	@Override
	public void run(ApplicationArguments args) throws Exception {

		if (logger.isInfoEnabled()) {
			logger.info("Starting booting core banking services platform...");
		}
		// Start
		try {
			Server server = new Server();
			server.start();
		} catch (Exception e) {
			e.printStackTrace();
			logger.error(e);
		}
		if (logger.isInfoEnabled()) {
			logger.info("Started booted bbw banking services platform...");
		}
	}

}

7、接收到系统B消息服务端

package com.com.test.server;

import com.com.test.bbw.BBWConfig;
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.ServerSocketChannel;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;


public class Server {
    private static final Log logger = LogFactory.getLog(Server.class);
    private int paySendPort;
    private String encoding;/* 编码方式 */
    private int timeout = 10000;/* 超时时间:秒 */
    private int conNum = 1;/* 保持连接数 */
    private ServerSocketChannel serverSocketChannel;

    private void bind() {
        //服务端要建立两个group,一个负责接收客户端的连接,一个负责处理数据传输
        //连接处理group
        EventLoopGroup boss = new NioEventLoopGroup();
        //事件处理group
        EventLoopGroup worker = new NioEventLoopGroup();
        ServerBootstrap bootstrap = new ServerBootstrap();
        // 绑定处理group
        bootstrap.group(boss, worker).channel(NioServerSocketChannel.class)
                //保持连接数
               // .option(ChannelOption.SO_BACKLOG, conNum)
                //有数据立即发送
                .option(ChannelOption.TCP_NODELAY, true)
                //保持连接
                .childOption(ChannelOption.SO_KEEPALIVE, false)
                //处理新连接
                .childHandler(new ChannelInitializer<SocketChannel>() {
                    @Override
                    protected void initChannel(SocketChannel sc) throws Exception {
                        // 增加任务处理
                        ChannelPipeline p = sc.pipeline();
                        p.addLast(
                                //使用了netty自带的编码器和解码器
                                new StringDecoder(),
                                new StringEncoder(),
                                //心跳检测,读超时,写超时,读写超时
                                //new IdleStateHandler(5, 0, 0, TimeUnit.SECONDS),
                                //自定义的处理器
                                new ServerHandler());
                    }
                });

        //绑定端口,同步等待成功
        ChannelFuture future;
        try {
            future = bootstrap.bind(paySendPort).sync();
            if (future.isSuccess()) {
                serverSocketChannel = (ServerSocketChannel) future.channel();
                logger.info("服务端启动成功,端口:" + paySendPort);
            } else {
                logger.info("服务端启动失败!");
            }
            //等待服务监听端口关闭,就是由于这里会将线程阻塞,导致无法发送信息,所以我这里开了线程
            future.channel().closeFuture().sync();
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //优雅地退出,释放线程池资源
            boss.shutdownGracefully();
            worker.shutdownGracefully();
        }
    }

    public void init() {
        this.paySendPort = Integer.parseInt(BBWConfig.getPaySendPort());
        this.encoding = BBWConfig.getEncoding();
    }

    public void start() {
        init();
        bind();
    }
}

 7、接收到系统B消息客户端,调用


TCPPayAdaptor转发到系统A


package com.com.test.server;

import com.com.test.bbw.TCPBBWAdaptor;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.util.CharsetUtil;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.util.StringUtils;

import java.io.UnsupportedEncodingException;

public class ServerHandler extends ChannelInboundHandlerAdapter {
	private static final Log logger = LogFactory.getLog(ServerHandler.class);
	/**
	 * 客户端与服务端创建连接的时候调用
	 */
	@Override
	public void channelActive(ChannelHandlerContext ctx) throws Exception {
		ctx.fireChannelActive();
	}

	/**
	 * 一旦建立连接第一个被执行
	 */
	@Override
	public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
	}

	/**
	 * 客户端与服务端断开连接时调用
	 */
	@Override
	public void channelInactive(ChannelHandlerContext ctx) throws Exception {
		ctx.close();
	}
	/**
	 * 服务端接收客户端发送过来的数据结束之后调用
	 */
	@Override
	public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
		ctx.flush();
	}

	/**
	 * 工程出现异常的时候调用
	 */
	@Override
	public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
		cause.printStackTrace();
		ctx.close();
	}

	/**
	 * 服务端处理客户端websocket请求的核心方法,这里接收了客户端发来的信息
	 */
	@Override
	public void channelRead(ChannelHandlerContext channelHandlerContext, Object info) throws Exception {
		logger.info("接收到了:" + info);
		try {
			final String msgL=(String)info;
			byte[] bytes = msgL.getBytes(CharsetUtil.UTF_8);
			ByteBuf buf = Unpooled.wrappedBuffer(bytes);
			byte[] req = new byte[buf.readableBytes()];
			buf.readBytes(req);
			String smsg = new String(req, CharsetUtil.UTF_8);
			if (!StringUtils.isEmpty(smsg)) {
				logger.debug("接收到报文为:" + smsg);
				new TCPBBWAdaptor().send(info);
			}
			ByteBuf buffer = Unpooled.buffer();
			buffer.writeBytes(req);
		} catch (Exception e) {
			logger.info("bbw返回发送异常");
		}
	}

}

8、启动类 

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.mongo.MongoAutoConfiguration;
import org.springframework.context.annotation.ComponentScan;

@ComponentScan(basePackages = {"com"})
@SuppressWarnings("all")
@SpringBootApplication(exclude= {DataSourceAutoConfiguration.class, MongoAutoConfiguration.class})
public class TestApplication {

	public static void main(String[] args) {
		SpringApplication.run(TestApplication.class, args);
	}

}


日志 logging.config=classpath:logback-boot.xml


BBWConfig.properties


#BBW服务器地址
bbwServerIp=127.0.0.1
#BBW服务器端口
bbwServerPort=4444
#支付接出端口Tcp
paySendPort=50039
#支付接入平台地址
payRcvServerIp=127.0.0.1
#支付平台接入端口tcp
payTcpRcvPort=50040
#支付平台接入http端口
payHttpRcvPort=9004
#AGENt重连次数警告consumer
warningNum=10
#报文编码格式
encoding=UTF8
#AGNET异常重连间隔(单位:毫秒)
reStartTime=10000
test=003090800C220000080000000040000000000000003535122313335102421203535301


logback-boot.xml

<?xml version="1.0" encoding="UTF-8"?>
<!-- 日志级别从低到高分为TRACE < DEBUG < INFO < WARN < ERROR < FATAL,如果设置为WARN,则低于WARN的信息都不会输出 -->
<!-- scan:当此属性设置为true时,配置文件如果发生改变,将会被重新加载,默认值为true -->
<!-- scanPeriod:设置监测配置文件是否有修改的时间间隔,如果没有给出时间单位,默认单位是毫秒。当scan为true时,此属性生效。默认的时间间隔为1分钟。 -->
<!-- debug:当此属性设置为true时,将打印出logback内部日志信息,实时查看logback运行状态。默认值为false。 -->
<configuration  scan="true" scanPeriod="10 seconds">

    <!--<include resource="org/springframework/boot/logging/logback/base.xml" />-->

    <contextName>logback</contextName>
    <!-- name的值是变量的名称,value的值时变量定义的值。通过定义的值会被插入到logger上下文中。定义变量后,可以使“${}”来使用变量。 -->
    <property name="log.path" value="D:/nmyslog/nmys" />

    <!-- 彩色日志 -->
    <!-- 彩色日志依赖的渲染类 -->
    <conversionRule conversionWord="clr" converterClass="org.springframework.boot.logging.logback.ColorConverter" />
    <conversionRule conversionWord="wex" converterClass="org.springframework.boot.logging.logback.WhitespaceThrowableProxyConverter" />
    <conversionRule conversionWord="wEx" converterClass="org.springframework.boot.logging.logback.ExtendedWhitespaceThrowableProxyConverter" />
    <!-- 彩色日志格式 -->
    <property name="CONSOLE_LOG_PATTERN" value="${CONSOLE_LOG_PATTERN:-%clr(%d{yyyy-MM-dd HH:mm:ss.SSS}){faint} %clr(${LOG_LEVEL_PATTERN:-%5p}) %clr(${PID:- }){magenta} %clr(---){faint} %clr([%15.15t]){faint} %clr(%-40.40logger{39}){cyan} %clr(:){faint} %m%n${LOG_EXCEPTION_CONVERSION_WORD:-%wEx}}"/>


    <!--输出到控制台-->
    <appender name="CONSOLE" class="ch.qos.logback.core.ConsoleAppender">
        <!--此日志appender是为开发使用,只配置最底级别,控制台输出的日志级别是大于或等于此级别的日志信息-->
        <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
            <level>info</level>
        </filter>
        <encoder>
            <Pattern>${CONSOLE_LOG_PATTERN}</Pattern>
            <!-- 设置字符集 -->
            <charset>UTF-8</charset>
        </encoder>
    </appender>


    <!--输出到文件-->

    <!-- 时间滚动输出 level为 DEBUG 日志 -->
    <appender name="DEBUG_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <!-- 正在记录的日志文件的路径及文件名 -->
        <file>${log.path}/log_debug.log</file>
        <!--日志文件输出格式-->
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
            <charset>UTF-8</charset> <!-- 设置字符集 -->
        </encoder>
        <!-- 日志记录器的滚动策略,按日期,按大小记录 -->
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <!-- 日志归档 -->
            <fileNamePattern>${log.path}/debug/log-debug-%d{yyyy-MM-dd}.%i.log</fileNamePattern>
            <timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <maxFileSize>100MB</maxFileSize>
            </timeBasedFileNamingAndTriggeringPolicy>
            <!--日志文件保留天数-->
            <maxHistory>15</maxHistory>
        </rollingPolicy>
        <!-- 此日志文件只记录debug级别的 -->
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>debug</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY</onMismatch>
        </filter>
    </appender>

    <!-- 时间滚动输出 level为 INFO 日志 -->
    <appender name="INFO_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <!-- 正在记录的日志文件的路径及文件名 -->
        <file>${log.path}/log_info.log</file>
        <!--日志文件输出格式-->
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
            <charset>UTF-8</charset>
        </encoder>
        <!-- 日志记录器的滚动策略,按日期,按大小记录 -->
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <!-- 每天日志归档路径以及格式 -->
            <fileNamePattern>${log.path}/info/log-info-%d{yyyy-MM-dd}.%i.log</fileNamePattern>
            <timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <maxFileSize>100MB</maxFileSize>
            </timeBasedFileNamingAndTriggeringPolicy>
            <!--日志文件保留天数-->
            <maxHistory>15</maxHistory>
        </rollingPolicy>
        <!-- 此日志文件只记录info级别的 -->
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>info</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY</onMismatch>
        </filter>
    </appender>

    <!-- 时间滚动输出 level为 WARN 日志 -->
    <appender name="WARN_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <!-- 正在记录的日志文件的路径及文件名 -->
        <file>${log.path}/log_warn.log</file>
        <!--日志文件输出格式-->
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
            <charset>UTF-8</charset> <!-- 此处设置字符集 -->
        </encoder>
        <!-- 日志记录器的滚动策略,按日期,按大小记录 -->
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <fileNamePattern>${log.path}/warn/log-warn-%d{yyyy-MM-dd}.%i.log</fileNamePattern>
            <timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <maxFileSize>100MB</maxFileSize>
            </timeBasedFileNamingAndTriggeringPolicy>
            <!--日志文件保留天数-->
            <maxHistory>15</maxHistory>
        </rollingPolicy>
        <!-- 此日志文件只记录warn级别的 -->
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>warn</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY</onMismatch>
        </filter>
    </appender>


    <!-- 时间滚动输出 level为 ERROR 日志 -->
    <appender name="ERROR_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
        <!-- 正在记录的日志文件的路径及文件名 -->
        <file>${log.path}/log_error.log</file>
        <!--日志文件输出格式-->
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
            <charset>UTF-8</charset> <!-- 此处设置字符集 -->
        </encoder>
        <!-- 日志记录器的滚动策略,按日期,按大小记录 -->
        <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
            <fileNamePattern>${log.path}/error/log-error-%d{yyyy-MM-dd}.%i.log</fileNamePattern>
            <timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
                <maxFileSize>100MB</maxFileSize>
            </timeBasedFileNamingAndTriggeringPolicy>
            <!--日志文件保留天数-->
            <maxHistory>15</maxHistory>
        </rollingPolicy>
        <!-- 此日志文件只记录ERROR级别的 -->
        <filter class="ch.qos.logback.classic.filter.LevelFilter">
            <level>ERROR</level>
            <onMatch>ACCEPT</onMatch>
            <onMismatch>DENY</onMismatch>
        </filter>
    </appender>

    <!--
        <logger>用来设置某一个包或者具体的某一个类的日志打印级别、
        以及指定<appender>。<logger>仅有一个name属性,
        一个可选的level和一个可选的addtivity属性。
        name:用来指定受此logger约束的某一个包或者具体的某一个类。
        level:用来设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF,
              还有一个特俗值INHERITED或者同义词NULL,代表强制执行上级的级别。
              如果未设置此属性,那么当前logger将会继承上级的级别。
        addtivity:是否向上级logger传递打印信息。默认是true。
    -->
    <!--<logger name="org.springframework.web" level="info"/>-->
    <!--<logger name="org.springframework.scheduling.annotation.ScheduledAnnotationBeanPostProcessor" level="INFO"/>-->
    <!--
        使用mybatis的时候,sql语句是debug下才会打印,而这里我们只配置了info,所以想要查看sql语句的话,有以下两种操作:
        第一种把<root level="info">改成<root level="DEBUG">这样就会打印sql,不过这样日志那边会出现很多其他消息
        第二种就是单独给dao下目录配置debug模式,代码如下,这样配置sql语句会打印,其他还是正常info级别:
     -->


    <!--
        root节点是必选节点,用来指定最基础的日志输出级别,只有一个level属性
        level:用来设置打印级别,大小写无关:TRACE, DEBUG, INFO, WARN, ERROR, ALL 和 OFF,
        不能设置为INHERITED或者同义词NULL。默认是DEBUG
        可以包含零个或多个元素,标识这个appender将会添加到这个logger。
    -->

    <!--开发环境:打印控制台-->
    <springProfile name="dev">
        <logger name="com.com.test.bbw" level="debug"/>
    </springProfile>

    <root level="info">
        <appender-ref ref="CONSOLE" />
        <appender-ref ref="DEBUG_FILE" />
        <appender-ref ref="INFO_FILE" />
        <appender-ref ref="WARN_FILE" />
        <appender-ref ref="ERROR_FILE" />
    </root>

    <!--生产环境:输出到文件-->
    <!--<springProfile name="pro">-->
        <!--<root level="info">-->
            <!--<appender-ref ref="CONSOLE" />-->
            <!--<appender-ref ref="DEBUG_FILE" />-->
            <!--<appender-ref ref="INFO_FILE" />-->
            <!--<appender-ref ref="ERROR_FILE" />-->
            <!--<appender-ref ref="WARN_FILE" />-->
        <!--</root>-->
    <!--</springProfile>-->

</configuration>

pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
	<modelVersion>4.0.0</modelVersion>
	<parent>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-parent</artifactId>
		<version>2.4.1</version>
		<relativePath/> <!-- lookup parent from repository -->
	</parent>
	<groupId>com.test</groupId>
	<artifactId>test</artifactId>
	<version>0.0.1-SNAPSHOT</version>
	<name>test</name>
	<description>Demo project for Spring Boot</description>

	<properties>
		<java.version>1.8</java.version>
	</properties>

	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-data-mongodb</artifactId>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-jdbc</artifactId>
		</dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-web</artifactId>
		</dependency>
		<dependency>
			<groupId>org.springframework.kafka</groupId>
			<artifactId>spring-kafka</artifactId>
		</dependency>
		<dependency>
			<groupId>io.netty</groupId>
			<artifactId>netty-all</artifactId>
		</dependency>
		<dependency>
			<groupId>org.msgpack</groupId>
			<artifactId>msgpack</artifactId>
			<version>0.6.12</version>
		</dependency>
		 <dependency>
              <groupId>ch.qos.logback</groupId>
              <artifactId>logback-classic</artifactId>
              <version>1.2.3</version>
              <scope>compile</scope>
            </dependency>
            <dependency>
              <groupId>org.apache.logging.log4j</groupId>
              <artifactId>log4j-to-slf4j</artifactId>
              <version>2.10.0</version>
              <scope>compile</scope>
            </dependency>
            <dependency>
              <groupId>org.slf4j</groupId>
              <artifactId>jul-to-slf4j</artifactId>
              <version>1.7.25</version>
              <scope>compile</scope>
            </dependency>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
		<dependency>
			<groupId>org.springframework.kafka</groupId>
			<artifactId>spring-kafka-test</artifactId>
			<scope>test</scope>
		</dependency>
	</dependencies>

	<build>
		<plugins>
			<plugin>
				<groupId>org.springframework.boot</groupId>
				<artifactId>spring-boot-maven-plugin</artifactId>
			</plugin>
		</plugins>
	</build>

</project>