Skip to content
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

Implement the endpoint GET /v2/block-headers #1638

Merged
merged 19 commits into from
Dec 20, 2024

Conversation

agodnic
Copy link
Contributor

@agodnic agodnic commented Nov 18, 2024

Summary

This pull request implements the endpoint GET /v2/block-headers (ported from AlgoNode#10).

In particular, this enables searching for blocks by different participation-related parameters (i.e.: proposer address, expired accounts, absent accounts).

Test Plan

The new endpoint has been tested manually, using the FNet dataset:

# no parameters
curl -Ss 'http://127.0.0.1:8980/v2/block-headers?limit=2' | jq

# filter by timestamp
curl -Ss 'http://127.0.0.1:8980/v2/block-headers?limit=2&after-time=2024-10-01T00:00:00Z' | jq

# filter by block proposer
curl -Ss 'http://127.0.0.1:8980/v2/block-headers?proposers=FNETC46DGTSSDHA6C54FWEGZ3Z5ADE4YAPYQTX5VEE2YGU5NFPAWTRANCE&limit=1' | jq

# filter by absent participation accounts
curl -Ss 'http://127.0.0.1:8980/v2/block-headers?absent=WXN65H5M7BRSABP545OES2NCMFGE7R7OICDZSXCTP5H5H2CHXCDKFQTF6U&limit=1' | jq

# filter by expired participation accounts
curl -Ss 'http://127.0.0.1:8980/v2/block-headers?expired=2D2HFM3EIYFNMBNLWZPK5E4SWVN6SR7RDSVCBQ4Z6HRHE26DBT6WQYO6XA&limit=1' | jq

So far there are no automated tests for this feature. Maybe the tests could look like these (mutatis mutandis):

@CLAassistant
Copy link

CLAassistant commented Nov 18, 2024

CLA assistant check
All committers have signed the CLA.

@agodnic agodnic changed the title [WIP] Implement the endpoint GET /v2/blocks Implement the endpoint GET /v2/blocks Nov 18, 2024
api/handlers.go Outdated Show resolved Hide resolved
@@ -853,6 +853,67 @@
}
}
},
"/v2/blocks": {
"get": {
"description": "Search for blocks. Blocks are returned in ascending round order. Transactions are not included in the output.",
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since there are no transactions, the endpoint might better be called blockheaders. I think that's what the similar algod endpoint under development is called.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The route was changed to GET /v2/block-headers, in a302e5b

api/indexer.oas2.json Outdated Show resolved Hide resolved
"type": "string",
"x-algorand-format": "Address"
},
"description": "Account(s) marked as expired, absent or as proposer in the block header's participation updates. This parameter accepts a comma separated list of addresses.",
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems like a weird catch-all. I guess the intent is for block explorers that want to show any relevant block for particular accounts?

Copy link
Contributor Author

@agodnic agodnic Nov 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe so. Maybe @urtho can give more context on how the participation parameter is intended to be used.

Performance-wise, it is the most problematic parameter. I can see the standard version of the indexer not including the participation parameter (I regard it as a low benefit / high cost feature)

Something similar happens with the updates parameter, maybe not as bad.

Moreover, the underlyng SQL query can be significantly simpler if these two parameters are not shipped.

Copy link
Contributor

@jannotti jannotti Nov 20, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there any meaningful advantage to it, compared to calling the endpoint three times, specifying the address for each of the three things? I would guess you're using an index for each, so it's basically three index lookups no matter how you slice it.

These lists (expired and suspended) are almost always empty. It seems like an index lookup would be super fast. Just do two of them?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is there any meaningful advantage to it, compared to calling the endpoint three times, specifying the address for each of the three things? I would guess you're using an index for each, so it's basically three index lookups no matter how you slice it.

These lists (expired and suspended) are almost always empty. It seems like an index lookup would be super fast. Just do two of them?

That's exactly right. The essential functionality is already being exposed by the "atomic" parameters (that is: proposer, absent and expired)

The data returned by the updates and participation parameters could be obtained by calling the endpoint multiple times using the "atomic" parameters.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I went overboard with my initial requirements and there has been no feedback so far.
I am totally for making the queries/implementation simpler, by only allowing querying for one of the cases (proposer/abset/expired) at a time.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed in 8cb90c1

@agodnic agodnic changed the title Implement the endpoint GET /v2/blocks Implement the endpoint GET /v2/block-headers Nov 21, 2024
Remove the `updates` and `participation` parameters from `GET
/v2/block-headers`.

The underlying SQL code is now simpler.
@@ -298,6 +308,94 @@ func txnRowToTransaction(row idb.TxnRow) (generated.Transaction, error) {
return txn, nil
}

func rowToBlock(blockHeader *sdk.BlockHeader) generated.Block {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I haven't written much indexer code, but this name seems a bit underspecified. Above, we see txnRowToTransaction, so this should probably be hdrRowToBlock. I'm not totally confident about this recommendation because I would have expected this to take some sort of idb type, rather than an sdk type.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's fair, the name hdrRowToBlock will generate less confusion when reading the code.

Addressed in eaefec7 and 8924c97

// decodeAddress returns the byte representation of the input string, or appends an error to errorArr
func decodeAddress(str *string, field string, errorArr []string) ([]byte, []string) {
// decodeAddress returns the sdk.Address representation of the input string, or appends an error to errorArr
func decodeAddress(str string, field string, errorArr []string) (sdk.Address, []string) {
Copy link
Contributor

@jannotti jannotti Nov 21, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see that decodeAddress already existed, so this PR is not to blame for the unusual way that multiple errors are accumulated, but I think there is a better way to handle multiple errors these days.

I would use sdk.DecodeAddress directly, and if it returns an error, append that into a []error instead of []string. Use fmt.Errorf with a %w to produce an error that wraps the error from sdk.DecodeAddress but also contains the str that would have been passed into decodeAddress. Then at the end of a function that may have accumulated errors, test its size just as you do, but then use errors.Join(errArr...) to combine them into a single error.

I'll leave it to others to decide if this is important enough to change. I believe it will make it easier to follow the error handling, but it isn't critical if this is a common pattern in indexer.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh, even nicer:
Join returns an error that wraps the given errors. Any nil error values are discarded. Join returns nil if every value in errs is nil.
therefore, you don't need to test the err that is returned from sdk.DecodeAddress, just append it to your error list. Then, at the bottom, don't check to see if the list has errors in it, just:

  err := errors.Join(errList...)
  if err != nil {
    return err
  }

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

orderedTrackingTypes[elems] = key
elems++
}
sort.Slice(orderedTrackingTypes, func(i, j int) bool { return orderedTrackingTypes[i] < orderedTrackingTypes[j] })
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

slices.Sort is a more convenient, and usually faster, way to sort.

Copy link
Contributor Author

@agodnic agodnic Nov 21, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed in 82e864b.

There are more instances of the sort.Slice function being used, but I don't want to create scope creep. Can totally do this change (potentially in a separate PR) if you guys want.

Comment on lines 741 to 743
creator, errorArr := decodeAddressToBytes(params.Creator, "creator", make([]string, 0))
if len(errorArr) != 0 {
return idb.AssetsQuery{}, errors.New(errUnableToParseAddress)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As an example of what I described earlier, I think this reads much better, without needing to understand the unusual error handling model of decodeAddress.

Suggested change
creator, errorArr := decodeAddressToBytes(params.Creator, "creator", make([]string, 0))
if len(errorArr) != 0 {
return idb.AssetsQuery{}, errors.New(errUnableToParseAddress)
creator, err := sdk.DecodeAddress(params.Creator)
if err {
return idb.AssetsQuery{}, fmt.Errorf("Unable to parse creator address: %v", err)

It's worth noting that the existing code supplies the word "creator" to decodeAddressToBytes in order to get a nicer error message, and then immediately throws that nice error message away in favor of a generic message.

You can also just use creator[:] in the code below, rather than have separate functions to decode to an sdk.Address, or to []byte.

Copy link
Contributor Author

@agodnic agodnic Nov 22, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change makes a lot of sense:

  • removes unnecessary code (there are two functions that pretty much do the same thing as sdk.DecodeAddress)
  • the logic in those functions is somewhat confusing to the reader
  • handling errors as []error is preferrable over []string

Commit 7f26e0f adds more context information to the errors being returned. If this is considered a breaking change, I can make it return the same exact old error messages.

I'm not a fan of returning multiple errors, I think it is not worth the extra complexity compared to a fail-fast approach (returning upon the first error encountered). In fact this could introduce bugs, since the control flow keeps advancing with variables that may be in an unknown state (I'd rather stop the control flow immediately upon finding something that doesn't check out).
But there may be a reason why multiple errors are being used that I'm not aware of.

There are other instances in the codebase that are handling multiple errors as []string instead of []error. I didn't change all of these to avoid scope creep, but could do so in a separate PR.

@agodnic agodnic marked this pull request as ready for review November 21, 2024 23:46
Remove the functions `decodeAddress` and `decodeAddressToBytes`.

Also, add more context information to the errors being returned.
Attempt at fixing `TestTimeouts/LookupAccountTransactions`
Change function `hdrRowToBlock` signature to be in line with other
similar functions.
@@ -2677,6 +2732,39 @@
}
},
"parameters": {
"proposer": {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we update proposer to proposers to follow conventions through the rest of the API? I think absent/expired are both fine.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@gmalouf: Yes we can!

Implemented in dff12aa

@gmalouf gmalouf linked an issue Dec 4, 2024 that may be closed by this pull request
In `GET /v2/block-headers`, rename the `proposer` parameter to `proposers` to follow conventions through the rest of the API.
@@ -3242,4 +3356,4 @@
"name": "search"
}
]
}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd rather we didn't gratuitously remove final newlines.

@gmalouf gmalouf merged commit a99e6f7 into algorand:main Dec 20, 2024
5 checks passed
@agodnic agodnic deleted the implement-get-v2-blocks branch December 20, 2024 19:59
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Incentives: Add a way to lookup the blocks proposed by an address
5 participants