2016-02-10 22:50:07 +00:00
|
|
|
package graphite
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/influxdata/telegraf"
|
|
|
|
)
|
|
|
|
|
2016-11-10 21:05:50 +00:00
|
|
|
// DefaultTemplate const
|
|
|
|
const DefaultTemplate = "host.tags.measurement.field"
|
2016-04-08 22:04:45 +00:00
|
|
|
|
2016-07-12 23:08:03 +00:00
|
|
|
var (
|
|
|
|
fieldDeleter = strings.NewReplacer(".FIELDNAME", "", "FIELDNAME.", "")
|
2016-09-05 13:30:40 +00:00
|
|
|
sanitizedChars = strings.NewReplacer("/", "-", "@", "-", "*", "-", " ", "_", "..", ".", `\`, "", ")", "_", "(", "_")
|
2016-07-12 23:08:03 +00:00
|
|
|
)
|
2016-04-08 22:04:45 +00:00
|
|
|
|
2016-11-10 21:05:50 +00:00
|
|
|
// SerializerGraphite struct
|
|
|
|
type SerializerGraphite struct {
|
2016-04-08 22:04:45 +00:00
|
|
|
Prefix string
|
|
|
|
Template string
|
2016-02-10 22:50:07 +00:00
|
|
|
}
|
|
|
|
|
2016-11-10 21:05:50 +00:00
|
|
|
// Serialize ([]string, error)
|
|
|
|
func (s *SerializerGraphite) Serialize(metric telegraf.Metric) ([]string, error) {
|
2016-02-10 22:50:07 +00:00
|
|
|
out := []string{}
|
2016-02-26 20:06:56 +00:00
|
|
|
|
2016-02-10 22:50:07 +00:00
|
|
|
// Convert UnixNano to Unix timestamps
|
|
|
|
timestamp := metric.UnixNano() / 1000000000
|
|
|
|
|
2016-07-12 23:08:03 +00:00
|
|
|
bucket := SerializeBucketName(metric.Name(), metric.Tags(), s.Template, s.Prefix)
|
2016-06-29 09:58:31 +00:00
|
|
|
if bucket == "" {
|
|
|
|
return out, nil
|
|
|
|
}
|
2016-04-08 22:04:45 +00:00
|
|
|
|
|
|
|
for fieldName, value := range metric.Fields() {
|
|
|
|
// Convert value to string
|
|
|
|
valueS := fmt.Sprintf("%#v", value)
|
|
|
|
point := fmt.Sprintf("%s %s %d",
|
|
|
|
// insert "field" section of template
|
2016-08-17 15:20:32 +00:00
|
|
|
sanitizedChars.Replace(InsertField(bucket, fieldName)),
|
|
|
|
sanitizedChars.Replace(valueS),
|
2016-02-26 20:06:56 +00:00
|
|
|
timestamp)
|
2016-04-08 22:04:45 +00:00
|
|
|
out = append(out, point)
|
2016-02-10 22:50:07 +00:00
|
|
|
}
|
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
|
2016-04-08 22:04:45 +00:00
|
|
|
// SerializeBucketName will take the given measurement name and tags and
|
2016-11-10 21:05:50 +00:00
|
|
|
// produce a graphite bucket. It will use the Serializer.Template
|
|
|
|
// to generate this, or DefaultTemplate.
|
2016-04-08 22:04:45 +00:00
|
|
|
//
|
|
|
|
// NOTE: SerializeBucketName replaces the "field" portion of the template with
|
|
|
|
// FIELDNAME. It is up to the user to replace this. This is so that
|
|
|
|
// SerializeBucketName can be called just once per measurement, rather than
|
2016-11-10 21:05:50 +00:00
|
|
|
// once per field. See Serializer.InsertField() function.
|
2016-07-12 23:08:03 +00:00
|
|
|
func SerializeBucketName(
|
2016-04-08 22:04:45 +00:00
|
|
|
measurement string,
|
|
|
|
tags map[string]string,
|
2016-07-12 23:08:03 +00:00
|
|
|
template string,
|
|
|
|
prefix string,
|
2016-04-08 22:04:45 +00:00
|
|
|
) string {
|
2016-07-12 21:31:08 +00:00
|
|
|
if template == "" {
|
2016-11-10 21:05:50 +00:00
|
|
|
template = DefaultTemplate
|
2016-04-08 22:04:45 +00:00
|
|
|
}
|
|
|
|
tagsCopy := make(map[string]string)
|
|
|
|
for k, v := range tags {
|
|
|
|
tagsCopy[k] = v
|
|
|
|
}
|
2016-02-26 20:06:56 +00:00
|
|
|
|
2016-04-08 22:04:45 +00:00
|
|
|
var out []string
|
2016-07-12 21:31:08 +00:00
|
|
|
templateParts := strings.Split(template, ".")
|
2016-04-08 22:04:45 +00:00
|
|
|
for _, templatePart := range templateParts {
|
|
|
|
switch templatePart {
|
|
|
|
case "measurement":
|
|
|
|
out = append(out, measurement)
|
|
|
|
case "tags":
|
|
|
|
// we will replace this later
|
|
|
|
out = append(out, "TAGS")
|
|
|
|
case "field":
|
|
|
|
// user of SerializeBucketName needs to replace this
|
|
|
|
out = append(out, "FIELDNAME")
|
|
|
|
default:
|
|
|
|
// This is a tag being applied
|
|
|
|
if tagvalue, ok := tagsCopy[templatePart]; ok {
|
2016-11-10 21:05:50 +00:00
|
|
|
if templatePart == "host" {
|
|
|
|
hostSplit := strings.Split(tagvalue, ".")
|
|
|
|
for i := len(hostSplit) - 1; i >= 0; i-- {
|
|
|
|
out = append(out, hostSplit[i])
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
out = append(out, strings.Replace(tagvalue, ".", "_", -1))
|
|
|
|
}
|
2016-04-08 22:04:45 +00:00
|
|
|
delete(tagsCopy, templatePart)
|
|
|
|
}
|
|
|
|
}
|
2016-02-26 20:06:56 +00:00
|
|
|
}
|
2016-04-08 22:04:45 +00:00
|
|
|
|
|
|
|
// insert remaining tags into output name
|
|
|
|
for i, templatePart := range out {
|
|
|
|
if templatePart == "TAGS" {
|
|
|
|
out[i] = buildTags(tagsCopy)
|
|
|
|
break
|
|
|
|
}
|
2016-02-26 20:06:56 +00:00
|
|
|
}
|
2016-04-08 22:04:45 +00:00
|
|
|
|
2016-06-29 09:58:31 +00:00
|
|
|
if len(out) == 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2016-07-12 23:08:03 +00:00
|
|
|
if prefix == "" {
|
2016-08-17 15:20:32 +00:00
|
|
|
return strings.Join(out, ".")
|
2016-04-08 22:04:45 +00:00
|
|
|
}
|
2016-08-17 15:20:32 +00:00
|
|
|
return prefix + "." + strings.Join(out, ".")
|
2016-02-26 20:06:56 +00:00
|
|
|
}
|
|
|
|
|
2016-04-08 22:04:45 +00:00
|
|
|
// InsertField takes the bucket string from SerializeBucketName and replaces the
|
|
|
|
// FIELDNAME portion. If fieldName == "value", it will simply delete the
|
|
|
|
// FIELDNAME portion.
|
|
|
|
func InsertField(bucket, fieldName string) string {
|
|
|
|
// if the field name is "value", then dont use it
|
|
|
|
if fieldName == "value" {
|
|
|
|
return fieldDeleter.Replace(bucket)
|
|
|
|
}
|
|
|
|
return strings.Replace(bucket, "FIELDNAME", fieldName, 1)
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildTags(tags map[string]string) string {
|
2016-02-10 22:50:07 +00:00
|
|
|
var keys []string
|
|
|
|
for k := range tags {
|
|
|
|
keys = append(keys, k)
|
|
|
|
}
|
|
|
|
sort.Strings(keys)
|
|
|
|
|
2016-11-10 21:05:50 +00:00
|
|
|
var tagStr string
|
|
|
|
var tagValue string
|
|
|
|
var reversedHost []string
|
2016-02-10 22:50:07 +00:00
|
|
|
for i, k := range keys {
|
2016-11-10 21:05:50 +00:00
|
|
|
if k == "host" {
|
|
|
|
hostSplit := strings.Split(tags[k], ".")
|
|
|
|
for i := len(hostSplit) - 1; i >= 0; i-- {
|
|
|
|
reversedHost = append(reversedHost, hostSplit[i])
|
|
|
|
}
|
|
|
|
tagValue = strings.Join(reversedHost, ".")
|
|
|
|
} else {
|
|
|
|
tagValue = strings.Replace(tags[k], ".", "_", -1)
|
|
|
|
}
|
2016-02-10 22:50:07 +00:00
|
|
|
if i == 0 {
|
2016-11-10 21:05:50 +00:00
|
|
|
tagStr += tagValue
|
2016-02-10 22:50:07 +00:00
|
|
|
} else {
|
2016-11-10 21:05:50 +00:00
|
|
|
tagStr += "." + tagValue
|
2016-02-10 22:50:07 +00:00
|
|
|
}
|
|
|
|
}
|
2016-11-10 21:05:50 +00:00
|
|
|
return tagStr
|
2016-02-10 22:50:07 +00:00
|
|
|
}
|