首頁  >  文章  >  資料庫  >  詳細介紹mysql 協定的錯誤包及解析

詳細介紹mysql 協定的錯誤包及解析

黄舟
黄舟原創
2017-03-07 14:06:161199瀏覽


git


https://github.com/sea-boat/mysql-protocol

概況

mysql客戶端與mysql服務端互動過程中,如果服務端遇到錯誤需要告知客戶端則回傳錯誤包。

mysql通訊封包結構

##string payload報文體,長度即為前面指定的payload長度
類型名字描述
#int5bdf4c78156c7953567bb5a0aef2fc53payload長度#按照the least significant byte first存儲,3個字節的payload和1個字節的序號組合成報文頭
intf35d6e602fd7d0f0edfa6f7d103c1b57#序號

TypeNameDescription#Nameheaderint2cc198a1d5eb0d3eb508d858c9f5cbdberror-codeif capabilities & CLIENT_PROTOCOL_41 {
#Payload


intf35d6e602fd7d0f0edfa6f7d103c1b57
[ff] header of the ERR packet

error_code

#string[1]
  1. sql_state_marker

    marker of the SQL State
    ##string[5]
  1. sql_state

  2. SQL State
}



###string######error_message######human readable error message##############更多詳情: http://dev .mysql.com/doc/internals/en/packet-ERR_Packet.html#######錯誤包操作###########錯誤包類別##########
/**
 * 
 * @author seaboat
 * @date 2016-09-25
 * @version 1.0
 * 
email: 849586227@qq.com
*
blog: http://www.php.cn/;/pre>
 * 

mysql error packet.

*/public class ErrorPacket extends MySQLPacket { public static final byte header = (byte) 0xff; private static final byte SQLSTATE_MARKER = (byte) '#'; private static final byte[] DEFAULT_SQLSTATE = "HY000".getBytes(); public int errno; public byte mark = SQLSTATE_MARKER; public byte[] sqlState = DEFAULT_SQLSTATE; public byte[] message; public void read(byte[] data) { MySQLMessage mm = new MySQLMessage(data); packetLength = mm.readUB3(); packetId = mm.read(); mm.read(); errno = mm.readUB2(); if (mm.hasRemaining() && (mm.read(mm.position()) == SQLSTATE_MARKER)) { mm.read(); sqlState = mm.readBytes(5); } message = mm.readBytes(); } public void write(ByteBuffer buffer) { int size = calcPacketSize(); BufferUtil.writeUB3(buffer, size); buffer.put(packetId); buffer.put(header); BufferUtil.writeUB2(buffer, errno); buffer.put(mark); buffer.put(sqlState); buffer.put(message); } @Override public int calcPacketSize() { int size = 9;// 1 + 2 + 1 + 5 if (message != null) { size += message.length; } return size; } @Override protected String getPacketInfo() { return "MySQL Error Packet"; } }
#########十六進位轉換工具#########
/**
 * 
 * @author seaboat
 * @date 2016-09-25
 * @version 1.0
 * 
email: 849586227@qq.com
*
blog: http://www.php.cn/;/pre>
 * 

hex transform util.

*/public class HexUtil { private final static byte[] hex = "0123456789ABCDEF".getBytes(); public static String Bytes2HexString(byte[] b) { byte[] buff = new byte[2 * b.length]; for (int i = 0; i < b.length; i++) { buff[2 * i] = hex[(b[i] >> 4) & 0x0f]; buff[2 * i + 1] = hex[b[i] & 0x0f]; } return new String(buff); } public static String str2HexStr(String str) { char[] chars = "0123456789ABCDEF".toCharArray(); StringBuilder sb = new StringBuilder(""); byte[] bs = str.getBytes(); int bit; for (int i = 0; i < bs.length; i++) { bit = (bs[i] & 0x0f0) >> 4; sb.append(chars[bit]); bit = bs[i] & 0x0f; sb.append(chars[bit]); } return sb.toString(); } }
##########錯誤包產生測試########
/**
 * 
 * @author seaboat
 * @date 2016-09-25
 * @version 1.0
 * 
email: 849586227@qq.com
*
blog: http://www.php.cn/;/pre>
 * 

test auth packet.

*/public class ErrorPacketTest { @Test public void produce() { ErrorPacket err = new ErrorPacket(); err.packetId = 1; err.errno = 32322; err.message = "sorry".getBytes(); ByteBuffer buffer = ByteBuffer.allocate(256); err.write(buffer); buffer.flip(); byte[] bytes = new byte[buffer.remaining()]; buffer.get(bytes, 0, bytes.length); String result = HexUtil.Bytes2HexString(bytes); System.out.println(result); assertTrue(Integer.valueOf(result.substring(0, 2), 16) == result .length() / 2 - 4); ErrorPacket err2 = new ErrorPacket(); err2.read(bytes); assertTrue(err2.errno == 32322); assertTrue(err2.message.length == "sorry".getBytes().length); } }
## # 以上就是詳細介紹mysql 協定的錯誤包及解析的內容,更多相關內容請關注PHP中文網(www.php.cn)! ################
陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn