feat(inputs.nats_consumer): Add receiver subject as tag (#13293)
This commit is contained in:
parent
1b74a25252
commit
ed72510fe3
|
|
@ -103,4 +103,4 @@ Which data you will get depends on the subjects you consume from nats
|
||||||
## Example Output
|
## Example Output
|
||||||
|
|
||||||
Depends on the nats subject input
|
Depends on the nats subject input
|
||||||
nats_consumer,host=[] value=1.9 1655972309339341000
|
nats_consumer,host=foo,subject=recvsubj value=1.9 1655972309339341000
|
||||||
|
|
|
||||||
|
|
@ -93,8 +93,6 @@ func (n *natsConsumer) natsErrHandler(c *nats.Conn, s *nats.Subscription, e erro
|
||||||
func (n *natsConsumer) Start(acc telegraf.Accumulator) error {
|
func (n *natsConsumer) Start(acc telegraf.Accumulator) error {
|
||||||
n.acc = acc.WithTracking(n.MaxUndeliveredMessages)
|
n.acc = acc.WithTracking(n.MaxUndeliveredMessages)
|
||||||
|
|
||||||
var connectErr error
|
|
||||||
|
|
||||||
options := []nats.Option{
|
options := []nats.Option{
|
||||||
nats.MaxReconnects(-1),
|
nats.MaxReconnects(-1),
|
||||||
nats.ErrorHandler(n.natsErrHandler),
|
nats.ErrorHandler(n.natsErrHandler),
|
||||||
|
|
@ -119,6 +117,7 @@ func (n *natsConsumer) Start(acc telegraf.Accumulator) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
if n.conn == nil || n.conn.IsClosed() {
|
if n.conn == nil || n.conn.IsClosed() {
|
||||||
|
var connectErr error
|
||||||
n.conn, connectErr = nats.Connect(strings.Join(n.Servers, ","), options...)
|
n.conn, connectErr = nats.Connect(strings.Join(n.Servers, ","), options...)
|
||||||
if connectErr != nil {
|
if connectErr != nil {
|
||||||
return connectErr
|
return connectErr
|
||||||
|
|
@ -219,7 +218,9 @@ func (n *natsConsumer) receiver(ctx context.Context) {
|
||||||
<-sem
|
<-sem
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
for _, m := range metrics {
|
||||||
|
m.AddTag("subject", msg.Subject)
|
||||||
|
}
|
||||||
n.acc.AddTrackingMetricGroup(metrics)
|
n.acc.AddTrackingMetricGroup(metrics)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -260,7 +261,6 @@ func init() {
|
||||||
inputs.Add("nats_consumer", func() telegraf.Input {
|
inputs.Add("nats_consumer", func() telegraf.Input {
|
||||||
return &natsConsumer{
|
return &natsConsumer{
|
||||||
Servers: []string{"nats://localhost:4222"},
|
Servers: []string{"nats://localhost:4222"},
|
||||||
Secure: false,
|
|
||||||
Subjects: []string{"telegraf"},
|
Subjects: []string{"telegraf"},
|
||||||
QueueGroup: "telegraf_consumers",
|
QueueGroup: "telegraf_consumers",
|
||||||
PendingBytesLimit: nats.DefaultSubPendingBytesLimit,
|
PendingBytesLimit: nats.DefaultSubPendingBytesLimit,
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,216 @@
|
||||||
|
package nats_consumer
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/nats-io/nats.go"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"github.com/testcontainers/testcontainers-go/wait"
|
||||||
|
|
||||||
|
"github.com/influxdata/telegraf"
|
||||||
|
"github.com/influxdata/telegraf/metric"
|
||||||
|
"github.com/influxdata/telegraf/plugins/parsers/influx"
|
||||||
|
"github.com/influxdata/telegraf/testutil"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestStartStop(t *testing.T) {
|
||||||
|
if testing.Short() {
|
||||||
|
t.Skip("Skipping integration test in short mode")
|
||||||
|
}
|
||||||
|
|
||||||
|
container := testutil.Container{
|
||||||
|
Image: "nats",
|
||||||
|
ExposedPorts: []string{"4222"},
|
||||||
|
WaitingFor: wait.ForLog("Server is ready"),
|
||||||
|
}
|
||||||
|
require.NoError(t, container.Start(), "failed to start container")
|
||||||
|
defer container.Terminate()
|
||||||
|
|
||||||
|
plugin := &natsConsumer{
|
||||||
|
Servers: []string{fmt.Sprintf("nats://%s:%s", container.Address, container.Ports["4222"])},
|
||||||
|
Subjects: []string{"telegraf"},
|
||||||
|
QueueGroup: "telegraf_consumers",
|
||||||
|
PendingBytesLimit: nats.DefaultSubPendingBytesLimit,
|
||||||
|
PendingMessageLimit: nats.DefaultSubPendingMsgsLimit,
|
||||||
|
MaxUndeliveredMessages: defaultMaxUndeliveredMessages,
|
||||||
|
Log: testutil.Logger{},
|
||||||
|
}
|
||||||
|
|
||||||
|
var acc testutil.Accumulator
|
||||||
|
require.NoError(t, plugin.Start(&acc))
|
||||||
|
plugin.Stop()
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSendReceive(t *testing.T) {
|
||||||
|
if testing.Short() {
|
||||||
|
t.Skip("Skipping integration test in short mode")
|
||||||
|
}
|
||||||
|
|
||||||
|
container := testutil.Container{
|
||||||
|
Image: "nats",
|
||||||
|
ExposedPorts: []string{"4222"},
|
||||||
|
WaitingFor: wait.ForLog("Server is ready"),
|
||||||
|
}
|
||||||
|
require.NoError(t, container.Start(), "failed to start container")
|
||||||
|
defer container.Terminate()
|
||||||
|
addr := fmt.Sprintf("nats://%s:%s", container.Address, container.Ports["4222"])
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
msgs map[string][]string
|
||||||
|
expected []telegraf.Metric
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "single message",
|
||||||
|
msgs: map[string][]string{
|
||||||
|
"telegraf": {"test,source=foo value=42i"},
|
||||||
|
},
|
||||||
|
expected: []telegraf.Metric{
|
||||||
|
metric.New(
|
||||||
|
"test",
|
||||||
|
map[string]string{
|
||||||
|
"source": "foo",
|
||||||
|
"subject": "telegraf",
|
||||||
|
},
|
||||||
|
map[string]interface{}{"value": int64(42)},
|
||||||
|
time.Unix(0, 0),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "multiple message",
|
||||||
|
msgs: map[string][]string{
|
||||||
|
"telegraf": {
|
||||||
|
"test,source=foo value=42i",
|
||||||
|
"test,source=bar value=23i",
|
||||||
|
},
|
||||||
|
"hitchhiker": {
|
||||||
|
"wale,part=front named=true",
|
||||||
|
"wale,part=back named=false",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
expected: []telegraf.Metric{
|
||||||
|
metric.New(
|
||||||
|
"test",
|
||||||
|
map[string]string{
|
||||||
|
"source": "foo",
|
||||||
|
"subject": "telegraf",
|
||||||
|
},
|
||||||
|
map[string]interface{}{"value": int64(42)},
|
||||||
|
time.Unix(0, 0),
|
||||||
|
),
|
||||||
|
metric.New(
|
||||||
|
"test",
|
||||||
|
map[string]string{
|
||||||
|
"source": "bar",
|
||||||
|
"subject": "telegraf",
|
||||||
|
},
|
||||||
|
map[string]interface{}{"value": int64(23)},
|
||||||
|
time.Unix(0, 0),
|
||||||
|
),
|
||||||
|
metric.New(
|
||||||
|
"wale",
|
||||||
|
map[string]string{
|
||||||
|
"part": "front",
|
||||||
|
"subject": "hitchhiker",
|
||||||
|
},
|
||||||
|
map[string]interface{}{"named": true},
|
||||||
|
time.Unix(0, 0),
|
||||||
|
),
|
||||||
|
metric.New(
|
||||||
|
"wale",
|
||||||
|
map[string]string{
|
||||||
|
"part": "back",
|
||||||
|
"subject": "hitchhiker",
|
||||||
|
},
|
||||||
|
map[string]interface{}{"named": false},
|
||||||
|
time.Unix(0, 0),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
subjects := make([]string, 0, len(tt.msgs))
|
||||||
|
for k := range tt.msgs {
|
||||||
|
subjects = append(subjects, k)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup the plugin
|
||||||
|
plugin := &natsConsumer{
|
||||||
|
Servers: []string{addr},
|
||||||
|
Subjects: subjects,
|
||||||
|
QueueGroup: "telegraf_consumers",
|
||||||
|
PendingBytesLimit: nats.DefaultSubPendingBytesLimit,
|
||||||
|
PendingMessageLimit: nats.DefaultSubPendingMsgsLimit,
|
||||||
|
MaxUndeliveredMessages: defaultMaxUndeliveredMessages,
|
||||||
|
Log: testutil.Logger{},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a line-protocol parser
|
||||||
|
parser := &influx.Parser{}
|
||||||
|
require.NoError(t, parser.Init())
|
||||||
|
plugin.SetParser(parser)
|
||||||
|
|
||||||
|
// Startup the plugin
|
||||||
|
var acc testutil.Accumulator
|
||||||
|
require.NoError(t, plugin.Start(&acc))
|
||||||
|
defer plugin.Stop()
|
||||||
|
|
||||||
|
// Send all messages to the topics (random order due to Golang map)
|
||||||
|
publisher := &sender{Addr: addr}
|
||||||
|
require.NoError(t, publisher.Connect())
|
||||||
|
defer publisher.Disconnect()
|
||||||
|
for topic, msgs := range tt.msgs {
|
||||||
|
for _, msg := range msgs {
|
||||||
|
require.NoError(t, publisher.Send(topic, msg))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
publisher.Disconnect()
|
||||||
|
|
||||||
|
// Wait for the metrics to be collected
|
||||||
|
require.Eventually(t, func() bool {
|
||||||
|
acc.Lock()
|
||||||
|
defer acc.Unlock()
|
||||||
|
return acc.NMetrics() >= uint64(len(tt.expected))
|
||||||
|
}, time.Second, 100*time.Millisecond)
|
||||||
|
|
||||||
|
actual := acc.GetTelegrafMetrics()
|
||||||
|
testutil.RequireMetricsEqual(t, tt.expected, actual, testutil.IgnoreTime(), testutil.SortMetrics())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type sender struct {
|
||||||
|
Addr string
|
||||||
|
|
||||||
|
Username string
|
||||||
|
Password string
|
||||||
|
|
||||||
|
conn *nats.Conn
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *sender) Connect() error {
|
||||||
|
conn, err := nats.Connect(s.Addr)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
s.conn = conn
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *sender) Disconnect() {
|
||||||
|
if s.conn != nil && !s.conn.IsClosed() {
|
||||||
|
_ = s.conn.Flush()
|
||||||
|
s.conn.Close()
|
||||||
|
}
|
||||||
|
s.conn = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *sender) Send(topic, msg string) error {
|
||||||
|
return s.conn.Publish(topic, []byte(msg))
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue