170 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			Go
		
	
	
	
| // selfstat is a package for tracking and collecting internal statistics
 | |
| // about telegraf. Metrics can be registered using this package, and then
 | |
| // incremented or set within your code. If the inputs.internal plugin is enabled,
 | |
| // then all registered stats will be collected as they would by any other input
 | |
| // plugin.
 | |
| package selfstat
 | |
| 
 | |
| import (
 | |
| 	"hash/fnv"
 | |
| 	"log"
 | |
| 	"sort"
 | |
| 	"sync"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/influxdata/telegraf"
 | |
| 	"github.com/influxdata/telegraf/metric"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	registry *rgstry
 | |
| )
 | |
| 
 | |
| // Stat is an interface for dealing with telegraf statistics collected
 | |
| // on itself.
 | |
| type Stat interface {
 | |
| 	// Name is the name of the measurement
 | |
| 	Name() string
 | |
| 
 | |
| 	// FieldName is the name of the measurement field
 | |
| 	FieldName() string
 | |
| 
 | |
| 	// Tags is a tag map. Each time this is called a new map is allocated.
 | |
| 	Tags() map[string]string
 | |
| 
 | |
| 	// Key is the unique measurement+tags key of the stat.
 | |
| 	Key() uint64
 | |
| 
 | |
| 	// Incr increments a regular stat by 'v'.
 | |
| 	// in the case of a timing stat, increment adds the timing to the cache.
 | |
| 	Incr(v int64)
 | |
| 
 | |
| 	// Set sets a regular stat to 'v'.
 | |
| 	// in the case of a timing stat, set adds the timing to the cache.
 | |
| 	Set(v int64)
 | |
| 
 | |
| 	// Get gets the value of the stat. In the case of timings, this returns
 | |
| 	// an average value of all timings received since the last call to Get().
 | |
| 	// If no timings were received, it returns the previous value.
 | |
| 	Get() int64
 | |
| }
 | |
| 
 | |
| // Register registers the given measurement, field, and tags in the selfstat
 | |
| // registry. If given an identical measurement, it will return the stat that's
 | |
| // already been registered.
 | |
| //
 | |
| // The returned Stat can be incremented by the consumer of Register(), and it's
 | |
| // value will be returned as a telegraf metric when Metrics() is called.
 | |
| func Register(measurement, field string, tags map[string]string) Stat {
 | |
| 	return registry.register(&stat{
 | |
| 		measurement: "internal_" + measurement,
 | |
| 		field:       field,
 | |
| 		tags:        tags,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // RegisterTiming registers the given measurement, field, and tags in the selfstat
 | |
| // registry. If given an identical measurement, it will return the stat that's
 | |
| // already been registered.
 | |
| //
 | |
| // Timing stats differ from regular stats in that they accumulate multiple
 | |
| // "timings" added to them, and will return the average when Get() is called.
 | |
| // After Get() is called, the average is cleared and the next timing returned
 | |
| // from Get() will only reflect timings added since the previous call to Get().
 | |
| // If Get() is called without receiving any new timings, then the previous value
 | |
| // is used.
 | |
| //
 | |
| // In other words, timings are an averaged metric that get cleared on each call
 | |
| // to Get().
 | |
| //
 | |
| // The returned Stat can be incremented by the consumer of Register(), and it's
 | |
| // value will be returned as a telegraf metric when Metrics() is called.
 | |
| func RegisterTiming(measurement, field string, tags map[string]string) Stat {
 | |
| 	return registry.register(&timingStat{
 | |
| 		measurement: "internal_" + measurement,
 | |
| 		field:       field,
 | |
| 		tags:        tags,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // Metrics returns all registered stats as telegraf metrics.
 | |
| func Metrics() []telegraf.Metric {
 | |
| 	registry.mu.Lock()
 | |
| 	now := time.Now()
 | |
| 	metrics := make([]telegraf.Metric, len(registry.stats))
 | |
| 	i := 0
 | |
| 	for _, stats := range registry.stats {
 | |
| 		if len(stats) > 0 {
 | |
| 			var tags map[string]string
 | |
| 			var name string
 | |
| 			fields := map[string]interface{}{}
 | |
| 			j := 0
 | |
| 			for fieldname, stat := range stats {
 | |
| 				if j == 0 {
 | |
| 					tags = stat.Tags()
 | |
| 					name = stat.Name()
 | |
| 				}
 | |
| 				fields[fieldname] = stat.Get()
 | |
| 				j++
 | |
| 			}
 | |
| 			metric, err := metric.New(name, tags, fields, now)
 | |
| 			if err != nil {
 | |
| 				log.Printf("E! Error creating selfstat metric: %s", err)
 | |
| 				continue
 | |
| 			}
 | |
| 			metrics[i] = metric
 | |
| 			i++
 | |
| 		}
 | |
| 	}
 | |
| 	registry.mu.Unlock()
 | |
| 	return metrics
 | |
| }
 | |
| 
 | |
| type rgstry struct {
 | |
| 	stats map[uint64]map[string]Stat
 | |
| 	mu    sync.Mutex
 | |
| }
 | |
| 
 | |
| func (r *rgstry) register(s Stat) Stat {
 | |
| 	r.mu.Lock()
 | |
| 	defer r.mu.Unlock()
 | |
| 	if stats, ok := r.stats[s.Key()]; ok {
 | |
| 		// measurement exists
 | |
| 		if stat, ok := stats[s.FieldName()]; ok {
 | |
| 			// field already exists, so don't create a new one
 | |
| 			return stat
 | |
| 		}
 | |
| 		r.stats[s.Key()][s.FieldName()] = s
 | |
| 		return s
 | |
| 	} else {
 | |
| 		// creating a new unique metric
 | |
| 		r.stats[s.Key()] = map[string]Stat{s.FieldName(): s}
 | |
| 		return s
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func key(measurement string, tags map[string]string) uint64 {
 | |
| 	h := fnv.New64a()
 | |
| 	h.Write([]byte(measurement))
 | |
| 
 | |
| 	tmp := make([]string, len(tags))
 | |
| 	i := 0
 | |
| 	for k, v := range tags {
 | |
| 		tmp[i] = k + v
 | |
| 		i++
 | |
| 	}
 | |
| 	sort.Strings(tmp)
 | |
| 
 | |
| 	for _, s := range tmp {
 | |
| 		h.Write([]byte(s))
 | |
| 	}
 | |
| 
 | |
| 	return h.Sum64()
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	registry = &rgstry{
 | |
| 		stats: make(map[uint64]map[string]Stat),
 | |
| 	}
 | |
| }
 |