opentelemetry-go-contrib/exporters/metric/datadog/example_test.go

172 lines
4.5 KiB
Go

// Copyright The OpenTelemetry Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package datadog_test
import (
"context"
"fmt"
"net"
"os"
"strings"
"time"
"github.com/DataDog/datadog-go/statsd"
"go.opentelemetry.io/contrib/exporters/metric/datadog"
"go.opentelemetry.io/otel/label"
"go.opentelemetry.io/otel/metric"
"go.opentelemetry.io/otel/metric/global"
controller "go.opentelemetry.io/otel/sdk/metric/controller/basic"
"go.opentelemetry.io/otel/sdk/metric/processor/basic"
"go.opentelemetry.io/otel/sdk/metric/selector/simple"
)
type TestUDPServer struct {
*net.UDPConn
}
func ExampleExporter() {
const testHostPort = ":8159"
selector := simple.NewWithInexpensiveDistribution()
exp, err := datadog.NewExporter(datadog.Options{
StatsAddr: testHostPort,
Tags: []string{"env:dev"},
StatsDOptions: []statsd.Option{statsd.WithoutTelemetry()},
})
if err != nil {
panic(err)
}
s, err := getTestServer(testHostPort)
if err != nil {
panic(err)
}
defer s.Close()
go func() {
defer exp.Close()
processor := basic.New(selector, exp)
pusher := controller.New(processor, controller.WithPusher(exp), controller.WithCollectPeriod(time.Second*10))
ctx := context.Background()
err := pusher.Start(ctx)
if err != nil {
panic(err)
}
defer func() { handleErr(pusher.Stop(ctx)) }()
global.SetMeterProvider(pusher.MeterProvider())
meter := global.Meter("marwandist")
m := metric.Must(meter).NewInt64ValueRecorder("myrecorder")
meter.RecordBatch(context.Background(), []label.KeyValue{label.Int("l", 1)},
m.Measurement(1), m.Measurement(50), m.Measurement(100))
}()
statsChan := make(chan []byte, 1)
timedOutChan, stopChan := make(chan struct{}), make(chan struct{})
defer close(stopChan)
go s.ReadPackets(statsChan, 500*time.Millisecond, timedOutChan, stopChan)
for {
select {
case d := <-statsChan:
// only look for "max" value, since we don't want to rely on
// specifics of OpenTelemetry aggregator calculations
// "max" is something that will always exist and always be the same
statLine := string(d)
if strings.HasPrefix(statLine, "myrecorder.max") {
fmt.Println(statLine)
return
}
case <-timedOutChan:
_, _ = fmt.Fprintln(os.Stderr, "Server timed out waiting for packets")
return
case <-time.After(1 * time.Second):
fmt.Println("no data received after 1 second")
return
}
}
// Output:
// myrecorder.max:100|g|#env:dev,l:1
//
}
func getTestServer(addr string) (*TestUDPServer, error) {
udpAddr, err := net.ResolveUDPAddr("udp", addr)
if err != nil {
return nil, err
}
conn, err := net.ListenUDP("udp", udpAddr)
if err != nil {
return nil, err
}
return &TestUDPServer{conn}, nil
}
// ReadPackets reads one "transmission" at a time from the UDP connection
//
// In the case of StatsD, there is one transmission per stat.
// If there is nothing on the connection for longer than maxIdleTime, the
// routine will return, assuming that everything has been sent
// doneChan is an output channel that is closed when ReadPackets returns
// stopChan is an input channel that tells ReadPackets to exit
func (s TestUDPServer) ReadPackets(
xferChan chan []byte,
maxIdleTime time.Duration,
doneChan chan<- struct{},
stopChan <-chan struct{}) {
const readTimeout = 50 * time.Millisecond
var timeouts int
buffer := make([]byte, 1500)
defer close(doneChan)
n := 0
for {
select {
case <-stopChan:
return
default:
_ = s.SetReadDeadline(time.Now().Add(readTimeout))
nn, _, err := s.ReadFrom(buffer[n:])
if err == nil {
timeouts = 0
data := make([]byte, nn)
_ = copy(data, buffer[n:n+nn])
xferChan <- data
n += nn
continue
} else {
if nerr, ok := err.(*net.OpError); ok && nerr.Timeout() {
timeouts++
if time.Duration(timeouts)*readTimeout > maxIdleTime {
return
}
continue
}
// give connection some time to connect
time.Sleep(2 * time.Millisecond)
}
}
}
}
func handleErr(err error) {
if err != nil {
fmt.Println("Encountered error: ", err.Error())
}
}