mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-11-21 14:44:00 +00:00
135 lines
4.1 KiB
Go
135 lines
4.1 KiB
Go
|
package netstorage
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/auth"
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/bytesutil"
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/consts"
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/encoding"
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompb"
|
||
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/storage"
|
||
|
xxhash "github.com/cespare/xxhash/v2"
|
||
|
jump "github.com/lithammer/go-jump-consistent-hash"
|
||
|
)
|
||
|
|
||
|
// InsertCtx is a generic context for inserting data
|
||
|
type InsertCtx struct {
|
||
|
Labels []prompb.Label
|
||
|
MetricNameBuf []byte
|
||
|
|
||
|
bufs [][]byte
|
||
|
labelsBuf []byte
|
||
|
sizeBuf [8]byte
|
||
|
}
|
||
|
|
||
|
// Reset resets ctx.
|
||
|
func (ctx *InsertCtx) Reset() {
|
||
|
for _, label := range ctx.Labels {
|
||
|
label.Name = nil
|
||
|
label.Value = nil
|
||
|
}
|
||
|
ctx.Labels = ctx.Labels[:0]
|
||
|
ctx.MetricNameBuf = ctx.MetricNameBuf[:0]
|
||
|
|
||
|
if ctx.bufs == nil {
|
||
|
ctx.bufs = make([][]byte, len(storageNodes))
|
||
|
}
|
||
|
for i := range ctx.bufs {
|
||
|
ctx.bufs[i] = ctx.bufs[i][:0]
|
||
|
}
|
||
|
ctx.labelsBuf = ctx.labelsBuf[:0]
|
||
|
}
|
||
|
|
||
|
// AddLabel adds (name, value) label to ctx.Labels.
|
||
|
//
|
||
|
// name and value must exist until ctx.Labels is used.
|
||
|
func (ctx *InsertCtx) AddLabel(name, value string) {
|
||
|
labels := ctx.Labels
|
||
|
if cap(labels) > len(labels) {
|
||
|
labels = labels[:len(labels)+1]
|
||
|
} else {
|
||
|
labels = append(labels, prompb.Label{})
|
||
|
}
|
||
|
label := &labels[len(labels)-1]
|
||
|
|
||
|
// Do not copy name and value contents for performance reasons.
|
||
|
// This reduces GC overhead on the number of objects and allocations.
|
||
|
label.Name = bytesutil.ToUnsafeBytes(name)
|
||
|
label.Value = bytesutil.ToUnsafeBytes(value)
|
||
|
|
||
|
ctx.Labels = labels
|
||
|
}
|
||
|
|
||
|
// WriteDataPoint writes (timestamp, value) data point with the given at and labels to ctx buffer.
|
||
|
func (ctx *InsertCtx) WriteDataPoint(at *auth.Token, labels []prompb.Label, timestamp int64, value float64) error {
|
||
|
ctx.MetricNameBuf = storage.MarshalMetricNameRaw(ctx.MetricNameBuf[:0], at.AccountID, at.ProjectID, labels)
|
||
|
storageNodeIdx := ctx.GetStorageNodeIdx(at, labels)
|
||
|
return ctx.WriteDataPointExt(at, storageNodeIdx, ctx.MetricNameBuf, timestamp, value)
|
||
|
}
|
||
|
|
||
|
// WriteDataPointExt writes the given metricNameRaw with (timestmap, value) to ctx buffer with the given storageNodeIdx.
|
||
|
func (ctx *InsertCtx) WriteDataPointExt(at *auth.Token, storageNodeIdx int, metricNameRaw []byte, timestamp int64, value float64) error {
|
||
|
buf := ctx.bufs[storageNodeIdx]
|
||
|
sn := storageNodes[storageNodeIdx]
|
||
|
bufNew := storage.MarshalMetricRow(buf, metricNameRaw, timestamp, value)
|
||
|
if len(bufNew) >= consts.MaxInsertPacketSize {
|
||
|
// Send buf to storageNode, since it is too big.
|
||
|
if err := sn.sendWithFallback(buf, ctx.sizeBuf[:]); err != nil {
|
||
|
return fmt.Errorf("cannot send %d bytes to storageNodes: %s", len(buf), err)
|
||
|
}
|
||
|
buf = storage.MarshalMetricRow(bufNew[:0], metricNameRaw, timestamp, value)
|
||
|
} else {
|
||
|
buf = bufNew
|
||
|
}
|
||
|
ctx.bufs[storageNodeIdx] = buf
|
||
|
sn.RowsPushed.Inc()
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// FlushBufs flushes ctx bufs to remote storage nodes.
|
||
|
func (ctx *InsertCtx) FlushBufs() error {
|
||
|
// Send per-storageNode bufs.
|
||
|
sizeBuf := ctx.sizeBuf[:]
|
||
|
for i, buf := range ctx.bufs {
|
||
|
if len(buf) == 0 {
|
||
|
continue
|
||
|
}
|
||
|
sn := storageNodes[i]
|
||
|
if err := sn.sendWithFallback(buf, sizeBuf); err != nil {
|
||
|
return fmt.Errorf("cannot send data to storageNodes: %s", err)
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// GetStorageNodeIdx returns storage node index for the given at and labels.
|
||
|
//
|
||
|
// The returned index must be passed to WriteDataPoint.
|
||
|
func (ctx *InsertCtx) GetStorageNodeIdx(at *auth.Token, labels []prompb.Label) int {
|
||
|
if len(storageNodes) == 1 {
|
||
|
// Fast path - only a single storage node.
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
buf := ctx.labelsBuf[:0]
|
||
|
buf = encoding.MarshalUint32(buf, at.AccountID)
|
||
|
buf = encoding.MarshalUint32(buf, at.ProjectID)
|
||
|
for i := range labels {
|
||
|
label := &labels[i]
|
||
|
buf = marshalBytesFast(buf, label.Name)
|
||
|
buf = marshalBytesFast(buf, label.Value)
|
||
|
}
|
||
|
h := xxhash.Sum64(buf)
|
||
|
ctx.labelsBuf = buf
|
||
|
|
||
|
idx := int(jump.Hash(h, int32(len(storageNodes))))
|
||
|
return idx
|
||
|
}
|
||
|
|
||
|
func marshalBytesFast(dst []byte, s []byte) []byte {
|
||
|
dst = encoding.MarshalUint16(dst, uint16(len(s)))
|
||
|
dst = append(dst, s...)
|
||
|
return dst
|
||
|
}
|