package server import ( "fmt" . "github.com/siddontang/go-mysql/mysql" ) func (c *Conn) writeOK(r *Result) error { if r == nil { r = &Result{} } r.Status |= c.status data := make([]byte, 4, 32) data = append(data, OK_HEADER) data = append(data, PutLengthEncodedInt(r.AffectedRows)...) data = append(data, PutLengthEncodedInt(r.InsertId)...) if c.capability&CLIENT_PROTOCOL_41 > 0 { data = append(data, byte(r.Status), byte(r.Status>>8)) data = append(data, 0, 0) } return c.WritePacket(data) } func (c *Conn) writeError(e error) error { var m *MyError var ok bool if m, ok = e.(*MyError); !ok { m = NewError(ER_UNKNOWN_ERROR, e.Error()) } data := make([]byte, 4, 16+len(m.Message)) data = append(data, ERR_HEADER) data = append(data, byte(m.Code), byte(m.Code>>8)) if c.capability&CLIENT_PROTOCOL_41 > 0 { data = append(data, '#') data = append(data, m.State...) } data = append(data, m.Message...) return c.WritePacket(data) } func (c *Conn) writeEOF() error { data := make([]byte, 4, 9) data = append(data, EOF_HEADER) if c.capability&CLIENT_PROTOCOL_41 > 0 { data = append(data, 0, 0) data = append(data, byte(c.status), byte(c.status>>8)) } return c.WritePacket(data) } // see: https://dev.mysql.com/doc/dev/mysql-server/latest/page_protocol_connection_phase_packets_protocol_auth_switch_request.html func (c *Conn) writeAuthSwitchRequest(newAuthPluginName string) error { data := make([]byte, 4) data = append(data, EOF_HEADER) data = append(data, []byte(newAuthPluginName)...) data = append(data, 0x00) rnd, err := RandomBuf(20) if err != nil { return err } // new auth data c.salt = rnd data = append(data, c.salt...) // the online doc states it's a string.EOF, however, the actual MySQL server add a \NUL to the end, without it, the // official MySQL client will fail. data = append(data, 0x00) return c.WritePacket(data) } // see: https://dev.mysql.com/doc/dev/mysql-server/latest/page_protocol_connection_phase_packets_protocol_auth_switch_response.html func (c *Conn) readAuthSwitchRequestResponse() ([]byte, error) { data, err := c.ReadPacket() if err != nil { return nil, err } if len(data) == 1 && data[0] == 0x00 { // \NUL return make([]byte, 0), nil } return data, nil } func (c *Conn) writeAuthMoreDataPubkey() error { data := make([]byte, 4) data = append(data, MORE_DATE_HEADER) data = append(data, c.serverConf.pubKey...) return c.WritePacket(data) } func (c *Conn) writeAuthMoreDataFullAuth() error { data := make([]byte, 4) data = append(data, MORE_DATE_HEADER) data = append(data, CACHE_SHA2_FULL_AUTH) return c.WritePacket(data) } func (c *Conn) writeAuthMoreDataFastAuth() error { data := make([]byte, 4) data = append(data, MORE_DATE_HEADER) data = append(data, CACHE_SHA2_FAST_AUTH) return c.WritePacket(data) } func (c *Conn) writeResultset(r *Resultset) error { columnLen := PutLengthEncodedInt(uint64(len(r.Fields))) data := make([]byte, 4, 1024) data = append(data, columnLen...) if err := c.WritePacket(data); err != nil { return err } for _, v := range r.Fields { data = data[0:4] data = append(data, v.Dump()...) if err := c.WritePacket(data); err != nil { return err } } if err := c.writeEOF(); err != nil { return err } for _, v := range r.RowDatas { data = data[0:4] data = append(data, v...) if err := c.WritePacket(data); err != nil { return err } } if err := c.writeEOF(); err != nil { return err } return nil } func (c *Conn) writeFieldList(fs []*Field) error { data := make([]byte, 4, 1024) for _, v := range fs { data = data[0:4] data = append(data, v.Dump()...) if err := c.WritePacket(data); err != nil { return err } } if err := c.writeEOF(); err != nil { return err } return nil } type noResponse struct{} func (c *Conn) writeValue(value interface{}) error { switch v := value.(type) { case noResponse: return nil case error: return c.writeError(v) case nil: return c.writeOK(nil) case *Result: if v != nil && v.Resultset != nil { return c.writeResultset(v.Resultset) } else { return c.writeOK(v) } case []*Field: return c.writeFieldList(v) case *Stmt: return c.writePrepare(v) default: return fmt.Errorf("invalid response type %T", value) } }