-
Notifications
You must be signed in to change notification settings - Fork 180
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
[Access] Improvements for SubscribeTransactionStatuses
statuses handling
#6736
base: master
Are you sure you want to change the base?
Changes from all commits
2e8d95b
de3851a
dfb40cf
edb0636
23a2214
e3a43a0
ef8136d
543fb9e
9ee3d6a
b49bb7c
2baa1a3
8c0c247
2ecd739
8f0e956
35d3da9
86e6096
5ba75c8
2a1f2f8
40a3323
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 |
---|---|---|
|
@@ -36,7 36,7 @@ type backendSubscribeTransactions struct { | |
type TransactionSubscriptionMetadata struct { | ||
*access.TransactionResult | ||
txReferenceBlockID flow.Identifier | ||
blockWithTx *flow.Header | ||
blockWithTx *flow.Block | ||
txExecuted bool | ||
eventEncodingVersion entities.EventEncodingVersion | ||
} | ||
|
@@ -126,7 126,7 @@ func (b *backendSubscribeTransactions) getTransactionStatusResponse(txInfo *Tran | |
// When a block with the transaction is available, it is possible to receive a new transaction status while | ||
// searching for the transaction result. Otherwise, it remains unchanged. So, if the old and new transaction | ||
// statuses are the same, the current transaction status should be retrieved. | ||
txInfo.Status, err = b.txLocalDataProvider.DeriveTransactionStatus(txInfo.blockWithTx.Height, txInfo.txExecuted) | ||
txInfo.Status, err = b.txLocalDataProvider.DeriveTransactionStatus(txInfo.BlockHeight, txInfo.txExecuted) | ||
} | ||
if err != nil { | ||
if !errors.Is(err, state.ErrUnknownSnapshotReference) { | ||
|
@@ -229,7 229,7 @@ func (b *backendSubscribeTransactions) checkBlockReady(height uint64) error { | |
func (b *backendSubscribeTransactions) searchForTransactionBlockInfo( | ||
height uint64, | ||
txInfo *TransactionSubscriptionMetadata, | ||
) (*flow.Header, flow.Identifier, uint64, flow.Identifier, error) { | ||
) (*flow.Block, flow.Identifier, uint64, flow.Identifier, error) { | ||
block, err := b.txLocalDataProvider.blocks.ByHeight(height) | ||
if err != nil { | ||
return nil, flow.ZeroID, 0, flow.ZeroID, fmt.Errorf("error looking up block: %w", err) | ||
|
@@ -241,43 241,40 @@ func (b *backendSubscribeTransactions) searchForTransactionBlockInfo( | |
} | ||
|
||
if collectionID != flow.ZeroID { | ||
return block.Header, block.ID(), height, collectionID, nil | ||
return block, block.ID(), height, collectionID, nil | ||
} | ||
|
||
return nil, flow.ZeroID, 0, flow.ZeroID, nil | ||
} | ||
|
||
// searchForTransactionResult searches for the transaction result of a block. It retrieves the execution result for the specified block ID. | ||
// Expected errors: | ||
// - codes.Internal if an internal error occurs while retrieving execution result. | ||
// searchForTransactionResult searches for the transaction result of a block. It retrieves the transaction result from | ||
// storage and, in case of failure, attempts to fetch the transaction result directly from the execution node. | ||
// This is necessary to ensure data availability despite sync storage latency. | ||
// | ||
// No errors expected during normal operations. | ||
func (b *backendSubscribeTransactions) searchForTransactionResult( | ||
ctx context.Context, | ||
txInfo *TransactionSubscriptionMetadata, | ||
) (*access.TransactionResult, error) { | ||
_, err := b.executionResults.ByBlockID(txInfo.BlockID) | ||
txResult, err := b.backendTransactions.GetTransactionResultFromStorage(ctx, txInfo.blockWithTx, txInfo.TransactionID, txInfo.eventEncodingVersion) | ||
peterargue marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if err != nil { | ||
Guitarheroua marked this conversation as resolved.
Show resolved
Hide resolved
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. at this point you can receive multiple sentinel errors and I assume an exception as well. We are basically sweeping all errors under one if statement but I would suggest to explicitly filter if it's an expected error. Otherwise we might get into a situation where a node observes a critical failure(lets say a storage failure where the badger DB is corrupted) but we are still trying to operate on best-effort scenario which is unacceptable. |
||
if errors.Is(err, storage.ErrNotFound) { | ||
return nil, nil | ||
} | ||
return nil, fmt.Errorf("failed to get execution result for block %s: %w", txInfo.BlockID, err) | ||
} | ||
|
||
txResult, err := b.backendTransactions.GetTransactionResult( | ||
ctx, | ||
txInfo.TransactionID, | ||
txInfo.BlockID, | ||
txInfo.CollectionID, | ||
txInfo.eventEncodingVersion, | ||
) | ||
// If any error occurs with local storage - request transaction result from EN | ||
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. as I said in previously comment, "any" error is a bit to extreme |
||
txResult, err = b.backendTransactions.GetTransactionResultFromExecutionNode( | ||
ctx, | ||
txInfo.blockWithTx, | ||
txInfo.TransactionID, | ||
txInfo.eventEncodingVersion, | ||
) | ||
|
||
if err != nil { | ||
// if either the storage or execution node reported no results or there were not enough execution results | ||
if status.Code(err) == codes.NotFound { | ||
// No result yet, indicate that it has not been executed | ||
return nil, nil | ||
if err != nil { | ||
// if either the execution node reported no results | ||
if status.Code(err) == codes.NotFound { | ||
// No result yet, indicate that it has not been executed | ||
return nil, nil | ||
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. This convention is not explained in the documentation, usually if there is no error then a result has to be available, returning |
||
} | ||
// Other Error trying to retrieve the result, return with err | ||
return nil, err | ||
} | ||
// Other Error trying to retrieve the result, return with err | ||
return nil, err | ||
} | ||
|
||
return txResult, nil | ||
|
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.
I have checked underlying code path and couldn't find a place why we essentially need a
flow.Block
. My suggestion here would be to stick toflow.Header
and actually refactor usages of this field to acceptflow.Header
instead offlow.Block
. I think that we should carry only the minimal needed information since extra info increases mental complexity and makes the APIs more convoluted and less intuitive.Suppose I see a method signature:
when I see such signature I assume that it does something with the block payload since it passes a block rather than a header and the difference between block and header is exactly the payload, in fact it is misleading since it utilizes only fields of
flow.Header
. To avoid such misleading APIs I would propose to make a refactoring to clean this.