452 lines
11 KiB
Go
452 lines
11 KiB
Go
// Package subscriber implements the subscriber service
|
|
// to forward incoming data to remote services.
|
|
package subscriber // import "github.com/influxdata/influxdb/services/subscriber"
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/influxdata/influxdb/coordinator"
|
|
"github.com/influxdata/influxdb/models"
|
|
"github.com/influxdata/influxdb/monitor"
|
|
"github.com/influxdata/influxdb/services/meta"
|
|
"github.com/uber-go/zap"
|
|
)
|
|
|
|
// Statistics for the Subscriber service.
|
|
const (
|
|
statCreateFailures = "createFailures"
|
|
statPointsWritten = "pointsWritten"
|
|
statWriteFailures = "writeFailures"
|
|
)
|
|
|
|
// PointsWriter is an interface for writing points to a subscription destination.
|
|
// Only WritePoints() needs to be satisfied. PointsWriter implementations
|
|
// must be goroutine safe.
|
|
type PointsWriter interface {
|
|
WritePoints(p *coordinator.WritePointsRequest) error
|
|
}
|
|
|
|
// subEntry is a unique set that identifies a given subscription.
|
|
type subEntry struct {
|
|
db string
|
|
rp string
|
|
name string
|
|
}
|
|
|
|
// Service manages forking the incoming data from InfluxDB
|
|
// to defined third party destinations.
|
|
// Subscriptions are defined per database and retention policy.
|
|
type Service struct {
|
|
MetaClient interface {
|
|
Databases() []meta.DatabaseInfo
|
|
WaitForDataChanged() chan struct{}
|
|
}
|
|
NewPointsWriter func(u url.URL) (PointsWriter, error)
|
|
Logger zap.Logger
|
|
update chan struct{}
|
|
stats *Statistics
|
|
points chan *coordinator.WritePointsRequest
|
|
wg sync.WaitGroup
|
|
closed bool
|
|
closing chan struct{}
|
|
mu sync.Mutex
|
|
conf Config
|
|
|
|
subs map[subEntry]chanWriter
|
|
subMu sync.RWMutex
|
|
}
|
|
|
|
// NewService returns a subscriber service with given settings
|
|
func NewService(c Config) *Service {
|
|
s := &Service{
|
|
Logger: zap.New(zap.NullEncoder()),
|
|
closed: true,
|
|
stats: &Statistics{},
|
|
conf: c,
|
|
}
|
|
s.NewPointsWriter = s.newPointsWriter
|
|
return s
|
|
}
|
|
|
|
// Open starts the subscription service.
|
|
func (s *Service) Open() error {
|
|
if !s.conf.Enabled {
|
|
return nil // Service disabled.
|
|
}
|
|
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
if s.MetaClient == nil {
|
|
return errors.New("no meta store")
|
|
}
|
|
|
|
s.closed = false
|
|
|
|
s.closing = make(chan struct{})
|
|
s.update = make(chan struct{})
|
|
s.points = make(chan *coordinator.WritePointsRequest, 100)
|
|
|
|
s.wg.Add(2)
|
|
go func() {
|
|
defer s.wg.Done()
|
|
s.run()
|
|
}()
|
|
go func() {
|
|
defer s.wg.Done()
|
|
s.waitForMetaUpdates()
|
|
}()
|
|
|
|
s.Logger.Info("opened service")
|
|
return nil
|
|
}
|
|
|
|
// Close terminates the subscription service.
|
|
// It will panic if called multiple times or without first opening the service.
|
|
func (s *Service) Close() error {
|
|
s.mu.Lock()
|
|
defer s.mu.Unlock()
|
|
|
|
if s.closed {
|
|
return nil // Already closed.
|
|
}
|
|
|
|
s.closed = true
|
|
|
|
close(s.points)
|
|
close(s.closing)
|
|
|
|
s.wg.Wait()
|
|
s.Logger.Info("closed service")
|
|
return nil
|
|
}
|
|
|
|
// WithLogger sets the logger on the service.
|
|
func (s *Service) WithLogger(log zap.Logger) {
|
|
s.Logger = log.With(zap.String("service", "subscriber"))
|
|
}
|
|
|
|
// Statistics maintains the statistics for the subscriber service.
|
|
type Statistics struct {
|
|
CreateFailures int64
|
|
PointsWritten int64
|
|
WriteFailures int64
|
|
}
|
|
|
|
// Statistics returns statistics for periodic monitoring.
|
|
func (s *Service) Statistics(tags map[string]string) []models.Statistic {
|
|
statistics := []models.Statistic{{
|
|
Name: "subscriber",
|
|
Tags: tags,
|
|
Values: map[string]interface{}{
|
|
statCreateFailures: atomic.LoadInt64(&s.stats.CreateFailures),
|
|
statPointsWritten: atomic.LoadInt64(&s.stats.PointsWritten),
|
|
statWriteFailures: atomic.LoadInt64(&s.stats.WriteFailures),
|
|
},
|
|
}}
|
|
|
|
s.subMu.RLock()
|
|
defer s.subMu.RUnlock()
|
|
|
|
for _, sub := range s.subs {
|
|
statistics = append(statistics, sub.Statistics(tags)...)
|
|
}
|
|
return statistics
|
|
}
|
|
|
|
func (s *Service) waitForMetaUpdates() {
|
|
for {
|
|
ch := s.MetaClient.WaitForDataChanged()
|
|
select {
|
|
case <-ch:
|
|
err := s.Update()
|
|
if err != nil {
|
|
s.Logger.Info(fmt.Sprint("error updating subscriptions: ", err))
|
|
}
|
|
case <-s.closing:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// Update will start new and stop deleted subscriptions.
|
|
func (s *Service) Update() error {
|
|
// signal update
|
|
select {
|
|
case s.update <- struct{}{}:
|
|
return nil
|
|
case <-s.closing:
|
|
return errors.New("service closed cannot update")
|
|
}
|
|
}
|
|
|
|
func (s *Service) createSubscription(se subEntry, mode string, destinations []string) (PointsWriter, error) {
|
|
var bm BalanceMode
|
|
switch mode {
|
|
case "ALL":
|
|
bm = ALL
|
|
case "ANY":
|
|
bm = ANY
|
|
default:
|
|
return nil, fmt.Errorf("unknown balance mode %q", mode)
|
|
}
|
|
writers := make([]PointsWriter, 0, len(destinations))
|
|
stats := make([]writerStats, 0, len(destinations))
|
|
// add only valid destinations
|
|
for _, dest := range destinations {
|
|
u, err := url.Parse(dest)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to parse destination: %s", dest)
|
|
}
|
|
w, err := s.NewPointsWriter(*u)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to create writer for destination: %s", dest)
|
|
}
|
|
writers = append(writers, w)
|
|
stats = append(stats, writerStats{dest: dest})
|
|
}
|
|
|
|
return &balancewriter{
|
|
bm: bm,
|
|
writers: writers,
|
|
stats: stats,
|
|
defaultTags: models.StatisticTags{
|
|
"database": se.db,
|
|
"retention_policy": se.rp,
|
|
"name": se.name,
|
|
"mode": mode,
|
|
},
|
|
}, nil
|
|
}
|
|
|
|
// Points returns a channel into which write point requests can be sent.
|
|
func (s *Service) Points() chan<- *coordinator.WritePointsRequest {
|
|
return s.points
|
|
}
|
|
|
|
// run read points from the points channel and writes them to the subscriptions.
|
|
func (s *Service) run() {
|
|
var wg sync.WaitGroup
|
|
s.subs = make(map[subEntry]chanWriter)
|
|
// Perform initial update
|
|
s.updateSubs(&wg)
|
|
for {
|
|
select {
|
|
case <-s.update:
|
|
s.updateSubs(&wg)
|
|
case p, ok := <-s.points:
|
|
if !ok {
|
|
// Close out all chanWriters
|
|
s.close(&wg)
|
|
return
|
|
}
|
|
for se, cw := range s.subs {
|
|
if p.Database == se.db && p.RetentionPolicy == se.rp {
|
|
select {
|
|
case cw.writeRequests <- p:
|
|
default:
|
|
atomic.AddInt64(&s.stats.WriteFailures, 1)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// close closes the existing channel writers.
|
|
func (s *Service) close(wg *sync.WaitGroup) {
|
|
s.subMu.Lock()
|
|
defer s.subMu.Unlock()
|
|
|
|
for _, cw := range s.subs {
|
|
cw.Close()
|
|
}
|
|
// Wait for them to finish
|
|
wg.Wait()
|
|
s.subs = nil
|
|
}
|
|
|
|
func (s *Service) updateSubs(wg *sync.WaitGroup) {
|
|
s.subMu.Lock()
|
|
defer s.subMu.Unlock()
|
|
|
|
if s.subs == nil {
|
|
s.subs = make(map[subEntry]chanWriter)
|
|
}
|
|
|
|
dbis := s.MetaClient.Databases()
|
|
allEntries := make(map[subEntry]bool, 0)
|
|
// Add in new subscriptions
|
|
for _, dbi := range dbis {
|
|
for _, rpi := range dbi.RetentionPolicies {
|
|
for _, si := range rpi.Subscriptions {
|
|
se := subEntry{
|
|
db: dbi.Name,
|
|
rp: rpi.Name,
|
|
name: si.Name,
|
|
}
|
|
allEntries[se] = true
|
|
if _, ok := s.subs[se]; ok {
|
|
continue
|
|
}
|
|
sub, err := s.createSubscription(se, si.Mode, si.Destinations)
|
|
if err != nil {
|
|
atomic.AddInt64(&s.stats.CreateFailures, 1)
|
|
s.Logger.Info(fmt.Sprintf("Subscription creation failed for '%s' with error: %s", si.Name, err))
|
|
continue
|
|
}
|
|
cw := chanWriter{
|
|
writeRequests: make(chan *coordinator.WritePointsRequest, s.conf.WriteBufferSize),
|
|
pw: sub,
|
|
pointsWritten: &s.stats.PointsWritten,
|
|
failures: &s.stats.WriteFailures,
|
|
logger: s.Logger,
|
|
}
|
|
for i := 0; i < s.conf.WriteConcurrency; i++ {
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
cw.Run()
|
|
}()
|
|
}
|
|
s.subs[se] = cw
|
|
s.Logger.Info(fmt.Sprintf("added new subscription for %s %s", se.db, se.rp))
|
|
}
|
|
}
|
|
}
|
|
|
|
// Remove deleted subs
|
|
for se := range s.subs {
|
|
if !allEntries[se] {
|
|
// Close the chanWriter
|
|
s.subs[se].Close()
|
|
|
|
// Remove it from the set
|
|
delete(s.subs, se)
|
|
s.Logger.Info(fmt.Sprintf("deleted old subscription for %s %s", se.db, se.rp))
|
|
}
|
|
}
|
|
}
|
|
|
|
// newPointsWriter returns a new PointsWriter from the given URL.
|
|
func (s *Service) newPointsWriter(u url.URL) (PointsWriter, error) {
|
|
switch u.Scheme {
|
|
case "udp":
|
|
return NewUDP(u.Host), nil
|
|
case "http":
|
|
return NewHTTP(u.String(), time.Duration(s.conf.HTTPTimeout))
|
|
case "https":
|
|
if s.conf.InsecureSkipVerify {
|
|
s.Logger.Info("WARNING: 'insecure-skip-verify' is true. This will skip all certificate verifications.")
|
|
}
|
|
return NewHTTPS(u.String(), time.Duration(s.conf.HTTPTimeout), s.conf.InsecureSkipVerify, s.conf.CaCerts)
|
|
default:
|
|
return nil, fmt.Errorf("unknown destination scheme %s", u.Scheme)
|
|
}
|
|
}
|
|
|
|
// chanWriter sends WritePointsRequest to a PointsWriter received over a channel.
|
|
type chanWriter struct {
|
|
writeRequests chan *coordinator.WritePointsRequest
|
|
pw PointsWriter
|
|
pointsWritten *int64
|
|
failures *int64
|
|
logger zap.Logger
|
|
}
|
|
|
|
// Close closes the chanWriter.
|
|
func (c chanWriter) Close() {
|
|
close(c.writeRequests)
|
|
}
|
|
|
|
func (c chanWriter) Run() {
|
|
for wr := range c.writeRequests {
|
|
err := c.pw.WritePoints(wr)
|
|
if err != nil {
|
|
c.logger.Info(err.Error())
|
|
atomic.AddInt64(c.failures, 1)
|
|
} else {
|
|
atomic.AddInt64(c.pointsWritten, int64(len(wr.Points)))
|
|
}
|
|
}
|
|
}
|
|
|
|
// Statistics returns statistics for periodic monitoring.
|
|
func (c chanWriter) Statistics(tags map[string]string) []models.Statistic {
|
|
if m, ok := c.pw.(monitor.Reporter); ok {
|
|
return m.Statistics(tags)
|
|
}
|
|
return []models.Statistic{}
|
|
}
|
|
|
|
// BalanceMode specifies what balance mode to use on a subscription.
|
|
type BalanceMode int
|
|
|
|
const (
|
|
// ALL indicates to send writes to all subscriber destinations.
|
|
ALL BalanceMode = iota
|
|
|
|
// ANY indicates to send writes to a single subscriber destination, round robin.
|
|
ANY
|
|
)
|
|
|
|
type writerStats struct {
|
|
dest string
|
|
failures int64
|
|
pointsWritten int64
|
|
}
|
|
|
|
// balances writes across PointsWriters according to BalanceMode
|
|
type balancewriter struct {
|
|
bm BalanceMode
|
|
writers []PointsWriter
|
|
stats []writerStats
|
|
defaultTags models.StatisticTags
|
|
i int
|
|
}
|
|
|
|
func (b *balancewriter) WritePoints(p *coordinator.WritePointsRequest) error {
|
|
var lastErr error
|
|
for range b.writers {
|
|
// round robin through destinations.
|
|
i := b.i
|
|
w := b.writers[i]
|
|
b.i = (b.i + 1) % len(b.writers)
|
|
|
|
// write points to destination.
|
|
err := w.WritePoints(p)
|
|
if err != nil {
|
|
lastErr = err
|
|
atomic.AddInt64(&b.stats[i].failures, 1)
|
|
} else {
|
|
atomic.AddInt64(&b.stats[i].pointsWritten, int64(len(p.Points)))
|
|
if b.bm == ANY {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return lastErr
|
|
}
|
|
|
|
// Statistics returns statistics for periodic monitoring.
|
|
func (b *balancewriter) Statistics(tags map[string]string) []models.Statistic {
|
|
statistics := make([]models.Statistic, len(b.stats))
|
|
for i := range b.stats {
|
|
subTags := b.defaultTags.Merge(tags)
|
|
subTags["destination"] = b.stats[i].dest
|
|
statistics[i] = models.Statistic{
|
|
Name: "subscriber",
|
|
Tags: subTags,
|
|
Values: map[string]interface{}{
|
|
statPointsWritten: atomic.LoadInt64(&b.stats[i].pointsWritten),
|
|
statWriteFailures: atomic.LoadInt64(&b.stats[i].failures),
|
|
},
|
|
}
|
|
}
|
|
return statistics
|
|
}
|