influxdb/monitor/service.go

504 lines
13 KiB
Go
Raw Normal View History

2016-12-30 20:34:53 +00:00
// Package monitor provides a service and associated functionality
// for InfluxDB to self-monitor internal statistics and diagnostics.
2016-02-10 18:30:52 +00:00
package monitor // import "github.com/influxdata/influxdb/monitor"
2015-09-01 03:17:13 +00:00
import (
"errors"
2015-09-01 03:17:13 +00:00
"expvar"
2016-05-03 23:34:17 +00:00
"fmt"
2015-09-01 03:17:13 +00:00
"os"
2015-09-04 20:14:38 +00:00
"runtime"
2015-09-01 03:17:13 +00:00
"sort"
"strconv"
"sync"
"time"
"github.com/influxdata/influxdb/logger"
"github.com/influxdata/influxdb/models"
"github.com/influxdata/influxdb/monitor/diagnostics"
"github.com/influxdata/influxdb/services/meta"
"go.uber.org/zap"
2015-09-01 03:17:13 +00:00
)
2015-11-22 18:42:34 +00:00
// Policy constants.
const (
2016-12-30 20:34:53 +00:00
// Name of the retention policy used by the monitor service.
MonitorRetentionPolicy = "monitor"
// Duration of the monitor retention policy.
MonitorRetentionPolicyDuration = 7 * 24 * time.Hour
2016-12-30 20:34:53 +00:00
// Default replication factor to set on the monitor retention policy.
MonitorRetentionPolicyReplicaN = 1
)
// Monitor represents an instance of the monitor system.
type Monitor struct {
2015-09-09 19:43:51 +00:00
// Build information for diagnostics.
2015-09-25 06:32:47 +00:00
Version string
Commit string
Branch string
BuildTime string
2015-09-09 19:43:51 +00:00
wg sync.WaitGroup
mu sync.RWMutex
2016-05-03 23:34:17 +00:00
globalTags map[string]string
diagRegistrations map[string]diagnostics.Client
reporter Reporter
done chan struct{}
storeCreated bool
storeEnabled bool
2015-09-01 03:17:13 +00:00
2017-01-13 17:43:38 +00:00
storeDatabase string
storeRetentionPolicy string
storeInterval time.Duration
2015-09-01 03:17:13 +00:00
2015-12-23 15:48:25 +00:00
MetaClient interface {
CreateDatabaseWithRetentionPolicy(name string, spec *meta.RetentionPolicySpec) (*meta.DatabaseInfo, error)
Database(name string) *meta.DatabaseInfo
}
// Writer for pushing stats back into the database.
PointsWriter PointsWriter
Logger *zap.Logger
2015-09-01 03:17:13 +00:00
}
2016-12-30 20:34:53 +00:00
// PointsWriter is a simplified interface for writing the points the monitor gathers.
type PointsWriter interface {
WritePoints(database, retentionPolicy string, points models.Points) error
}
// New returns a new instance of the monitor system.
func New(r Reporter, c Config) *Monitor {
return &Monitor{
2016-05-03 23:34:17 +00:00
globalTags: make(map[string]string),
2016-02-17 14:06:04 +00:00
diagRegistrations: make(map[string]diagnostics.Client),
reporter: r,
2016-02-17 14:06:04 +00:00
storeEnabled: c.StoreEnabled,
storeDatabase: c.StoreDatabase,
storeInterval: time.Duration(c.StoreInterval),
storeRetentionPolicy: MonitorRetentionPolicy,
Logger: zap.NewNop(),
2015-09-01 03:17:13 +00:00
}
}
2016-12-30 20:34:53 +00:00
// open returns whether the monitor service is open.
2016-05-03 23:34:17 +00:00
func (m *Monitor) open() bool {
m.mu.Lock()
defer m.mu.Unlock()
return m.done != nil
}
// Open opens the monitoring system, using the given clusterID, node ID, and hostname
// for identification purpose.
func (m *Monitor) Open() error {
2016-05-03 23:34:17 +00:00
if m.open() {
m.Logger.Info("Monitor is already open")
2016-05-03 23:34:17 +00:00
return nil
}
m.Logger.Info("Starting monitor service")
2015-09-01 03:17:13 +00:00
// Self-register various stats and diagnostics.
2015-09-09 19:43:51 +00:00
m.RegisterDiagnosticsClient("build", &build{
Version: m.Version,
Commit: m.Commit,
Branch: m.Branch,
2015-09-25 06:32:47 +00:00
Time: m.BuildTime,
2015-09-09 19:43:51 +00:00
})
m.RegisterDiagnosticsClient("runtime", &goRuntime{})
m.RegisterDiagnosticsClient("network", &network{})
m.RegisterDiagnosticsClient("system", &system{})
2015-09-01 03:17:13 +00:00
m.mu.Lock()
m.done = make(chan struct{})
m.mu.Unlock()
2015-09-01 03:17:13 +00:00
// If enabled, record stats in a InfluxDB system.
if m.storeEnabled {
hostname, _ := os.Hostname()
m.SetGlobalTag("hostname", hostname)
2015-09-01 03:17:13 +00:00
// Start periodic writes to system.
m.wg.Add(1)
go m.storeStatistics()
2015-09-01 03:17:13 +00:00
}
return nil
}
2018-10-08 19:53:29 +00:00
// Enabled returns true if any underlying Config is Enabled.
func (m *Monitor) Enabled() bool { return m.storeEnabled }
2018-10-08 19:53:29 +00:00
// WritePoints writes the points the monitor gathers.
func (m *Monitor) WritePoints(p models.Points) error {
if !m.storeEnabled {
return nil
}
if len(m.globalTags) > 0 {
for _, pp := range p {
pp.SetTags(pp.Tags().Merge(m.globalTags))
}
}
return m.writePoints(p)
}
func (m *Monitor) writePoints(p models.Points) error {
m.mu.RLock()
defer m.mu.RUnlock()
if err := m.PointsWriter.WritePoints(m.storeDatabase, m.storeRetentionPolicy, p); err != nil {
m.Logger.Info("failed to store statistics", zap.Error(err))
}
return nil
}
// Close closes the monitor system.
func (m *Monitor) Close() error {
2016-05-03 23:34:17 +00:00
if !m.open() {
m.Logger.Info("Monitor is already closed")
2016-05-03 23:34:17 +00:00
return nil
}
m.Logger.Info("Shutting down monitor service")
m.mu.Lock()
close(m.done)
m.mu.Unlock()
2016-04-29 18:08:00 +00:00
m.wg.Wait()
m.mu.Lock()
m.done = nil
m.mu.Unlock()
2016-04-29 18:08:00 +00:00
m.DeregisterDiagnosticsClient("build")
m.DeregisterDiagnosticsClient("runtime")
m.DeregisterDiagnosticsClient("network")
m.DeregisterDiagnosticsClient("system")
return nil
2015-09-01 03:17:13 +00:00
}
2016-05-03 23:34:17 +00:00
// SetGlobalTag can be used to set tags that will appear on all points
// written by the Monitor.
func (m *Monitor) SetGlobalTag(key string, value interface{}) {
m.mu.Lock()
m.globalTags[key] = fmt.Sprintf("%v", value)
m.mu.Unlock()
}
2016-12-30 20:34:53 +00:00
// RemoteWriterConfig represents the configuration of a remote writer.
2016-05-03 23:34:17 +00:00
type RemoteWriterConfig struct {
RemoteAddr string
NodeID string
Username string
Password string
ClusterID uint64
}
// SetPointsWriter can be used to set a writer for the monitoring points.
func (m *Monitor) SetPointsWriter(pw PointsWriter) error {
if !m.storeEnabled {
// not enabled, nothing to do
return nil
2016-05-03 23:34:17 +00:00
}
2016-06-01 16:46:01 +00:00
m.mu.Lock()
m.PointsWriter = pw
2016-05-03 23:34:17 +00:00
m.mu.Unlock()
2016-05-03 23:34:17 +00:00
// Subsequent calls to an already open Monitor are just a no-op.
return m.Open()
}
2016-12-30 20:34:53 +00:00
// WithLogger sets the logger for the Monitor.
func (m *Monitor) WithLogger(log *zap.Logger) {
m.Logger = log.With(zap.String("service", "monitor"))
2015-09-01 03:17:13 +00:00
}
// RegisterDiagnosticsClient registers a diagnostics client with the given name and tags.
func (m *Monitor) RegisterDiagnosticsClient(name string, client diagnostics.Client) {
m.mu.Lock()
defer m.mu.Unlock()
m.diagRegistrations[name] = client
m.Logger.Info("Registered diagnostics client", zap.String("name", name))
}
// DeregisterDiagnosticsClient deregisters a diagnostics client by name.
func (m *Monitor) DeregisterDiagnosticsClient(name string) {
m.mu.Lock()
defer m.mu.Unlock()
delete(m.diagRegistrations, name)
}
// Statistics returns the combined statistics for all expvar data. The given
// tags are added to each of the returned statistics.
2015-10-19 21:06:14 +00:00
func (m *Monitor) Statistics(tags map[string]string) ([]*Statistic, error) {
2015-11-22 18:42:34 +00:00
var statistics []*Statistic
2015-09-01 03:17:13 +00:00
expvar.Do(func(kv expvar.KeyValue) {
// Skip built-in expvar stats.
if kv.Key == "memstats" || kv.Key == "cmdline" {
return
2015-09-01 03:17:13 +00:00
}
2015-10-19 21:06:14 +00:00
statistic := &Statistic{
Statistic: models.NewStatistic(""),
}
// Add any supplied tags.
for k, v := range tags {
statistic.Tags[k] = v
}
// Every other top-level expvar value should be a map.
m, ok := kv.Value.(*expvar.Map)
if !ok {
return
}
m.Do(func(subKV expvar.KeyValue) {
switch subKV.Key {
case "name":
// straight to string name.
u, err := strconv.Unquote(subKV.Value.String())
if err != nil {
return
}
statistic.Name = u
case "tags":
// string-string tags map.
n := subKV.Value.(*expvar.Map)
n.Do(func(t expvar.KeyValue) {
u, err := strconv.Unquote(t.Value.String())
if err != nil {
return
}
statistic.Tags[t.Key] = u
})
case "values":
// string-interface map.
n := subKV.Value.(*expvar.Map)
n.Do(func(kv expvar.KeyValue) {
var f interface{}
var err error
switch v := kv.Value.(type) {
case *expvar.Float:
f, err = strconv.ParseFloat(v.String(), 64)
if err != nil {
return
}
case *expvar.Int:
f, err = strconv.ParseInt(v.String(), 10, 64)
if err != nil {
return
}
default:
return
}
statistic.Values[kv.Key] = f
})
}
})
// If a registered client has no field data, don't include it in the results
if len(statistic.Values) == 0 {
return
}
2015-09-04 20:14:38 +00:00
statistics = append(statistics, statistic)
})
2015-09-04 20:14:38 +00:00
// Add Go memstats.
2015-10-19 21:06:14 +00:00
statistic := &Statistic{
Statistic: models.NewStatistic("runtime"),
2015-09-04 20:14:38 +00:00
}
// Add any supplied tags to Go memstats
for k, v := range tags {
statistic.Tags[k] = v
}
2015-09-04 20:14:38 +00:00
var rt runtime.MemStats
runtime.ReadMemStats(&rt)
statistic.Values = map[string]interface{}{
"Alloc": int64(rt.Alloc),
"TotalAlloc": int64(rt.TotalAlloc),
"Sys": int64(rt.Sys),
"Lookups": int64(rt.Lookups),
"Mallocs": int64(rt.Mallocs),
"Frees": int64(rt.Frees),
"HeapAlloc": int64(rt.HeapAlloc),
"HeapSys": int64(rt.HeapSys),
"HeapIdle": int64(rt.HeapIdle),
"HeapInUse": int64(rt.HeapInuse),
"HeapReleased": int64(rt.HeapReleased),
"HeapObjects": int64(rt.HeapObjects),
"PauseTotalNs": int64(rt.PauseTotalNs),
"NumGC": int64(rt.NumGC),
"NumGoroutine": int64(runtime.NumGoroutine()),
}
statistics = append(statistics, statistic)
statistics = m.gatherStatistics(statistics, tags)
2015-09-01 03:17:13 +00:00
return statistics, nil
}
func (m *Monitor) gatherStatistics(statistics []*Statistic, tags map[string]string) []*Statistic {
m.mu.RLock()
defer m.mu.RUnlock()
if m.reporter != nil {
for _, s := range m.reporter.Statistics(tags) {
statistics = append(statistics, &Statistic{Statistic: s})
}
}
return statistics
}
2015-11-22 18:42:34 +00:00
// Diagnostics fetches diagnostic information for each registered
// diagnostic client. It skips any clients that return an error when
// retrieving their diagnostics.
func (m *Monitor) Diagnostics() (map[string]*diagnostics.Diagnostics, error) {
m.mu.Lock()
defer m.mu.Unlock()
diags := make(map[string]*diagnostics.Diagnostics, len(m.diagRegistrations))
for k, v := range m.diagRegistrations {
d, err := v.Diagnostics()
if err != nil {
continue
}
diags[k] = d
}
return diags, nil
}
// createInternalStorage ensures the internal storage has been created.
func (m *Monitor) createInternalStorage() {
if m.storeCreated {
return
}
if di := m.MetaClient.Database(m.storeDatabase); di == nil {
duration := MonitorRetentionPolicyDuration
replicaN := MonitorRetentionPolicyReplicaN
spec := meta.RetentionPolicySpec{
Name: MonitorRetentionPolicy,
Duration: &duration,
ReplicaN: &replicaN,
}
if _, err := m.MetaClient.CreateDatabaseWithRetentionPolicy(m.storeDatabase, &spec); err != nil {
m.Logger.Info("Failed to create storage", logger.Database(m.storeDatabase), zap.Error(err))
return
}
}
// Mark storage creation complete.
m.storeCreated = true
}
// waitUntilInterval waits until we are on an even interval for the duration.
func (m *Monitor) waitUntilInterval(d time.Duration) error {
now := time.Now()
until := now.Truncate(d).Add(d)
timer := time.NewTimer(until.Sub(now))
defer timer.Stop()
select {
case <-timer.C:
return nil
case <-m.done:
return errors.New("interrupted")
}
}
// storeStatistics writes the statistics to an InfluxDB system.
func (m *Monitor) storeStatistics() {
defer m.wg.Done()
m.Logger.Info("Storing statistics", logger.Database(m.storeDatabase), logger.RetentionPolicy(m.storeRetentionPolicy), logger.DurationLiteral("interval", m.storeInterval))
// Wait until an even interval to start recording monitor statistics.
// If we are interrupted before the interval for some reason, exit early.
if err := m.waitUntilInterval(m.storeInterval); err != nil {
return
}
tick := time.NewTicker(m.storeInterval)
2015-09-01 03:17:13 +00:00
defer tick.Stop()
2015-09-01 03:17:13 +00:00
for {
select {
case now := <-tick.C:
now = now.Truncate(m.storeInterval)
2016-05-03 23:34:17 +00:00
func() {
m.mu.Lock()
defer m.mu.Unlock()
m.createInternalStorage()
}()
stats, err := m.Statistics(m.globalTags)
if err != nil {
m.Logger.Info("Failed to retrieve registered statistics", zap.Error(err))
return
}
// Write all stats in batches
batch := make(models.Points, 0, 5000)
for _, s := range stats {
pt, err := models.NewPoint(s.Name, models.NewTags(s.Tags), s.Values, now)
if err != nil {
m.Logger.Info("Dropping point", zap.String("name", s.Name), zap.Error(err))
2016-05-03 23:34:17 +00:00
return
}
batch = append(batch, pt)
if len(batch) == cap(batch) {
m.writePoints(batch)
batch = batch[:0]
}
}
// Write the last batch
if len(batch) > 0 {
m.writePoints(batch)
}
case <-m.done:
m.Logger.Info("Terminating storage of statistics")
2015-09-01 03:17:13 +00:00
return
}
}
}
2015-10-19 21:06:14 +00:00
// Statistic represents the information returned by a single monitor client.
type Statistic struct {
models.Statistic
2015-09-01 03:17:13 +00:00
}
// ValueNames returns a sorted list of the value names, if any.
func (s *Statistic) ValueNames() []string {
2015-09-01 03:17:13 +00:00
a := make([]string, 0, len(s.Values))
2015-11-22 18:42:34 +00:00
for k := range s.Values {
2015-09-01 03:17:13 +00:00
a = append(a, k)
}
sort.Strings(a)
return a
}
2016-12-30 20:34:53 +00:00
// Statistics is a slice of sortable statistics.
type Statistics []*Statistic
2016-12-30 20:34:53 +00:00
// Len implements sort.Interface.
func (a Statistics) Len() int { return len(a) }
2016-12-30 20:34:53 +00:00
// Less implements sort.Interface.
func (a Statistics) Less(i, j int) bool {
return a[i].Name < a[j].Name
}
2016-12-30 20:34:53 +00:00
// Swap implements sort.Interface.
func (a Statistics) Swap(i, j int) { a[i], a[j] = a[j], a[i] }