2017-05-10 09:44:13 +00:00
|
|
|
// Copyright 2017 The Prometheus 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
|
2017-10-17 01:26:38 +00:00
|
|
|
// limitations under the License.
|
2017-05-10 09:44:13 +00:00
|
|
|
|
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
2017-10-04 19:04:15 +00:00
|
|
|
"context"
|
2017-05-10 09:44:13 +00:00
|
|
|
|
2021-06-11 16:17:59 +00:00
|
|
|
"github.com/go-kit/log"
|
|
|
|
"github.com/go-kit/log/level"
|
2017-10-18 11:08:14 +00:00
|
|
|
"github.com/prometheus/common/model"
|
2020-10-22 09:00:08 +00:00
|
|
|
|
2021-11-08 14:23:17 +00:00
|
|
|
"github.com/prometheus/prometheus/model/exemplar"
|
|
|
|
"github.com/prometheus/prometheus/model/labels"
|
2020-03-24 20:15:47 +00:00
|
|
|
tsdb_errors "github.com/prometheus/prometheus/tsdb/errors"
|
2017-05-10 09:44:13 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type fanout struct {
|
2017-08-11 18:45:52 +00:00
|
|
|
logger log.Logger
|
|
|
|
|
2017-07-12 14:50:26 +00:00
|
|
|
primary Storage
|
|
|
|
secondaries []Storage
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
2020-06-09 16:57:31 +00:00
|
|
|
// NewFanout returns a new fanout Storage, which proxies reads and writes
|
2017-05-10 09:44:13 +00:00
|
|
|
// through to multiple underlying storages.
|
2020-06-09 16:57:31 +00:00
|
|
|
//
|
|
|
|
// The difference between primary and secondary Storage is only for read (Querier) path and it goes as follows:
|
|
|
|
// * If the primary querier returns an error, then any of the Querier operations will fail.
|
|
|
|
// * If any secondary querier returns an error the result from that queries is discarded. The overall operation will succeed,
|
|
|
|
// and the error from the secondary querier will be returned as a warning.
|
|
|
|
//
|
|
|
|
// NOTE: In the case of Prometheus, it treats all remote storages as secondary / best effort.
|
2017-08-11 18:45:52 +00:00
|
|
|
func NewFanout(logger log.Logger, primary Storage, secondaries ...Storage) Storage {
|
2017-05-10 09:44:13 +00:00
|
|
|
return &fanout{
|
2017-08-11 18:45:52 +00:00
|
|
|
logger: logger,
|
2017-07-12 14:50:26 +00:00
|
|
|
primary: primary,
|
|
|
|
secondaries: secondaries,
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-18 11:08:14 +00:00
|
|
|
// StartTime implements the Storage interface.
|
|
|
|
func (f *fanout) StartTime() (int64, error) {
|
|
|
|
// StartTime of a fanout should be the earliest StartTime of all its storages,
|
|
|
|
// both primary and secondaries.
|
|
|
|
firstTime, err := f.primary.StartTime()
|
|
|
|
if err != nil {
|
|
|
|
return int64(model.Latest), err
|
|
|
|
}
|
|
|
|
|
2020-06-09 16:57:31 +00:00
|
|
|
for _, s := range f.secondaries {
|
|
|
|
t, err := s.StartTime()
|
2017-10-18 11:08:14 +00:00
|
|
|
if err != nil {
|
|
|
|
return int64(model.Latest), err
|
|
|
|
}
|
|
|
|
if t < firstTime {
|
|
|
|
firstTime = t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return firstTime, nil
|
|
|
|
}
|
|
|
|
|
2017-10-04 19:04:15 +00:00
|
|
|
func (f *fanout) Querier(ctx context.Context, mint, maxt int64) (Querier, error) {
|
2020-06-09 16:57:31 +00:00
|
|
|
primary, err := f.primary.Querier(ctx, mint, maxt)
|
2017-07-12 14:50:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-06-09 16:57:31 +00:00
|
|
|
secondaries := make([]Querier, 0, len(f.secondaries))
|
2017-07-12 14:50:26 +00:00
|
|
|
for _, storage := range f.secondaries {
|
2017-10-04 19:04:15 +00:00
|
|
|
querier, err := storage.Querier(ctx, mint, maxt)
|
2017-05-10 09:44:13 +00:00
|
|
|
if err != nil {
|
2020-06-09 16:57:31 +00:00
|
|
|
// Close already open Queriers, append potential errors to returned error.
|
2020-10-28 15:24:58 +00:00
|
|
|
errs := tsdb_errors.NewMulti(err, primary.Close())
|
2020-06-09 16:57:31 +00:00
|
|
|
for _, q := range secondaries {
|
|
|
|
errs.Add(q.Close())
|
2020-03-24 20:15:47 +00:00
|
|
|
}
|
2020-06-09 16:57:31 +00:00
|
|
|
return nil, errs.Err()
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
2020-06-09 16:57:31 +00:00
|
|
|
secondaries = append(secondaries, querier)
|
|
|
|
}
|
2020-07-31 15:03:02 +00:00
|
|
|
return NewMergeQuerier([]Querier{primary}, secondaries, ChainedSeriesMerge), nil
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
2020-06-24 13:41:52 +00:00
|
|
|
func (f *fanout) ChunkQuerier(ctx context.Context, mint, maxt int64) (ChunkQuerier, error) {
|
|
|
|
primary, err := f.primary.ChunkQuerier(ctx, mint, maxt)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
secondaries := make([]ChunkQuerier, 0, len(f.secondaries))
|
|
|
|
for _, storage := range f.secondaries {
|
|
|
|
querier, err := storage.ChunkQuerier(ctx, mint, maxt)
|
|
|
|
if err != nil {
|
|
|
|
// Close already open Queriers, append potential errors to returned error.
|
2020-10-28 15:24:58 +00:00
|
|
|
errs := tsdb_errors.NewMulti(err, primary.Close())
|
2020-06-24 13:41:52 +00:00
|
|
|
for _, q := range secondaries {
|
|
|
|
errs.Add(q.Close())
|
|
|
|
}
|
|
|
|
return nil, errs.Err()
|
|
|
|
}
|
|
|
|
secondaries = append(secondaries, querier)
|
|
|
|
}
|
2020-07-31 15:03:02 +00:00
|
|
|
return NewMergeChunkQuerier([]ChunkQuerier{primary}, secondaries, NewCompactingChunkSeriesMerger(ChainedSeriesMerge)), nil
|
2020-06-24 13:41:52 +00:00
|
|
|
}
|
|
|
|
|
2020-07-24 14:10:51 +00:00
|
|
|
func (f *fanout) Appender(ctx context.Context) Appender {
|
|
|
|
primary := f.primary.Appender(ctx)
|
2017-07-12 14:50:26 +00:00
|
|
|
secondaries := make([]Appender, 0, len(f.secondaries))
|
|
|
|
for _, storage := range f.secondaries {
|
2020-07-24 14:10:51 +00:00
|
|
|
secondaries = append(secondaries, storage.Appender(ctx))
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
return &fanoutAppender{
|
2017-08-11 18:45:52 +00:00
|
|
|
logger: f.logger,
|
2017-07-12 14:50:26 +00:00
|
|
|
primary: primary,
|
|
|
|
secondaries: secondaries,
|
2020-02-06 15:58:38 +00:00
|
|
|
}
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the storage and all its underlying resources.
|
|
|
|
func (f *fanout) Close() error {
|
2020-10-28 15:24:58 +00:00
|
|
|
errs := tsdb_errors.NewMulti(f.primary.Close())
|
2020-06-09 16:57:31 +00:00
|
|
|
for _, s := range f.secondaries {
|
|
|
|
errs.Add(s.Close())
|
2017-07-12 14:50:26 +00:00
|
|
|
}
|
2020-06-09 16:57:31 +00:00
|
|
|
return errs.Err()
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// fanoutAppender implements Appender.
|
|
|
|
type fanoutAppender struct {
|
2017-08-11 18:45:52 +00:00
|
|
|
logger log.Logger
|
|
|
|
|
2017-07-12 14:50:26 +00:00
|
|
|
primary Appender
|
|
|
|
secondaries []Appender
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
2021-11-06 10:10:04 +00:00
|
|
|
func (f *fanoutAppender) Append(ref SeriesRef, l labels.Labels, t int64, v float64) (SeriesRef, error) {
|
2021-02-18 12:07:00 +00:00
|
|
|
ref, err := f.primary.Append(ref, l, t, v)
|
2017-07-12 14:50:26 +00:00
|
|
|
if err != nil {
|
|
|
|
return ref, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, appender := range f.secondaries {
|
2021-02-18 12:07:00 +00:00
|
|
|
if _, err := appender.Append(ref, l, t, v); err != nil {
|
2017-09-07 12:14:41 +00:00
|
|
|
return 0, err
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
}
|
2017-07-12 14:50:26 +00:00
|
|
|
return ref, nil
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
2021-11-06 10:10:04 +00:00
|
|
|
func (f *fanoutAppender) AppendExemplar(ref SeriesRef, l labels.Labels, e exemplar.Exemplar) (SeriesRef, error) {
|
2021-03-16 09:47:45 +00:00
|
|
|
ref, err := f.primary.AppendExemplar(ref, l, e)
|
|
|
|
if err != nil {
|
|
|
|
return ref, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, appender := range f.secondaries {
|
|
|
|
if _, err := appender.AppendExemplar(ref, l, e); err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ref, nil
|
|
|
|
}
|
|
|
|
|
2017-07-12 14:50:26 +00:00
|
|
|
func (f *fanoutAppender) Commit() (err error) {
|
|
|
|
err = f.primary.Commit()
|
|
|
|
|
|
|
|
for _, appender := range f.secondaries {
|
|
|
|
if err == nil {
|
|
|
|
err = appender.Commit()
|
|
|
|
} else {
|
|
|
|
if rollbackErr := appender.Rollback(); rollbackErr != nil {
|
2017-08-11 18:45:52 +00:00
|
|
|
level.Error(f.logger).Log("msg", "Squashed rollback error on commit", "err", rollbackErr)
|
2017-07-12 14:50:26 +00:00
|
|
|
}
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
}
|
2017-07-12 14:50:26 +00:00
|
|
|
return
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
|
2017-07-12 14:50:26 +00:00
|
|
|
func (f *fanoutAppender) Rollback() (err error) {
|
|
|
|
err = f.primary.Rollback()
|
|
|
|
|
|
|
|
for _, appender := range f.secondaries {
|
|
|
|
rollbackErr := appender.Rollback()
|
|
|
|
if err == nil {
|
|
|
|
err = rollbackErr
|
|
|
|
} else if rollbackErr != nil {
|
2017-08-11 18:45:52 +00:00
|
|
|
level.Error(f.logger).Log("msg", "Squashed rollback error on rollback", "err", rollbackErr)
|
2017-05-10 09:44:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|