2020-04-29 17:40:33 +00:00
|
|
|
// Package kafka defines an implementation of Database interface
|
|
|
|
// which exports streaming data using Kafka for data analysis.
|
2019-12-06 02:05:58 +00:00
|
|
|
package kafka
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2020-08-27 18:13:32 +00:00
|
|
|
fssz "github.com/ferranbt/fastssz"
|
2019-12-06 02:05:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db/iface"
|
2021-07-06 15:34:05 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/proto/interfaces"
|
2019-12-06 02:05:58 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/traceutil"
|
|
|
|
"go.opencensus.io/trace"
|
2021-05-26 16:19:54 +00:00
|
|
|
jsonpb "google.golang.org/protobuf/encoding/protojson"
|
2019-12-06 02:05:58 +00:00
|
|
|
"gopkg.in/confluentinc/confluent-kafka-go.v1/kafka"
|
2020-05-31 06:44:34 +00:00
|
|
|
_ "gopkg.in/confluentinc/confluent-kafka-go.v1/kafka/librdkafka" // Required for c++ kafka library.
|
2021-01-20 21:03:46 +00:00
|
|
|
"gopkg.in/errgo.v2/fmt/errors"
|
2019-12-06 02:05:58 +00:00
|
|
|
)
|
|
|
|
|
2020-10-10 00:36:48 +00:00
|
|
|
var _ iface.Database = (*Exporter)(nil)
|
2021-05-26 16:19:54 +00:00
|
|
|
var marshaler = jsonpb.MarshalOptions{}
|
2019-12-06 02:05:58 +00:00
|
|
|
|
|
|
|
// Exporter wraps a database interface and exports certain objects to kafka topics.
|
|
|
|
type Exporter struct {
|
|
|
|
db iface.Database
|
|
|
|
p *kafka.Producer
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wrap the db with kafka exporter. If the feature flag is not enabled, this service does not wrap
|
|
|
|
// the database, but returns the underlying database pointer itself.
|
|
|
|
func Wrap(db iface.Database) (iface.Database, error) {
|
|
|
|
if featureconfig.Get().KafkaBootstrapServers == "" {
|
2020-06-23 13:51:00 +00:00
|
|
|
log.Debug("Empty Kafka bootstrap servers list, database was not wrapped with Kafka exporter")
|
2019-12-06 02:05:58 +00:00
|
|
|
return db, nil
|
|
|
|
}
|
|
|
|
p, err := kafka.NewProducer(&kafka.ConfigMap{"bootstrap.servers": featureconfig.Get().KafkaBootstrapServers})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Exporter{db: db, p: p}, nil
|
|
|
|
}
|
|
|
|
|
2021-05-26 16:19:54 +00:00
|
|
|
func (e Exporter) publish(ctx context.Context, topic string, msg interfaces.SignedBeaconBlock) error {
|
2019-12-06 02:05:58 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "kafka.publish")
|
|
|
|
defer span.End()
|
|
|
|
|
2021-05-26 16:19:54 +00:00
|
|
|
var err error
|
|
|
|
var buf []byte
|
|
|
|
if buf, err = marshaler.Marshal(msg.Proto()); err != nil {
|
2019-12-06 02:05:58 +00:00
|
|
|
traceutil.AnnotateError(span, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-27 18:13:32 +00:00
|
|
|
var key [32]byte
|
|
|
|
if v, ok := msg.(fssz.HashRoot); ok {
|
|
|
|
key, err = v.HashTreeRoot()
|
|
|
|
} else {
|
2021-01-20 21:03:46 +00:00
|
|
|
err = errors.New("object does not follow hash tree root interface")
|
2020-08-27 18:13:32 +00:00
|
|
|
}
|
2019-12-06 02:05:58 +00:00
|
|
|
if err != nil {
|
|
|
|
traceutil.AnnotateError(span, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := e.p.Produce(&kafka.Message{
|
|
|
|
TopicPartition: kafka.TopicPartition{
|
|
|
|
Topic: &topic,
|
|
|
|
},
|
2021-05-26 16:19:54 +00:00
|
|
|
Value: buf,
|
2019-12-06 02:05:58 +00:00
|
|
|
Key: key[:],
|
|
|
|
}, nil); err != nil {
|
|
|
|
traceutil.AnnotateError(span, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes kafka producer and underlying db.
|
|
|
|
func (e Exporter) Close() error {
|
|
|
|
e.p.Close()
|
|
|
|
return e.db.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
// SaveBlock publishes to the kafka topic for beacon blocks.
|
2021-05-26 16:19:54 +00:00
|
|
|
func (e Exporter) SaveBlock(ctx context.Context, block interfaces.SignedBeaconBlock) error {
|
2019-12-06 02:05:58 +00:00
|
|
|
go func() {
|
|
|
|
if err := e.publish(ctx, "beacon_block", block); err != nil {
|
|
|
|
log.WithError(err).Error("Failed to publish block")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return e.db.SaveBlock(ctx, block)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SaveBlocks publishes to the kafka topic for beacon blocks.
|
2021-05-26 16:19:54 +00:00
|
|
|
func (e Exporter) SaveBlocks(ctx context.Context, blocks []interfaces.SignedBeaconBlock) error {
|
2019-12-06 02:05:58 +00:00
|
|
|
go func() {
|
|
|
|
for _, block := range blocks {
|
|
|
|
if err := e.publish(ctx, "beacon_block", block); err != nil {
|
|
|
|
log.WithError(err).Error("Failed to publish block")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
return e.db.SaveBlocks(ctx, blocks)
|
|
|
|
}
|