-
Notifications
You must be signed in to change notification settings - Fork 5.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
4852b5c
commit b96dd8c
Showing
2 changed files
with
135 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,134 @@ | ||
package graphite | ||
|
||
import ( | ||
"errors" | ||
"fmt" | ||
"github.com/influxdb/influxdb/client/v2" | ||
"github.com/influxdb/telegraf/outputs" | ||
"log" | ||
"math/rand" | ||
"net" | ||
"strings" | ||
"time" | ||
) | ||
|
||
type Graphite struct { | ||
// URL is only for backwards compatability | ||
Servers []string | ||
Prefix string | ||
Timeout int | ||
conns []net.Conn | ||
} | ||
|
||
var sampleConfig = ` | ||
# TCP raw endpoint for your graphite instance. | ||
servers = ["mygraphiteserver:2003"] # default "localhost:2003" | ||
# Prefix metrics name | ||
prefix = "" # default "" | ||
# Timeout in second | ||
timeout = 2 # default 2 | ||
` | ||
|
||
func (g *Graphite) Connect() error { | ||
// Set default values | ||
if g.Timeout <= 0 { | ||
g.Timeout = 2 | ||
} | ||
if len(g.Servers) == 0 { | ||
g.Servers = append(g.Servers, "localhost:2003") | ||
} | ||
// Get Connections | ||
var conns []net.Conn | ||
for _, server := range g.Servers { | ||
conn, err := net.DialTimeout("tcp", server, time.Duration(g.Timeout)*time.Second) | ||
if err == nil { | ||
conns = append(conns, conn) | ||
} | ||
} | ||
g.conns = conns | ||
return nil | ||
} | ||
|
||
func (g *Graphite) Close() error { | ||
// Closing all connections | ||
for _, conn := range g.conns { | ||
conn.Close() | ||
} | ||
return nil | ||
} | ||
|
||
func (g *Graphite) SampleConfig() string { | ||
return sampleConfig | ||
} | ||
|
||
func (g *Graphite) Description() string { | ||
return "Configuration for Graphite server to send metrics to using TCP raw protocol" | ||
} | ||
|
||
// Choose a random server in the cluster to write to until a successful write | ||
// occurs, logging each unsuccessful. If all servers fail, return error. | ||
func (g *Graphite) Write(points []*client.Point) error { | ||
// Prepare data | ||
var bp []string | ||
for _, point := range points { | ||
// Get name | ||
names := strings.SplitN(point.Name(), "_", 2) | ||
var name string | ||
if len(names) > 1 { | ||
name = names[1] | ||
} else { | ||
name = point.Name() | ||
} | ||
// Convert UnixNano to Unix timestamps | ||
timestamp := point.UnixNano() / 1000000000 | ||
|
||
for field_name, value := range point.Fields() { | ||
var graphitePoint string | ||
if name == field_name { | ||
graphitePoint = fmt.Sprintf("%s.%s %s %d\n", | ||
strings.Replace(point.Tags()["host"], ".", "_", -1), | ||
strings.Replace(name, ".", "_", -1), | ||
value, | ||
timestamp) | ||
} else { | ||
graphitePoint = fmt.Sprintf("%s.%s.%s %s %d\n", | ||
strings.Replace(point.Tags()["host"], ".", "_", -1), | ||
strings.Replace(name, ".", "_", -1), | ||
strings.Replace(field_name, ".", "_", -1), | ||
value, | ||
timestamp) | ||
} | ||
if g.Prefix != "" { | ||
graphitePoint = fmt.Sprintf("%s.%s", g.Prefix, graphitePoint) | ||
} | ||
bp = append(bp, graphitePoint) | ||
//fmt.Printf(graphitePoint) | ||
} | ||
} | ||
graphitePoints := strings.Join(bp, "\n") | ||
|
||
// This will get set to nil if a successful write occurs | ||
err := errors.New("Could not write to any Graphite server in cluster\n") | ||
|
||
// Send data to a random server | ||
p := rand.Perm(len(g.conns)) | ||
for _, n := range p { | ||
if _, e := fmt.Fprintf(g.conns[n], graphitePoints); e != nil { | ||
// Error | ||
log.Println("ERROR: " + err.Error()) | ||
// Let's try the next one | ||
} else { | ||
// Success | ||
err = nil | ||
break | ||
} | ||
} | ||
|
||
return err | ||
} | ||
|
||
func init() { | ||
outputs.Add("graphite", func() outputs.Output { | ||
return &Graphite{} | ||
}) | ||
} |