1 Star 0 Fork 0

sqos/beats

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
mysql.go 16.01 KB
一键复制 编辑 原始数据 按行查看 历史
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562
package main
import (
"bytes"
"encoding/csv"
"labix.org/v2/mgo/bson"
"strings"
"time"
)
// Packet types
const (
MYSQL_CMD_QUERY = 3
)
type MysqlMessage struct {
start int
end int
Ts time.Time
IsRequest bool
PacketLength uint32
Seq uint8
Typ uint8
NumberOfRows int
NumberOfFields int
Size uint64
Fields []string
Rows [][]string
Tables string
IsOK bool
AffectedRows int
InsertId int
IsError bool
ErrorCode int
ErrorInfo string
Query string
Stream_id uint32
Direction uint8
IsTruncated bool
Tuple *IpPortTuple
CmdlineTuple *CmdlineTuple
Raw []byte
}
type MysqlTransaction struct {
Type string
tuple TcpTuple
Src Endpoint
Dst Endpoint
ResponseTime int32
Ts int64
JsTs time.Time
ts time.Time
Mysql bson.M
Request_raw string
Response_raw string
timer *time.Timer
}
type MysqlStream struct {
tcpStream *TcpStream
data []byte
parseOffset int
parseState int
isClient bool
message *MysqlMessage
}
const (
TransactionsHashSize = 2 ^ 16
TransactionTimeout = 10 * 1e9
)
const (
MysqlStateStart = iota
MysqlStateEatMessage
MysqlStateEatFields
MysqlStateEatRows
)
var mysqlTransactionsMap = make(map[TcpTuple]*MysqlTransaction, TransactionsHashSize)
func (stream *MysqlStream) PrepareForNewMessage() {
stream.data = stream.data[stream.message.end:]
stream.parseState = MysqlStateStart
stream.parseOffset = 0
stream.isClient = false
stream.message = nil
}
func mysqlMessageParser(s *MysqlStream) (bool, bool) {
m := s.message
for s.parseOffset < len(s.data) {
switch s.parseState {
case MysqlStateStart:
m.start = s.parseOffset
if len(s.data[s.parseOffset:]) < 5 {
WARN("MySQL Message too short. Ignore it.")
return false, false
}
hdr := s.data[s.parseOffset : s.parseOffset+5]
m.PacketLength = uint32(hdr[0]) | uint32(hdr[1])<<8 | uint32(hdr[2])<<16
m.Seq = uint8(hdr[3])
m.Typ = uint8(hdr[4])
DEBUG("mysqldetailed", "MySQL Header: Packet length %d, Seq %d, Type=%d", m.PacketLength, m.Seq, m.Typ)
if m.Seq == 0 {
// starts Command Phase
if m.Typ == MYSQL_CMD_QUERY {
// parse request
m.IsRequest = true
m.start = s.parseOffset
s.parseState = MysqlStateEatMessage
} else {
// ignore command
s.parseState = MysqlStateEatMessage
}
if !s.isClient {
s.isClient = true
}
} else if !s.isClient {
// parse response
m.IsRequest = false
if uint8(hdr[4]) == 0x00 {
DEBUG("mysqldetailed", "Received OK response")
m.start = s.parseOffset
s.parseState = MysqlStateEatMessage
m.IsOK = true
} else if uint8(hdr[4]) == 0xff {
DEBUG("mysqldetailed", "Received ERR response")
m.start = s.parseOffset
s.parseState = MysqlStateEatMessage
m.IsError = true
} else if m.PacketLength == 1 {
DEBUG("mysqldetailed", "Query response. Number of fields %d", uint8(hdr[4]))
m.NumberOfFields = int(hdr[4])
m.start = s.parseOffset
s.parseOffset += 5
s.parseState = MysqlStateEatFields
} else {
// something else. ignore
s.parseState = MysqlStateEatMessage
}
} else {
// something else, not expected
WARN("Unexpected MySQL message of type %d received.", m.Typ)
return false, false
}
break
case MysqlStateEatMessage:
if len(s.data[s.parseOffset:]) >= int(m.PacketLength)+4 {
s.parseOffset += 4 //header
s.parseOffset += int(m.PacketLength)
m.end = s.parseOffset
if m.IsRequest {
m.Query = string(s.data[m.start+5 : m.end])
return true, true
} else if m.IsOK {
m.AffectedRows = int(s.data[m.start+5])
m.InsertId = int(s.data[m.start+6])
return true, true
} else if m.IsError {
m.ErrorCode = int(uint16(s.data[m.start+6])<<8 | uint16(s.data[m.start+7]))
m.ErrorInfo = string(s.data[m.start+9:m.start+14]) + ": " + string(s.data[m.start+15:])
return true, true
}
} else {
// wait for more
return true, false
}
break
case MysqlStateEatFields:
if len(s.data[s.parseOffset:]) < 4 {
// wait for more
return true, false
}
hdr := s.data[s.parseOffset : s.parseOffset+4]
m.PacketLength = uint32(hdr[0]) | uint32(hdr[1])<<8 | uint32(hdr[2])<<16
m.Seq = uint8(hdr[3])
DEBUG("mysqldetailed", "Fields: packet length %d, packet number %d", m.PacketLength, m.Seq)
if len(s.data[s.parseOffset:]) >= int(m.PacketLength)+4 {
s.parseOffset += 4 // header
if uint8(s.data[s.parseOffset]) == 0xfe {
DEBUG("mysqldetailed", "Received EOF packet")
// EOF marker
s.parseOffset += int(m.PacketLength)
s.parseState = MysqlStateEatRows
} else {
_ /* catalog */, off := read_lstring(s.data, s.parseOffset)
db /*schema */, off := read_lstring(s.data, off)
table /* table */, off := read_lstring(s.data, off)
db_table := string(db) + "." + string(table)
if len(m.Tables) == 0 {
m.Tables = db_table
} else if !strings.Contains(m.Tables, db_table) {
m.Tables = m.Tables + ", " + db_table
}
DEBUG("mysqldetailed", "db=%s, table=%s", db, table)
s.parseOffset += int(m.PacketLength)
// go to next field
}
} else {
// wait for more
return true, false
}
break
case MysqlStateEatRows:
if len(s.data[s.parseOffset:]) < 4 {
// wait for more
return true, false
}
hdr := s.data[s.parseOffset : s.parseOffset+4]
m.PacketLength = uint32(hdr[0]) | uint32(hdr[1])<<8 | uint32(hdr[2])<<16
m.Seq = uint8(hdr[3])
DEBUG("mysqldetailed", "Rows: packet length %d, packet number %d", m.PacketLength, m.Seq)
if len(s.data[s.parseOffset:]) >= int(m.PacketLength)+4 {
s.parseOffset += 4 //header
if uint8(s.data[s.parseOffset]) == 0xfe {
DEBUG("mysqldetailed", "Received EOF packet")
// EOF marker
s.parseOffset += int(m.PacketLength)
if m.end == 0 {
m.end = s.parseOffset
} else {
m.IsTruncated = true
}
m.Size = uint64(s.parseOffset - m.start)
if !m.IsError {
// in case the response was sent successfully
m.IsOK = true
}
return true, true
} else {
s.parseOffset += int(m.PacketLength)
if m.end == 0 && s.parseOffset > MAX_PAYLOAD_SIZE {
// only send up to here, but read until the end
m.end = s.parseOffset
}
m.NumberOfRows += 1
// go to next row
}
} else {
// wait for more
return true, false
}
break
}
}
return true, false
}
func ParseMysql(pkt *Packet, tcp *TcpStream, dir uint8) {
defer RECOVER("ParseMysql exception")
if tcp.mysqlData[dir] == nil {
tcp.mysqlData[dir] = &MysqlStream{
tcpStream: tcp,
data: pkt.payload,
message: &MysqlMessage{Ts: pkt.ts},
}
} else {
// concatenate bytes
tcp.mysqlData[dir].data = append(tcp.mysqlData[dir].data, pkt.payload...)
}
stream := tcp.mysqlData[dir]
for len(stream.data) > 0 {
if stream.message == nil {
stream.message = &MysqlMessage{Ts: pkt.ts}
}
ok, complete := mysqlMessageParser(tcp.mysqlData[dir])
if !ok {
// drop this tcp stream. Will retry parsing with the next
// segment in it
tcp.mysqlData[dir] = nil
DEBUG("mysql", "Ignore MySQL message. Drop tcp stream. Try parsing with the next segment")
return
}
if complete {
// all ok, ship it
msg := stream.data[stream.message.start:stream.message.end]
handleMysql(stream.message, tcp, dir, msg)
// and reset message
stream.PrepareForNewMessage()
} else {
// wait for more data
break
}
}
}
var handleMysql = func(m *MysqlMessage, tcp *TcpStream,
dir uint8, raw_msg []byte) {
m.Stream_id = tcp.id
m.Tuple = tcp.tuple
m.Direction = dir
m.CmdlineTuple = procWatcher.FindProcessesTuple(tcp.tuple)
m.Raw = raw_msg
if m.IsRequest {
receivedMysqlRequest(m)
} else {
receivedMysqlResponse(m)
}
}
func receivedMysqlRequest(msg *MysqlMessage) {
// Add it to the HT
tuple := TcpTuple{
Src_ip: msg.Tuple.Src_ip,
Dst_ip: msg.Tuple.Dst_ip,
Src_port: msg.Tuple.Src_port,
Dst_port: msg.Tuple.Dst_port,
stream_id: msg.Stream_id,
}
trans := mysqlTransactionsMap[tuple]
if trans != nil {
if len(trans.Mysql) != 0 {
WARN("Two requests without a Response. Dropping old request: %s", trans.Mysql)
}
} else {
trans = &MysqlTransaction{Type: "mysql", tuple: tuple}
mysqlTransactionsMap[tuple] = trans
}
trans.ts = msg.Ts
trans.Ts = int64(trans.ts.UnixNano() / 1000) // transactions have microseconds resolution
trans.JsTs = msg.Ts
trans.Src = Endpoint{
Ip: Ipv4_Ntoa(tuple.Src_ip),
Port: tuple.Src_port,
Proc: string(msg.CmdlineTuple.Src),
}
trans.Dst = Endpoint{
Ip: Ipv4_Ntoa(tuple.Dst_ip),
Port: tuple.Dst_port,
Proc: string(msg.CmdlineTuple.Dst),
}
index := strings.Index(msg.Query, " ")
var method string
if index > 0 {
method = strings.ToUpper(msg.Query[:index])
} else {
method = strings.ToUpper(msg.Query)
}
trans.Mysql = bson.M{
"query": msg.Query,
"query.raw": msg.Query,
"method": method,
}
// save Raw message
trans.Request_raw = msg.Query
if trans.timer != nil {
trans.timer.Stop()
}
trans.timer = time.AfterFunc(TransactionTimeout, func() { trans.Expire() })
}
func receivedMysqlResponse(msg *MysqlMessage) {
tuple := TcpTuple{
Src_ip: msg.Tuple.Src_ip,
Dst_ip: msg.Tuple.Dst_ip,
Src_port: msg.Tuple.Src_port,
Dst_port: msg.Tuple.Dst_port,
stream_id: msg.Stream_id,
}
trans := mysqlTransactionsMap[tuple]
if trans == nil {
WARN("Response from unknown transaction. Ignoring.")
return
}
// check if the request was received
if len(trans.Mysql) == 0 {
WARN("Response from unknown transaction. Ignoring.")
return
}
// save json details
trans.Mysql = bson_concat(trans.Mysql, bson.M{
"isok": msg.IsOK,
"affected_rows": msg.AffectedRows,
"insert_id": msg.InsertId,
"tables": msg.Tables,
"num_rows": msg.NumberOfRows,
"size": msg.Size,
"num_fields": msg.NumberOfFields,
"iserror": msg.IsError,
"error_code": msg.ErrorCode,
"error_message": msg.ErrorInfo,
})
trans.ResponseTime = int32(msg.Ts.Sub(trans.ts).Nanoseconds() / 1e6) // resp_time in milliseconds
// save Raw message
if len(msg.Raw) > 0 {
fields, rows := parseMysqlResponse(msg.Raw)
trans.Response_raw = dumpInCSVFormat(fields, rows)
}
err := Publisher.PublishMysqlTransaction(trans)
if err != nil {
WARN("Publish failure: %s", err)
}
DEBUG("mysql", "Mysql transaction completed: %s", trans.Mysql)
DEBUG("mysql", "%s", trans.Response_raw)
// remove from map
delete(mysqlTransactionsMap, trans.tuple)
if trans.timer != nil {
trans.timer.Stop()
}
}
func (trans *MysqlTransaction) Expire() {
// TODO: Here we need to PUBLISH an incomplete/timeout transaction
// remove from map
delete(mysqlTransactionsMap, trans.tuple)
}
func dumpInCSVFormat(fields []string, rows [][]string) string {
var buf bytes.Buffer
writer := csv.NewWriter(&buf)
for i, field := range fields {
fields[i] = strings.Replace(field, "\n", "\\n", -1)
}
if len(fields) > 0 {
writer.Write(fields)
}
for _, row := range rows {
for i, field := range row {
field = strings.Replace(field, "\n", "\\n", -1)
field = strings.Replace(field, "\r", "\\r", -1)
row[i] = field
}
writer.Write(row)
}
writer.Flush()
csv := buf.String()
return csv
}
func parseMysqlResponse(data []byte) ([]string, [][]string) {
length := read_length(data, 0)
if length < 1 {
WARN("Warning: Skipping empty Response")
return []string{}, [][]string{}
}
fields := []string{}
rows := [][]string{}
if uint8(data[4]) == 0x00 {
// OK response
} else if uint8(data[4]) == 0xff {
// Error response
} else {
offset := 5
// Read fields
for {
length = read_length(data, offset)
if uint8(data[offset+4]) == 0xfe {
// EOF
offset += length + 4
break
}
_ /* catalog */, off := read_lstring(data, offset+4)
_ /*database*/, off = read_lstring(data, off)
_ /*table*/, off = read_lstring(data, off)
_ /*org table*/, off = read_lstring(data, off)
name, off := read_lstring(data, off)
_ /* org name */, off = read_lstring(data, off)
fields = append(fields, string(name))
offset += length + 4
}
// Read rows
for offset < len(data) {
var row []string
if uint8(data[offset+4]) == 0xfe {
// EOF
offset += length + 4
break
}
length = read_length(data, offset)
off := offset + 4 // skip length + packet number
start := off
for off < start+length {
var text []byte
if uint8(data[off]) == 0xfb {
text = []byte("NULL")
off++
} else {
text, off = read_lstring(data, off)
}
row = append(row, string(text))
}
rows = append(rows, row)
offset += length + 4
}
}
return fields, rows
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/sqos/beats.git
git@gitee.com:sqos/beats.git
sqos
beats
beats
v0.2.1

搜索帮助