Add mysql plugin
This commit is contained in:
parent
60a49243cf
commit
00c99ec373
|
@ -0,0 +1,134 @@
|
||||||
|
package mysql
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
_ "github.com/go-sql-driver/mysql"
|
||||||
|
"github.com/influxdb/tivan/plugins"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Server struct {
|
||||||
|
Address string
|
||||||
|
}
|
||||||
|
|
||||||
|
type Mysql struct {
|
||||||
|
Disabled bool
|
||||||
|
Servers []*Server
|
||||||
|
}
|
||||||
|
|
||||||
|
var localhost = &Server{}
|
||||||
|
|
||||||
|
func (m *Mysql) Gather(acc plugins.Accumulator) error {
|
||||||
|
if m.Disabled {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(m.Servers) == 0 {
|
||||||
|
// if we can't get stats in this case, thats fine, don't report
|
||||||
|
// an error.
|
||||||
|
m.gatherServer(localhost, acc)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, serv := range m.Servers {
|
||||||
|
err := m.gatherServer(serv, acc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
type mapping struct {
|
||||||
|
onServer string
|
||||||
|
inExport string
|
||||||
|
}
|
||||||
|
|
||||||
|
var mappings = []*mapping{
|
||||||
|
{
|
||||||
|
onServer: "Bytes_",
|
||||||
|
inExport: "mysql_bytes_",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onServer: "Com_",
|
||||||
|
inExport: "mysql_commands_",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onServer: "Handler_",
|
||||||
|
inExport: "mysql_handler_",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onServer: "Innodb_",
|
||||||
|
inExport: "mysql_innodb_",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
onServer: "Threads_",
|
||||||
|
inExport: "mysql_threads_",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Mysql) gatherServer(serv *Server, acc plugins.Accumulator) error {
|
||||||
|
db, err := sql.Open("mysql", serv.Address)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
|
rows, err := db.Query(`SHOW /*!50002 GLOBAL */ STATUS`)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for rows.Next() {
|
||||||
|
var name string
|
||||||
|
var val interface{}
|
||||||
|
|
||||||
|
err = rows.Scan(&name, &val)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
var found bool
|
||||||
|
|
||||||
|
for _, mapped := range mappings {
|
||||||
|
if strings.HasPrefix(name, mapped.onServer) {
|
||||||
|
i, _ := strconv.Atoi(string(val.([]byte)))
|
||||||
|
acc.Add(mapped.inExport+name[len(mapped.onServer):], i, nil)
|
||||||
|
found = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if found {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch name {
|
||||||
|
case "Queries":
|
||||||
|
i, err := strconv.ParseInt(string(val.([]byte)), 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
acc.Add("mysql_queries", i, nil)
|
||||||
|
case "Slow_queries":
|
||||||
|
i, err := strconv.ParseInt(string(val.([]byte)), 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
acc.Add("mysql_slow_queries", i, nil)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
plugins.Add("mysql", func() plugins.Plugin {
|
||||||
|
return &Mysql{}
|
||||||
|
})
|
||||||
|
}
|
|
@ -0,0 +1,68 @@
|
||||||
|
package mysql
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/influxdb/tivan/testutil"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMysqlGeneratesMetrics(t *testing.T) {
|
||||||
|
m := &Mysql{
|
||||||
|
Servers: []*Server{
|
||||||
|
{
|
||||||
|
Address: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
var acc testutil.Accumulator
|
||||||
|
|
||||||
|
err := m.Gather(&acc)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
prefixes := []struct {
|
||||||
|
prefix string
|
||||||
|
count int
|
||||||
|
}{
|
||||||
|
{"mysql_commands", 141},
|
||||||
|
{"mysql_handler", 18},
|
||||||
|
{"mysql_bytes", 2},
|
||||||
|
{"mysql_innodb", 51},
|
||||||
|
{"mysql_threads", 4},
|
||||||
|
}
|
||||||
|
|
||||||
|
intMetrics := []string{
|
||||||
|
"mysql_queries",
|
||||||
|
"mysql_slow_queries",
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, prefix := range prefixes {
|
||||||
|
var count int
|
||||||
|
|
||||||
|
for _, p := range acc.Points {
|
||||||
|
if strings.HasPrefix(p.Name, prefix.prefix) {
|
||||||
|
count++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, prefix.count, count)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, metric := range intMetrics {
|
||||||
|
assert.True(t, acc.HasIntValue(metric))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMysqlDefaultsToLocal(t *testing.T) {
|
||||||
|
m := &Mysql{}
|
||||||
|
|
||||||
|
var acc testutil.Accumulator
|
||||||
|
|
||||||
|
err := m.Gather(&acc)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
assert.True(t, len(acc.Points) > 0)
|
||||||
|
}
|
Loading…
Reference in New Issue