Adding upstream version 1.34.4.
Signed-off-by: Daniel Baumann <daniel@debian.org>
This commit is contained in:
parent
e393c3af3f
commit
4978089aab
4963 changed files with 677545 additions and 0 deletions
53
plugins/outputs/postgresql/utils/column.go
Normal file
53
plugins/outputs/postgresql/utils/column.go
Normal file
|
@ -0,0 +1,53 @@
|
|||
package utils
|
||||
|
||||
// This is split out from the 'postgresql' package as its depended upon by both the 'postgresql' and
|
||||
// 'postgresql/template' packages.
|
||||
|
||||
import (
|
||||
"sort"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// ColumnRole specifies the role of a column in a metric.
|
||||
// It helps map the columns to the DB.
|
||||
type ColumnRole int
|
||||
|
||||
const (
|
||||
TimeColType ColumnRole = iota + 1
|
||||
TagsIDColType
|
||||
TagColType
|
||||
FieldColType
|
||||
)
|
||||
|
||||
type Column struct {
|
||||
Name string
|
||||
// the data type of each column should have in the db. used when checking
|
||||
// if the schema matches or it needs updates
|
||||
Type string
|
||||
// the role each column has, helps properly map the metric to the db
|
||||
Role ColumnRole
|
||||
}
|
||||
|
||||
// ColumnList implements sort.Interface.
|
||||
// Columns are sorted first into groups of time,tag_id,tags,fields, and then alphabetically within
|
||||
// each group.
|
||||
type ColumnList []Column
|
||||
|
||||
func (cl ColumnList) Len() int {
|
||||
return len(cl)
|
||||
}
|
||||
|
||||
func (cl ColumnList) Less(i, j int) bool {
|
||||
if cl[i].Role != cl[j].Role {
|
||||
return cl[i].Role < cl[j].Role
|
||||
}
|
||||
return strings.ToLower(cl[i].Name) < strings.ToLower(cl[j].Name)
|
||||
}
|
||||
|
||||
func (cl ColumnList) Swap(i, j int) {
|
||||
cl[i], cl[j] = cl[j], cl[i]
|
||||
}
|
||||
|
||||
func (cl ColumnList) Sort() {
|
||||
sort.Sort(cl)
|
||||
}
|
117
plugins/outputs/postgresql/utils/utils.go
Normal file
117
plugins/outputs/postgresql/utils/utils.go
Normal file
|
@ -0,0 +1,117 @@
|
|||
package utils
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"hash/fnv"
|
||||
"strings"
|
||||
"sync/atomic"
|
||||
|
||||
"github.com/jackc/pgx/v4"
|
||||
|
||||
"github.com/influxdata/telegraf"
|
||||
)
|
||||
|
||||
func TagListToJSON(tagList []*telegraf.Tag) []byte {
|
||||
tags := make(map[string]string, len(tagList))
|
||||
for _, tag := range tagList {
|
||||
tags[tag.Key] = tag.Value
|
||||
}
|
||||
//nolint:errcheck // unable to propagate error
|
||||
bs, _ := json.Marshal(tags)
|
||||
return bs
|
||||
}
|
||||
|
||||
func FieldListToJSON(fieldList []*telegraf.Field) ([]byte, error) {
|
||||
fields := make(map[string]interface{}, len(fieldList))
|
||||
for _, field := range fieldList {
|
||||
fields[field.Key] = field.Value
|
||||
}
|
||||
return json.Marshal(fields)
|
||||
}
|
||||
|
||||
// QuoteIdentifier returns a sanitized string safe to use in SQL as an identifier
|
||||
func QuoteIdentifier(name string) string {
|
||||
return pgx.Identifier{name}.Sanitize()
|
||||
}
|
||||
|
||||
// QuoteLiteral returns a sanitized string safe to use in sql as a string literal
|
||||
func QuoteLiteral(name string) string {
|
||||
return "'" + strings.Replace(name, "'", "''", -1) + "'"
|
||||
}
|
||||
|
||||
// FullTableName returns a sanitized table name with its schema (if supplied)
|
||||
func FullTableName(schema, name string) pgx.Identifier {
|
||||
if schema != "" {
|
||||
return pgx.Identifier{schema, name}
|
||||
}
|
||||
|
||||
return pgx.Identifier{name}
|
||||
}
|
||||
|
||||
// PGXLogger makes telegraf.Logger compatible with pgx.Logger
|
||||
type PGXLogger struct {
|
||||
telegraf.Logger
|
||||
}
|
||||
|
||||
func (l PGXLogger) Log(_ context.Context, level pgx.LogLevel, msg string, data map[string]interface{}) {
|
||||
switch level {
|
||||
case pgx.LogLevelError:
|
||||
l.Errorf("PG %s - %+v", msg, data)
|
||||
case pgx.LogLevelWarn:
|
||||
l.Warnf("PG %s - %+v", msg, data)
|
||||
case pgx.LogLevelInfo, pgx.LogLevelNone:
|
||||
l.Infof("PG %s - %+v", msg, data)
|
||||
case pgx.LogLevelDebug, pgx.LogLevelTrace:
|
||||
l.Debugf("PG %s - %+v", msg, data)
|
||||
default:
|
||||
l.Debugf("PG %s - %+v", msg, data)
|
||||
}
|
||||
}
|
||||
|
||||
func GetTagID(metric telegraf.Metric) int64 {
|
||||
hash := fnv.New64a()
|
||||
for _, tag := range metric.TagList() {
|
||||
hash.Write([]byte(tag.Key))
|
||||
hash.Write([]byte{0})
|
||||
hash.Write([]byte(tag.Value))
|
||||
hash.Write([]byte{0})
|
||||
}
|
||||
// Convert to int64 as postgres does not support uint64
|
||||
return int64(hash.Sum64())
|
||||
}
|
||||
|
||||
// WaitGroup is similar to sync.WaitGroup, but allows interruptible waiting (e.g. a timeout).
|
||||
type WaitGroup struct {
|
||||
count int32
|
||||
done chan struct{}
|
||||
}
|
||||
|
||||
func NewWaitGroup() *WaitGroup {
|
||||
return &WaitGroup{
|
||||
done: make(chan struct{}),
|
||||
}
|
||||
}
|
||||
|
||||
func (wg *WaitGroup) Add(i int32) {
|
||||
select {
|
||||
case <-wg.done:
|
||||
panic("use of an already-done WaitGroup")
|
||||
default:
|
||||
}
|
||||
atomic.AddInt32(&wg.count, i)
|
||||
}
|
||||
|
||||
func (wg *WaitGroup) Done() {
|
||||
i := atomic.AddInt32(&wg.count, -1)
|
||||
if i == 0 {
|
||||
close(wg.done)
|
||||
}
|
||||
if i < 0 {
|
||||
panic("too many Done() calls")
|
||||
}
|
||||
}
|
||||
|
||||
func (wg *WaitGroup) C() <-chan struct{} {
|
||||
return wg.done
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue