connection.go 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. package binlog
  2. import (
  3. "bytes"
  4. "database/sql"
  5. "database/sql/driver"
  6. "encoding/binary"
  7. "encoding/json"
  8. "fmt"
  9. "net"
  10. "time"
  11. )
  12. const NullByte byte = '\x00'
  13. const MaxPacketSize = 16777216
  14. // MySQL Packet Data Types
  15. const TypeNullTerminatedString = int(0)
  16. const TypeFixedString = int(1)
  17. const TypeFixedInt = int(2)
  18. const TypeLenEncodedInt = int(3)
  19. type Config struct {
  20. Host string `json:"host"`
  21. Port int `json:"port"`
  22. User string `json:"user"`
  23. Pass string `json:"password"`
  24. Database string `json:"database"`
  25. SSL bool `json:"ssl"`
  26. VerifyCert bool `json:"verify_cert"`
  27. Timeout time.Duration
  28. }
  29. func newBinlogConfig(dsn string) (*Config, error) {
  30. var err error
  31. config := Config{}
  32. err = json.Unmarshal([]byte(dsn), &config)
  33. return &config, err
  34. }
  35. type Conn struct {
  36. Config *Config
  37. tcpConn *net.TCPConn
  38. Handshake *HandshakePacket
  39. }
  40. func newBinlogConn(config *Config) Conn {
  41. return Conn{
  42. Config: config,
  43. }
  44. }
  45. func (c Conn) Prepare(query string) (driver.Stmt, error) {
  46. return nil, nil
  47. }
  48. func (c Conn) Close() error {
  49. return nil
  50. }
  51. func (c Conn) Begin() (driver.Tx, error) {
  52. return nil, nil
  53. }
  54. type Driver struct{}
  55. func (d Driver) Open(dsn string) (driver.Conn, error) {
  56. config, err := newBinlogConfig(dsn)
  57. if nil != err {
  58. return nil, err
  59. }
  60. blConn := newBinlogConn(config)
  61. dialer := net.Dialer{Timeout: blConn.Config.Timeout}
  62. addr := fmt.Sprintf("%s:%d", blConn.Config.Host, blConn.Config.Port)
  63. c, err := dialer.Dial("tcp", addr)
  64. blConn.tcpConn = c.(*net.TCPConn)
  65. if err != nil {
  66. netErr, ok := err.(net.Error)
  67. if ok && netErr.Temporary() {
  68. fmt.Printf("Error: %s", netErr.Error())
  69. return nil, err
  70. }
  71. }
  72. hsp, err := blConn.handshakePacket()
  73. blConn.Handshake = hsp
  74. resp := blConn.handshakeResponse()
  75. b := resp.encode(&blConn)
  76. fmt.Printf("%d", b)
  77. _, err = blConn.tcpConn.Write(b)
  78. return blConn, err
  79. }
  80. func init() {
  81. sql.Register("mysql-binlog", &Driver{})
  82. }
  83. func (c *Conn) getBytes(l uint64) ([]byte, error) {
  84. b := make([]byte, l)
  85. _, err := c.tcpConn.Read(b)
  86. return b, err
  87. }
  88. func (c *Conn) consumeBytes(l uint64) error {
  89. b := make([]byte, l)
  90. _, err := c.tcpConn.Read(b)
  91. return err
  92. }
  93. func (c *Conn) getInt(t int, l uint64) (uint64, error) {
  94. var v uint64
  95. var err error = nil
  96. switch t {
  97. case TypeFixedInt:
  98. v, err = c.popFixedInt(l)
  99. default:
  100. v = 0
  101. }
  102. if err != nil {
  103. return 0, err
  104. }
  105. return v, nil
  106. }
  107. func (c *Conn) getString(t int, l uint64) (string, error) {
  108. var v string
  109. var err error = nil
  110. switch t {
  111. case TypeFixedString:
  112. v, err = c.popFixedString(l)
  113. case TypeNullTerminatedString:
  114. v, err = c.popNullTerminatedString()
  115. default:
  116. v = ""
  117. }
  118. if err != nil {
  119. return "", err
  120. }
  121. return v, nil
  122. }
  123. func (c *Conn) readBytes(l uint64) (*bytes.Buffer, error) {
  124. b := make([]byte, l)
  125. _, err := c.tcpConn.Read(b)
  126. if err != nil {
  127. return nil, err
  128. }
  129. return bytes.NewBuffer(b), nil
  130. }
  131. func (c *Conn) readToNull() (*bytes.Buffer, error) {
  132. var s []byte
  133. for {
  134. bA := make([]byte, 1)
  135. _, err := c.tcpConn.Read(bA)
  136. if err != nil {
  137. return nil, err
  138. }
  139. b := bA[0]
  140. if b == NullByte {
  141. break
  142. } else {
  143. s = append(s, b)
  144. }
  145. }
  146. return bytes.NewBuffer(s), nil
  147. }
  148. func (c *Conn) popNullTerminatedString() (string, error) {
  149. b, err := c.readToNull()
  150. if err != nil {
  151. return "", err
  152. }
  153. return string(b.Bytes()), nil
  154. }
  155. func (c *Conn) popFixedString(l uint64) (string, error) {
  156. b, err := c.readBytes(l)
  157. if err != nil {
  158. return "", err
  159. }
  160. return string(b.Bytes()), nil
  161. }
  162. func (c *Conn) popFixedInt(l uint64) (uint64, error) {
  163. b, err := c.readBytes(l)
  164. if err != nil {
  165. return 0, err
  166. }
  167. var i uint64
  168. i, err = binary.ReadUvarint(b)
  169. return i, err
  170. }