influxdb/telegraf.go

323 lines
11 KiB
Go
Raw Normal View History

package influxdb
2018-10-09 00:45:42 +00:00
import (
"context"
2018-10-16 00:38:36 +00:00
"encoding/json"
"fmt"
"regexp"
2018-10-16 00:38:36 +00:00
"github.com/BurntSushi/toml"
"github.com/influxdata/influxdb/v2/kit/platform"
"github.com/influxdata/influxdb/v2/kit/platform/errors"
"github.com/influxdata/influxdb/v2/telegraf/plugins"
"github.com/influxdata/influxdb/v2/telegraf/plugins/inputs"
"github.com/influxdata/influxdb/v2/telegraf/plugins/outputs"
2018-10-09 00:45:42 +00:00
)
const (
ErrTelegrafConfigInvalidOrgID = "invalid org ID" // ErrTelegrafConfigInvalidOrgID is the error message for a missing or invalid organization ID.
ErrTelegrafConfigNotFound = "telegraf configuration not found" // ErrTelegrafConfigNotFound is the error message for a missing telegraf config.
ErrTelegrafPluginNameUnmatch = "the telegraf plugin is name %s doesn't match the config %s"
ErrNoTelegrafPlugins = "there is no telegraf plugin in the config"
ErrUnsupportTelegrafPluginType = "unsupported telegraf plugin type %s"
ErrUnsupportTelegrafPluginName = "unsupported telegraf plugin %s, type %s"
)
// ops for buckets error and buckets op logs.
var (
OpFindTelegrafConfigByID = "FindTelegrafConfigByID"
OpFindTelegrafConfigs = "FindTelegrafConfigs"
OpCreateTelegrafConfig = "CreateTelegrafConfig"
OpUpdateTelegrafConfig = "UpdateTelegrafConfig"
OpDeleteTelegrafConfig = "DeleteTelegrafConfig"
)
2018-10-09 00:45:42 +00:00
// TelegrafConfigStore represents a service for managing telegraf config data.
type TelegrafConfigStore interface {
// FindTelegrafConfigByID returns a single telegraf config by ID.
FindTelegrafConfigByID(ctx context.Context, id platform.ID) (*TelegrafConfig, error)
2018-10-09 00:45:42 +00:00
// FindTelegrafConfigs returns a list of telegraf configs that match filter and the total count of matching telegraf configs.
// Additional options provide pagination & sorting.
FindTelegrafConfigs(ctx context.Context, filter TelegrafConfigFilter, opt ...FindOptions) ([]*TelegrafConfig, int, error)
2018-10-09 00:45:42 +00:00
2018-10-16 00:38:36 +00:00
// CreateTelegrafConfig creates a new telegraf config and sets b.ID with the new identifier.
CreateTelegrafConfig(ctx context.Context, tc *TelegrafConfig, userID platform.ID) error
2018-10-09 00:45:42 +00:00
// UpdateTelegrafConfig updates a single telegraf config.
// Returns the new telegraf config after update.
UpdateTelegrafConfig(ctx context.Context, id platform.ID, tc *TelegrafConfig, userID platform.ID) (*TelegrafConfig, error)
2018-10-09 00:45:42 +00:00
// DeleteTelegrafConfig removes a telegraf config by ID.
DeleteTelegrafConfig(ctx context.Context, id platform.ID) error
2018-10-09 00:45:42 +00:00
}
// TelegrafConfigFilter represents a set of filter that restrict the returned telegraf configs.
type TelegrafConfigFilter struct {
OrgID *platform.ID
2019-04-15 19:25:48 +00:00
Organization *string
}
2018-10-09 00:45:42 +00:00
// TelegrafConfig stores telegraf config for one telegraf instance.
type TelegrafConfig struct {
ID platform.ID `json:"id,omitempty"` // ID of this config object.
OrgID platform.ID `json:"orgID,omitempty"` // OrgID is the id of the owning organization.
Name string `json:"name,omitempty"` // Name of this config object.
Description string `json:"description,omitempty"` // Decription of this config object.
Config string `json:"config,omitempty"` // ConfigTOML contains the raw toml config.
Metadata map[string]interface{} `json:"metadata,omitempty"` // Metadata for the config.
2018-10-16 00:38:36 +00:00
}
var pluginCount = regexp.MustCompilePOSIX(`\[\[(inputs\..*|outputs\..*|aggregators\..*|processors\..*)\]\]`)
// CountPlugins returns a map of the number of times each plugin is used.
func (tc *TelegrafConfig) CountPlugins() map[string]float64 {
plugins := map[string]float64{}
founds := pluginCount.FindAllStringSubmatch(tc.Config, -1)
for _, v := range founds {
if len(v) < 2 {
continue
}
plugins[v[1]]++
}
return plugins
}
// UnmarshalJSON implement the json.Unmarshaler interface.
// Gets called when reading from the kv db. mostly legacy so loading old/stored configs still work.
// May not remove for a while. Primarily will get hit when user views/downloads config.
func (tc *TelegrafConfig) UnmarshalJSON(b []byte) error {
tcd := new(telegrafConfigDecode)
2018-10-16 00:38:36 +00:00
if err := json.Unmarshal(b, tcd); err != nil {
return err
}
2018-10-09 00:45:42 +00:00
orgID := tcd.OrgID
if orgID == nil || !orgID.Valid() {
orgID = tcd.OrganizationID
}
2018-10-09 00:45:42 +00:00
if tcd.ID != nil {
tc.ID = *tcd.ID
}
2018-10-16 00:38:36 +00:00
if orgID != nil {
tc.OrgID = *orgID
}
2018-10-09 00:45:42 +00:00
tc.Name = tcd.Name
tc.Description = tcd.Description
2018-10-09 00:45:42 +00:00
// Prefer new structure; use full toml config.
tc.Config = tcd.Config
tc.Metadata = tcd.Metadata
2018-10-09 00:45:42 +00:00
if tcd.Plugins != nil {
// legacy, remove after some moons. or a migration.
if len(tcd.Plugins) > 0 {
bkts, conf, err := decodePluginRaw(tcd)
if err != nil {
return err
}
tc.Config = plugins.AgentConfig + conf
tc.Metadata = map[string]interface{}{"buckets": bkts}
} else if c, ok := plugins.GetPlugin("output", "influxdb_v2"); ok {
// Handles legacy adding of default plugins (agent and output).
tc.Config = plugins.AgentConfig + c.Config
tc.Metadata = map[string]interface{}{
"buckets": []string{},
}
}
} else if tcd.Metadata == nil || len(tcd.Metadata) == 0 {
// Get buckets from the config.
m, err := parseMetadata(tc.Config)
if err != nil {
return err
}
2018-10-16 00:38:36 +00:00
tc.Metadata = m
}
return nil
2018-10-16 00:38:36 +00:00
}
type buckets []string
2018-10-17 19:51:35 +00:00
func (t *buckets) UnmarshalTOML(data interface{}) error {
dataOk, ok := data.(map[string]interface{})
2018-10-17 19:51:35 +00:00
if !ok {
return &errors.Error{
Code: errors.EEmptyValue,
Msg: "no config to get buckets",
}
2018-10-17 19:51:35 +00:00
}
bkts := []string{}
2018-10-17 19:51:35 +00:00
for tp, ps := range dataOk {
if tp != "outputs" {
2018-10-17 19:51:35 +00:00
continue
}
plugins, ok := ps.(map[string]interface{})
if !ok {
return &errors.Error{
Code: errors.EEmptyValue,
Msg: "no plugins in config to get buckets",
2018-10-17 19:51:35 +00:00
}
}
for name, configDataArray := range plugins {
if name != "influxdb_v2" {
2018-10-17 19:51:35 +00:00
continue
}
config, ok := configDataArray.([]map[string]interface{})
if !ok {
return &errors.Error{
Code: errors.EEmptyValue,
Msg: "influxdb_v2 output has no config",
}
}
for i := range config {
if b, ok := config[i]["bucket"]; ok {
bkts = append(bkts, b.(string))
2018-10-17 19:51:35 +00:00
}
}
}
}
*t = bkts
2018-10-17 19:51:35 +00:00
return nil
}
func parseMetadata(cfg string) (map[string]interface{}, error) {
bs := []string{}
this := &buckets{}
_, err := toml.Decode(cfg, this)
if err != nil {
return nil, err
2018-10-17 19:51:35 +00:00
}
for _, i := range *this {
if i != "" {
bs = append(bs, i)
}
2018-10-17 19:51:35 +00:00
}
return map[string]interface{}{"buckets": bs}, nil
2018-10-16 00:38:36 +00:00
}
// return bucket, config, error
func decodePluginRaw(tcd *telegrafConfigDecode) ([]string, string, error) {
2018-10-16 00:38:36 +00:00
op := "unmarshal telegraf config raw plugin"
ps := ""
bucket := []string{}
for _, pr := range tcd.Plugins {
var tpFn func() plugins.Config
2018-10-16 00:38:36 +00:00
var ok bool
2018-10-16 00:38:36 +00:00
switch pr.Type {
case "input":
tpFn, ok = availableInputPlugins[pr.Name]
case "output":
tpFn, ok = availableOutputPlugins[pr.Name]
2018-10-16 00:38:36 +00:00
default:
return nil, "", &errors.Error{
Code: errors.EInvalid,
2018-10-16 00:38:36 +00:00
Op: op,
Msg: fmt.Sprintf(ErrUnsupportTelegrafPluginType, pr.Type),
2018-10-16 00:38:36 +00:00
}
}
if !ok {
// This removes the validation (and does not create toml) for new "input" plugins
// but keeps in place the existing behavior for certain "input" plugins
if pr.Type == "output" {
return nil, "", &errors.Error{
Code: errors.EInvalid,
Op: op,
Msg: fmt.Sprintf(ErrUnsupportTelegrafPluginName, pr.Name, pr.Type),
}
2018-10-16 00:38:36 +00:00
}
continue
}
config := tpFn()
// if pr.Config if empty, make it a blank obj,
// so it will still go to the unmarshalling process to validate.
if pr.Config == nil || len(string(pr.Config)) == 0 {
pr.Config = []byte("{}")
}
if err := json.Unmarshal(pr.Config, config); err != nil {
return nil, "", &errors.Error{
Code: errors.EInvalid,
Err: err,
Op: op,
2018-10-16 00:38:36 +00:00
}
}
if pr.Name == "influxdb_v2" {
if b := config.(*outputs.InfluxDBV2).Bucket; b != "" {
bucket = []string{b}
}
2018-10-16 00:38:36 +00:00
}
ps += config.TOML()
2018-10-16 00:38:36 +00:00
}
return bucket, ps, nil
}
// telegrafConfigDecode is the helper struct for json decoding. legacy.
type telegrafConfigDecode struct {
ID *platform.ID `json:"id,omitempty"`
OrganizationID *platform.ID `json:"organizationID,omitempty"`
OrgID *platform.ID `json:"orgID,omitempty"`
Name string `json:"name,omitempty"`
Description string `json:"description,omitempty"`
Config string `json:"config,omitempty"`
Plugins []telegrafPluginDecode `json:"plugins,omitempty"`
Metadata map[string]interface{} `json:"metadata,omitempty"`
}
// telegrafPluginDecode is the helper struct for json decoding. legacy.
type telegrafPluginDecode struct {
Type string `json:"type,omitempty"` // Type of the plugin.
Name string `json:"name,omitempty"` // Name of the plugin.
Alias string `json:"alias,omitempty"` // Alias of the plugin.
Description string `json:"description,omitempty"` // Description of the plugin.
Config json.RawMessage `json:"config,omitempty"` // Config is the currently stored plugin configuration.
2018-10-16 00:38:36 +00:00
}
var availableInputPlugins = map[string](func() plugins.Config){
"cpu": func() plugins.Config { return &inputs.CPUStats{} },
"disk": func() plugins.Config { return &inputs.DiskStats{} },
"diskio": func() plugins.Config { return &inputs.DiskIO{} },
"docker": func() plugins.Config { return &inputs.Docker{} },
"file": func() plugins.Config { return &inputs.File{} },
"kernel": func() plugins.Config { return &inputs.Kernel{} },
"kubernetes": func() plugins.Config { return &inputs.Kubernetes{} },
"logparser": func() plugins.Config { return &inputs.LogParserPlugin{} },
"mem": func() plugins.Config { return &inputs.MemStats{} },
"net_response": func() plugins.Config { return &inputs.NetResponse{} },
"net": func() plugins.Config { return &inputs.NetIOStats{} },
"nginx": func() plugins.Config { return &inputs.Nginx{} },
"processes": func() plugins.Config { return &inputs.Processes{} },
"procstat": func() plugins.Config { return &inputs.Procstat{} },
"prometheus": func() plugins.Config { return &inputs.Prometheus{} },
"redis": func() plugins.Config { return &inputs.Redis{} },
"swap": func() plugins.Config { return &inputs.SwapStats{} },
"syslog": func() plugins.Config { return &inputs.Syslog{} },
"system": func() plugins.Config { return &inputs.SystemStats{} },
"tail": func() plugins.Config { return &inputs.Tail{} },
2018-10-16 00:38:36 +00:00
}
var availableOutputPlugins = map[string](func() plugins.Config){
"file": func() plugins.Config { return &outputs.File{} },
"influxdb_v2": func() plugins.Config { return &outputs.InfluxDBV2{} },
2018-10-16 00:38:36 +00:00
}