Skip to content

Commit

Permalink
Poll node in batch to get validator indices (#2730)
Browse files Browse the repository at this point in the history
* Query validator indices in batch

* chore: add log

* Set PUBKEYS_PER_REQUEST to 10

* Loop batches sequentially instead of Promise.all()

* Review PR

Co-authored-by: dapplion <[email protected]>
  • Loading branch information
twoeths and dapplion authored Jun 28, 2021
1 parent bbc6f32 commit 866da32
Show file tree
Hide file tree
Showing 3 changed files with 78 additions and 2 deletions.
22 changes: 20 additions & 2 deletions packages/validator/src/services/indices.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,13 @@ import {ILogger} from "@chainsafe/lodestar-utils";
import {toHexString} from "@chainsafe/ssz";
import {Api} from "@chainsafe/lodestar-api";
import {ValidatorStore} from "./validatorStore";
import {batchItems} from "../util/batch";

/**
* URLs have a limitation on size, adding an unbounded num of pubkeys will break the request.
* For reasoning on the specific number see: https://github.com/ChainSafe/lodestar/pull/2730#issuecomment-866749083
*/
const PUBKEYS_PER_REQUEST = 10;

// To assist with readability
type PubkeyHex = string;
Expand Down Expand Up @@ -58,9 +65,21 @@ export class IndicesService {
}

// Query the remote BN to resolve a pubkey to a validator index.
// support up to 1000 pubkeys per poll
const pubkeysHex = pubkeysToPoll.map((pubkey) => toHexString(pubkey));
const validatorsState = await this.api.beacon.getStateValidators("head", {indices: pubkeysHex});
const pubkeysHexBatches = batchItems(pubkeysHex, {batchSize: PUBKEYS_PER_REQUEST});

const newIndices: number[] = [];
for (const pubkeysHexBatch of pubkeysHexBatches) {
const validatorIndicesArr = await this.fetchValidatorIndices(pubkeysHexBatch);
newIndices.push(...validatorIndicesArr);
}
this.logger.info("Discovered new validators", {count: newIndices.length});
return newIndices;
}

private async fetchValidatorIndices(pubkeysHex: string[]): Promise<ValidatorIndex[]> {
const validatorsState = await this.api.beacon.getStateValidators("head", {indices: pubkeysHex});
const newIndices = [];
for (const validatorState of validatorsState.data) {
const pubkeyHex = toHexString(validatorState.validator.pubkey);
Expand All @@ -71,7 +90,6 @@ export class IndicesService {
newIndices.push(validatorState.index);
}
}

return newIndices;
}
}
16 changes: 16 additions & 0 deletions packages/validator/src/util/batch.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
/**
* Divide pubkeys into batches, each batch contains at most 5 http requests,
* each request can work on at most 40 pubkeys.
*/
export function batchItems<T>(items: T[], opts: {batchSize: number; maxBatches?: number}): T[][] {
const batches: T[][] = [];
const maxBatches = opts.maxBatches ?? Math.ceil(items.length / opts.batchSize);

for (let i = 0; i < maxBatches; i++) {
const batch = items.slice(opts.batchSize * i, opts.batchSize * (i + 1));
if (batch.length === 0) break;
batches.push(batch);
}

return batches;
}
42 changes: 42 additions & 0 deletions packages/validator/test/unit/utils/batch.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import {expect} from "chai";
import {batchItems} from "../../../src/util/batch";

describe("util / batch", function () {
const testCases: {items: string[]; expected: string[][]}[] = [
{items: [], expected: []},
{items: ["1"], expected: [["1"]]},
{items: ["1", "2"], expected: [["1", "2"]]},
{items: ["1", "2", "3"], expected: [["1", "2"], ["3"]]},
{
items: ["1", "2", "3", "4"],
expected: [
["1", "2"],
["3", "4"],
],
},
{items: ["1", "2", "3", "4", "5"], expected: [["1", "2"], ["3", "4"], ["5"]]},
{
items: ["1", "2", "3", "4", "5", "6"],
expected: [
["1", "2"],
["3", "4"],
["5", "6"],
],
},
// Ignore item 7 since it's over the max
{
items: ["1", "2", "3", "4", "5", "6", "7"],
expected: [
["1", "2"],
["3", "4"],
["5", "6"],
],
},
];

for (const {items: pubkeys, expected} of testCases) {
it(`Batch ${pubkeys.length} items`, () => {
expect(batchItems(pubkeys, {batchSize: 2, maxBatches: 3})).to.deep.equal(expected);
});
}
});

0 comments on commit 866da32

Please sign in to comment.