172 lines
4.5 KiB
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())
|
|
}
|
|
}
|