2018-10-05 11:43:56 +00:00
|
|
|
package storage
|
|
|
|
|
2018-11-01 18:44:56 +00:00
|
|
|
import (
|
2019-02-25 16:11:20 +00:00
|
|
|
"context"
|
2020-07-02 15:15:09 +00:00
|
|
|
"fmt"
|
2020-07-01 15:57:00 +00:00
|
|
|
"time"
|
2019-02-25 16:11:20 +00:00
|
|
|
|
2020-07-01 15:57:00 +00:00
|
|
|
"github.com/influxdata/influxdb/v2"
|
2020-07-28 22:59:11 +00:00
|
|
|
"github.com/influxdata/influxdb/v2/models"
|
2018-11-01 18:44:56 +00:00
|
|
|
)
|
2018-10-05 11:43:56 +00:00
|
|
|
|
|
|
|
// PointsWriter describes the ability to write points into a storage engine.
|
|
|
|
type PointsWriter interface {
|
2020-07-23 17:28:57 +00:00
|
|
|
WritePoints(ctx context.Context, orgID influxdb.ID, bucketID influxdb.ID, points []models.Point) error
|
2018-10-05 11:43:56 +00:00
|
|
|
}
|
2019-08-02 16:29:27 +00:00
|
|
|
|
2020-07-01 15:57:00 +00:00
|
|
|
// LoggingPointsWriter wraps an underlying points writer but writes logs to
|
|
|
|
// another bucket when an error occurs.
|
|
|
|
type LoggingPointsWriter struct {
|
|
|
|
// Wrapped points writer. Errored writes from here will be logged.
|
|
|
|
Underlying PointsWriter
|
|
|
|
|
|
|
|
// Service used to look up logging bucket.
|
2020-07-02 15:15:09 +00:00
|
|
|
BucketFinder BucketFinder
|
2020-07-01 15:57:00 +00:00
|
|
|
|
|
|
|
// Name of the bucket to log to.
|
|
|
|
LogBucketName string
|
|
|
|
}
|
|
|
|
|
|
|
|
// WritePoints writes points to the underlying PointsWriter. Logs on error.
|
2020-07-23 17:28:57 +00:00
|
|
|
func (w *LoggingPointsWriter) WritePoints(ctx context.Context, orgID influxdb.ID, bucketID influxdb.ID, p []models.Point) error {
|
2020-07-01 15:57:00 +00:00
|
|
|
if len(p) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write to underlying writer and exit immediately if successful.
|
2020-07-23 17:28:57 +00:00
|
|
|
err := w.Underlying.WritePoints(ctx, orgID, bucketID, p)
|
2020-07-01 15:57:00 +00:00
|
|
|
if err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Attempt to lookup log bucket.
|
2020-07-02 15:15:09 +00:00
|
|
|
bkts, n, e := w.BucketFinder.FindBuckets(ctx, influxdb.BucketFilter{
|
|
|
|
OrganizationID: &orgID,
|
|
|
|
Name: &w.LogBucketName,
|
|
|
|
})
|
2020-07-01 15:57:00 +00:00
|
|
|
if e != nil {
|
|
|
|
return e
|
2020-07-02 15:15:09 +00:00
|
|
|
} else if n == 0 {
|
|
|
|
return fmt.Errorf("logging bucket not found: %q", w.LogBucketName)
|
2020-07-01 15:57:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Log error to bucket.
|
|
|
|
pt, e := models.NewPoint(
|
2020-04-24 14:58:24 +00:00
|
|
|
"write_errors",
|
|
|
|
nil,
|
2020-07-01 15:57:00 +00:00
|
|
|
models.Fields{"error": err.Error()},
|
|
|
|
time.Now(),
|
|
|
|
)
|
|
|
|
if e != nil {
|
|
|
|
return e
|
|
|
|
}
|
2020-04-24 14:58:24 +00:00
|
|
|
if e := w.Underlying.WritePoints(ctx, orgID, bkts[0].ID, []models.Point{pt}); e != nil {
|
2020-07-01 15:57:00 +00:00
|
|
|
return e
|
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-08-02 16:29:27 +00:00
|
|
|
type BufferedPointsWriter struct {
|
2020-08-03 22:15:00 +00:00
|
|
|
buf []models.Point
|
|
|
|
orgID influxdb.ID
|
|
|
|
bucketID influxdb.ID
|
|
|
|
n int
|
|
|
|
wr PointsWriter
|
|
|
|
err error
|
2019-08-02 16:29:27 +00:00
|
|
|
}
|
|
|
|
|
2020-08-03 22:15:00 +00:00
|
|
|
func NewBufferedPointsWriter(orgID influxdb.ID, bucketID influxdb.ID, size int, pointswriter PointsWriter) *BufferedPointsWriter {
|
2019-08-02 16:29:27 +00:00
|
|
|
return &BufferedPointsWriter{
|
2020-08-03 22:15:00 +00:00
|
|
|
buf: make([]models.Point, size),
|
|
|
|
orgID: orgID,
|
|
|
|
bucketID: bucketID,
|
|
|
|
wr: pointswriter,
|
2019-08-02 16:29:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WritePoints writes the points to the underlying PointsWriter.
|
2020-08-03 22:15:00 +00:00
|
|
|
func (b *BufferedPointsWriter) WritePoints(ctx context.Context, p []models.Point) error {
|
2019-08-02 16:29:27 +00:00
|
|
|
for len(p) > b.Available() && b.err == nil {
|
|
|
|
if b.Buffered() == 0 {
|
|
|
|
// Large write, empty buffer.
|
|
|
|
// Write directly from p to avoid copy.
|
2020-08-03 22:15:00 +00:00
|
|
|
b.err = b.wr.WritePoints(ctx, b.orgID, b.bucketID, p)
|
2019-08-03 04:52:33 +00:00
|
|
|
return b.err
|
2019-08-02 16:29:27 +00:00
|
|
|
}
|
2019-08-03 04:52:33 +00:00
|
|
|
n := copy(b.buf[b.n:], p)
|
|
|
|
b.n += n
|
|
|
|
b.err = b.Flush(ctx)
|
2019-08-02 16:29:27 +00:00
|
|
|
p = p[n:]
|
|
|
|
}
|
|
|
|
if b.err != nil {
|
|
|
|
return b.err
|
|
|
|
}
|
|
|
|
b.n += copy(b.buf[b.n:], p)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Available returns how many models.Points are unused in the buffer.
|
|
|
|
func (b *BufferedPointsWriter) Available() int { return len(b.buf) - b.n }
|
|
|
|
|
|
|
|
// Buffered returns the number of models.Points that have been written into the current buffer.
|
2019-08-03 04:52:33 +00:00
|
|
|
func (b *BufferedPointsWriter) Buffered() int { return b.n }
|
2019-08-02 16:29:27 +00:00
|
|
|
|
|
|
|
// Flush writes any buffered data to the underlying PointsWriter.
|
|
|
|
func (b *BufferedPointsWriter) Flush(ctx context.Context) error {
|
|
|
|
if b.err != nil {
|
|
|
|
return b.err
|
|
|
|
}
|
|
|
|
if b.n == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-03 22:15:00 +00:00
|
|
|
b.err = b.wr.WritePoints(ctx, b.orgID, b.bucketID, b.buf[:b.n])
|
2019-08-02 16:29:27 +00:00
|
|
|
if b.err != nil {
|
|
|
|
return b.err
|
|
|
|
}
|
|
|
|
b.n = 0
|
|
|
|
return nil
|
|
|
|
}
|