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 }