ponder-with-flashblocks 0.16.3

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 (614) hide show
  1. package/CHANGELOG.md +3427 -0
  2. package/README.md +186 -0
  3. package/dist/esm/bin/commands/codegen.js +46 -0
  4. package/dist/esm/bin/commands/codegen.js.map +1 -0
  5. package/dist/esm/bin/commands/createViews.js +196 -0
  6. package/dist/esm/bin/commands/createViews.js.map +1 -0
  7. package/dist/esm/bin/commands/dev.js +430 -0
  8. package/dist/esm/bin/commands/dev.js.map +1 -0
  9. package/dist/esm/bin/commands/list.js +148 -0
  10. package/dist/esm/bin/commands/list.js.map +1 -0
  11. package/dist/esm/bin/commands/prune.js +223 -0
  12. package/dist/esm/bin/commands/prune.js.map +1 -0
  13. package/dist/esm/bin/commands/serve.js +198 -0
  14. package/dist/esm/bin/commands/serve.js.map +1 -0
  15. package/dist/esm/bin/commands/start.js +253 -0
  16. package/dist/esm/bin/commands/start.js.map +1 -0
  17. package/dist/esm/bin/isolatedController.js +200 -0
  18. package/dist/esm/bin/isolatedController.js.map +1 -0
  19. package/dist/esm/bin/isolatedWorker.js +146 -0
  20. package/dist/esm/bin/isolatedWorker.js.map +1 -0
  21. package/dist/esm/bin/ponder.js +137 -0
  22. package/dist/esm/bin/ponder.js.map +1 -0
  23. package/dist/esm/bin/utils/codegen.js +25 -0
  24. package/dist/esm/bin/utils/codegen.js.map +1 -0
  25. package/dist/esm/bin/utils/exit.js +100 -0
  26. package/dist/esm/bin/utils/exit.js.map +1 -0
  27. package/dist/esm/build/config.js +746 -0
  28. package/dist/esm/build/config.js.map +1 -0
  29. package/dist/esm/build/factory.js +76 -0
  30. package/dist/esm/build/factory.js.map +1 -0
  31. package/dist/esm/build/index.js +567 -0
  32. package/dist/esm/build/index.js.map +1 -0
  33. package/dist/esm/build/plugin.js +53 -0
  34. package/dist/esm/build/plugin.js.map +1 -0
  35. package/dist/esm/build/pre.js +83 -0
  36. package/dist/esm/build/pre.js.map +1 -0
  37. package/dist/esm/build/schema.js +202 -0
  38. package/dist/esm/build/schema.js.map +1 -0
  39. package/dist/esm/build/stacktrace.js +137 -0
  40. package/dist/esm/build/stacktrace.js.map +1 -0
  41. package/dist/esm/client/index.js +441 -0
  42. package/dist/esm/client/index.js.map +1 -0
  43. package/dist/esm/config/address.js +2 -0
  44. package/dist/esm/config/address.js.map +1 -0
  45. package/dist/esm/config/eventFilter.js +2 -0
  46. package/dist/esm/config/eventFilter.js.map +1 -0
  47. package/dist/esm/config/index.js +2 -0
  48. package/dist/esm/config/index.js.map +1 -0
  49. package/dist/esm/config/utilityTypes.js +2 -0
  50. package/dist/esm/config/utilityTypes.js.map +1 -0
  51. package/dist/esm/database/actions.js +445 -0
  52. package/dist/esm/database/actions.js.map +1 -0
  53. package/dist/esm/database/index.js +604 -0
  54. package/dist/esm/database/index.js.map +1 -0
  55. package/dist/esm/database/queryBuilder.js +314 -0
  56. package/dist/esm/database/queryBuilder.js.map +1 -0
  57. package/dist/esm/drizzle/bigint.js +38 -0
  58. package/dist/esm/drizzle/bigint.js.map +1 -0
  59. package/dist/esm/drizzle/bytes.js +47 -0
  60. package/dist/esm/drizzle/bytes.js.map +1 -0
  61. package/dist/esm/drizzle/hex.js +40 -0
  62. package/dist/esm/drizzle/hex.js.map +1 -0
  63. package/dist/esm/drizzle/index.js +28 -0
  64. package/dist/esm/drizzle/index.js.map +1 -0
  65. package/dist/esm/drizzle/json.js +123 -0
  66. package/dist/esm/drizzle/json.js.map +1 -0
  67. package/dist/esm/drizzle/kit/index.js +927 -0
  68. package/dist/esm/drizzle/kit/index.js.map +1 -0
  69. package/dist/esm/drizzle/onchain.js +184 -0
  70. package/dist/esm/drizzle/onchain.js.map +1 -0
  71. package/dist/esm/drizzle/text.js +61 -0
  72. package/dist/esm/drizzle/text.js.map +1 -0
  73. package/dist/esm/graphql/graphiql.html.js +59 -0
  74. package/dist/esm/graphql/graphiql.html.js.map +1 -0
  75. package/dist/esm/graphql/index.js +934 -0
  76. package/dist/esm/graphql/index.js.map +1 -0
  77. package/dist/esm/graphql/json.js +42 -0
  78. package/dist/esm/graphql/json.js.map +1 -0
  79. package/dist/esm/graphql/middleware.js +83 -0
  80. package/dist/esm/graphql/middleware.js.map +1 -0
  81. package/dist/esm/index.js +9 -0
  82. package/dist/esm/index.js.map +1 -0
  83. package/dist/esm/indexing/addStackTrace.js +54 -0
  84. package/dist/esm/indexing/addStackTrace.js.map +1 -0
  85. package/dist/esm/indexing/client.js +675 -0
  86. package/dist/esm/indexing/client.js.map +1 -0
  87. package/dist/esm/indexing/index.js +663 -0
  88. package/dist/esm/indexing/index.js.map +1 -0
  89. package/dist/esm/indexing/profile.js +584 -0
  90. package/dist/esm/indexing/profile.js.map +1 -0
  91. package/dist/esm/indexing-store/cache.js +666 -0
  92. package/dist/esm/indexing-store/cache.js.map +1 -0
  93. package/dist/esm/indexing-store/index.js +461 -0
  94. package/dist/esm/indexing-store/index.js.map +1 -0
  95. package/dist/esm/indexing-store/profile.js +428 -0
  96. package/dist/esm/indexing-store/profile.js.map +1 -0
  97. package/dist/esm/indexing-store/utils.js +111 -0
  98. package/dist/esm/indexing-store/utils.js.map +1 -0
  99. package/dist/esm/internal/common.js +2 -0
  100. package/dist/esm/internal/common.js.map +1 -0
  101. package/dist/esm/internal/errors.js +300 -0
  102. package/dist/esm/internal/errors.js.map +1 -0
  103. package/dist/esm/internal/logger.js +178 -0
  104. package/dist/esm/internal/logger.js.map +1 -0
  105. package/dist/esm/internal/metrics.js +1049 -0
  106. package/dist/esm/internal/metrics.js.map +1 -0
  107. package/dist/esm/internal/options.js +73 -0
  108. package/dist/esm/internal/options.js.map +1 -0
  109. package/dist/esm/internal/shutdown.js +24 -0
  110. package/dist/esm/internal/shutdown.js.map +1 -0
  111. package/dist/esm/internal/telemetry.js +200 -0
  112. package/dist/esm/internal/telemetry.js.map +1 -0
  113. package/dist/esm/internal/types.js +2 -0
  114. package/dist/esm/internal/types.js.map +1 -0
  115. package/dist/esm/rpc/actions.js +988 -0
  116. package/dist/esm/rpc/actions.js.map +1 -0
  117. package/dist/esm/rpc/http.js +130 -0
  118. package/dist/esm/rpc/http.js.map +1 -0
  119. package/dist/esm/rpc/index.js +751 -0
  120. package/dist/esm/rpc/index.js.map +1 -0
  121. package/dist/esm/runtime/events.js +664 -0
  122. package/dist/esm/runtime/events.js.map +1 -0
  123. package/dist/esm/runtime/filter.js +443 -0
  124. package/dist/esm/runtime/filter.js.map +1 -0
  125. package/dist/esm/runtime/fragments.js +478 -0
  126. package/dist/esm/runtime/fragments.js.map +1 -0
  127. package/dist/esm/runtime/historical.js +950 -0
  128. package/dist/esm/runtime/historical.js.map +1 -0
  129. package/dist/esm/runtime/index.js +316 -0
  130. package/dist/esm/runtime/index.js.map +1 -0
  131. package/dist/esm/runtime/isolated.js +463 -0
  132. package/dist/esm/runtime/isolated.js.map +1 -0
  133. package/dist/esm/runtime/multichain.js +510 -0
  134. package/dist/esm/runtime/multichain.js.map +1 -0
  135. package/dist/esm/runtime/omnichain.js +545 -0
  136. package/dist/esm/runtime/omnichain.js.map +1 -0
  137. package/dist/esm/runtime/realtime.js +724 -0
  138. package/dist/esm/runtime/realtime.js.map +1 -0
  139. package/dist/esm/server/error.js +56 -0
  140. package/dist/esm/server/error.js.map +1 -0
  141. package/dist/esm/server/index.js +121 -0
  142. package/dist/esm/server/index.js.map +1 -0
  143. package/dist/esm/sync-historical/index.js +711 -0
  144. package/dist/esm/sync-historical/index.js.map +1 -0
  145. package/dist/esm/sync-realtime/bloom.js +76 -0
  146. package/dist/esm/sync-realtime/bloom.js.map +1 -0
  147. package/dist/esm/sync-realtime/index.js +928 -0
  148. package/dist/esm/sync-realtime/index.js.map +1 -0
  149. package/dist/esm/sync-store/encode.js +105 -0
  150. package/dist/esm/sync-store/encode.js.map +1 -0
  151. package/dist/esm/sync-store/index.js +885 -0
  152. package/dist/esm/sync-store/index.js.map +1 -0
  153. package/dist/esm/sync-store/migrations.js +1595 -0
  154. package/dist/esm/sync-store/migrations.js.map +1 -0
  155. package/dist/esm/sync-store/schema.js +181 -0
  156. package/dist/esm/sync-store/schema.js.map +1 -0
  157. package/dist/esm/types/db.js +2 -0
  158. package/dist/esm/types/db.js.map +1 -0
  159. package/dist/esm/types/eth.js +2 -0
  160. package/dist/esm/types/eth.js.map +1 -0
  161. package/dist/esm/types/utils.js +2 -0
  162. package/dist/esm/types/utils.js.map +1 -0
  163. package/dist/esm/types/virtual.js +2 -0
  164. package/dist/esm/types/virtual.js.map +1 -0
  165. package/dist/esm/ui/app.js +157 -0
  166. package/dist/esm/ui/app.js.map +1 -0
  167. package/dist/esm/ui/index.js +29 -0
  168. package/dist/esm/ui/index.js.map +1 -0
  169. package/dist/esm/ui/patch.js +140 -0
  170. package/dist/esm/ui/patch.js.map +1 -0
  171. package/dist/esm/utils/abi.js +55 -0
  172. package/dist/esm/utils/abi.js.map +1 -0
  173. package/dist/esm/utils/bigint.js +37 -0
  174. package/dist/esm/utils/bigint.js.map +1 -0
  175. package/dist/esm/utils/chains.js +21 -0
  176. package/dist/esm/utils/chains.js.map +1 -0
  177. package/dist/esm/utils/checkpoint.js +139 -0
  178. package/dist/esm/utils/checkpoint.js.map +1 -0
  179. package/dist/esm/utils/chunk.js +8 -0
  180. package/dist/esm/utils/chunk.js.map +1 -0
  181. package/dist/esm/utils/copy.js +129 -0
  182. package/dist/esm/utils/copy.js.map +1 -0
  183. package/dist/esm/utils/date.js +27 -0
  184. package/dist/esm/utils/date.js.map +1 -0
  185. package/dist/esm/utils/debug.js +2 -0
  186. package/dist/esm/utils/debug.js.map +1 -0
  187. package/dist/esm/utils/decodeAbiParameters.js +290 -0
  188. package/dist/esm/utils/decodeAbiParameters.js.map +1 -0
  189. package/dist/esm/utils/decodeEventLog.js +75 -0
  190. package/dist/esm/utils/decodeEventLog.js.map +1 -0
  191. package/dist/esm/utils/dedupe.js +29 -0
  192. package/dist/esm/utils/dedupe.js.map +1 -0
  193. package/dist/esm/utils/duplicates.js +19 -0
  194. package/dist/esm/utils/duplicates.js.map +1 -0
  195. package/dist/esm/utils/estimate.js +6 -0
  196. package/dist/esm/utils/estimate.js.map +1 -0
  197. package/dist/esm/utils/finality.js +38 -0
  198. package/dist/esm/utils/finality.js.map +1 -0
  199. package/dist/esm/utils/format.js +20 -0
  200. package/dist/esm/utils/format.js.map +1 -0
  201. package/dist/esm/utils/generators.js +121 -0
  202. package/dist/esm/utils/generators.js.map +1 -0
  203. package/dist/esm/utils/hash.js +11 -0
  204. package/dist/esm/utils/hash.js.map +1 -0
  205. package/dist/esm/utils/interval.js +171 -0
  206. package/dist/esm/utils/interval.js.map +1 -0
  207. package/dist/esm/utils/lowercase.js +7 -0
  208. package/dist/esm/utils/lowercase.js.map +1 -0
  209. package/dist/esm/utils/mutex.js +26 -0
  210. package/dist/esm/utils/mutex.js.map +1 -0
  211. package/dist/esm/utils/never.js +4 -0
  212. package/dist/esm/utils/never.js.map +1 -0
  213. package/dist/esm/utils/offset.js +101 -0
  214. package/dist/esm/utils/offset.js.map +1 -0
  215. package/dist/esm/utils/order.js +18 -0
  216. package/dist/esm/utils/order.js.map +1 -0
  217. package/dist/esm/utils/partition.js +46 -0
  218. package/dist/esm/utils/partition.js.map +1 -0
  219. package/dist/esm/utils/pg.js +175 -0
  220. package/dist/esm/utils/pg.js.map +1 -0
  221. package/dist/esm/utils/pglite.js +80 -0
  222. package/dist/esm/utils/pglite.js.map +1 -0
  223. package/dist/esm/utils/port.js +30 -0
  224. package/dist/esm/utils/port.js.map +1 -0
  225. package/dist/esm/utils/print.js +23 -0
  226. package/dist/esm/utils/print.js.map +1 -0
  227. package/dist/esm/utils/promiseAllSettledWithThrow.js +19 -0
  228. package/dist/esm/utils/promiseAllSettledWithThrow.js.map +1 -0
  229. package/dist/esm/utils/promiseWithResolvers.js +13 -0
  230. package/dist/esm/utils/promiseWithResolvers.js.map +1 -0
  231. package/dist/esm/utils/queue.js +150 -0
  232. package/dist/esm/utils/queue.js.map +1 -0
  233. package/dist/esm/utils/range.js +8 -0
  234. package/dist/esm/utils/range.js.map +1 -0
  235. package/dist/esm/utils/result.js +10 -0
  236. package/dist/esm/utils/result.js.map +1 -0
  237. package/dist/esm/utils/sql-parse.js +1326 -0
  238. package/dist/esm/utils/sql-parse.js.map +1 -0
  239. package/dist/esm/utils/timer.js +9 -0
  240. package/dist/esm/utils/timer.js.map +1 -0
  241. package/dist/esm/utils/truncate.js +15 -0
  242. package/dist/esm/utils/truncate.js.map +1 -0
  243. package/dist/esm/utils/wait.js +10 -0
  244. package/dist/esm/utils/wait.js.map +1 -0
  245. package/dist/esm/utils/zipper.js +67 -0
  246. package/dist/esm/utils/zipper.js.map +1 -0
  247. package/dist/types/bin/commands/codegen.d.ts +5 -0
  248. package/dist/types/bin/commands/codegen.d.ts.map +1 -0
  249. package/dist/types/bin/commands/createViews.d.ts +8 -0
  250. package/dist/types/bin/commands/createViews.d.ts.map +1 -0
  251. package/dist/types/bin/commands/dev.d.ts +5 -0
  252. package/dist/types/bin/commands/dev.d.ts.map +1 -0
  253. package/dist/types/bin/commands/list.d.ts +5 -0
  254. package/dist/types/bin/commands/list.d.ts.map +1 -0
  255. package/dist/types/bin/commands/prune.d.ts +5 -0
  256. package/dist/types/bin/commands/prune.d.ts.map +1 -0
  257. package/dist/types/bin/commands/serve.d.ts +5 -0
  258. package/dist/types/bin/commands/serve.d.ts.map +1 -0
  259. package/dist/types/bin/commands/start.d.ts +19 -0
  260. package/dist/types/bin/commands/start.d.ts.map +1 -0
  261. package/dist/types/bin/isolatedController.d.ts +13 -0
  262. package/dist/types/bin/isolatedController.d.ts.map +1 -0
  263. package/dist/types/bin/isolatedWorker.d.ts +9 -0
  264. package/dist/types/bin/isolatedWorker.d.ts.map +1 -0
  265. package/dist/types/bin/ponder.d.ts +37 -0
  266. package/dist/types/bin/ponder.d.ts.map +1 -0
  267. package/dist/types/bin/utils/codegen.d.ts +6 -0
  268. package/dist/types/bin/utils/codegen.d.ts.map +1 -0
  269. package/dist/types/bin/utils/exit.d.ts +10 -0
  270. package/dist/types/bin/utils/exit.d.ts.map +1 -0
  271. package/dist/types/build/config.d.ts +97 -0
  272. package/dist/types/build/config.d.ts.map +1 -0
  273. package/dist/types/build/factory.d.ts +13 -0
  274. package/dist/types/build/factory.d.ts.map +1 -0
  275. package/dist/types/build/index.d.ts +84 -0
  276. package/dist/types/build/index.d.ts.map +1 -0
  277. package/dist/types/build/plugin.d.ts +4 -0
  278. package/dist/types/build/plugin.d.ts.map +1 -0
  279. package/dist/types/build/pre.d.ts +29 -0
  280. package/dist/types/build/pre.d.ts.map +1 -0
  281. package/dist/types/build/schema.d.ts +20 -0
  282. package/dist/types/build/schema.d.ts.map +1 -0
  283. package/dist/types/build/stacktrace.d.ts +13 -0
  284. package/dist/types/build/stacktrace.d.ts.map +1 -0
  285. package/dist/types/client/index.d.ts +27 -0
  286. package/dist/types/client/index.d.ts.map +1 -0
  287. package/dist/types/config/address.d.ts +24 -0
  288. package/dist/types/config/address.d.ts.map +1 -0
  289. package/dist/types/config/eventFilter.d.ts +18 -0
  290. package/dist/types/config/eventFilter.d.ts.map +1 -0
  291. package/dist/types/config/index.d.ts +150 -0
  292. package/dist/types/config/index.d.ts.map +1 -0
  293. package/dist/types/config/utilityTypes.d.ts +43 -0
  294. package/dist/types/config/utilityTypes.d.ts.map +1 -0
  295. package/dist/types/database/actions.d.ts +99 -0
  296. package/dist/types/database/actions.d.ts.map +1 -0
  297. package/dist/types/database/index.d.ts +493 -0
  298. package/dist/types/database/index.d.ts.map +1 -0
  299. package/dist/types/database/queryBuilder.d.ts +65 -0
  300. package/dist/types/database/queryBuilder.d.ts.map +1 -0
  301. package/dist/types/drizzle/bigint.d.ts +25 -0
  302. package/dist/types/drizzle/bigint.d.ts.map +1 -0
  303. package/dist/types/drizzle/bytes.d.ts +31 -0
  304. package/dist/types/drizzle/bytes.d.ts.map +1 -0
  305. package/dist/types/drizzle/hex.d.ts +25 -0
  306. package/dist/types/drizzle/hex.d.ts.map +1 -0
  307. package/dist/types/drizzle/index.d.ts +6 -0
  308. package/dist/types/drizzle/index.d.ts.map +1 -0
  309. package/dist/types/drizzle/json.d.ts +51 -0
  310. package/dist/types/drizzle/json.d.ts.map +1 -0
  311. package/dist/types/drizzle/kit/index.d.ts +187 -0
  312. package/dist/types/drizzle/kit/index.d.ts.map +1 -0
  313. package/dist/types/drizzle/onchain.d.ts +298 -0
  314. package/dist/types/drizzle/onchain.d.ts.map +1 -0
  315. package/dist/types/drizzle/text.d.ts +29 -0
  316. package/dist/types/drizzle/text.d.ts.map +1 -0
  317. package/dist/types/graphql/graphiql.html.d.ts +2 -0
  318. package/dist/types/graphql/graphiql.html.d.ts.map +1 -0
  319. package/dist/types/graphql/index.d.ts +12 -0
  320. package/dist/types/graphql/index.d.ts.map +1 -0
  321. package/dist/types/graphql/json.d.ts +3 -0
  322. package/dist/types/graphql/json.d.ts.map +1 -0
  323. package/dist/types/graphql/middleware.d.ts +29 -0
  324. package/dist/types/graphql/middleware.d.ts.map +1 -0
  325. package/dist/types/index.d.ts +23 -0
  326. package/dist/types/index.d.ts.map +1 -0
  327. package/dist/types/indexing/addStackTrace.d.ts +3 -0
  328. package/dist/types/indexing/addStackTrace.d.ts.map +1 -0
  329. package/dist/types/indexing/client.d.ts +154 -0
  330. package/dist/types/indexing/client.d.ts.map +1 -0
  331. package/dist/types/indexing/index.d.ts +72 -0
  332. package/dist/types/indexing/index.d.ts.map +1 -0
  333. package/dist/types/indexing/profile.d.ts +16 -0
  334. package/dist/types/indexing/profile.d.ts.map +1 -0
  335. package/dist/types/indexing-store/cache.d.ts +115 -0
  336. package/dist/types/indexing-store/cache.d.ts.map +1 -0
  337. package/dist/types/indexing-store/index.d.ts +24 -0
  338. package/dist/types/indexing-store/index.d.ts.map +1 -0
  339. package/dist/types/indexing-store/profile.d.ts +7 -0
  340. package/dist/types/indexing-store/profile.d.ts.map +1 -0
  341. package/dist/types/indexing-store/utils.d.ts +19 -0
  342. package/dist/types/indexing-store/utils.d.ts.map +1 -0
  343. package/dist/types/internal/common.d.ts +15 -0
  344. package/dist/types/internal/common.d.ts.map +1 -0
  345. package/dist/types/internal/errors.d.ts +101 -0
  346. package/dist/types/internal/errors.d.ts.map +1 -0
  347. package/dist/types/internal/logger.d.ts +37 -0
  348. package/dist/types/internal/logger.d.ts.map +1 -0
  349. package/dist/types/internal/metrics.d.ts +120 -0
  350. package/dist/types/internal/metrics.d.ts.map +1 -0
  351. package/dist/types/internal/options.d.ts +62 -0
  352. package/dist/types/internal/options.d.ts.map +1 -0
  353. package/dist/types/internal/shutdown.d.ts +8 -0
  354. package/dist/types/internal/shutdown.d.ts.map +1 -0
  355. package/dist/types/internal/telemetry.d.ts +43 -0
  356. package/dist/types/internal/telemetry.d.ts.map +1 -0
  357. package/dist/types/internal/types.d.ts +434 -0
  358. package/dist/types/internal/types.d.ts.map +1 -0
  359. package/dist/types/rpc/actions.d.ts +360 -0
  360. package/dist/types/rpc/actions.d.ts.map +1 -0
  361. package/dist/types/rpc/http.d.ts +17 -0
  362. package/dist/types/rpc/http.d.ts.map +1 -0
  363. package/dist/types/rpc/index.d.ts +43 -0
  364. package/dist/types/rpc/index.d.ts.map +1 -0
  365. package/dist/types/runtime/events.d.ts +40 -0
  366. package/dist/types/runtime/events.d.ts.map +1 -0
  367. package/dist/types/runtime/filter.d.ts +87 -0
  368. package/dist/types/runtime/filter.d.ts.map +1 -0
  369. package/dist/types/runtime/fragments.d.ts +30 -0
  370. package/dist/types/runtime/fragments.d.ts.map +1 -0
  371. package/dist/types/runtime/historical.d.ts +123 -0
  372. package/dist/types/runtime/historical.d.ts.map +1 -0
  373. package/dist/types/runtime/index.d.ts +89 -0
  374. package/dist/types/runtime/index.d.ts.map +1 -0
  375. package/dist/types/runtime/isolated.d.ts +14 -0
  376. package/dist/types/runtime/isolated.d.ts.map +1 -0
  377. package/dist/types/runtime/multichain.d.ts +13 -0
  378. package/dist/types/runtime/multichain.d.ts.map +1 -0
  379. package/dist/types/runtime/omnichain.d.ts +23 -0
  380. package/dist/types/runtime/omnichain.d.ts.map +1 -0
  381. package/dist/types/runtime/realtime.d.ts +93 -0
  382. package/dist/types/runtime/realtime.d.ts.map +1 -0
  383. package/dist/types/server/error.d.ts +5 -0
  384. package/dist/types/server/error.d.ts.map +1 -0
  385. package/dist/types/server/index.d.ts +13 -0
  386. package/dist/types/server/index.d.ts.map +1 -0
  387. package/dist/types/sync-historical/index.d.ts +36 -0
  388. package/dist/types/sync-historical/index.d.ts.map +1 -0
  389. package/dist/types/sync-realtime/bloom.d.ts +18 -0
  390. package/dist/types/sync-realtime/bloom.d.ts.map +1 -0
  391. package/dist/types/sync-realtime/index.d.ts +47 -0
  392. package/dist/types/sync-realtime/index.d.ts.map +1 -0
  393. package/dist/types/sync-store/encode.d.ts +25 -0
  394. package/dist/types/sync-store/encode.d.ts.map +1 -0
  395. package/dist/types/sync-store/index.d.ts +135 -0
  396. package/dist/types/sync-store/index.d.ts.map +1 -0
  397. package/dist/types/sync-store/migrations.d.ts +8 -0
  398. package/dist/types/sync-store/migrations.d.ts.map +1 -0
  399. package/dist/types/sync-store/schema.d.ts +1828 -0
  400. package/dist/types/sync-store/schema.d.ts.map +1 -0
  401. package/dist/types/types/db.d.ts +213 -0
  402. package/dist/types/types/db.d.ts.map +1 -0
  403. package/dist/types/types/eth.d.ts +196 -0
  404. package/dist/types/types/eth.d.ts.map +1 -0
  405. package/dist/types/types/utils.d.ts +38 -0
  406. package/dist/types/types/utils.d.ts.map +1 -0
  407. package/dist/types/types/virtual.d.ts +99 -0
  408. package/dist/types/types/virtual.d.ts.map +1 -0
  409. package/dist/types/ui/app.d.ts +22 -0
  410. package/dist/types/ui/app.d.ts.map +1 -0
  411. package/dist/types/ui/index.d.ts +5 -0
  412. package/dist/types/ui/index.d.ts.map +1 -0
  413. package/dist/types/ui/patch.d.ts +7 -0
  414. package/dist/types/ui/patch.d.ts.map +1 -0
  415. package/dist/types/utils/abi.d.ts +23 -0
  416. package/dist/types/utils/abi.d.ts.map +1 -0
  417. package/dist/types/utils/bigint.d.ts +15 -0
  418. package/dist/types/utils/bigint.d.ts.map +1 -0
  419. package/dist/types/utils/chains.d.ts +42 -0
  420. package/dist/types/utils/chains.d.ts.map +1 -0
  421. package/dist/types/utils/checkpoint.d.ts +52 -0
  422. package/dist/types/utils/checkpoint.d.ts.map +1 -0
  423. package/dist/types/utils/chunk.d.ts +2 -0
  424. package/dist/types/utils/chunk.d.ts.map +1 -0
  425. package/dist/types/utils/copy.d.ts +16 -0
  426. package/dist/types/utils/copy.d.ts.map +1 -0
  427. package/dist/types/utils/date.d.ts +7 -0
  428. package/dist/types/utils/date.d.ts.map +1 -0
  429. package/dist/types/utils/debug.d.ts +105 -0
  430. package/dist/types/utils/debug.d.ts.map +1 -0
  431. package/dist/types/utils/decodeAbiParameters.d.ts +28 -0
  432. package/dist/types/utils/decodeAbiParameters.d.ts.map +1 -0
  433. package/dist/types/utils/decodeEventLog.d.ts +12 -0
  434. package/dist/types/utils/decodeEventLog.d.ts.map +1 -0
  435. package/dist/types/utils/dedupe.d.ts +20 -0
  436. package/dist/types/utils/dedupe.d.ts.map +1 -0
  437. package/dist/types/utils/duplicates.d.ts +7 -0
  438. package/dist/types/utils/duplicates.d.ts.map +1 -0
  439. package/dist/types/utils/estimate.d.ts +11 -0
  440. package/dist/types/utils/estimate.d.ts.map +1 -0
  441. package/dist/types/utils/finality.d.ts +12 -0
  442. package/dist/types/utils/finality.d.ts.map +1 -0
  443. package/dist/types/utils/format.d.ts +3 -0
  444. package/dist/types/utils/format.d.ts.map +1 -0
  445. package/dist/types/utils/generators.d.ts +42 -0
  446. package/dist/types/utils/generators.d.ts.map +1 -0
  447. package/dist/types/utils/hash.d.ts +11 -0
  448. package/dist/types/utils/hash.d.ts.map +1 -0
  449. package/dist/types/utils/interval.d.ts +53 -0
  450. package/dist/types/utils/interval.d.ts.map +1 -0
  451. package/dist/types/utils/lowercase.d.ts +5 -0
  452. package/dist/types/utils/lowercase.d.ts.map +1 -0
  453. package/dist/types/utils/mutex.d.ts +5 -0
  454. package/dist/types/utils/mutex.d.ts.map +1 -0
  455. package/dist/types/utils/never.d.ts +2 -0
  456. package/dist/types/utils/never.d.ts.map +1 -0
  457. package/dist/types/utils/offset.d.ts +8 -0
  458. package/dist/types/utils/offset.d.ts.map +1 -0
  459. package/dist/types/utils/order.d.ts +2 -0
  460. package/dist/types/utils/order.d.ts.map +1 -0
  461. package/dist/types/utils/partition.d.ts +22 -0
  462. package/dist/types/utils/partition.d.ts.map +1 -0
  463. package/dist/types/utils/pg.d.ts +8 -0
  464. package/dist/types/utils/pg.d.ts.map +1 -0
  465. package/dist/types/utils/pglite.d.ts +25 -0
  466. package/dist/types/utils/pglite.d.ts.map +1 -0
  467. package/dist/types/utils/port.d.ts +5 -0
  468. package/dist/types/utils/port.d.ts.map +1 -0
  469. package/dist/types/utils/print.d.ts +2 -0
  470. package/dist/types/utils/print.d.ts.map +1 -0
  471. package/dist/types/utils/promiseAllSettledWithThrow.d.ts +8 -0
  472. package/dist/types/utils/promiseAllSettledWithThrow.d.ts.map +1 -0
  473. package/dist/types/utils/promiseWithResolvers.d.ts +10 -0
  474. package/dist/types/utils/promiseWithResolvers.d.ts.map +1 -0
  475. package/dist/types/utils/queue.d.ts +33 -0
  476. package/dist/types/utils/queue.d.ts.map +1 -0
  477. package/dist/types/utils/range.d.ts +8 -0
  478. package/dist/types/utils/range.d.ts.map +1 -0
  479. package/dist/types/utils/result.d.ts +17 -0
  480. package/dist/types/utils/result.d.ts.map +1 -0
  481. package/dist/types/utils/sql-parse.d.ts +21 -0
  482. package/dist/types/utils/sql-parse.d.ts.map +1 -0
  483. package/dist/types/utils/timer.d.ts +6 -0
  484. package/dist/types/utils/timer.d.ts.map +1 -0
  485. package/dist/types/utils/truncate.d.ts +9 -0
  486. package/dist/types/utils/truncate.d.ts.map +1 -0
  487. package/dist/types/utils/wait.d.ts +6 -0
  488. package/dist/types/utils/wait.d.ts.map +1 -0
  489. package/dist/types/utils/zipper.d.ts +36 -0
  490. package/dist/types/utils/zipper.d.ts.map +1 -0
  491. package/package.json +114 -0
  492. package/src/bin/commands/codegen.ts +56 -0
  493. package/src/bin/commands/createViews.ts +318 -0
  494. package/src/bin/commands/dev.ts +490 -0
  495. package/src/bin/commands/list.ts +208 -0
  496. package/src/bin/commands/prune.ts +322 -0
  497. package/src/bin/commands/serve.ts +236 -0
  498. package/src/bin/commands/start.ts +319 -0
  499. package/src/bin/isolatedController.ts +300 -0
  500. package/src/bin/isolatedWorker.ts +192 -0
  501. package/src/bin/ponder.ts +208 -0
  502. package/src/bin/utils/codegen.ts +32 -0
  503. package/src/bin/utils/exit.ts +112 -0
  504. package/src/build/config.ts +1142 -0
  505. package/src/build/factory.ts +122 -0
  506. package/src/build/index.ts +790 -0
  507. package/src/build/plugin.ts +58 -0
  508. package/src/build/pre.ts +114 -0
  509. package/src/build/schema.ts +358 -0
  510. package/src/build/stacktrace.ts +137 -0
  511. package/src/client/index.ts +551 -0
  512. package/src/config/address.ts +32 -0
  513. package/src/config/eventFilter.ts +33 -0
  514. package/src/config/index.ts +246 -0
  515. package/src/config/utilityTypes.ts +152 -0
  516. package/src/database/actions.ts +873 -0
  517. package/src/database/index.ts +1029 -0
  518. package/src/database/queryBuilder.ts +537 -0
  519. package/src/drizzle/bigint.ts +57 -0
  520. package/src/drizzle/bytes.ts +68 -0
  521. package/src/drizzle/hex.ts +58 -0
  522. package/src/drizzle/index.ts +40 -0
  523. package/src/drizzle/json.ts +159 -0
  524. package/src/drizzle/kit/index.ts +1348 -0
  525. package/src/drizzle/onchain.ts +476 -0
  526. package/src/drizzle/text.ts +77 -0
  527. package/src/graphql/graphiql.html.ts +59 -0
  528. package/src/graphql/index.ts +1351 -0
  529. package/src/graphql/json.ts +62 -0
  530. package/src/graphql/middleware.ts +115 -0
  531. package/src/index.ts +139 -0
  532. package/src/indexing/addStackTrace.ts +69 -0
  533. package/src/indexing/client.ts +1184 -0
  534. package/src/indexing/index.ts +976 -0
  535. package/src/indexing/profile.ts +771 -0
  536. package/src/indexing-store/cache.ts +1057 -0
  537. package/src/indexing-store/index.ts +628 -0
  538. package/src/indexing-store/profile.ts +557 -0
  539. package/src/indexing-store/utils.ts +162 -0
  540. package/src/internal/common.ts +15 -0
  541. package/src/internal/errors.ts +228 -0
  542. package/src/internal/logger.ts +252 -0
  543. package/src/internal/metrics.ts +1030 -0
  544. package/src/internal/options.ts +130 -0
  545. package/src/internal/shutdown.ts +32 -0
  546. package/src/internal/telemetry.ts +303 -0
  547. package/src/internal/types.ts +597 -0
  548. package/src/rpc/actions.ts +1344 -0
  549. package/src/rpc/http.ts +164 -0
  550. package/src/rpc/index.ts +961 -0
  551. package/src/runtime/events.ts +875 -0
  552. package/src/runtime/filter.ts +664 -0
  553. package/src/runtime/fragments.ts +674 -0
  554. package/src/runtime/historical.ts +1509 -0
  555. package/src/runtime/index.ts +569 -0
  556. package/src/runtime/isolated.ts +778 -0
  557. package/src/runtime/multichain.ts +860 -0
  558. package/src/runtime/omnichain.ts +920 -0
  559. package/src/runtime/realtime.ts +1166 -0
  560. package/src/server/error.ts +68 -0
  561. package/src/server/index.ts +173 -0
  562. package/src/sync-historical/index.ts +1072 -0
  563. package/src/sync-realtime/bloom.ts +102 -0
  564. package/src/sync-realtime/index.ts +1312 -0
  565. package/src/sync-store/encode.ts +153 -0
  566. package/src/sync-store/index.ts +1633 -0
  567. package/src/sync-store/migrations.ts +1801 -0
  568. package/src/sync-store/schema.ts +248 -0
  569. package/src/types/db.ts +292 -0
  570. package/src/types/eth.ts +216 -0
  571. package/src/types/utils.ts +47 -0
  572. package/src/types/virtual.ts +244 -0
  573. package/src/types.d.ts +38 -0
  574. package/src/ui/app.ts +207 -0
  575. package/src/ui/index.ts +37 -0
  576. package/src/ui/patch.ts +204 -0
  577. package/src/utils/abi.ts +103 -0
  578. package/src/utils/bigint.ts +41 -0
  579. package/src/utils/chains.ts +22 -0
  580. package/src/utils/checkpoint.ts +203 -0
  581. package/src/utils/chunk.ts +7 -0
  582. package/src/utils/copy.ts +151 -0
  583. package/src/utils/date.ts +26 -0
  584. package/src/utils/debug.ts +110 -0
  585. package/src/utils/decodeAbiParameters.ts +428 -0
  586. package/src/utils/decodeEventLog.ts +100 -0
  587. package/src/utils/dedupe.ts +32 -0
  588. package/src/utils/duplicates.ts +19 -0
  589. package/src/utils/estimate.ts +27 -0
  590. package/src/utils/finality.ts +40 -0
  591. package/src/utils/format.ts +22 -0
  592. package/src/utils/generators.ts +157 -0
  593. package/src/utils/hash.ts +22 -0
  594. package/src/utils/interval.ts +212 -0
  595. package/src/utils/lowercase.ts +6 -0
  596. package/src/utils/mutex.ts +33 -0
  597. package/src/utils/never.ts +3 -0
  598. package/src/utils/offset.ts +133 -0
  599. package/src/utils/order.ts +16 -0
  600. package/src/utils/partition.ts +53 -0
  601. package/src/utils/pg.ts +230 -0
  602. package/src/utils/pglite.ts +97 -0
  603. package/src/utils/port.ts +34 -0
  604. package/src/utils/print.ts +31 -0
  605. package/src/utils/promiseAllSettledWithThrow.ts +27 -0
  606. package/src/utils/promiseWithResolvers.ts +20 -0
  607. package/src/utils/queue.ts +258 -0
  608. package/src/utils/range.ts +8 -0
  609. package/src/utils/result.ts +26 -0
  610. package/src/utils/sql-parse.ts +1477 -0
  611. package/src/utils/timer.ts +8 -0
  612. package/src/utils/truncate.ts +15 -0
  613. package/src/utils/wait.ts +8 -0
  614. package/src/utils/zipper.ts +80 -0
@@ -0,0 +1,1509 @@
1
+ import type { Database } from "@/database/index.js";
2
+ import type { Common } from "@/internal/common.js";
3
+ import { ShutdownError } from "@/internal/errors.js";
4
+ import type {
5
+ Chain,
6
+ CrashRecoveryCheckpoint,
7
+ Event,
8
+ EventCallback,
9
+ IndexingBuild,
10
+ RawEvent,
11
+ SyncBlock,
12
+ } from "@/internal/types.js";
13
+ import { eth_getBlockByNumber } from "@/rpc/actions.js";
14
+ import type { Rpc } from "@/rpc/index.js";
15
+ import { buildEvents, decodeEvents } from "@/runtime/events.js";
16
+ import { createHistoricalSync } from "@/sync-historical/index.js";
17
+ import { type SyncStore, createSyncStore } from "@/sync-store/index.js";
18
+ import {
19
+ MAX_CHECKPOINT,
20
+ ZERO_CHECKPOINT,
21
+ decodeCheckpoint,
22
+ encodeCheckpoint,
23
+ min,
24
+ } from "@/utils/checkpoint.js";
25
+ import { estimate } from "@/utils/estimate.js";
26
+ import { formatPercentage } from "@/utils/format.js";
27
+ import {
28
+ bufferAsyncGenerator,
29
+ createCallbackGenerator,
30
+ mergeAsyncGenerators,
31
+ } from "@/utils/generators.js";
32
+ import { type Interval, intervalSum } from "@/utils/interval.js";
33
+ import { partition } from "@/utils/partition.js";
34
+ import { promiseWithResolvers } from "@/utils/promiseWithResolvers.js";
35
+ import { startClock } from "@/utils/timer.js";
36
+ import { hexToNumber, numberToHex } from "viem";
37
+ import {
38
+ type CachedIntervals,
39
+ type ChildAddresses,
40
+ type SyncProgress,
41
+ getRequiredIntervals,
42
+ getRequiredIntervalsWithFilters,
43
+ } from "./index.js";
44
+ import { getOmnichainCheckpoint } from "./omnichain.js";
45
+
46
+ export async function* getHistoricalEventsOmnichain(params: {
47
+ common: Common;
48
+ indexingBuild: Pick<IndexingBuild, "eventCallbacks" | "chains" | "rpcs">;
49
+ crashRecoveryCheckpoint: CrashRecoveryCheckpoint;
50
+ perChainSync: Map<
51
+ Chain,
52
+ {
53
+ syncProgress: SyncProgress;
54
+ childAddresses: ChildAddresses;
55
+ cachedIntervals: CachedIntervals;
56
+ }
57
+ >;
58
+ database: Database;
59
+ }): AsyncGenerator<
60
+ | {
61
+ type: "events";
62
+ result: {
63
+ chainId: number;
64
+ events: Event[];
65
+ checkpoint: string;
66
+ blockRange: [number, number];
67
+ }[];
68
+ }
69
+ | { type: "pending"; result: Event[] }
70
+ > {
71
+ let pendingEvents: Event[] = [];
72
+ let isCatchup = false;
73
+ const perChainCursor = new Map<Chain, string>();
74
+
75
+ while (true) {
76
+ const eventGenerators = Array.from(params.perChainSync.entries()).map(
77
+ async function* ([
78
+ chain,
79
+ { syncProgress, childAddresses, cachedIntervals },
80
+ ]) {
81
+ const rpc =
82
+ params.indexingBuild.rpcs[
83
+ params.indexingBuild.chains.findIndex((c) => c.id === chain.id)
84
+ ]!;
85
+
86
+ const eventCallbacks =
87
+ params.indexingBuild.eventCallbacks[
88
+ params.indexingBuild.chains.findIndex((c) => c.id === chain.id)
89
+ ]!;
90
+
91
+ const crashRecoveryCheckpoint = params.crashRecoveryCheckpoint?.find(
92
+ ({ chainId }) => chainId === chain.id,
93
+ )?.checkpoint;
94
+
95
+ const to = min(
96
+ syncProgress.getCheckpoint({ tag: "finalized" }),
97
+ syncProgress.getCheckpoint({ tag: "end" }),
98
+ );
99
+ const omnichainTo = min(
100
+ getOmnichainCheckpoint({
101
+ perChainSync: params.perChainSync,
102
+ tag: "finalized",
103
+ }),
104
+ getOmnichainCheckpoint({
105
+ perChainSync: params.perChainSync,
106
+ tag: "end",
107
+ }),
108
+ );
109
+ let from: string;
110
+
111
+ if (isCatchup === false) {
112
+ // In order to speed up the "extract" phase when there is a crash recovery,
113
+ // the beginning cursor is moved forwards. This only works when `crashRecoveryCheckpoint`
114
+ // is defined.
115
+
116
+ if (crashRecoveryCheckpoint === undefined) {
117
+ from = syncProgress.getCheckpoint({ tag: "start" });
118
+ } else if (
119
+ Number(decodeCheckpoint(crashRecoveryCheckpoint).chainId) ===
120
+ chain.id
121
+ ) {
122
+ from = crashRecoveryCheckpoint;
123
+ } else {
124
+ const fromBlock = await createSyncStore({
125
+ common: params.common,
126
+ qb: params.database.syncQB,
127
+ }).getSafeCrashRecoveryBlock({
128
+ chainId: chain.id,
129
+ timestamp: Number(
130
+ decodeCheckpoint(crashRecoveryCheckpoint).blockTimestamp,
131
+ ),
132
+ });
133
+
134
+ if (fromBlock === undefined) {
135
+ from = syncProgress.getCheckpoint({ tag: "start" });
136
+ } else {
137
+ from = encodeCheckpoint({
138
+ ...ZERO_CHECKPOINT,
139
+ blockNumber: fromBlock.number,
140
+ blockTimestamp: fromBlock.timestamp,
141
+ chainId: BigInt(chain.id),
142
+ });
143
+ }
144
+ }
145
+ } else {
146
+ // Previous iterations `to` value
147
+ const cursor = perChainCursor.get(chain)!;
148
+
149
+ // Yield pending events from previous iterations. Note that it is possible for
150
+ // previous pending events to still be pending after the catchup.
151
+
152
+ const events = pendingEvents.filter(
153
+ (event) =>
154
+ event.chain.id === chain.id && event.checkpoint <= omnichainTo,
155
+ );
156
+
157
+ pendingEvents = pendingEvents.filter(
158
+ (event) =>
159
+ (event.chain.id === chain.id &&
160
+ event.checkpoint <= omnichainTo) === false,
161
+ );
162
+
163
+ if (events.length > 0) {
164
+ if (omnichainTo >= cursor) {
165
+ const blockRange = [
166
+ Number(decodeCheckpoint(events[0]!.checkpoint).blockNumber),
167
+ Number(decodeCheckpoint(cursor).blockNumber),
168
+ ] satisfies [number, number];
169
+
170
+ yield { events, checkpoint: cursor, blockRange };
171
+ } else {
172
+ const checkpoint = events[events.length - 1]!.checkpoint;
173
+
174
+ const blockRange = [
175
+ Number(decodeCheckpoint(events[0]!.checkpoint).blockNumber),
176
+ Number(decodeCheckpoint(checkpoint).blockNumber),
177
+ ] satisfies [number, number];
178
+
179
+ yield { events, checkpoint, blockRange };
180
+ }
181
+ }
182
+
183
+ from = encodeCheckpoint({
184
+ ...ZERO_CHECKPOINT,
185
+ blockTimestamp: decodeCheckpoint(cursor).blockTimestamp,
186
+ chainId: decodeCheckpoint(cursor).chainId,
187
+ blockNumber: decodeCheckpoint(cursor).blockNumber + 1n,
188
+ });
189
+
190
+ if (from > to) return;
191
+ }
192
+
193
+ params.common.logger.info({
194
+ msg: "Started backfill indexing",
195
+ chain: chain.name,
196
+ chain_id: chain.id,
197
+ block_range: JSON.stringify([
198
+ Number(decodeCheckpoint(from).blockNumber),
199
+ Number(decodeCheckpoint(to).blockNumber),
200
+ ]),
201
+ });
202
+
203
+ const eventGenerator = getLocalEventGenerator({
204
+ common: params.common,
205
+ chain,
206
+ rpc,
207
+ eventCallbacks,
208
+ childAddresses,
209
+ syncProgress,
210
+ cachedIntervals,
211
+ from,
212
+ to,
213
+ limit:
214
+ Math.round(
215
+ params.common.options.syncEventsQuerySize /
216
+ (params.indexingBuild.chains.length + 1),
217
+ ) + 6,
218
+ database: params.database,
219
+ isCatchup,
220
+ });
221
+
222
+ for await (let {
223
+ events: rawEvents,
224
+ checkpoint,
225
+ blockRange,
226
+ } of eventGenerator) {
227
+ const endClock = startClock();
228
+
229
+ let events = decodeEvents(
230
+ params.common,
231
+ chain,
232
+ eventCallbacks,
233
+ rawEvents,
234
+ );
235
+
236
+ params.common.logger.trace({
237
+ msg: "Decoded events",
238
+ chain: chain.name,
239
+ chain_id: chain.id,
240
+ event_count: events.length,
241
+ duration: endClock(),
242
+ });
243
+
244
+ params.common.metrics.ponder_historical_extract_duration.inc(
245
+ { step: "decode" },
246
+ endClock(),
247
+ );
248
+
249
+ // Removes events that have a checkpoint earlier than (or equal to)
250
+ // the crash recovery checkpoint.
251
+
252
+ if (crashRecoveryCheckpoint) {
253
+ const [left, right] = partition(
254
+ events,
255
+ (event) => event.checkpoint <= crashRecoveryCheckpoint,
256
+ );
257
+ events = right;
258
+
259
+ if (left.length > 0) {
260
+ params.common.logger.trace({
261
+ msg: "Filtered events before crash recovery checkpoint",
262
+ chain: chain.name,
263
+ chain_id: chain.id,
264
+ event_count: left.length,
265
+ checkpoint: crashRecoveryCheckpoint,
266
+ });
267
+ }
268
+ }
269
+
270
+ // Sort out any events between the omnichain finalized checkpoint and the single-chain
271
+ // finalized checkpoint and add them to pendingEvents. These events are synced during
272
+ // the historical phase, but must be indexed in the realtime phase because events
273
+ // synced in realtime on other chains might be ordered before them.
274
+
275
+ if (checkpoint > omnichainTo) {
276
+ const [left, right] = partition(
277
+ events,
278
+ (event) => event.checkpoint <= omnichainTo,
279
+ );
280
+ events = left;
281
+ pendingEvents = pendingEvents.concat(right);
282
+
283
+ params.common.logger.trace({
284
+ msg: "Filtered pending events",
285
+ chain: chain.name,
286
+ chain_id: chain.id,
287
+ event_count: right.length,
288
+ checkpoint: omnichainTo,
289
+ });
290
+
291
+ if (left.length > 0) {
292
+ checkpoint = left[left.length - 1]!.checkpoint;
293
+ blockRange[1] = Number(
294
+ decodeCheckpoint(left[left.length - 1]!.checkpoint).blockNumber,
295
+ );
296
+
297
+ yield { events, checkpoint, blockRange };
298
+ }
299
+ } else {
300
+ yield { events, checkpoint, blockRange };
301
+ }
302
+ }
303
+
304
+ perChainCursor.set(chain, to);
305
+ },
306
+ );
307
+
308
+ const eventGenerator = mergeAsyncGeneratorsWithEventOrder(eventGenerators);
309
+
310
+ for await (const mergeResults of eventGenerator) {
311
+ yield { type: "events", result: mergeResults };
312
+ }
313
+
314
+ const context = {
315
+ logger: params.common.logger.child({ action: "refetch_finalized_block" }),
316
+ retryNullBlockRequest: true,
317
+ };
318
+
319
+ const endClock = startClock();
320
+
321
+ const finalizedBlocks = await Promise.all(
322
+ params.indexingBuild.chains.map((chain, i) => {
323
+ const rpc = params.indexingBuild.rpcs[i]!;
324
+
325
+ return eth_getBlockByNumber(rpc, ["latest", false], context)
326
+ .then((latest) =>
327
+ eth_getBlockByNumber(
328
+ rpc,
329
+ [
330
+ numberToHex(
331
+ Math.max(
332
+ hexToNumber(latest.number) - chain.finalityBlockCount,
333
+ 0,
334
+ ),
335
+ ),
336
+ false,
337
+ ],
338
+ context,
339
+ ),
340
+ )
341
+ .then((finalizedBlock) => {
342
+ const finalizedBlockNumber = hexToNumber(finalizedBlock.number);
343
+ params.common.logger.debug({
344
+ msg: "Refetched finalized block for backfill cutover",
345
+ chain: chain.name,
346
+ chain_id: chain.id,
347
+ finalized_block: finalizedBlockNumber,
348
+ duration: endClock(),
349
+ });
350
+
351
+ return finalizedBlock;
352
+ });
353
+ }),
354
+ );
355
+
356
+ let shouldCatchup = false;
357
+
358
+ for (let i = 0; i < params.indexingBuild.chains.length; i++) {
359
+ const chain = params.indexingBuild.chains[i]!;
360
+ const oldFinalizedBlock =
361
+ params.perChainSync.get(chain)!.syncProgress.finalized;
362
+ const newFinalizedBlock = finalizedBlocks[i]!;
363
+
364
+ if (
365
+ hexToNumber(newFinalizedBlock.number) -
366
+ hexToNumber(oldFinalizedBlock.number) >
367
+ chain.finalityBlockCount
368
+ ) {
369
+ shouldCatchup = true;
370
+ break;
371
+ }
372
+ }
373
+
374
+ if (shouldCatchup === false) break;
375
+
376
+ for (let i = 0; i < params.indexingBuild.chains.length; i++) {
377
+ const chain = params.indexingBuild.chains[i]!;
378
+ const finalizedBlock = finalizedBlocks[i]!;
379
+
380
+ params.perChainSync.get(chain)!.syncProgress.finalized = finalizedBlock;
381
+ }
382
+
383
+ isCatchup = true;
384
+ }
385
+
386
+ yield { type: "pending", result: pendingEvents };
387
+ }
388
+
389
+ export async function* getHistoricalEventsMultichain(params: {
390
+ common: Common;
391
+ indexingBuild: Pick<IndexingBuild, "eventCallbacks" | "chains" | "rpcs">;
392
+ crashRecoveryCheckpoint: CrashRecoveryCheckpoint;
393
+ perChainSync: Map<
394
+ Chain,
395
+ {
396
+ syncProgress: SyncProgress;
397
+ childAddresses: ChildAddresses;
398
+ cachedIntervals: CachedIntervals;
399
+ }
400
+ >;
401
+ database: Database;
402
+ }) {
403
+ let isCatchup = false;
404
+ let lastUnfinalizedRefetch = Date.now();
405
+ const perChainCursor = new Map<Chain, string>();
406
+
407
+ while (true) {
408
+ const eventGenerators = Array.from(params.perChainSync.entries()).map(
409
+ async function* ([
410
+ chain,
411
+ { syncProgress, childAddresses, cachedIntervals },
412
+ ]) {
413
+ const rpc =
414
+ params.indexingBuild.rpcs[
415
+ params.indexingBuild.chains.findIndex((c) => c.id === chain.id)
416
+ ]!;
417
+
418
+ const eventCallbacks =
419
+ params.indexingBuild.eventCallbacks[
420
+ params.indexingBuild.chains.findIndex((c) => c.id === chain.id)
421
+ ]!;
422
+
423
+ const crashRecoveryCheckpoint = params.crashRecoveryCheckpoint?.find(
424
+ ({ chainId }) => chainId === chain.id,
425
+ )?.checkpoint;
426
+
427
+ const to = min(
428
+ syncProgress.getCheckpoint({ tag: "finalized" }),
429
+ syncProgress.getCheckpoint({ tag: "end" }),
430
+ );
431
+ let from: string;
432
+
433
+ if (isCatchup === false) {
434
+ // In order to speed up the "extract" phase when there is a crash recovery,
435
+ // the beginning cursor is moved forwards. This only works when `crashRecoveryCheckpoint`
436
+ // is defined.
437
+
438
+ if (crashRecoveryCheckpoint === undefined) {
439
+ from = syncProgress.getCheckpoint({ tag: "start" });
440
+ } else if (
441
+ Number(decodeCheckpoint(crashRecoveryCheckpoint).chainId) ===
442
+ chain.id
443
+ ) {
444
+ from = crashRecoveryCheckpoint;
445
+ } else {
446
+ const fromBlock = await createSyncStore({
447
+ common: params.common,
448
+ qb: params.database.syncQB,
449
+ }).getSafeCrashRecoveryBlock({
450
+ chainId: chain.id,
451
+ timestamp: Number(
452
+ decodeCheckpoint(crashRecoveryCheckpoint).blockTimestamp,
453
+ ),
454
+ });
455
+
456
+ if (fromBlock === undefined) {
457
+ from = syncProgress.getCheckpoint({ tag: "start" });
458
+ } else {
459
+ from = encodeCheckpoint({
460
+ ...ZERO_CHECKPOINT,
461
+ blockNumber: fromBlock.number,
462
+ blockTimestamp: fromBlock.timestamp,
463
+ chainId: BigInt(chain.id),
464
+ });
465
+ }
466
+ }
467
+ } else {
468
+ const cursor = perChainCursor.get(chain)!;
469
+
470
+ from = encodeCheckpoint({
471
+ ...ZERO_CHECKPOINT,
472
+ blockTimestamp: decodeCheckpoint(cursor).blockTimestamp,
473
+ chainId: decodeCheckpoint(cursor).chainId,
474
+ blockNumber: decodeCheckpoint(cursor).blockNumber + 1n,
475
+ });
476
+
477
+ if (from > to) return;
478
+ }
479
+
480
+ params.common.logger.info({
481
+ msg: "Started backfill indexing",
482
+ chain: chain.name,
483
+ chain_id: chain.id,
484
+ block_range: JSON.stringify([
485
+ Number(decodeCheckpoint(from).blockNumber),
486
+ Number(decodeCheckpoint(to).blockNumber),
487
+ ]),
488
+ });
489
+
490
+ const eventGenerator = getLocalEventGenerator({
491
+ common: params.common,
492
+ chain,
493
+ rpc,
494
+ eventCallbacks,
495
+ childAddresses,
496
+ syncProgress,
497
+ cachedIntervals,
498
+ from,
499
+ to,
500
+ limit:
501
+ Math.round(
502
+ params.common.options.syncEventsQuerySize /
503
+ (params.indexingBuild.chains.length + 1),
504
+ ) + 6,
505
+ database: params.database,
506
+ isCatchup,
507
+ });
508
+
509
+ for await (const {
510
+ events: rawEvents,
511
+ checkpoint,
512
+ blockRange,
513
+ } of eventGenerator) {
514
+ const endClock = startClock();
515
+
516
+ let events = decodeEvents(
517
+ params.common,
518
+ chain,
519
+ eventCallbacks,
520
+ rawEvents,
521
+ );
522
+
523
+ params.common.logger.trace({
524
+ msg: "Decoded events",
525
+ chain: chain.name,
526
+ chain_id: chain.id,
527
+ event_count: events.length,
528
+ duration: endClock(),
529
+ });
530
+
531
+ params.common.metrics.ponder_historical_extract_duration.inc(
532
+ { step: "decode" },
533
+ endClock(),
534
+ );
535
+
536
+ // Removes events that have a checkpoint earlier than (or equal to)
537
+ // the crash recovery checkpoint.
538
+
539
+ if (crashRecoveryCheckpoint) {
540
+ const [left, right] = partition(
541
+ events,
542
+ (event) => event.checkpoint <= crashRecoveryCheckpoint,
543
+ );
544
+ events = right;
545
+
546
+ if (left.length > 0) {
547
+ params.common.logger.trace({
548
+ msg: "Filtered events before crash recovery checkpoint",
549
+ chain: chain.name,
550
+ chain_id: chain.id,
551
+ event_count: left.length,
552
+ checkpoint: crashRecoveryCheckpoint,
553
+ });
554
+ }
555
+ }
556
+
557
+ yield { chainId: chain.id, events, checkpoint, blockRange };
558
+ }
559
+
560
+ perChainCursor.set(chain, to);
561
+ },
562
+ );
563
+
564
+ yield* mergeAsyncGenerators(eventGenerators);
565
+
566
+ if (Date.now() - lastUnfinalizedRefetch < 30_000) {
567
+ break;
568
+ }
569
+ lastUnfinalizedRefetch = Date.now();
570
+
571
+ const context = {
572
+ logger: params.common.logger.child({ action: "refetch_finalized_block" }),
573
+ retryNullBlockRequest: true,
574
+ };
575
+
576
+ const endClock = startClock();
577
+
578
+ const finalizedBlocks = await Promise.all(
579
+ params.indexingBuild.chains.map((chain, i) => {
580
+ const rpc = params.indexingBuild.rpcs[i]!;
581
+
582
+ return eth_getBlockByNumber(rpc, ["latest", false], context)
583
+ .then((latest) =>
584
+ eth_getBlockByNumber(
585
+ rpc,
586
+ [
587
+ numberToHex(
588
+ Math.max(
589
+ hexToNumber(latest.number) - chain.finalityBlockCount,
590
+ 0,
591
+ ),
592
+ ),
593
+ false,
594
+ ],
595
+ context,
596
+ ),
597
+ )
598
+ .then((finalizedBlock) => {
599
+ const finalizedBlockNumber = hexToNumber(finalizedBlock.number);
600
+ params.common.logger.debug({
601
+ msg: "Refetched finalized block for backfill cutover",
602
+ chain: chain.name,
603
+ chain_id: chain.id,
604
+ finalized_block: finalizedBlockNumber,
605
+ duration: endClock(),
606
+ });
607
+
608
+ return finalizedBlock;
609
+ });
610
+ }),
611
+ );
612
+
613
+ let shouldCatchup = false;
614
+
615
+ for (let i = 0; i < params.indexingBuild.chains.length; i++) {
616
+ const chain = params.indexingBuild.chains[i]!;
617
+ const oldFinalizedBlock =
618
+ params.perChainSync.get(chain)!.syncProgress.finalized;
619
+ const newFinalizedBlock = finalizedBlocks[i]!;
620
+
621
+ if (
622
+ hexToNumber(newFinalizedBlock.number) -
623
+ hexToNumber(oldFinalizedBlock.number) >
624
+ chain.finalityBlockCount
625
+ ) {
626
+ shouldCatchup = true;
627
+ break;
628
+ }
629
+ }
630
+
631
+ if (shouldCatchup === false) break;
632
+
633
+ for (let i = 0; i < params.indexingBuild.chains.length; i++) {
634
+ const chain = params.indexingBuild.chains[i]!;
635
+ const finalizedBlock = finalizedBlocks[i]!;
636
+
637
+ params.perChainSync.get(chain)!.syncProgress.finalized = finalizedBlock;
638
+ }
639
+
640
+ isCatchup = true;
641
+ }
642
+ }
643
+
644
+ export async function* getHistoricalEventsIsolated(params: {
645
+ common: Common;
646
+ indexingBuild: Pick<IndexingBuild, "eventCallbacks" | "chains" | "rpcs">;
647
+ crashRecoveryCheckpoint: CrashRecoveryCheckpoint;
648
+ chain: Chain;
649
+ syncProgress: SyncProgress;
650
+ childAddresses: ChildAddresses;
651
+ cachedIntervals: CachedIntervals;
652
+ database: Database;
653
+ }) {
654
+ let isCatchup = false;
655
+ let lastUnfinalizedRefetch = Date.now();
656
+ let cursor: string | undefined;
657
+
658
+ while (true) {
659
+ const rpc =
660
+ params.indexingBuild.rpcs[
661
+ params.indexingBuild.chains.findIndex((c) => c.id === params.chain.id)
662
+ ]!;
663
+
664
+ const eventCallbacks =
665
+ params.indexingBuild.eventCallbacks[
666
+ params.indexingBuild.chains.findIndex((c) => c.id === params.chain.id)
667
+ ]!;
668
+
669
+ const crashRecoveryCheckpoint = params.crashRecoveryCheckpoint?.find(
670
+ ({ chainId }) => chainId === params.chain.id,
671
+ )?.checkpoint;
672
+
673
+ const to = min(
674
+ params.syncProgress.getCheckpoint({ tag: "finalized" }),
675
+ params.syncProgress.getCheckpoint({ tag: "end" }),
676
+ );
677
+ let from: string;
678
+
679
+ if (isCatchup === false) {
680
+ // In order to speed up the "extract" phase when there is a crash recovery,
681
+ // the beginning cursor is moved forwards. This only works when `crashRecoveryCheckpoint`
682
+ // is defined.
683
+
684
+ if (crashRecoveryCheckpoint === undefined) {
685
+ from = params.syncProgress.getCheckpoint({ tag: "start" });
686
+ } else if (
687
+ Number(decodeCheckpoint(crashRecoveryCheckpoint).chainId) ===
688
+ params.chain.id
689
+ ) {
690
+ from = crashRecoveryCheckpoint;
691
+ } else {
692
+ from = params.syncProgress.getCheckpoint({ tag: "start" });
693
+ }
694
+ } else {
695
+ from = encodeCheckpoint({
696
+ ...ZERO_CHECKPOINT,
697
+ blockTimestamp: decodeCheckpoint(cursor!).blockTimestamp,
698
+ chainId: decodeCheckpoint(cursor!).chainId,
699
+ blockNumber: decodeCheckpoint(cursor!).blockNumber + 1n,
700
+ });
701
+
702
+ if (from > to) return;
703
+ }
704
+
705
+ params.common.logger.info({
706
+ msg: "Started backfill indexing",
707
+ chain: params.chain.name,
708
+ chain_id: params.chain.id,
709
+ block_range: JSON.stringify([
710
+ Number(decodeCheckpoint(from).blockNumber),
711
+ Number(decodeCheckpoint(to).blockNumber),
712
+ ]),
713
+ });
714
+
715
+ const eventGenerator = getLocalEventGenerator({
716
+ common: params.common,
717
+ chain: params.chain,
718
+ rpc,
719
+ eventCallbacks,
720
+ childAddresses: params.childAddresses,
721
+ syncProgress: params.syncProgress,
722
+ cachedIntervals: params.cachedIntervals,
723
+ from,
724
+ to,
725
+ limit:
726
+ Math.round(
727
+ params.common.options.syncEventsQuerySize /
728
+ (params.indexingBuild.chains.length + 1),
729
+ ) + 6,
730
+ database: params.database,
731
+ isCatchup,
732
+ });
733
+
734
+ for await (const {
735
+ events: rawEvents,
736
+ checkpoint,
737
+ blockRange,
738
+ } of eventGenerator) {
739
+ const endClock = startClock();
740
+
741
+ let events = decodeEvents(
742
+ params.common,
743
+ params.chain,
744
+ eventCallbacks,
745
+ rawEvents,
746
+ );
747
+
748
+ params.common.logger.trace({
749
+ msg: "Decoded events",
750
+ chain: params.chain.name,
751
+ chain_id: params.chain.id,
752
+ event_count: events.length,
753
+ duration: endClock(),
754
+ });
755
+
756
+ params.common.metrics.ponder_historical_extract_duration.inc(
757
+ { step: "decode" },
758
+ endClock(),
759
+ );
760
+
761
+ // Removes events that have a checkpoint earlier than (or equal to)
762
+ // the crash recovery checkpoint.
763
+
764
+ if (crashRecoveryCheckpoint) {
765
+ const [left, right] = partition(
766
+ events,
767
+ (event) => event.checkpoint <= crashRecoveryCheckpoint,
768
+ );
769
+ events = right;
770
+
771
+ if (left.length > 0) {
772
+ params.common.logger.trace({
773
+ msg: "Filtered events before crash recovery checkpoint",
774
+ chain: params.chain.name,
775
+ chain_id: params.chain.id,
776
+ event_count: left.length,
777
+ checkpoint: crashRecoveryCheckpoint,
778
+ });
779
+ }
780
+ }
781
+
782
+ yield { chainId: params.chain.id, events, checkpoint, blockRange };
783
+ }
784
+
785
+ cursor = to;
786
+
787
+ if (Date.now() - lastUnfinalizedRefetch < 30_000) {
788
+ break;
789
+ }
790
+ lastUnfinalizedRefetch = Date.now();
791
+
792
+ const context = {
793
+ logger: params.common.logger.child({ action: "refetch_finalized_block" }),
794
+ };
795
+
796
+ const endClock = startClock();
797
+
798
+ const finalizedBlock = await eth_getBlockByNumber(
799
+ rpc,
800
+ ["latest", false],
801
+ context,
802
+ ).then((latest) =>
803
+ eth_getBlockByNumber(
804
+ rpc,
805
+ [
806
+ numberToHex(
807
+ Math.max(
808
+ hexToNumber(latest.number) - params.chain.finalityBlockCount,
809
+ 0,
810
+ ),
811
+ ),
812
+ false,
813
+ ],
814
+ context,
815
+ ),
816
+ );
817
+
818
+ const finalizedBlockNumber = hexToNumber(finalizedBlock.number);
819
+ params.common.logger.debug({
820
+ msg: "Refetched finalized block for backfill cutover",
821
+ chain: params.chain.name,
822
+ chain_id: params.chain.id,
823
+ finalized_block: finalizedBlockNumber,
824
+ duration: endClock(),
825
+ });
826
+
827
+ if (
828
+ hexToNumber(finalizedBlock.number) -
829
+ hexToNumber(params.syncProgress.finalized.number) <=
830
+ params.chain.finalityBlockCount
831
+ ) {
832
+ break;
833
+ }
834
+
835
+ params.syncProgress.finalized = finalizedBlock;
836
+ isCatchup = true;
837
+ }
838
+ }
839
+
840
+ export async function refetchHistoricalEvents(params: {
841
+ common: Common;
842
+ indexingBuild: Pick<IndexingBuild, "eventCallbacks" | "chains">;
843
+ perChainSync: Map<Chain, { childAddresses: ChildAddresses }>;
844
+ events: Event[];
845
+ syncStore: SyncStore;
846
+ }): Promise<Event[]> {
847
+ const events: Event[] = new Array(params.events.length);
848
+
849
+ for (const chain of params.indexingBuild.chains) {
850
+ const { childAddresses } = params.perChainSync.get(chain)!;
851
+
852
+ // Note: All filters are refetched, no matter if they are resolved or not.
853
+ const eventCallbacks =
854
+ params.indexingBuild.eventCallbacks[
855
+ params.indexingBuild.chains.findIndex((c) => c.id === chain.id)
856
+ ]!;
857
+
858
+ const chainEvents = params.events.filter(
859
+ (event) => event.chain.id === chain.id,
860
+ );
861
+
862
+ if (chainEvents.length === 0) continue;
863
+
864
+ const rawEvents = await refetchLocalEvents({
865
+ common: params.common,
866
+ chain,
867
+ childAddresses,
868
+ eventCallbacks,
869
+ events: chainEvents,
870
+ syncStore: params.syncStore,
871
+ });
872
+
873
+ const endClock = startClock();
874
+
875
+ const refetchedEvents = decodeEvents(
876
+ params.common,
877
+ chain,
878
+ eventCallbacks,
879
+ rawEvents,
880
+ );
881
+
882
+ params.common.logger.trace({
883
+ msg: "Decoded events",
884
+ chain: chain.name,
885
+ chain_id: chain.id,
886
+ event_count: events.length,
887
+ duration: endClock(),
888
+ });
889
+
890
+ params.common.metrics.ponder_historical_extract_duration.inc(
891
+ { step: "decode" },
892
+ endClock(),
893
+ );
894
+
895
+ let i = 0;
896
+ let j = 0;
897
+
898
+ while (i < params.events.length && j < refetchedEvents.length) {
899
+ if (params.events[i]?.chain.id === chain.id) {
900
+ events[i] = refetchedEvents[j]!;
901
+ i++;
902
+ j++;
903
+ } else {
904
+ i++;
905
+ }
906
+ }
907
+ }
908
+
909
+ return events;
910
+ }
911
+
912
+ export async function refetchLocalEvents(params: {
913
+ common: Common;
914
+ chain: Chain;
915
+ childAddresses: ChildAddresses;
916
+ eventCallbacks: EventCallback[];
917
+ events: Event[];
918
+ syncStore: SyncStore;
919
+ }): Promise<RawEvent[]> {
920
+ const from = params.events[0]!.checkpoint;
921
+ const to = params.events[params.events.length - 1]!.checkpoint;
922
+
923
+ const fromBlock = Number(decodeCheckpoint(from).blockNumber);
924
+ const toBlock = Number(decodeCheckpoint(to).blockNumber);
925
+ let cursor = fromBlock;
926
+
927
+ let events: RawEvent[] | undefined;
928
+ while (cursor <= toBlock) {
929
+ const queryEndClock = startClock();
930
+
931
+ const {
932
+ blocks,
933
+ logs,
934
+ transactions,
935
+ transactionReceipts,
936
+ traces,
937
+ cursor: queryCursor,
938
+ } = await params.syncStore.getEventData({
939
+ filters: params.eventCallbacks.map(({ filter }) => filter),
940
+ fromBlock: cursor,
941
+ toBlock,
942
+ chainId: params.chain.id,
943
+ limit: params.events.length,
944
+ });
945
+
946
+ const endClock = startClock();
947
+ const rawEvents = buildEvents({
948
+ eventCallbacks: params.eventCallbacks,
949
+ blocks,
950
+ logs,
951
+ transactions,
952
+ transactionReceipts,
953
+ traces,
954
+ childAddresses: params.childAddresses,
955
+ chainId: params.chain.id,
956
+ });
957
+
958
+ params.common.logger.trace({
959
+ msg: "Constructed events from block data",
960
+ chain: params.chain.name,
961
+ chain_id: params.chain.id,
962
+ block_range: JSON.stringify([cursor, queryCursor]),
963
+ event_count: rawEvents.length,
964
+ duration: endClock(),
965
+ });
966
+
967
+ params.common.metrics.ponder_historical_extract_duration.inc(
968
+ { step: "build" },
969
+ endClock(),
970
+ );
971
+
972
+ params.common.logger.debug({
973
+ msg: "Queried backfill JSON-RPC data from database",
974
+ chain: params.chain.name,
975
+ chain_id: params.chain.id,
976
+ block_range: JSON.stringify([cursor, queryCursor]),
977
+ event_count: rawEvents.length,
978
+ duration: queryEndClock(),
979
+ });
980
+
981
+ await new Promise(setImmediate);
982
+
983
+ if (events === undefined) {
984
+ events = rawEvents;
985
+ } else {
986
+ events.push(...rawEvents);
987
+ }
988
+
989
+ cursor = queryCursor + 1;
990
+ }
991
+
992
+ return events!;
993
+ }
994
+
995
+ export async function* getLocalEventGenerator(params: {
996
+ common: Common;
997
+ chain: Chain;
998
+ rpc: Rpc;
999
+ eventCallbacks: EventCallback[];
1000
+ childAddresses: ChildAddresses;
1001
+ syncProgress: SyncProgress;
1002
+ cachedIntervals: CachedIntervals;
1003
+ from: string;
1004
+ to: string;
1005
+ limit: number;
1006
+ database: Database;
1007
+ isCatchup: boolean;
1008
+ }) {
1009
+ const syncStore = createSyncStore({
1010
+ common: params.common,
1011
+ qb: params.database.syncQB,
1012
+ });
1013
+
1014
+ const fromBlock = Number(decodeCheckpoint(params.from).blockNumber);
1015
+ const toBlock = Number(decodeCheckpoint(params.to).blockNumber);
1016
+ let cursor = fromBlock;
1017
+
1018
+ const localSyncGenerator = getLocalSyncGenerator(params);
1019
+
1020
+ for await (const syncCursor of bufferAsyncGenerator(
1021
+ localSyncGenerator,
1022
+ Number.POSITIVE_INFINITY,
1023
+ )) {
1024
+ while (cursor <= Math.min(syncCursor, toBlock)) {
1025
+ const queryEndClock = startClock();
1026
+
1027
+ const {
1028
+ blocks,
1029
+ logs,
1030
+ transactions,
1031
+ transactionReceipts,
1032
+ traces,
1033
+ cursor: queryCursor,
1034
+ } = await syncStore.getEventData({
1035
+ filters: params.eventCallbacks.map(({ filter }) => filter),
1036
+ fromBlock: cursor,
1037
+ toBlock: Math.min(syncCursor, toBlock),
1038
+ chainId: params.chain.id,
1039
+ limit: params.limit,
1040
+ });
1041
+
1042
+ const endClock = startClock();
1043
+ const events = buildEvents({
1044
+ eventCallbacks: params.eventCallbacks,
1045
+ blocks,
1046
+ logs,
1047
+ transactions,
1048
+ transactionReceipts,
1049
+ traces,
1050
+ childAddresses: params.childAddresses,
1051
+ chainId: params.chain.id,
1052
+ });
1053
+
1054
+ params.common.logger.trace({
1055
+ msg: "Constructed events from block data",
1056
+ chain: params.chain.name,
1057
+ chain_id: params.chain.id,
1058
+ block_range: JSON.stringify([cursor, queryCursor]),
1059
+ event_count: events.length,
1060
+ duration: endClock(),
1061
+ });
1062
+
1063
+ params.common.metrics.ponder_historical_extract_duration.inc(
1064
+ { step: "build" },
1065
+ endClock(),
1066
+ );
1067
+
1068
+ params.common.logger.debug({
1069
+ msg: "Queried backfill JSON-RPC data from database",
1070
+ chain: params.chain.name,
1071
+ chain_id: params.chain.id,
1072
+ block_range: JSON.stringify([cursor, queryCursor]),
1073
+ event_count: events.length,
1074
+ duration: queryEndClock(),
1075
+ });
1076
+
1077
+ await new Promise(setImmediate);
1078
+
1079
+ const blockRange = [cursor, queryCursor] satisfies [number, number];
1080
+
1081
+ cursor = queryCursor + 1;
1082
+ if (cursor >= toBlock) {
1083
+ yield { events, checkpoint: params.to, blockRange };
1084
+ } else if (blocks.length > 0) {
1085
+ const checkpoint = encodeCheckpoint({
1086
+ ...MAX_CHECKPOINT,
1087
+ blockTimestamp: blocks[blocks.length - 1]!.timestamp,
1088
+ chainId: BigInt(params.chain.id),
1089
+ blockNumber: blocks[blocks.length - 1]!.number,
1090
+ });
1091
+ yield { events, checkpoint, blockRange };
1092
+ }
1093
+ }
1094
+ }
1095
+ }
1096
+
1097
+ export async function* getLocalSyncGenerator(params: {
1098
+ common: Common;
1099
+ chain: Chain;
1100
+ rpc: Rpc;
1101
+ eventCallbacks: EventCallback[];
1102
+ syncProgress: SyncProgress;
1103
+ childAddresses: ChildAddresses;
1104
+ cachedIntervals: CachedIntervals;
1105
+ database: Database;
1106
+ isCatchup: boolean;
1107
+ }) {
1108
+ const backfillEndClock = startClock();
1109
+ const label = { chain: params.chain.name };
1110
+
1111
+ let first = hexToNumber(params.syncProgress.start.number);
1112
+ const last =
1113
+ params.syncProgress.end === undefined
1114
+ ? params.syncProgress.finalized
1115
+ : hexToNumber(params.syncProgress.end.number) >
1116
+ hexToNumber(params.syncProgress.finalized.number)
1117
+ ? params.syncProgress.finalized
1118
+ : params.syncProgress.end;
1119
+
1120
+ // Estimate optimal range (blocks) to sync at a time, eventually to be used to
1121
+ // determine `interval` passed to `historicalSync.sync()`.
1122
+ let estimateRange = 25;
1123
+
1124
+ // Handle two special cases:
1125
+ // 1. `syncProgress.start` > `syncProgress.finalized`
1126
+ // 2. `cached` is defined
1127
+
1128
+ if (
1129
+ hexToNumber(params.syncProgress.start.number) >
1130
+ hexToNumber(params.syncProgress.finalized.number)
1131
+ ) {
1132
+ params.syncProgress.current = params.syncProgress.finalized;
1133
+
1134
+ params.common.logger.info({
1135
+ msg: "Skipped fetching backfill JSON-RPC data (chain only requires live indexing)",
1136
+ chain: params.chain.name,
1137
+ chain_id: params.chain.id,
1138
+ finalized_block: hexToNumber(params.syncProgress.finalized.number),
1139
+ start_block: hexToNumber(params.syncProgress.start.number),
1140
+ });
1141
+
1142
+ params.common.metrics.ponder_sync_block.set(
1143
+ label,
1144
+ hexToNumber(params.syncProgress.current.number),
1145
+ );
1146
+ params.common.metrics.ponder_sync_block_timestamp.set(
1147
+ label,
1148
+ hexToNumber(params.syncProgress.current.timestamp),
1149
+ );
1150
+ params.common.metrics.ponder_historical_total_blocks.set(label, 0);
1151
+ params.common.metrics.ponder_historical_cached_blocks.set(label, 0);
1152
+
1153
+ return;
1154
+ }
1155
+
1156
+ const totalInterval = [
1157
+ hexToNumber(params.syncProgress.start.number),
1158
+ hexToNumber(last!.number),
1159
+ ] satisfies Interval;
1160
+
1161
+ const requiredIntervals = getRequiredIntervals({
1162
+ filters: params.eventCallbacks.map(({ filter }) => filter),
1163
+ interval: totalInterval,
1164
+ cachedIntervals: params.cachedIntervals,
1165
+ });
1166
+
1167
+ const required = intervalSum(requiredIntervals);
1168
+ const total = totalInterval[1] - totalInterval[0] + 1;
1169
+
1170
+ params.common.metrics.ponder_historical_total_blocks.set(label, total);
1171
+ params.common.metrics.ponder_historical_cached_blocks.set(
1172
+ label,
1173
+ total - required,
1174
+ );
1175
+
1176
+ // Handle cache hit
1177
+ if (params.syncProgress.current !== undefined) {
1178
+ params.common.metrics.ponder_sync_block.set(
1179
+ label,
1180
+ hexToNumber(params.syncProgress.current.number),
1181
+ );
1182
+ params.common.metrics.ponder_sync_block_timestamp.set(
1183
+ label,
1184
+ hexToNumber(params.syncProgress.current.timestamp),
1185
+ );
1186
+
1187
+ // `getEvents` can make progress without calling `sync`, so immediately "yield"
1188
+ yield hexToNumber(params.syncProgress.current.number);
1189
+
1190
+ if (
1191
+ hexToNumber(params.syncProgress.current.number) ===
1192
+ hexToNumber(last!.number)
1193
+ ) {
1194
+ if (params.isCatchup === false) {
1195
+ params.common.logger.info({
1196
+ msg: "Skipped fetching backfill JSON-RPC data (cache contains all required data)",
1197
+ chain: params.chain.name,
1198
+ chain_id: params.chain.id,
1199
+ cached_block: hexToNumber(params.syncProgress.current.number),
1200
+ cache_rate: "100%",
1201
+ });
1202
+ }
1203
+ return;
1204
+ } else if (params.isCatchup === false) {
1205
+ params.common.logger.info({
1206
+ msg: "Started fetching backfill JSON-RPC data",
1207
+ chain: params.chain.name,
1208
+ chain_id: params.chain.id,
1209
+ cached_block: hexToNumber(params.syncProgress.current.number),
1210
+ cache_rate: formatPercentage((total - required) / total),
1211
+ });
1212
+ }
1213
+
1214
+ first = hexToNumber(params.syncProgress.current.number) + 1;
1215
+ } else {
1216
+ params.common.logger.info({
1217
+ msg: "Started fetching backfill JSON-RPC data",
1218
+ chain: params.chain.name,
1219
+ chain_id: params.chain.id,
1220
+ cache_rate: "0%",
1221
+ });
1222
+ }
1223
+
1224
+ const historicalSync = createHistoricalSync(params);
1225
+
1226
+ const { callback: intervalCallback, generator: intervalGenerator } =
1227
+ createCallbackGenerator<{
1228
+ interval: Interval;
1229
+ promise: Promise<void>;
1230
+ }>();
1231
+
1232
+ intervalCallback({
1233
+ interval: [
1234
+ first,
1235
+ Math.min(first + estimateRange, hexToNumber(last.number)),
1236
+ ],
1237
+ promise: Promise.resolve(),
1238
+ });
1239
+
1240
+ /**
1241
+ * @returns `true` if any data was inserted into the database.
1242
+ */
1243
+ async function syncInterval({
1244
+ interval,
1245
+ promise,
1246
+ }: { interval: Interval; promise: Promise<void> }): Promise<boolean> {
1247
+ const endClock = startClock();
1248
+
1249
+ const isSyncComplete = interval[1] === hexToNumber(last.number);
1250
+ const {
1251
+ intervals: requiredIntervals,
1252
+ factoryIntervals: requiredFactoryIntervals,
1253
+ } = getRequiredIntervalsWithFilters({
1254
+ interval,
1255
+ filters: params.eventCallbacks.map(({ filter }) => filter),
1256
+ cachedIntervals: params.cachedIntervals,
1257
+ });
1258
+
1259
+ let closestToTipBlock: SyncBlock | undefined;
1260
+ if (requiredIntervals.length > 0 || requiredFactoryIntervals.length > 0) {
1261
+ const pwr = promiseWithResolvers<void>();
1262
+
1263
+ const durationTimer = setTimeout(
1264
+ () => {
1265
+ params.common.logger.warn({
1266
+ msg: "Fetching backfill JSON-RPC data is taking longer than expected",
1267
+ chain: params.chain.name,
1268
+ chain_id: params.chain.id,
1269
+ block_range: JSON.stringify(interval),
1270
+ duration: endClock(),
1271
+ });
1272
+ },
1273
+ params.common.options.command === "dev" ? 10_000 : 50_000,
1274
+ );
1275
+
1276
+ closestToTipBlock = await params.database.syncQB
1277
+ .transaction(async (tx) => {
1278
+ const syncStore = createSyncStore({ common: params.common, qb: tx });
1279
+ const logs = await historicalSync.syncBlockRangeData({
1280
+ interval,
1281
+ requiredIntervals,
1282
+ requiredFactoryIntervals,
1283
+ syncStore,
1284
+ });
1285
+
1286
+ // Wait for the previous interval to complete `syncBlockData`.
1287
+ await promise;
1288
+
1289
+ if (isSyncComplete === false) {
1290
+ // Queue the next interval
1291
+ intervalCallback({
1292
+ interval: [
1293
+ Math.min(interval[1] + 1, hexToNumber(last.number)),
1294
+ Math.min(
1295
+ interval[1] + 1 + estimateRange,
1296
+ hexToNumber(last.number),
1297
+ ),
1298
+ ],
1299
+ promise: pwr.promise,
1300
+ });
1301
+ }
1302
+
1303
+ const closestToTipBlock = await historicalSync.syncBlockData({
1304
+ interval,
1305
+ requiredIntervals,
1306
+ logs,
1307
+ syncStore,
1308
+ });
1309
+ if (params.chain.disableCache === false) {
1310
+ await syncStore.insertIntervals({
1311
+ intervals: requiredIntervals,
1312
+ factoryIntervals: requiredFactoryIntervals,
1313
+ chainId: params.chain.id,
1314
+ });
1315
+ }
1316
+
1317
+ return closestToTipBlock;
1318
+ })
1319
+ .catch((error) => {
1320
+ if (error instanceof ShutdownError) {
1321
+ throw error;
1322
+ }
1323
+
1324
+ params.common.logger.warn({
1325
+ msg: "Failed to fetch backfill JSON-RPC data",
1326
+ chain: params.chain.name,
1327
+ chain_id: params.chain.id,
1328
+ block_range: JSON.stringify(interval),
1329
+ duration: endClock(),
1330
+ error,
1331
+ });
1332
+ throw error;
1333
+ });
1334
+
1335
+ clearTimeout(durationTimer);
1336
+
1337
+ const duration = endClock();
1338
+
1339
+ // Use the duration and interval of the last call to `sync` to update estimate
1340
+ estimateRange = estimate({
1341
+ from: interval[0],
1342
+ to: interval[1],
1343
+ target: params.common.options.command === "dev" ? 2_000 : 10_000,
1344
+ result: duration,
1345
+ min: 25,
1346
+ max: 100_000,
1347
+ prev: estimateRange,
1348
+ maxIncrease: 1.5,
1349
+ });
1350
+
1351
+ params.common.logger.trace({
1352
+ msg: "Updated block range estimate for fetching backfill JSON-RPC data",
1353
+ chain: params.chain.name,
1354
+ chain_id: params.chain.id,
1355
+ range: estimateRange,
1356
+ });
1357
+
1358
+ // Resolve promise so the next interval can continue.
1359
+ pwr.resolve();
1360
+ } else {
1361
+ // Wait for the previous interval to complete `syncBlockData`.
1362
+ await promise;
1363
+
1364
+ if (isSyncComplete === false) {
1365
+ // Queue the next interval
1366
+ intervalCallback({
1367
+ interval: [
1368
+ Math.min(interval[1] + 1, hexToNumber(last.number)),
1369
+ Math.min(interval[1] + 1 + estimateRange, hexToNumber(last.number)),
1370
+ ],
1371
+ promise: Promise.resolve(),
1372
+ });
1373
+ }
1374
+ }
1375
+
1376
+ if (interval[1] === hexToNumber(last.number)) {
1377
+ params.syncProgress.current = last;
1378
+ }
1379
+
1380
+ if (closestToTipBlock) {
1381
+ params.common.metrics.ponder_sync_block.set(
1382
+ label,
1383
+ hexToNumber(closestToTipBlock.number),
1384
+ );
1385
+ params.common.metrics.ponder_sync_block_timestamp.set(
1386
+ label,
1387
+ hexToNumber(closestToTipBlock.timestamp),
1388
+ );
1389
+ } else {
1390
+ params.common.metrics.ponder_sync_block.set(label, interval[1]);
1391
+ }
1392
+
1393
+ params.common.metrics.ponder_historical_completed_blocks.inc(
1394
+ label,
1395
+ interval[1] - interval[0] + 1,
1396
+ );
1397
+
1398
+ return requiredIntervals.length > 0;
1399
+ }
1400
+
1401
+ const { callback, generator } =
1402
+ createCallbackGenerator<IteratorResult<number>>();
1403
+
1404
+ (async () => {
1405
+ for await (const { interval, promise } of intervalGenerator) {
1406
+ // Note: this relies on the invariant that `syncInterval`
1407
+ // will always resolve promises in the order it was called.
1408
+ syncInterval({ interval, promise }).then((didInsertData) => {
1409
+ const isDone = interval[1] === hexToNumber(last.number);
1410
+ if (didInsertData || isDone) {
1411
+ callback({ value: interval[1], done: false });
1412
+ }
1413
+
1414
+ if (isDone) {
1415
+ callback({ value: undefined, done: true });
1416
+ }
1417
+ });
1418
+ }
1419
+ })();
1420
+
1421
+ for await (const result of generator) {
1422
+ if (result.done) break;
1423
+ yield result.value;
1424
+ }
1425
+
1426
+ params.common.logger.info({
1427
+ msg: "Finished fetching backfill JSON-RPC data",
1428
+ chain: params.chain.name,
1429
+ chain_id: params.chain.id,
1430
+ duration: backfillEndClock(),
1431
+ });
1432
+ }
1433
+
1434
+ /**
1435
+ * Merges multiple event generators into a single generator while preserving
1436
+ * the order of events.
1437
+ *
1438
+ * @param generators - Generators to merge.
1439
+ * @returns A single generator that yields events from all generators.
1440
+ */
1441
+ export async function* mergeAsyncGeneratorsWithEventOrder(
1442
+ generators: AsyncGenerator<{
1443
+ events: Event[];
1444
+ checkpoint: string;
1445
+ blockRange: [number, number];
1446
+ }>[],
1447
+ ): AsyncGenerator<
1448
+ {
1449
+ chainId: number;
1450
+ events: Event[];
1451
+ checkpoint: string;
1452
+ blockRange: [number, number];
1453
+ }[]
1454
+ > {
1455
+ const results = await Promise.all(generators.map((gen) => gen.next()));
1456
+
1457
+ while (results.some((res) => res.done !== true)) {
1458
+ const supremum = min(
1459
+ ...results.map((res) => (res.done ? undefined : res.value.checkpoint)),
1460
+ );
1461
+
1462
+ const mergedResults: {
1463
+ chainId: number;
1464
+ events: Event[];
1465
+ checkpoint: string;
1466
+ blockRange: [number, number];
1467
+ }[] = [];
1468
+
1469
+ for (const result of results) {
1470
+ if (result.done === false) {
1471
+ const [left, right] = partition(
1472
+ result.value.events,
1473
+ (event) => event.checkpoint <= supremum,
1474
+ );
1475
+
1476
+ const event = left[left.length - 1];
1477
+
1478
+ if (event) {
1479
+ const blockRange = [
1480
+ result.value.blockRange[0],
1481
+ right.length > 0
1482
+ ? Number(decodeCheckpoint(event.checkpoint).blockNumber)
1483
+ : result.value.blockRange[1],
1484
+ ] satisfies [number, number];
1485
+
1486
+ mergedResults.push({
1487
+ events: left,
1488
+ chainId: event.chain.id,
1489
+ checkpoint:
1490
+ right.length > 0 ? event.checkpoint : result.value.checkpoint,
1491
+ blockRange,
1492
+ });
1493
+ }
1494
+
1495
+ result.value.events = right;
1496
+ }
1497
+ }
1498
+
1499
+ const index = results.findIndex(
1500
+ (res) => res.done === false && res.value.checkpoint === supremum,
1501
+ );
1502
+
1503
+ const resultPromise = generators[index]!.next();
1504
+ if (mergedResults.length > 0) {
1505
+ yield mergedResults;
1506
+ }
1507
+ results[index] = await resultPromise;
1508
+ }
1509
+ }