123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129 |
- package xugu
- import "encoding/binary"
- var BUFF_SIZE = 8 * 1024
- // EncryptBuff 假设的加密函数
- func EncryptBuff(pconn *xugusqlConn, data []byte) {
- // 加密逻辑,根据实际情况实现
- }
- func sockSendCommand0(conn *xugusqlConn, sql string) (int, error) {
- cmdLen := len(sql)
- //发送问号字符 ?
- ret, err := RhSend(conn, []byte("?"), cmdLen)
- if err != nil {
- return 0, err
- }
- // Comand_Len
- lengthInt32 := uint32(conn.sendBuff.Len())
- ret, err = rhSendInt32(conn, lengthInt32)
- if err != nil {
- return 0, err
- }
- // Comand_str
- ret, err = RhSend(conn, []byte(sql), cmdLen+1)
- if err != nil {
- return 0, err
- }
- // Param_num
- ret, err = rhSendInt32(conn, 0)
- if err != nil {
- return 0, err
- }
- return ret, nil
- }
- // RhSend Go 版本的 rh_send 函数
- // TODO :加密连接还没有实现
- func RhSend(pconn *xugusqlConn, buff []byte, length int) (int, error) {
- pconn.mu.Lock()
- defer pconn.mu.Unlock()
- //没有内容则不发送
- if length == 0 {
- return 0, nil
- }
- //连接里是否有加密
- if pconn.turingDataR != nil && pconn.useSSL {
- if length > 2*1024 {
- EncryptBuff(pconn, buff)
- if _, err := pconn.conn.Write(pconn.sendBuff.Bytes()); err != nil {
- pconn.sendBuff.Reset()
- return 0, err
- }
- pconn.sendBuff.Reset()
- _, err := pconn.conn.Write(buff)
- return len(buff), err
- } else if pconn.sendBuff.Len()+length >= BUFF_SIZE {
- if _, err := pconn.conn.Write(pconn.sendBuff.Bytes()); err != nil {
- pconn.sendBuff.Reset()
- return 0, err
- }
- pconn.sendBuff.Reset()
- _, err := pconn.sendBuff.Write(buff)
- if err != nil {
- return 0, err
- }
- EncryptBuff(pconn, pconn.sendBuff.Bytes())
- return len(buff), nil
- } else {
- pconn.sendBuff.Write(buff)
- EncryptBuff(pconn, pconn.sendBuff.Bytes()[pconn.sendBuff.Len()-length:])
- return length, nil
- }
- } else {
- /*如果数据长度大于 2048,先发送缓冲区中的数据,然后直接发送新的大数据块。*/
- if length > 2048 {
- if _, err := pconn.conn.Write(pconn.sendBuff.Bytes()); err != nil {
- //将缓冲区重置为空
- pconn.sendBuff.Reset()
- return 0, err
- }
- //将缓冲区重置为空
- pconn.sendBuff.Reset()
- //发送新的大数据块
- _, err := pconn.conn.Write(buff)
- return length, err
- } else if pconn.sendBuff.Len()+length >= BUFF_SIZE {
- //缓冲区空间不足:
- /*- 如果当前缓冲区中的数据加上新数据的长度超过 `8 * 1024` 字节,
- 则先发送缓冲区中的数据,然后重置缓冲区,将新数据拷贝到缓冲区中。*/
- if _, err := pconn.conn.Write(pconn.sendBuff.Bytes()); err != nil {
- pconn.sendBuff.Reset()
- return 0, err
- }
- pconn.sendBuff.Reset()
- pconn.sendBuff.Write(buff)
- return length, nil
- } else {
- //TODO: 这里c代码里是添加到缓存buff里面 ,不直接发送
- pconn.sendBuff.Write(buff)
- return length, nil
- }
- }
- }
- func rhSendInt32(pconn *xugusqlConn, i uint32) (int, error) {
- // 创建一个 4 字节的缓冲区,用于存储网络字节序的 int32 值
- var networkBytes [4]byte
- // 将 int32 值转换为网络字节序(大端字节序)并写入缓冲区
- binary.BigEndian.PutUint32(networkBytes[:], i)
- return RhSend(pconn, networkBytes[:], 4)
- }
- func RhFlush(pconn *xugusqlConn) (int, error) {
- if pconn.sendBuff.Len() > 0 {
- if _, err := pconn.conn.Write(pconn.sendBuff.Bytes()); err != nil {
- pconn.sendBuff.Reset()
- return 0, err
- }
- pconn.sendBuff.Reset()
- }
- return 0, nil
- }
|