-
Notifications
You must be signed in to change notification settings - Fork 198
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Created blob verifier worker. #708
Changes from all commits
eaa5bc5
24c942c
97ee14c
60d5dad
51d38ad
aef5d55
b7df5a2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,256 @@ | ||
package workers | ||
|
||
import ( | ||
"context" | ||
"github.com/Layr-Labs/eigenda/api/clients" | ||
"github.com/Layr-Labs/eigenda/api/grpc/disperser" | ||
"github.com/Layr-Labs/eigenda/tools/traffic/config" | ||
"github.com/Layr-Labs/eigenda/tools/traffic/metrics" | ||
"github.com/Layr-Labs/eigenda/tools/traffic/table" | ||
"github.com/Layr-Labs/eigensdk-go/logging" | ||
"math/rand" | ||
"sync" | ||
"time" | ||
) | ||
|
||
// BlobStatusTracker periodically polls the disperser service to verify the status of blobs that were recently written. | ||
// When blobs become confirmed, the status tracker updates the blob blobsToRead accordingly. | ||
// This is a thread safe data structure. | ||
type BlobStatusTracker struct { | ||
|
||
// The context for the generator. All work should cease when this context is cancelled. | ||
ctx *context.Context | ||
|
||
// Tracks the number of active goroutines within the generator. | ||
waitGroup *sync.WaitGroup | ||
|
||
// All logs should be written using this logger. | ||
logger logging.Logger | ||
|
||
// config contains the configuration for the generator. | ||
config *config.WorkerConfig | ||
|
||
// Contains confirmed blobs. Blobs are added here when they are confirmed by the disperser service. | ||
confirmedBlobs *table.BlobStore | ||
|
||
// The disperser client used to monitor the disperser service. | ||
disperser clients.DisperserClient | ||
|
||
// The keys of blobs that have not yet been confirmed by the disperser service. | ||
unconfirmedBlobs []*UnconfirmedKey | ||
|
||
// Newly added keys that require verification. | ||
keyChannel chan *UnconfirmedKey | ||
|
||
blobsInFlightMetric metrics.GaugeMetric | ||
getStatusLatencyMetric metrics.LatencyMetric | ||
confirmationLatencyMetric metrics.LatencyMetric | ||
getStatusErrorCountMetric metrics.CountMetric | ||
unknownCountMetric metrics.CountMetric | ||
processingCountMetric metrics.CountMetric | ||
dispersingCountMetric metrics.CountMetric | ||
failedCountMetric metrics.CountMetric | ||
insufficientSignaturesCountMetric metrics.CountMetric | ||
confirmedCountMetric metrics.CountMetric | ||
finalizedCountMetric metrics.CountMetric | ||
} | ||
|
||
// NewBlobStatusTracker creates a new BlobStatusTracker instance. | ||
func NewBlobStatusTracker( | ||
ctx *context.Context, | ||
waitGroup *sync.WaitGroup, | ||
logger logging.Logger, | ||
config *config.WorkerConfig, | ||
keyChannel chan *UnconfirmedKey, | ||
table *table.BlobStore, | ||
disperser clients.DisperserClient, | ||
generatorMetrics metrics.Metrics) BlobStatusTracker { | ||
|
||
return BlobStatusTracker{ | ||
ctx: ctx, | ||
waitGroup: waitGroup, | ||
logger: logger, | ||
config: config, | ||
keyChannel: keyChannel, | ||
confirmedBlobs: table, | ||
disperser: disperser, | ||
unconfirmedBlobs: make([]*UnconfirmedKey, 0), | ||
blobsInFlightMetric: generatorMetrics.NewGaugeMetric("blobs_in_flight"), | ||
getStatusLatencyMetric: generatorMetrics.NewLatencyMetric("get_status"), | ||
confirmationLatencyMetric: generatorMetrics.NewLatencyMetric("confirmation"), | ||
getStatusErrorCountMetric: generatorMetrics.NewCountMetric("get_status_ERROR"), | ||
unknownCountMetric: generatorMetrics.NewCountMetric("get_status_UNKNOWN"), | ||
processingCountMetric: generatorMetrics.NewCountMetric("get_status_PROCESSING"), | ||
dispersingCountMetric: generatorMetrics.NewCountMetric("get_status_DISPERSING"), | ||
failedCountMetric: generatorMetrics.NewCountMetric("get_status_FAILED"), | ||
insufficientSignaturesCountMetric: generatorMetrics.NewCountMetric("get_status_INSUFFICIENT_SIGNATURES"), | ||
confirmedCountMetric: generatorMetrics.NewCountMetric("get_status_CONFIRMED"), | ||
finalizedCountMetric: generatorMetrics.NewCountMetric("get_status_FINALIZED"), | ||
} | ||
} | ||
|
||
// Start begins the status goroutine, which periodically polls | ||
// the disperser service to verify the status of blobs. | ||
func (tracker *BlobStatusTracker) Start() { | ||
tracker.waitGroup.Add(1) | ||
go tracker.monitor() | ||
} | ||
|
||
// monitor periodically polls the disperser service to verify the status of blobs. | ||
func (tracker *BlobStatusTracker) monitor() { | ||
ticker := time.NewTicker(tracker.config.TrackerInterval) | ||
for { | ||
select { | ||
case <-(*tracker.ctx).Done(): | ||
tracker.waitGroup.Done() | ||
return | ||
case key := <-tracker.keyChannel: | ||
tracker.unconfirmedBlobs = append(tracker.unconfirmedBlobs, key) | ||
case <-ticker.C: | ||
tracker.poll() | ||
} | ||
} | ||
} | ||
|
||
// poll checks all unconfirmed keys to see if they have been confirmed by the disperser service. | ||
// If a Key is confirmed, it is added to the blob confirmedBlobs and removed from the list of unconfirmed keys. | ||
func (tracker *BlobStatusTracker) poll() { | ||
|
||
// FUTURE WORK If the number of unconfirmed blobs is high and the time to confirm is high, this is not efficient. | ||
// Revisit this method if there are performance problems. | ||
|
||
nonFinalBlobs := make([]*UnconfirmedKey, 0) | ||
for _, key := range tracker.unconfirmedBlobs { | ||
|
||
blobStatus, err := tracker.getBlobStatus(key) | ||
if err != nil { | ||
tracker.logger.Error("failed to get blob status: ", "err:", err) | ||
// There was an error getting status. Try again later. | ||
nonFinalBlobs = append(nonFinalBlobs, key) | ||
continue | ||
} | ||
|
||
tracker.updateStatusMetrics(blobStatus.Status) | ||
if isBlobStatusTerminal(blobStatus.Status) { | ||
if isBlobStatusConfirmed(blobStatus.Status) { | ||
tracker.forwardToReader(key, blobStatus) | ||
} | ||
} else { | ||
// try again later | ||
nonFinalBlobs = append(nonFinalBlobs, key) | ||
} | ||
} | ||
tracker.unconfirmedBlobs = nonFinalBlobs | ||
tracker.blobsInFlightMetric.Set(float64(len(tracker.unconfirmedBlobs))) | ||
} | ||
|
||
// isBlobStatusTerminal returns true if the status is a terminal status. | ||
func isBlobStatusTerminal(status disperser.BlobStatus) bool { | ||
switch status { | ||
case disperser.BlobStatus_FAILED: | ||
return true | ||
case disperser.BlobStatus_INSUFFICIENT_SIGNATURES: | ||
return true | ||
case disperser.BlobStatus_CONFIRMED: | ||
// Technically this isn't terminal, as confirmed blobs eventually should become finalized. | ||
// But it is terminal from the status tracker's perspective, since we stop tracking the blob | ||
// once it becomes either confirmed or finalized. | ||
return true | ||
case disperser.BlobStatus_FINALIZED: | ||
return true | ||
default: | ||
return false | ||
} | ||
} | ||
|
||
// isBlobStatusConfirmed returns true if the status is a confirmed status. | ||
func isBlobStatusConfirmed(status disperser.BlobStatus) bool { | ||
switch status { | ||
case disperser.BlobStatus_CONFIRMED: | ||
return true | ||
case disperser.BlobStatus_FINALIZED: | ||
return true | ||
default: | ||
return false | ||
} | ||
} | ||
|
||
// updateStatusMetrics updates the metrics for the reported status of a blob. | ||
func (tracker *BlobStatusTracker) updateStatusMetrics(status disperser.BlobStatus) { | ||
switch status { | ||
case disperser.BlobStatus_UNKNOWN: | ||
tracker.unknownCountMetric.Increment() | ||
case disperser.BlobStatus_PROCESSING: | ||
tracker.processingCountMetric.Increment() | ||
case disperser.BlobStatus_DISPERSING: | ||
tracker.dispersingCountMetric.Increment() | ||
case disperser.BlobStatus_FAILED: | ||
tracker.failedCountMetric.Increment() | ||
case disperser.BlobStatus_INSUFFICIENT_SIGNATURES: | ||
tracker.insufficientSignaturesCountMetric.Increment() | ||
case disperser.BlobStatus_CONFIRMED: | ||
tracker.confirmedCountMetric.Increment() | ||
case disperser.BlobStatus_FINALIZED: | ||
tracker.finalizedCountMetric.Increment() | ||
default: | ||
tracker.logger.Error("unknown blob status", "status:", status) | ||
} | ||
} | ||
|
||
// getBlobStatus gets the status of a blob from the disperser service. Returns nil if there was an error | ||
// getting the status. | ||
func (tracker *BlobStatusTracker) getBlobStatus(key *UnconfirmedKey) (*disperser.BlobStatusReply, error) { | ||
ctxTimeout, cancel := context.WithTimeout(*tracker.ctx, tracker.config.GetBlobStatusTimeout) | ||
defer cancel() | ||
|
||
status, err := metrics.InvokeAndReportLatency[*disperser.BlobStatusReply](tracker.getStatusLatencyMetric, | ||
func() (*disperser.BlobStatusReply, error) { | ||
return tracker.disperser.GetBlobStatus(ctxTimeout, key.Key) | ||
}) | ||
|
||
if err != nil { | ||
tracker.getStatusErrorCountMetric.Increment() | ||
return nil, err | ||
} | ||
|
||
return status, nil | ||
} | ||
|
||
// forwardToReader forwards a blob to the reader. Only called once the blob is ready to be read. | ||
func (tracker *BlobStatusTracker) forwardToReader(key *UnconfirmedKey, status *disperser.BlobStatusReply) { | ||
batchHeaderHash := [32]byte(status.GetInfo().BlobVerificationProof.BatchMetadata.BatchHeaderHash) | ||
blobIndex := status.GetInfo().BlobVerificationProof.GetBlobIndex() | ||
|
||
confirmationTime := time.Now() | ||
confirmationLatency := confirmationTime.Sub(key.SubmissionTime) | ||
tracker.confirmationLatencyMetric.ReportLatency(confirmationLatency) | ||
|
||
requiredDownloads := tracker.config.RequiredDownloads | ||
var downloadCount int32 | ||
if requiredDownloads < 0 { | ||
// Allow unlimited downloads. | ||
downloadCount = -1 | ||
} else if requiredDownloads == 0 { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. doesn't look like this block will ever be triggered There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. fixed |
||
// Do not download blob. | ||
return | ||
} else if requiredDownloads < 1 { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. same with this condition. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. fixed There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not sure if it's fixed. If There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
The |
||
// Download blob with probability equal to requiredDownloads. | ||
if rand.Float64() < requiredDownloads { | ||
// Download the blob once. | ||
downloadCount = 1 | ||
} else { | ||
// Do not download blob. | ||
return | ||
} | ||
} else { | ||
// Download blob requiredDownloads times. | ||
downloadCount = int32(requiredDownloads) | ||
} | ||
|
||
blobMetadata, err := table.NewBlobMetadata(key.Key, key.Checksum, key.Size, uint(blobIndex), batchHeaderHash, int(downloadCount)) | ||
if err != nil { | ||
tracker.logger.Error("failed to create blob metadata", "err:", err) | ||
return | ||
} | ||
tracker.confirmedBlobs.Add(blobMetadata) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
CONFIRMED isn't a terminal state as it can be transitioned to FINALIZED or FAILED (if reorg'd)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Correct, although the behavior is intentional in this context. Added a comment to clarify.