@lodestar/beacon-node 1.39.0-dev.aceb5b7416 → 1.39.0-dev.b37f2bd1bd

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (248) hide show
  1. package/lib/api/impl/beacon/blocks/utils.js +1 -1
  2. package/lib/api/impl/beacon/blocks/utils.js.map +1 -1
  3. package/lib/chain/chain.d.ts +4 -1
  4. package/lib/chain/chain.d.ts.map +1 -1
  5. package/lib/chain/chain.js +12 -2
  6. package/lib/chain/chain.js.map +1 -1
  7. package/lib/chain/genesis/genesis.d.ts +51 -0
  8. package/lib/chain/genesis/genesis.d.ts.map +1 -0
  9. package/lib/chain/genesis/genesis.js +123 -0
  10. package/lib/chain/genesis/genesis.js.map +1 -0
  11. package/lib/chain/genesis/interface.d.ts +13 -0
  12. package/lib/chain/genesis/interface.d.ts.map +1 -0
  13. package/lib/chain/genesis/interface.js +2 -0
  14. package/lib/chain/genesis/interface.js.map +1 -0
  15. package/lib/chain/initState.d.ts +14 -1
  16. package/lib/chain/initState.d.ts.map +1 -1
  17. package/lib/chain/initState.js +62 -1
  18. package/lib/chain/initState.js.map +1 -1
  19. package/lib/chain/interface.d.ts +2 -0
  20. package/lib/chain/interface.d.ts.map +1 -1
  21. package/lib/chain/interface.js.map +1 -1
  22. package/lib/chain/prepareNextSlot.d.ts +4 -0
  23. package/lib/chain/prepareNextSlot.d.ts.map +1 -1
  24. package/lib/chain/prepareNextSlot.js +21 -1
  25. package/lib/chain/prepareNextSlot.js.map +1 -1
  26. package/lib/chain/produceBlock/produceBlockBody.d.ts +1 -0
  27. package/lib/chain/produceBlock/produceBlockBody.d.ts.map +1 -1
  28. package/lib/chain/produceBlock/produceBlockBody.js +9 -6
  29. package/lib/chain/produceBlock/produceBlockBody.js.map +1 -1
  30. package/lib/db/beacon.d.ts +7 -1
  31. package/lib/db/beacon.d.ts.map +1 -1
  32. package/lib/db/beacon.js +12 -1
  33. package/lib/db/beacon.js.map +1 -1
  34. package/lib/db/buckets.d.ts +6 -0
  35. package/lib/db/buckets.d.ts.map +1 -1
  36. package/lib/db/buckets.js +7 -6
  37. package/lib/db/buckets.js.map +1 -1
  38. package/lib/db/interface.d.ts +7 -1
  39. package/lib/db/interface.d.ts.map +1 -1
  40. package/lib/db/repositories/blockArchiveIndex.d.ts +2 -2
  41. package/lib/db/repositories/blockArchiveIndex.d.ts.map +1 -1
  42. package/lib/db/repositories/depositDataRoot.d.ts +22 -0
  43. package/lib/db/repositories/depositDataRoot.d.ts.map +1 -0
  44. package/lib/db/repositories/depositDataRoot.js +62 -0
  45. package/lib/db/repositories/depositDataRoot.js.map +1 -0
  46. package/lib/db/repositories/depositEvent.d.ts +13 -0
  47. package/lib/db/repositories/depositEvent.d.ts.map +1 -0
  48. package/lib/db/repositories/depositEvent.js +27 -0
  49. package/lib/db/repositories/depositEvent.js.map +1 -0
  50. package/lib/db/repositories/eth1Data.d.ts +13 -0
  51. package/lib/db/repositories/eth1Data.d.ts.map +1 -0
  52. package/lib/db/repositories/eth1Data.js +26 -0
  53. package/lib/db/repositories/eth1Data.js.map +1 -0
  54. package/lib/db/repositories/index.d.ts +3 -0
  55. package/lib/db/repositories/index.d.ts.map +1 -1
  56. package/lib/db/repositories/index.js +3 -0
  57. package/lib/db/repositories/index.js.map +1 -1
  58. package/lib/db/single/index.d.ts +3 -0
  59. package/lib/db/single/index.d.ts.map +1 -0
  60. package/lib/db/single/index.js +3 -0
  61. package/lib/db/single/index.js.map +1 -0
  62. package/lib/db/single/preGenesisState.d.ts +16 -0
  63. package/lib/db/single/preGenesisState.d.ts.map +1 -0
  64. package/lib/db/single/preGenesisState.js +29 -0
  65. package/lib/db/single/preGenesisState.js.map +1 -0
  66. package/lib/db/single/preGenesisStateLastProcessedBlock.d.ts +14 -0
  67. package/lib/db/single/preGenesisStateLastProcessedBlock.d.ts.map +1 -0
  68. package/lib/db/single/preGenesisStateLastProcessedBlock.js +27 -0
  69. package/lib/db/single/preGenesisStateLastProcessedBlock.js.map +1 -0
  70. package/lib/eth1/errors.d.ts +66 -0
  71. package/lib/eth1/errors.d.ts.map +1 -0
  72. package/lib/eth1/errors.js +27 -0
  73. package/lib/eth1/errors.js.map +1 -0
  74. package/lib/eth1/eth1DataCache.d.ts +19 -0
  75. package/lib/eth1/eth1DataCache.d.ts.map +1 -0
  76. package/lib/eth1/eth1DataCache.js +19 -0
  77. package/lib/eth1/eth1DataCache.js.map +1 -0
  78. package/lib/eth1/eth1DepositDataTracker.d.ts +80 -0
  79. package/lib/eth1/eth1DepositDataTracker.d.ts.map +1 -0
  80. package/lib/eth1/eth1DepositDataTracker.js +317 -0
  81. package/lib/eth1/eth1DepositDataTracker.js.map +1 -0
  82. package/lib/eth1/eth1DepositsCache.d.ts +42 -0
  83. package/lib/eth1/eth1DepositsCache.d.ts.map +1 -0
  84. package/lib/eth1/eth1DepositsCache.js +119 -0
  85. package/lib/eth1/eth1DepositsCache.js.map +1 -0
  86. package/lib/eth1/index.d.ts +31 -0
  87. package/lib/eth1/index.d.ts.map +1 -0
  88. package/lib/eth1/index.js +71 -0
  89. package/lib/eth1/index.js.map +1 -0
  90. package/lib/eth1/interface.d.ts +74 -0
  91. package/lib/eth1/interface.d.ts.map +1 -0
  92. package/lib/eth1/interface.js +8 -0
  93. package/lib/eth1/interface.js.map +1 -0
  94. package/lib/eth1/options.d.ts +22 -0
  95. package/lib/eth1/options.d.ts.map +1 -0
  96. package/lib/eth1/options.js +8 -0
  97. package/lib/eth1/options.js.map +1 -0
  98. package/lib/eth1/provider/eth1Provider.d.ts +39 -0
  99. package/lib/eth1/provider/eth1Provider.d.ts.map +1 -0
  100. package/lib/eth1/provider/eth1Provider.js +147 -0
  101. package/lib/eth1/provider/eth1Provider.js.map +1 -0
  102. package/lib/{execution/engine → eth1/provider}/jsonRpcHttpClient.d.ts +1 -1
  103. package/lib/eth1/provider/jsonRpcHttpClient.d.ts.map +1 -0
  104. package/lib/eth1/provider/jsonRpcHttpClient.js.map +1 -0
  105. package/lib/eth1/provider/jwt.d.ts.map +1 -0
  106. package/lib/eth1/provider/jwt.js.map +1 -0
  107. package/lib/eth1/provider/utils.d.ts +65 -0
  108. package/lib/eth1/provider/utils.d.ts.map +1 -0
  109. package/lib/eth1/provider/utils.js +120 -0
  110. package/lib/eth1/provider/utils.js.map +1 -0
  111. package/lib/eth1/stream.d.ts +15 -0
  112. package/lib/eth1/stream.d.ts.map +1 -0
  113. package/lib/eth1/stream.js +54 -0
  114. package/lib/eth1/stream.js.map +1 -0
  115. package/lib/eth1/utils/depositContract.d.ts +14 -0
  116. package/lib/eth1/utils/depositContract.d.ts.map +1 -0
  117. package/lib/eth1/utils/depositContract.js +33 -0
  118. package/lib/eth1/utils/depositContract.js.map +1 -0
  119. package/lib/eth1/utils/deposits.d.ts +8 -0
  120. package/lib/eth1/utils/deposits.d.ts.map +1 -0
  121. package/lib/eth1/utils/deposits.js +47 -0
  122. package/lib/eth1/utils/deposits.js.map +1 -0
  123. package/lib/eth1/utils/eth1Data.d.ts +22 -0
  124. package/lib/eth1/utils/eth1Data.d.ts.map +1 -0
  125. package/lib/eth1/utils/eth1Data.js +77 -0
  126. package/lib/eth1/utils/eth1Data.js.map +1 -0
  127. package/lib/eth1/utils/eth1DepositEvent.d.ts +7 -0
  128. package/lib/eth1/utils/eth1DepositEvent.d.ts.map +1 -0
  129. package/lib/eth1/utils/eth1DepositEvent.js +13 -0
  130. package/lib/eth1/utils/eth1DepositEvent.js.map +1 -0
  131. package/lib/eth1/utils/eth1Vote.d.ts +17 -0
  132. package/lib/eth1/utils/eth1Vote.d.ts.map +1 -0
  133. package/lib/eth1/utils/eth1Vote.js +111 -0
  134. package/lib/eth1/utils/eth1Vote.js.map +1 -0
  135. package/lib/eth1/utils/groupDepositEventsByBlock.d.ts +9 -0
  136. package/lib/eth1/utils/groupDepositEventsByBlock.d.ts.map +1 -0
  137. package/lib/eth1/utils/groupDepositEventsByBlock.js +17 -0
  138. package/lib/eth1/utils/groupDepositEventsByBlock.js.map +1 -0
  139. package/lib/eth1/utils/optimizeNextBlockDiffForGenesis.d.ts +10 -0
  140. package/lib/eth1/utils/optimizeNextBlockDiffForGenesis.d.ts.map +1 -0
  141. package/lib/eth1/utils/optimizeNextBlockDiffForGenesis.js +14 -0
  142. package/lib/eth1/utils/optimizeNextBlockDiffForGenesis.js.map +1 -0
  143. package/lib/execution/engine/http.d.ts +1 -1
  144. package/lib/execution/engine/http.d.ts.map +1 -1
  145. package/lib/execution/engine/http.js +3 -2
  146. package/lib/execution/engine/http.js.map +1 -1
  147. package/lib/execution/engine/index.d.ts.map +1 -1
  148. package/lib/execution/engine/index.js +1 -1
  149. package/lib/execution/engine/index.js.map +1 -1
  150. package/lib/execution/engine/interface.d.ts +1 -1
  151. package/lib/execution/engine/interface.d.ts.map +1 -1
  152. package/lib/execution/engine/interface.js.map +1 -1
  153. package/lib/execution/engine/mock.d.ts.map +1 -1
  154. package/lib/execution/engine/mock.js +1 -1
  155. package/lib/execution/engine/mock.js.map +1 -1
  156. package/lib/execution/engine/payloadIdCache.d.ts +1 -1
  157. package/lib/execution/engine/payloadIdCache.d.ts.map +1 -1
  158. package/lib/execution/engine/types.d.ts +1 -1
  159. package/lib/execution/engine/types.d.ts.map +1 -1
  160. package/lib/execution/engine/types.js +1 -1
  161. package/lib/execution/engine/types.js.map +1 -1
  162. package/lib/execution/engine/utils.d.ts +2 -64
  163. package/lib/execution/engine/utils.d.ts.map +1 -1
  164. package/lib/execution/engine/utils.js +2 -91
  165. package/lib/execution/engine/utils.js.map +1 -1
  166. package/lib/index.d.ts +2 -1
  167. package/lib/index.d.ts.map +1 -1
  168. package/lib/index.js +2 -1
  169. package/lib/index.js.map +1 -1
  170. package/lib/metrics/metrics/lodestar.d.ts +35 -0
  171. package/lib/metrics/metrics/lodestar.d.ts.map +1 -1
  172. package/lib/metrics/metrics/lodestar.js +90 -0
  173. package/lib/metrics/metrics/lodestar.js.map +1 -1
  174. package/lib/node/nodejs.d.ts.map +1 -1
  175. package/lib/node/nodejs.js +9 -0
  176. package/lib/node/nodejs.js.map +1 -1
  177. package/lib/node/options.d.ts +2 -0
  178. package/lib/node/options.d.ts.map +1 -1
  179. package/lib/node/options.js +2 -0
  180. package/lib/node/options.js.map +1 -1
  181. package/lib/node/utils/interop/deposits.d.ts +1 -2
  182. package/lib/node/utils/interop/deposits.d.ts.map +1 -1
  183. package/lib/node/utils/interop/deposits.js.map +1 -1
  184. package/lib/node/utils/interop/state.d.ts +1 -1
  185. package/lib/node/utils/interop/state.d.ts.map +1 -1
  186. package/lib/node/utils/state.d.ts +7 -1
  187. package/lib/node/utils/state.d.ts.map +1 -1
  188. package/lib/node/utils/state.js +14 -1
  189. package/lib/node/utils/state.js.map +1 -1
  190. package/package.json +20 -14
  191. package/src/api/impl/beacon/blocks/utils.ts +1 -1
  192. package/src/chain/chain.ts +15 -1
  193. package/src/chain/genesis/genesis.ts +190 -0
  194. package/src/chain/genesis/interface.ts +14 -0
  195. package/src/chain/initState.ts +97 -1
  196. package/src/chain/interface.ts +2 -0
  197. package/src/chain/prepareNextSlot.ts +28 -1
  198. package/src/chain/produceBlock/produceBlockBody.ts +10 -6
  199. package/src/db/beacon.ts +15 -0
  200. package/src/db/buckets.ts +7 -6
  201. package/src/db/interface.ts +13 -0
  202. package/src/db/repositories/depositDataRoot.ts +80 -0
  203. package/src/db/repositories/depositEvent.ts +32 -0
  204. package/src/db/repositories/eth1Data.ts +33 -0
  205. package/src/db/repositories/index.ts +3 -0
  206. package/src/db/single/index.ts +2 -0
  207. package/src/db/single/preGenesisState.ts +37 -0
  208. package/src/db/single/preGenesisStateLastProcessedBlock.ts +34 -0
  209. package/src/eth1/errors.ts +40 -0
  210. package/src/eth1/eth1DataCache.ts +26 -0
  211. package/src/eth1/eth1DepositDataTracker.ts +410 -0
  212. package/src/eth1/eth1DepositsCache.ts +141 -0
  213. package/src/eth1/index.ts +94 -0
  214. package/src/eth1/interface.ts +87 -0
  215. package/src/eth1/options.ts +28 -0
  216. package/src/eth1/provider/eth1Provider.ts +229 -0
  217. package/src/{execution/engine → eth1/provider}/jsonRpcHttpClient.ts +1 -1
  218. package/src/eth1/provider/utils.ts +136 -0
  219. package/src/eth1/stream.ts +75 -0
  220. package/src/eth1/utils/depositContract.ts +37 -0
  221. package/src/eth1/utils/deposits.ts +70 -0
  222. package/src/eth1/utils/eth1Data.ts +100 -0
  223. package/src/eth1/utils/eth1DepositEvent.ts +12 -0
  224. package/src/eth1/utils/eth1Vote.ts +142 -0
  225. package/src/eth1/utils/groupDepositEventsByBlock.ts +19 -0
  226. package/src/eth1/utils/optimizeNextBlockDiffForGenesis.ts +18 -0
  227. package/src/execution/engine/http.ts +9 -8
  228. package/src/execution/engine/index.ts +1 -1
  229. package/src/execution/engine/interface.ts +1 -1
  230. package/src/execution/engine/mock.ts +2 -1
  231. package/src/execution/engine/payloadIdCache.ts +1 -1
  232. package/src/execution/engine/types.ts +9 -9
  233. package/src/execution/engine/utils.ts +5 -111
  234. package/src/index.ts +2 -1
  235. package/src/metrics/metrics/lodestar.ts +92 -0
  236. package/src/node/nodejs.ts +9 -0
  237. package/src/node/options.ts +3 -0
  238. package/src/node/utils/interop/deposits.ts +1 -3
  239. package/src/node/utils/interop/state.ts +1 -1
  240. package/src/node/utils/state.ts +18 -3
  241. package/lib/execution/engine/jsonRpcHttpClient.d.ts.map +0 -1
  242. package/lib/execution/engine/jsonRpcHttpClient.js.map +0 -1
  243. package/lib/execution/engine/jwt.d.ts.map +0 -1
  244. package/lib/execution/engine/jwt.js.map +0 -1
  245. /package/lib/{execution/engine → eth1/provider}/jsonRpcHttpClient.js +0 -0
  246. /package/lib/{execution/engine → eth1/provider}/jwt.d.ts +0 -0
  247. /package/lib/{execution/engine → eth1/provider}/jwt.js +0 -0
  248. /package/src/{execution/engine → eth1/provider}/jwt.ts +0 -0
@@ -0,0 +1,120 @@
1
+ import { bigIntToBytes, bytesToBigInt, fromHex, fromHexInto, toHex } from "@lodestar/utils";
2
+ import { ErrorParseJson } from "./jsonRpcHttpClient.js";
3
+ export const rootHexRegex = /^0x[a-fA-F0-9]{64}$/;
4
+ export function numberToHex(n) {
5
+ return "0x" + n.toString(16);
6
+ }
7
+ export function isJsonRpcTruncatedError(error) {
8
+ return (
9
+ // Truncated responses usually get as 200 but since it's truncated the JSON will be invalid
10
+ error instanceof ErrorParseJson ||
11
+ // Otherwise guess Infura error message of too many events
12
+ (error instanceof Error && error.message.includes("query returned more than 10000 results")) ||
13
+ // Nethermind enforces limits on JSON RPC batch calls
14
+ (error instanceof Error && error.message.toLowerCase().includes("batch size limit exceeded")));
15
+ }
16
+ export function bytesToHex(bytes) {
17
+ // Handle special case in Ethereum hex formating where hex values may include a single letter
18
+ // 0x0, 0x1 are valid values
19
+ if (bytes.length === 1 && bytes[0] <= 0xf) {
20
+ return "0x" + bytes[0].toString(16);
21
+ }
22
+ return toHex(bytes);
23
+ }
24
+ /**
25
+ * QUANTITY as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API
26
+ *
27
+ * When encoding QUANTITIES (integers, numbers): encode as hex, prefix with “0x”, the most compact representation (slight exception: zero should be represented as “0x0”). Examples:
28
+ * - 0x41 (65 in decimal)
29
+ * - 0x400 (1024 in decimal)
30
+ * - WRONG: 0x (should always have at least one digit - zero is “0x0”)
31
+ * - WRONG: 0x0400 (no leading zeroes allowed)
32
+ * - WRONG: ff (must be prefixed 0x)
33
+ */
34
+ export function numToQuantity(num) {
35
+ return "0x" + num.toString(16);
36
+ }
37
+ /**
38
+ * QUANTITY as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API
39
+ */
40
+ export function quantityToNum(hex, id = "") {
41
+ const num = parseInt(hex, 16);
42
+ if (Number.isNaN(num) || num < 0)
43
+ throw Error(`Invalid hex decimal ${id} '${hex}'`);
44
+ return num;
45
+ }
46
+ /**
47
+ * QUANTITY as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API.
48
+ * Typesafe fn to convert hex string to bigint. The BigInt constructor param is any
49
+ */
50
+ export function quantityToBigint(hex, id = "") {
51
+ try {
52
+ return BigInt(hex);
53
+ }
54
+ catch (e) {
55
+ throw Error(`Invalid hex bigint ${id} '${hex}': ${e.message}`);
56
+ }
57
+ }
58
+ /**
59
+ * QUANTITY as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API.
60
+ */
61
+ export function quantityToBytes(hex) {
62
+ const bn = quantityToBigint(hex);
63
+ return bigIntToBytes(bn, 32, "le");
64
+ }
65
+ /**
66
+ * QUANTITY as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API.
67
+ * Compress a 32 ByteVector into a QUANTITY
68
+ */
69
+ export function bytesToQuantity(bytes) {
70
+ const bn = bytesToBigInt(bytes, "le");
71
+ return numToQuantity(bn);
72
+ }
73
+ /**
74
+ * DATA as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API
75
+ *
76
+ * When encoding UNFORMATTED DATA (byte arrays, account addresses, hashes, bytecode arrays): encode as hex, prefix with
77
+ * “0x”, two hex digits per byte. Examples:
78
+ *
79
+ * - 0x41 (size 1, “A”)
80
+ * - 0x004200 (size 3, “\0B\0”)
81
+ * - 0x (size 0, “”)
82
+ * - WRONG: 0xf0f0f (must be even number of digits)
83
+ * - WRONG: 004200 (must be prefixed 0x)
84
+ */
85
+ export function bytesToData(bytes) {
86
+ return toHex(bytes);
87
+ }
88
+ /**
89
+ * DATA as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API
90
+ */
91
+ export function dataToBytes(hex, fixedLength) {
92
+ try {
93
+ const bytes = fromHex(hex);
94
+ if (fixedLength != null && bytes.length !== fixedLength) {
95
+ throw Error(`Wrong data length ${bytes.length} expected ${fixedLength}`);
96
+ }
97
+ return bytes;
98
+ }
99
+ catch (e) {
100
+ e.message = `Invalid hex string: ${e.message}`;
101
+ throw e;
102
+ }
103
+ }
104
+ /**
105
+ * Convert DATA into a preallocated buffer
106
+ * fromHexInto will throw if buffer's length is not the same as the decoded hex length
107
+ */
108
+ export function dataIntoBytes(hex, buffer) {
109
+ fromHexInto(hex, buffer);
110
+ return buffer;
111
+ }
112
+ /**
113
+ * DATA as defined in ethereum execution layer JSON RPC https://eth.wiki/json-rpc/API
114
+ */
115
+ export function dataToRootHex(hex, id = "") {
116
+ if (!rootHexRegex.test(hex))
117
+ throw Error(`Invalid hex root ${id} '${hex}'`);
118
+ return hex;
119
+ }
120
+ //# sourceMappingURL=utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../src/eth1/provider/utils.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ import { phase0 } from "@lodestar/types";
2
+ import { BatchDepositEvents, Eth1Block, IEth1Provider, IEth1StreamParams } from "./interface.js";
3
+ /**
4
+ * Phase 1 of genesis building.
5
+ * Not enough validators, only stream deposits
6
+ * @param signal Abort stream returning after a while loop cycle. Aborts internal sleep
7
+ */
8
+ export declare function getDepositsStream(fromBlock: number, provider: IEth1Provider, params: IEth1StreamParams, signal?: AbortSignal): AsyncGenerator<BatchDepositEvents>;
9
+ /**
10
+ * Phase 2 of genesis building.
11
+ * There are enough validators, stream deposits and blocks
12
+ * @param signal Abort stream returning after a while loop cycle. Aborts internal sleep
13
+ */
14
+ export declare function getDepositsAndBlockStreamForGenesis(fromBlock: number, provider: IEth1Provider, params: IEth1StreamParams, signal?: AbortSignal): AsyncGenerator<[phase0.DepositEvent[], Eth1Block]>;
15
+ //# sourceMappingURL=stream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream.d.ts","sourceRoot":"","sources":["../../src/eth1/stream.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,MAAM,EAAC,MAAM,iBAAiB,CAAC;AAEvC,OAAO,EAAC,kBAAkB,EAAE,SAAS,EAAE,aAAa,EAAE,iBAAiB,EAAC,MAAM,gBAAgB,CAAC;AAK/F;;;;GAIG;AACH,wBAAuB,iBAAiB,CACtC,SAAS,EAAE,MAAM,EACjB,QAAQ,EAAE,aAAa,EACvB,MAAM,EAAE,iBAAiB,EACzB,MAAM,CAAC,EAAE,WAAW,GACnB,cAAc,CAAC,kBAAkB,CAAC,CAgBpC;AAED;;;;GAIG;AACH,wBAAuB,mCAAmC,CACxD,SAAS,EAAE,MAAM,EACjB,QAAQ,EAAE,aAAa,EACvB,MAAM,EAAE,iBAAiB,EACzB,MAAM,CAAC,EAAE,WAAW,GACnB,cAAc,CAAC,CAAC,MAAM,CAAC,YAAY,EAAE,EAAE,SAAS,CAAC,CAAC,CAwBpD"}
@@ -0,0 +1,54 @@
1
+ import { sleep } from "@lodestar/utils";
2
+ import { parseEth1Block } from "./provider/eth1Provider.js";
3
+ import { groupDepositEventsByBlock } from "./utils/groupDepositEventsByBlock.js";
4
+ import { optimizeNextBlockDiffForGenesis } from "./utils/optimizeNextBlockDiffForGenesis.js";
5
+ /**
6
+ * Phase 1 of genesis building.
7
+ * Not enough validators, only stream deposits
8
+ * @param signal Abort stream returning after a while loop cycle. Aborts internal sleep
9
+ */
10
+ export async function* getDepositsStream(fromBlock, provider, params, signal) {
11
+ fromBlock = Math.max(fromBlock, provider.deployBlock);
12
+ while (true) {
13
+ const remoteFollowBlock = await getRemoteFollowBlock(provider, params);
14
+ const toBlock = Math.min(remoteFollowBlock, fromBlock + params.maxBlocksPerPoll);
15
+ const logs = await provider.getDepositEvents(fromBlock, toBlock);
16
+ for (const batchedDeposits of groupDepositEventsByBlock(logs)) {
17
+ yield batchedDeposits;
18
+ }
19
+ fromBlock = toBlock;
20
+ // If reached head, sleep for an eth1 block. Throws if signal is aborted
21
+ await sleep(toBlock >= remoteFollowBlock ? params.SECONDS_PER_ETH1_BLOCK * 1000 : 10, signal);
22
+ }
23
+ }
24
+ /**
25
+ * Phase 2 of genesis building.
26
+ * There are enough validators, stream deposits and blocks
27
+ * @param signal Abort stream returning after a while loop cycle. Aborts internal sleep
28
+ */
29
+ export async function* getDepositsAndBlockStreamForGenesis(fromBlock, provider, params, signal) {
30
+ fromBlock = Math.max(fromBlock, provider.deployBlock);
31
+ fromBlock = Math.min(fromBlock, await getRemoteFollowBlock(provider, params));
32
+ let toBlock = fromBlock; // First, fetch only the first block
33
+ while (true) {
34
+ const [logs, blockRaw] = await Promise.all([
35
+ provider.getDepositEvents(fromBlock, toBlock),
36
+ provider.getBlockByNumber(toBlock),
37
+ ]);
38
+ if (!blockRaw)
39
+ throw Error(`No block found for number ${toBlock}`);
40
+ const block = parseEth1Block(blockRaw);
41
+ yield [logs, block];
42
+ const remoteFollowBlock = await getRemoteFollowBlock(provider, params);
43
+ const nextBlockDiff = optimizeNextBlockDiffForGenesis(block, params);
44
+ fromBlock = toBlock;
45
+ toBlock = Math.min(remoteFollowBlock, fromBlock + Math.min(nextBlockDiff, params.maxBlocksPerPoll));
46
+ // If reached head, sleep for an eth1 block. Throws if signal is aborted
47
+ await sleep(toBlock >= remoteFollowBlock ? params.SECONDS_PER_ETH1_BLOCK * 1000 : 10, signal);
48
+ }
49
+ }
50
+ async function getRemoteFollowBlock(provider, params) {
51
+ const remoteHighestBlock = await provider.getBlockNumber();
52
+ return Math.max(remoteHighestBlock - params.ETH1_FOLLOW_DISTANCE, 0);
53
+ }
54
+ //# sourceMappingURL=stream.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"stream.js","sourceRoot":"","sources":["../../src/eth1/stream.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ import { phase0 } from "@lodestar/types";
2
+ /**
3
+ * Precomputed topics of DepositEvent logs
4
+ */
5
+ export declare const depositEventTopics: string[];
6
+ /**
7
+ * Parse DepositEvent log
8
+ */
9
+ export declare function parseDepositLog(log: {
10
+ blockNumber: number;
11
+ data: string;
12
+ topics: string[];
13
+ }): phase0.DepositEvent;
14
+ //# sourceMappingURL=depositContract.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"depositContract.d.ts","sourceRoot":"","sources":["../../../src/eth1/utils/depositContract.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,MAAM,EAAM,MAAM,iBAAiB,CAAC;AAQ5C;;GAEG;AACH,eAAO,MAAM,kBAAkB,UAA2D,CAAC;AAE3F;;GAEG;AACH,wBAAgB,eAAe,CAAC,GAAG,EAAE;IAAC,WAAW,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,EAAE,CAAA;CAAC,GAAG,MAAM,CAAC,YAAY,CAc/G"}
@@ -0,0 +1,33 @@
1
+ import { Interface } from "@ethersproject/abi";
2
+ import { ssz } from "@lodestar/types";
3
+ import { fromHex } from "@lodestar/utils";
4
+ const depositEventFragment = "event DepositEvent(bytes pubkey, bytes withdrawal_credentials, bytes amount, bytes signature, bytes index)";
5
+ const depositContractInterface = new Interface([depositEventFragment]);
6
+ /**
7
+ * Precomputed topics of DepositEvent logs
8
+ */
9
+ export const depositEventTopics = [depositContractInterface.getEventTopic("DepositEvent")];
10
+ /**
11
+ * Parse DepositEvent log
12
+ */
13
+ export function parseDepositLog(log) {
14
+ const event = depositContractInterface.parseLog(log);
15
+ const values = event.args;
16
+ if (values === undefined)
17
+ throw Error(`DepositEvent at ${log.blockNumber} has no values`);
18
+ return {
19
+ blockNumber: log.blockNumber,
20
+ index: parseHexNumLittleEndian(values.index),
21
+ depositData: {
22
+ pubkey: fromHex(values.pubkey),
23
+ withdrawalCredentials: fromHex(values.withdrawal_credentials),
24
+ amount: parseHexNumLittleEndian(values.amount),
25
+ signature: fromHex(values.signature),
26
+ },
27
+ };
28
+ }
29
+ function parseHexNumLittleEndian(hex) {
30
+ // Can't use parseInt() because amount is a hex string in little endian
31
+ return ssz.UintNum64.deserialize(fromHex(hex));
32
+ }
33
+ //# sourceMappingURL=depositContract.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"depositContract.js","sourceRoot":"","sources":["../../../src/eth1/utils/depositContract.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,SAAS,EAAC,MAAM,oBAAoB,CAAC;AAC7C,OAAO,EAAS,GAAG,EAAC,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAAC,OAAO,EAAC,MAAM,iBAAiB,CAAC;AAExC,MAAM,oBAAoB,GACxB,4GAA4G,CAAC;AAE/G,MAAM,wBAAwB,GAAG,IAAI,SAAS,CAAC,CAAC,oBAAoB,CAAC,CAAC,CAAC;AAEvE;;GAEG;AACH,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,wBAAwB,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC,CAAC;AAE3F;;GAEG;AACH,MAAM,UAAU,eAAe,CAAC,GAA0D;IACxF,MAAM,KAAK,GAAG,wBAAwB,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IACrD,MAAM,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC;IAC1B,IAAI,MAAM,KAAK,SAAS;QAAE,MAAM,KAAK,CAAC,mBAAmB,GAAG,CAAC,WAAW,gBAAgB,CAAC,CAAC;IAC1F,OAAO;QACL,WAAW,EAAE,GAAG,CAAC,WAAW;QAC5B,KAAK,EAAE,uBAAuB,CAAC,MAAM,CAAC,KAAK,CAAC;QAC5C,WAAW,EAAE;YACX,MAAM,EAAE,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC;YAC9B,qBAAqB,EAAE,OAAO,CAAC,MAAM,CAAC,sBAAsB,CAAC;YAC7D,MAAM,EAAE,uBAAuB,CAAC,MAAM,CAAC,MAAM,CAAC;YAC9C,SAAS,EAAE,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC;SACrC;KACF,CAAC;AACJ,CAAC;AAED,SAAS,uBAAuB,CAAC,GAAW;IAC1C,uEAAuE;IACvE,OAAO,GAAG,CAAC,SAAS,CAAC,WAAW,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC;AACjD,CAAC"}
@@ -0,0 +1,8 @@
1
+ import { FilterOptions } from "@lodestar/db";
2
+ import { CachedBeaconStateAllForks } from "@lodestar/state-transition";
3
+ import { phase0 } from "@lodestar/types";
4
+ import { DepositTree } from "../../db/repositories/depositDataRoot.js";
5
+ export type DepositGetter<T> = (indexRange: FilterOptions<number>, eth1Data: phase0.Eth1Data) => Promise<T[]>;
6
+ export declare function getDeposits<T>(state: CachedBeaconStateAllForks, eth1Data: phase0.Eth1Data, depositsGetter: DepositGetter<T>): Promise<T[]>;
7
+ export declare function getDepositsWithProofs(depositEvents: phase0.DepositEvent[], depositRootTree: DepositTree, eth1Data: phase0.Eth1Data): phase0.Deposit[];
8
+ //# sourceMappingURL=deposits.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deposits.d.ts","sourceRoot":"","sources":["../../../src/eth1/utils/deposits.ts"],"names":[],"mappings":"AACA,OAAO,EAAC,aAAa,EAAC,MAAM,cAAc,CAAC;AAC3C,OAAO,EAAC,yBAAyB,EAAsB,MAAM,4BAA4B,CAAC;AAC1F,OAAO,EAAC,MAAM,EAAM,MAAM,iBAAiB,CAAC;AAE5C,OAAO,EAAC,WAAW,EAAC,MAAM,0CAA0C,CAAC;AAGrE,MAAM,MAAM,aAAa,CAAC,CAAC,IAAI,CAAC,UAAU,EAAE,aAAa,CAAC,MAAM,CAAC,EAAE,QAAQ,EAAE,MAAM,CAAC,QAAQ,KAAK,OAAO,CAAC,CAAC,EAAE,CAAC,CAAC;AAE9G,wBAAsB,WAAW,CAAC,CAAC,EAEjC,KAAK,EAAE,yBAAyB,EAChC,QAAQ,EAAE,MAAM,CAAC,QAAQ,EACzB,cAAc,EAAE,aAAa,CAAC,CAAC,CAAC,GAC/B,OAAO,CAAC,CAAC,EAAE,CAAC,CA0Bd;AAED,wBAAgB,qBAAqB,CACnC,aAAa,EAAE,MAAM,CAAC,YAAY,EAAE,EACpC,eAAe,EAAE,WAAW,EAC5B,QAAQ,EAAE,MAAM,CAAC,QAAQ,GACxB,MAAM,CAAC,OAAO,EAAE,CAsBlB"}
@@ -0,0 +1,47 @@
1
+ import { Tree, toGindex } from "@chainsafe/persistent-merkle-tree";
2
+ import { getEth1DepositCount } from "@lodestar/state-transition";
3
+ import { ssz } from "@lodestar/types";
4
+ import { toRootHex } from "@lodestar/utils";
5
+ import { Eth1Error, Eth1ErrorCode } from "../errors.js";
6
+ export async function getDeposits(
7
+ // eth1_deposit_index represents the next deposit index to be added
8
+ state, eth1Data, depositsGetter) {
9
+ const depositIndex = state.eth1DepositIndex;
10
+ const depositCount = eth1Data.depositCount;
11
+ if (depositIndex > depositCount) {
12
+ throw new Eth1Error({ code: Eth1ErrorCode.DEPOSIT_INDEX_TOO_HIGH, depositIndex, depositCount });
13
+ }
14
+ const depositsLen = getEth1DepositCount(state, eth1Data);
15
+ if (depositsLen === 0) {
16
+ return []; // If depositsLen === 0, we can return early since no deposit with be returned from depositsGetter
17
+ }
18
+ const indexRange = { gte: depositIndex, lt: depositIndex + depositsLen };
19
+ const deposits = await depositsGetter(indexRange, eth1Data);
20
+ if (deposits.length < depositsLen) {
21
+ throw new Eth1Error({ code: Eth1ErrorCode.NOT_ENOUGH_DEPOSITS, len: deposits.length, expectedLen: depositsLen });
22
+ }
23
+ if (deposits.length > depositsLen) {
24
+ throw new Eth1Error({ code: Eth1ErrorCode.TOO_MANY_DEPOSITS, len: deposits.length, expectedLen: depositsLen });
25
+ }
26
+ return deposits;
27
+ }
28
+ export function getDepositsWithProofs(depositEvents, depositRootTree, eth1Data) {
29
+ // Get tree at this particular depositCount to compute correct proofs
30
+ const viewAtDepositCount = depositRootTree.sliceTo(eth1Data.depositCount - 1);
31
+ const depositRoot = viewAtDepositCount.hashTreeRoot();
32
+ if (!ssz.Root.equals(depositRoot, eth1Data.depositRoot)) {
33
+ throw new Eth1Error({
34
+ code: Eth1ErrorCode.WRONG_DEPOSIT_ROOT,
35
+ root: toRootHex(depositRoot),
36
+ expectedRoot: toRootHex(eth1Data.depositRoot),
37
+ });
38
+ }
39
+ // Already commited for .hashTreeRoot()
40
+ const treeAtDepositCount = new Tree(viewAtDepositCount.node);
41
+ const depositTreeDepth = viewAtDepositCount.type.depth;
42
+ return depositEvents.map((log) => ({
43
+ proof: treeAtDepositCount.getSingleProof(toGindex(depositTreeDepth, BigInt(log.index))),
44
+ data: log.depositData,
45
+ }));
46
+ }
47
+ //# sourceMappingURL=deposits.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deposits.js","sourceRoot":"","sources":["../../../src/eth1/utils/deposits.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ import { Root, phase0 } from "@lodestar/types";
2
+ import { DepositTree } from "../../db/repositories/depositDataRoot.js";
3
+ import { Eth1Block } from "../interface.js";
4
+ type BlockNumber = number;
5
+ /**
6
+ * Appends partial eth1 data (depositRoot, depositCount) in a sequence of blocks
7
+ * eth1 data deposit is inferred from sparse eth1 data obtained from the deposit logs
8
+ */
9
+ export declare function getEth1DataForBlocks(blocks: Eth1Block[], depositDescendingStream: AsyncIterable<phase0.DepositEvent>, depositRootTree: DepositTree, lastProcessedDepositBlockNumber: BlockNumber | null): Promise<(phase0.Eth1Data & Eth1Block)[]>;
10
+ /**
11
+ * Collect depositCount by blockNumber from a stream matching a block number range
12
+ * For a given blockNumber it's depositCount is equal to the index + 1 of the
13
+ * closest deposit event whose deposit.blockNumber <= blockNumber
14
+ * @returns array ascending by blockNumber
15
+ */
16
+ export declare function getDepositsByBlockNumber(fromBlock: BlockNumber, toBlock: BlockNumber, depositEventDescendingStream: AsyncIterable<phase0.DepositEvent>): Promise<phase0.DepositEvent[]>;
17
+ /**
18
+ * Precompute a map of depositCount => depositRoot from a depositRootTree filled beforehand
19
+ */
20
+ export declare function getDepositRootByDepositCount(depositCounts: number[], depositRootTree: DepositTree): Map<number, Root>;
21
+ export {};
22
+ //# sourceMappingURL=eth1Data.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1Data.d.ts","sourceRoot":"","sources":["../../../src/eth1/utils/eth1Data.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,IAAI,EAAE,MAAM,EAAC,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EAAC,WAAW,EAAC,MAAM,0CAA0C,CAAC;AAGrE,OAAO,EAAC,SAAS,EAAC,MAAM,iBAAiB,CAAC;AAE1C,KAAK,WAAW,GAAG,MAAM,CAAC;AAE1B;;;GAGG;AACH,wBAAsB,oBAAoB,CACxC,MAAM,EAAE,SAAS,EAAE,EACnB,uBAAuB,EAAE,aAAa,CAAC,MAAM,CAAC,YAAY,CAAC,EAC3D,eAAe,EAAE,WAAW,EAC5B,+BAA+B,EAAE,WAAW,GAAG,IAAI,GAClD,OAAO,CAAC,CAAC,MAAM,CAAC,QAAQ,GAAG,SAAS,CAAC,EAAE,CAAC,CAkC1C;AAED;;;;;GAKG;AACH,wBAAsB,wBAAwB,CAC5C,SAAS,EAAE,WAAW,EACtB,OAAO,EAAE,WAAW,EACpB,4BAA4B,EAAE,aAAa,CAAC,MAAM,CAAC,YAAY,CAAC,GAC/D,OAAO,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC,CAahC;AAED;;GAEG;AACH,wBAAgB,4BAA4B,CAAC,aAAa,EAAE,MAAM,EAAE,EAAE,eAAe,EAAE,WAAW,GAAG,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAkBrH"}
@@ -0,0 +1,77 @@
1
+ import { binarySearchLte } from "../../util/binarySearch.js";
2
+ import { Eth1Error, Eth1ErrorCode } from "../errors.js";
3
+ /**
4
+ * Appends partial eth1 data (depositRoot, depositCount) in a sequence of blocks
5
+ * eth1 data deposit is inferred from sparse eth1 data obtained from the deposit logs
6
+ */
7
+ export async function getEth1DataForBlocks(blocks, depositDescendingStream, depositRootTree, lastProcessedDepositBlockNumber) {
8
+ // Exclude blocks for which there is no valid eth1 data deposit
9
+ if (lastProcessedDepositBlockNumber !== null) {
10
+ blocks = blocks.filter((block) => block.blockNumber <= lastProcessedDepositBlockNumber);
11
+ }
12
+ // A valid block can be constructed using previous `state.eth1Data`, don't throw
13
+ if (blocks.length === 0) {
14
+ return [];
15
+ }
16
+ // Collect the latest deposit of each blockNumber in a block number range
17
+ const fromBlock = blocks[0].blockNumber;
18
+ const toBlock = blocks.at(-1)?.blockNumber;
19
+ const depositsByBlockNumber = await getDepositsByBlockNumber(fromBlock, toBlock, depositDescendingStream);
20
+ if (depositsByBlockNumber.length === 0) {
21
+ throw new Eth1Error({ code: Eth1ErrorCode.NO_DEPOSITS_FOR_BLOCK_RANGE, fromBlock, toBlock });
22
+ }
23
+ // Precompute a map of depositCount => depositRoot (from depositRootTree)
24
+ const depositCounts = depositsByBlockNumber.map((event) => event.index + 1);
25
+ const depositRootByDepositCount = getDepositRootByDepositCount(depositCounts, depositRootTree);
26
+ const eth1Datas = [];
27
+ for (const block of blocks) {
28
+ const deposit = binarySearchLte(depositsByBlockNumber, block.blockNumber, (event) => event.blockNumber);
29
+ const depositCount = deposit.index + 1;
30
+ const depositRoot = depositRootByDepositCount.get(depositCount);
31
+ if (depositRoot === undefined) {
32
+ throw new Eth1Error({ code: Eth1ErrorCode.NO_DEPOSIT_ROOT, depositCount });
33
+ }
34
+ eth1Datas.push({ ...block, depositCount, depositRoot });
35
+ }
36
+ return eth1Datas;
37
+ }
38
+ /**
39
+ * Collect depositCount by blockNumber from a stream matching a block number range
40
+ * For a given blockNumber it's depositCount is equal to the index + 1 of the
41
+ * closest deposit event whose deposit.blockNumber <= blockNumber
42
+ * @returns array ascending by blockNumber
43
+ */
44
+ export async function getDepositsByBlockNumber(fromBlock, toBlock, depositEventDescendingStream) {
45
+ const depositCountMap = new Map();
46
+ // Take blocks until the block under the range lower bound (included)
47
+ for await (const deposit of depositEventDescendingStream) {
48
+ if (deposit.blockNumber <= toBlock && !depositCountMap.has(deposit.blockNumber)) {
49
+ depositCountMap.set(deposit.blockNumber, deposit);
50
+ }
51
+ if (deposit.blockNumber < fromBlock) {
52
+ break;
53
+ }
54
+ }
55
+ return Array.from(depositCountMap.values()).sort((a, b) => a.blockNumber - b.blockNumber);
56
+ }
57
+ /**
58
+ * Precompute a map of depositCount => depositRoot from a depositRootTree filled beforehand
59
+ */
60
+ export function getDepositRootByDepositCount(depositCounts, depositRootTree) {
61
+ // Unique + sort numerically in descending order
62
+ depositCounts = [...new Set(depositCounts)].sort((a, b) => b - a);
63
+ if (depositCounts.length > 0) {
64
+ const maxIndex = depositCounts[0] - 1;
65
+ const treeLength = depositRootTree.length - 1;
66
+ if (maxIndex > treeLength) {
67
+ throw new Eth1Error({ code: Eth1ErrorCode.NOT_ENOUGH_DEPOSIT_ROOTS, index: maxIndex, treeLength });
68
+ }
69
+ }
70
+ const depositRootByDepositCount = new Map();
71
+ for (const depositCount of depositCounts) {
72
+ depositRootTree = depositRootTree.sliceTo(depositCount - 1);
73
+ depositRootByDepositCount.set(depositCount, depositRootTree.hashTreeRoot());
74
+ }
75
+ return depositRootByDepositCount;
76
+ }
77
+ //# sourceMappingURL=eth1Data.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1Data.js","sourceRoot":"","sources":["../../../src/eth1/utils/eth1Data.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * Assert that an array of deposits are consecutive and ascending
3
+ */
4
+ export declare function assertConsecutiveDeposits(depositEvents: {
5
+ index: number;
6
+ }[]): void;
7
+ //# sourceMappingURL=eth1DepositEvent.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1DepositEvent.d.ts","sourceRoot":"","sources":["../../../src/eth1/utils/eth1DepositEvent.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,wBAAgB,yBAAyB,CAAC,aAAa,EAAE;IAAC,KAAK,EAAE,MAAM,CAAA;CAAC,EAAE,GAAG,IAAI,CAQhF"}
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Assert that an array of deposits are consecutive and ascending
3
+ */
4
+ export function assertConsecutiveDeposits(depositEvents) {
5
+ for (let i = 0; i < depositEvents.length - 1; i++) {
6
+ const indexLeft = depositEvents[i].index;
7
+ const indexRight = depositEvents[i + 1].index;
8
+ if (indexLeft !== indexRight - 1) {
9
+ throw Error(`Non consecutive deposits. deposit[${i}] = ${indexLeft}, deposit[${i + 1}] ${indexRight}`);
10
+ }
11
+ }
12
+ }
13
+ //# sourceMappingURL=eth1DepositEvent.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1DepositEvent.js","sourceRoot":"","sources":["../../../src/eth1/utils/eth1DepositEvent.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,MAAM,UAAU,yBAAyB,CAAC,aAAgC;IACxE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;QAClD,MAAM,SAAS,GAAG,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QACzC,MAAM,UAAU,GAAG,aAAa,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC;QAC9C,IAAI,SAAS,KAAK,UAAU,GAAG,CAAC,EAAE,CAAC;YACjC,MAAM,KAAK,CAAC,qCAAqC,CAAC,OAAO,SAAS,aAAa,CAAC,GAAG,CAAC,KAAK,UAAU,EAAE,CAAC,CAAC;QACzG,CAAC;IACH,CAAC;AACH,CAAC"}
@@ -0,0 +1,17 @@
1
+ import { ChainForkConfig } from "@lodestar/config";
2
+ import { BeaconStateAllForks } from "@lodestar/state-transition";
3
+ import { phase0 } from "@lodestar/types";
4
+ export type Eth1DataGetter = ({ timestampRange, }: {
5
+ timestampRange: {
6
+ gte: number;
7
+ lte: number;
8
+ };
9
+ }) => Promise<phase0.Eth1Data[]>;
10
+ export declare function getEth1VotesToConsider(config: ChainForkConfig, state: BeaconStateAllForks, eth1DataGetter: Eth1DataGetter): Promise<phase0.Eth1Data[]>;
11
+ export declare function pickEth1Vote(state: BeaconStateAllForks, votesToConsider: phase0.Eth1Data[]): phase0.Eth1Data;
12
+ /**
13
+ * Serialize eth1Data types to a unique string ID. It is only used for comparison.
14
+ */
15
+ export declare function fastSerializeEth1Data(eth1Data: phase0.Eth1Data): string;
16
+ export declare function votingPeriodStartTime(config: ChainForkConfig, state: BeaconStateAllForks): number;
17
+ //# sourceMappingURL=eth1Vote.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1Vote.d.ts","sourceRoot":"","sources":["../../../src/eth1/utils/eth1Vote.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,eAAe,EAAC,MAAM,kBAAkB,CAAC;AAEjD,OAAO,EAAC,mBAAmB,EAAwC,MAAM,4BAA4B,CAAC;AACtG,OAAO,EAAU,MAAM,EAAC,MAAM,iBAAiB,CAAC;AAGhD,MAAM,MAAM,cAAc,GAAG,CAAC,EAC5B,cAAc,GACf,EAAE;IACD,cAAc,EAAE;QAAC,GAAG,EAAE,MAAM,CAAC;QAAC,GAAG,EAAE,MAAM,CAAA;KAAC,CAAC;CAC5C,KAAK,OAAO,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC;AAEjC,wBAAsB,sBAAsB,CAC1C,MAAM,EAAE,eAAe,EACvB,KAAK,EAAE,mBAAmB,EAC1B,cAAc,EAAE,cAAc,GAC7B,OAAO,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAyB5B;AAED,wBAAgB,YAAY,CAAC,KAAK,EAAE,mBAAmB,EAAE,eAAe,EAAE,MAAM,CAAC,QAAQ,EAAE,GAAG,MAAM,CAAC,QAAQ,CA+C5G;AAyCD;;GAEG;AACH,wBAAgB,qBAAqB,CAAC,QAAQ,EAAE,MAAM,CAAC,QAAQ,GAAG,MAAM,CAEvE;AAED,wBAAgB,qBAAqB,CAAC,MAAM,EAAE,eAAe,EAAE,KAAK,EAAE,mBAAmB,GAAG,MAAM,CAGjG"}
@@ -0,0 +1,111 @@
1
+ import { EPOCHS_PER_ETH1_VOTING_PERIOD, SLOTS_PER_EPOCH, isForkPostElectra } from "@lodestar/params";
2
+ import { computeTimeAtSlot } from "@lodestar/state-transition";
3
+ import { toRootHex } from "@lodestar/utils";
4
+ export async function getEth1VotesToConsider(config, state, eth1DataGetter) {
5
+ const fork = config.getForkName(state.slot);
6
+ if (isForkPostElectra(fork)) {
7
+ const { eth1DepositIndex, depositRequestsStartIndex } = state;
8
+ if (eth1DepositIndex === Number(depositRequestsStartIndex)) {
9
+ return state.eth1DataVotes.getAllReadonly();
10
+ }
11
+ }
12
+ const periodStart = votingPeriodStartTime(config, state);
13
+ const { SECONDS_PER_ETH1_BLOCK, ETH1_FOLLOW_DISTANCE } = config;
14
+ // Modified version of the spec function to fetch the required range directly from the DB
15
+ return (await eth1DataGetter({
16
+ timestampRange: {
17
+ // Spec v0.12.2
18
+ // is_candidate_block =
19
+ // block.timestamp + SECONDS_PER_ETH1_BLOCK * ETH1_FOLLOW_DISTANCE <= period_start &&
20
+ // block.timestamp + SECONDS_PER_ETH1_BLOCK * ETH1_FOLLOW_DISTANCE * 2 >= period_start
21
+ lte: periodStart - SECONDS_PER_ETH1_BLOCK * ETH1_FOLLOW_DISTANCE,
22
+ gte: periodStart - SECONDS_PER_ETH1_BLOCK * ETH1_FOLLOW_DISTANCE * 2,
23
+ },
24
+ })).filter((eth1Data) => eth1Data.depositCount >= state.eth1Data.depositCount);
25
+ }
26
+ export function pickEth1Vote(state, votesToConsider) {
27
+ const votesToConsiderKeys = new Set();
28
+ for (const eth1Data of votesToConsider) {
29
+ votesToConsiderKeys.add(getEth1DataKey(eth1Data));
30
+ }
31
+ const eth1DataHashToEth1Data = new Map();
32
+ const eth1DataVoteCountByRoot = new Map();
33
+ const eth1DataVotesOrder = [];
34
+ // BeaconStateAllForks is always represented as a tree with a hashing cache.
35
+ // To check equality its cheaper to use hashTreeRoot as keys.
36
+ // However `votesToConsider` is an array of values since those are read from DB.
37
+ // TODO: Optimize cache of known votes, to prevent re-hashing stored values.
38
+ // Note: for low validator counts it's not very important, since this runs once per proposal
39
+ const eth1DataVotes = state.eth1DataVotes.getAllReadonly();
40
+ for (const eth1DataVote of eth1DataVotes) {
41
+ const rootHex = getEth1DataKey(eth1DataVote);
42
+ if (votesToConsiderKeys.has(rootHex)) {
43
+ const prevVoteCount = eth1DataVoteCountByRoot.get(rootHex);
44
+ eth1DataVoteCountByRoot.set(rootHex, 1 + (prevVoteCount ?? 0));
45
+ // Cache eth1DataVote to root Map only once per root
46
+ if (prevVoteCount === undefined) {
47
+ eth1DataHashToEth1Data.set(rootHex, eth1DataVote);
48
+ eth1DataVotesOrder.push(rootHex);
49
+ }
50
+ }
51
+ }
52
+ const eth1DataRootsMaxVotes = getKeysWithMaxValue(eth1DataVoteCountByRoot);
53
+ // No votes, vote for the last valid vote
54
+ if (eth1DataRootsMaxVotes.length === 0) {
55
+ return votesToConsider.at(-1) ?? state.eth1Data;
56
+ }
57
+ // If there's a single winning vote with a majority vote that one
58
+ if (eth1DataRootsMaxVotes.length === 1) {
59
+ return eth1DataHashToEth1Data.get(eth1DataRootsMaxVotes[0]) ?? state.eth1Data;
60
+ }
61
+ // If there are multiple winning votes, vote for the latest one
62
+ const latestMostVotedRoot = eth1DataVotesOrder[Math.max(...eth1DataRootsMaxVotes.map((root) => eth1DataVotesOrder.indexOf(root)))];
63
+ return eth1DataHashToEth1Data.get(latestMostVotedRoot) ?? state.eth1Data;
64
+ }
65
+ /**
66
+ * Returns the array of keys with max value. May return 0, 1 or more keys
67
+ */
68
+ function getKeysWithMaxValue(map) {
69
+ const entries = Array.from(map.entries());
70
+ let keysMax = [];
71
+ let valueMax = -Infinity;
72
+ for (const [key, value] of entries) {
73
+ if (value > valueMax) {
74
+ keysMax = [key];
75
+ valueMax = value;
76
+ }
77
+ else if (value === valueMax) {
78
+ keysMax.push(key);
79
+ }
80
+ }
81
+ return keysMax;
82
+ }
83
+ /**
84
+ * Key-ed by fastSerializeEth1Data(). votesToConsider is read from DB as struct and always has a length of 2048.
85
+ * `state.eth1DataVotes` has a length between 0 and ETH1_FOLLOW_DISTANCE with an equal probability of each value.
86
+ * So to get the average faster time to key both votesToConsider and state.eth1DataVotes it's better to use
87
+ * fastSerializeEth1Data(). However, a long term solution is to cache valid votes in memory and prevent having
88
+ * to recompute their key on every proposal.
89
+ *
90
+ * With `fastSerializeEth1Data()`: avg time 20 ms/op
91
+ * ✓ pickEth1Vote - no votes 233.0587 ops/s 4.290764 ms/op - 121 runs 1.02 s
92
+ * ✓ pickEth1Vote - max votes 29.21546 ops/s 34.22845 ms/op - 25 runs 1.38 s
93
+ *
94
+ * With `toHexString(ssz.phase0.Eth1Data.hashTreeRoot(eth1Data))`: avg time 23 ms/op
95
+ * ✓ pickEth1Vote - no votes 46.12341 ops/s 21.68096 ms/op - 133 runs 3.40 s
96
+ * ✓ pickEth1Vote - max votes 37.89912 ops/s 26.38583 ms/op - 29 runs 1.27 s
97
+ */
98
+ function getEth1DataKey(eth1Data) {
99
+ return fastSerializeEth1Data(eth1Data);
100
+ }
101
+ /**
102
+ * Serialize eth1Data types to a unique string ID. It is only used for comparison.
103
+ */
104
+ export function fastSerializeEth1Data(eth1Data) {
105
+ return toRootHex(eth1Data.blockHash) + eth1Data.depositCount.toString(16) + toRootHex(eth1Data.depositRoot);
106
+ }
107
+ export function votingPeriodStartTime(config, state) {
108
+ const eth1VotingPeriodStartSlot = state.slot - (state.slot % (EPOCHS_PER_ETH1_VOTING_PERIOD * SLOTS_PER_EPOCH));
109
+ return computeTimeAtSlot(config, eth1VotingPeriodStartSlot, state.genesisTime);
110
+ }
111
+ //# sourceMappingURL=eth1Vote.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eth1Vote.js","sourceRoot":"","sources":["../../../src/eth1/utils/eth1Vote.ts"],"names":[],"mappings":"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"}