mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-12-19 15:06:25 +01:00
d5c180e680
It is better developing vmctl tool in VictoriaMetrics repository, so it could be released together with the rest of vmutils tools such as vmalert, vmagent, vmbackup, vmrestore and vmauth.
122 lines
2.7 KiB
Go
122 lines
2.7 KiB
Go
package client
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
// UDPPayloadSize is a reasonable default payload size for UDP packets that
|
|
// could be travelling over the internet.
|
|
UDPPayloadSize = 512
|
|
)
|
|
|
|
// UDPConfig is the config data needed to create a UDP Client.
|
|
type UDPConfig struct {
|
|
// Addr should be of the form "host:port"
|
|
// or "[ipv6-host%zone]:port".
|
|
Addr string
|
|
|
|
// PayloadSize is the maximum size of a UDP client message, optional
|
|
// Tune this based on your network. Defaults to UDPPayloadSize.
|
|
PayloadSize int
|
|
}
|
|
|
|
// NewUDPClient returns a client interface for writing to an InfluxDB UDP
|
|
// service from the given config.
|
|
func NewUDPClient(conf UDPConfig) (Client, error) {
|
|
var udpAddr *net.UDPAddr
|
|
udpAddr, err := net.ResolveUDPAddr("udp", conf.Addr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
conn, err := net.DialUDP("udp", nil, udpAddr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
payloadSize := conf.PayloadSize
|
|
if payloadSize == 0 {
|
|
payloadSize = UDPPayloadSize
|
|
}
|
|
|
|
return &udpclient{
|
|
conn: conn,
|
|
payloadSize: payloadSize,
|
|
}, nil
|
|
}
|
|
|
|
// Close releases the udpclient's resources.
|
|
func (uc *udpclient) Close() error {
|
|
return uc.conn.Close()
|
|
}
|
|
|
|
type udpclient struct {
|
|
conn io.WriteCloser
|
|
payloadSize int
|
|
}
|
|
|
|
func (uc *udpclient) Write(bp BatchPoints) error {
|
|
var b = make([]byte, 0, uc.payloadSize) // initial buffer size, it will grow as needed
|
|
var d, _ = time.ParseDuration("1" + bp.Precision())
|
|
|
|
var delayedError error
|
|
|
|
var checkBuffer = func(n int) {
|
|
if len(b) > 0 && len(b)+n > uc.payloadSize {
|
|
if _, err := uc.conn.Write(b); err != nil {
|
|
delayedError = err
|
|
}
|
|
b = b[:0]
|
|
}
|
|
}
|
|
|
|
for _, p := range bp.Points() {
|
|
p.pt.Round(d)
|
|
pointSize := p.pt.StringSize() + 1 // include newline in size
|
|
//point := p.pt.RoundedString(d) + "\n"
|
|
|
|
checkBuffer(pointSize)
|
|
|
|
if p.Time().IsZero() || pointSize <= uc.payloadSize {
|
|
b = p.pt.AppendString(b)
|
|
b = append(b, '\n')
|
|
continue
|
|
}
|
|
|
|
points := p.pt.Split(uc.payloadSize - 1) // account for newline character
|
|
for _, sp := range points {
|
|
checkBuffer(sp.StringSize() + 1)
|
|
b = sp.AppendString(b)
|
|
b = append(b, '\n')
|
|
}
|
|
}
|
|
|
|
if len(b) > 0 {
|
|
if _, err := uc.conn.Write(b); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return delayedError
|
|
}
|
|
|
|
func (uc *udpclient) Query(q Query) (*Response, error) {
|
|
return nil, fmt.Errorf("querying via UDP is not supported")
|
|
}
|
|
|
|
func (uc *udpclient) QueryCtx(ctx context.Context, q Query) (*Response, error) {
|
|
return nil, fmt.Errorf("querying via UDP is not supported")
|
|
}
|
|
|
|
func (uc *udpclient) QueryAsChunk(q Query) (*ChunkedResponse, error) {
|
|
return nil, fmt.Errorf("querying via UDP is not supported")
|
|
}
|
|
|
|
func (uc *udpclient) Ping(timeout time.Duration) (time.Duration, string, error) {
|
|
return 0, "", nil
|
|
}
|