@hyperlane-xyz/rebalancer 0.1.0-beta.5a8bd28ab

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 (202) hide show
  1. package/README.md +178 -0
  2. package/dist/config/RebalancerConfig.d.ts +12 -0
  3. package/dist/config/RebalancerConfig.d.ts.map +1 -0
  4. package/dist/config/RebalancerConfig.js +29 -0
  5. package/dist/config/RebalancerConfig.js.map +1 -0
  6. package/dist/config/RebalancerConfig.test.d.ts +2 -0
  7. package/dist/config/RebalancerConfig.test.d.ts.map +1 -0
  8. package/dist/config/RebalancerConfig.test.js +325 -0
  9. package/dist/config/RebalancerConfig.test.js.map +1 -0
  10. package/dist/core/Rebalancer.d.ts +23 -0
  11. package/dist/core/Rebalancer.d.ts.map +1 -0
  12. package/dist/core/Rebalancer.js +290 -0
  13. package/dist/core/Rebalancer.js.map +1 -0
  14. package/dist/core/RebalancerService.d.ts +115 -0
  15. package/dist/core/RebalancerService.d.ts.map +1 -0
  16. package/dist/core/RebalancerService.js +227 -0
  17. package/dist/core/RebalancerService.js.map +1 -0
  18. package/dist/core/WithInflightGuard.d.ts +20 -0
  19. package/dist/core/WithInflightGuard.d.ts.map +1 -0
  20. package/dist/core/WithInflightGuard.js +47 -0
  21. package/dist/core/WithInflightGuard.js.map +1 -0
  22. package/dist/core/WithInflightGuard.test.d.ts +2 -0
  23. package/dist/core/WithInflightGuard.test.d.ts.map +1 -0
  24. package/dist/core/WithInflightGuard.test.js +64 -0
  25. package/dist/core/WithInflightGuard.test.js.map +1 -0
  26. package/dist/core/WithSemaphore.d.ts +22 -0
  27. package/dist/core/WithSemaphore.d.ts.map +1 -0
  28. package/dist/core/WithSemaphore.js +67 -0
  29. package/dist/core/WithSemaphore.js.map +1 -0
  30. package/dist/core/WithSemaphore.test.d.ts +2 -0
  31. package/dist/core/WithSemaphore.test.d.ts.map +1 -0
  32. package/dist/core/WithSemaphore.test.js +83 -0
  33. package/dist/core/WithSemaphore.test.js.map +1 -0
  34. package/dist/factories/RebalancerContextFactory.d.ts +41 -0
  35. package/dist/factories/RebalancerContextFactory.d.ts.map +1 -0
  36. package/dist/factories/RebalancerContextFactory.js +115 -0
  37. package/dist/factories/RebalancerContextFactory.js.map +1 -0
  38. package/dist/index.d.ts +33 -0
  39. package/dist/index.d.ts.map +1 -0
  40. package/dist/index.js +35 -0
  41. package/dist/index.js.map +1 -0
  42. package/dist/interfaces/IMetrics.d.ts +5 -0
  43. package/dist/interfaces/IMetrics.d.ts.map +1 -0
  44. package/dist/interfaces/IMetrics.js +2 -0
  45. package/dist/interfaces/IMetrics.js.map +1 -0
  46. package/dist/interfaces/IMonitor.d.ts +51 -0
  47. package/dist/interfaces/IMonitor.d.ts.map +1 -0
  48. package/dist/interfaces/IMonitor.js +14 -0
  49. package/dist/interfaces/IMonitor.js.map +1 -0
  50. package/dist/interfaces/IRebalancer.d.ts +15 -0
  51. package/dist/interfaces/IRebalancer.d.ts.map +1 -0
  52. package/dist/interfaces/IRebalancer.js +2 -0
  53. package/dist/interfaces/IRebalancer.js.map +1 -0
  54. package/dist/interfaces/IStrategy.d.ts +11 -0
  55. package/dist/interfaces/IStrategy.d.ts.map +1 -0
  56. package/dist/interfaces/IStrategy.js +2 -0
  57. package/dist/interfaces/IStrategy.js.map +1 -0
  58. package/dist/metrics/Metrics.d.ts +31 -0
  59. package/dist/metrics/Metrics.d.ts.map +1 -0
  60. package/dist/metrics/Metrics.js +302 -0
  61. package/dist/metrics/Metrics.js.map +1 -0
  62. package/dist/metrics/PriceGetter.d.ts +10 -0
  63. package/dist/metrics/PriceGetter.d.ts.map +1 -0
  64. package/dist/metrics/PriceGetter.js +41 -0
  65. package/dist/metrics/PriceGetter.js.map +1 -0
  66. package/dist/metrics/scripts/metrics.d.ts +14 -0
  67. package/dist/metrics/scripts/metrics.d.ts.map +1 -0
  68. package/dist/metrics/scripts/metrics.js +198 -0
  69. package/dist/metrics/scripts/metrics.js.map +1 -0
  70. package/dist/metrics/types.d.ts +24 -0
  71. package/dist/metrics/types.d.ts.map +1 -0
  72. package/dist/metrics/types.js +2 -0
  73. package/dist/metrics/types.js.map +1 -0
  74. package/dist/metrics/utils/metrics.d.ts +12 -0
  75. package/dist/metrics/utils/metrics.d.ts.map +1 -0
  76. package/dist/metrics/utils/metrics.js +28 -0
  77. package/dist/metrics/utils/metrics.js.map +1 -0
  78. package/dist/monitor/Monitor.d.ts +26 -0
  79. package/dist/monitor/Monitor.d.ts.map +1 -0
  80. package/dist/monitor/Monitor.js +116 -0
  81. package/dist/monitor/Monitor.js.map +1 -0
  82. package/dist/service.d.ts +3 -0
  83. package/dist/service.d.ts.map +1 -0
  84. package/dist/service.js +125 -0
  85. package/dist/service.js.map +1 -0
  86. package/dist/strategy/BaseStrategy.d.ts +34 -0
  87. package/dist/strategy/BaseStrategy.d.ts.map +1 -0
  88. package/dist/strategy/BaseStrategy.js +127 -0
  89. package/dist/strategy/BaseStrategy.js.map +1 -0
  90. package/dist/strategy/MinAmountStrategy.d.ts +27 -0
  91. package/dist/strategy/MinAmountStrategy.d.ts.map +1 -0
  92. package/dist/strategy/MinAmountStrategy.js +103 -0
  93. package/dist/strategy/MinAmountStrategy.js.map +1 -0
  94. package/dist/strategy/MinAmountStrategy.test.d.ts +2 -0
  95. package/dist/strategy/MinAmountStrategy.test.d.ts.map +1 -0
  96. package/dist/strategy/MinAmountStrategy.test.js +472 -0
  97. package/dist/strategy/MinAmountStrategy.test.js.map +1 -0
  98. package/dist/strategy/StrategyFactory.d.ts +16 -0
  99. package/dist/strategy/StrategyFactory.d.ts.map +1 -0
  100. package/dist/strategy/StrategyFactory.js +25 -0
  101. package/dist/strategy/StrategyFactory.js.map +1 -0
  102. package/dist/strategy/StrategyFactory.test.d.ts +2 -0
  103. package/dist/strategy/StrategyFactory.test.d.ts.map +1 -0
  104. package/dist/strategy/StrategyFactory.test.js +80 -0
  105. package/dist/strategy/StrategyFactory.test.js.map +1 -0
  106. package/dist/strategy/WeightedStrategy.d.ts +23 -0
  107. package/dist/strategy/WeightedStrategy.d.ts.map +1 -0
  108. package/dist/strategy/WeightedStrategy.js +61 -0
  109. package/dist/strategy/WeightedStrategy.js.map +1 -0
  110. package/dist/strategy/WeightedStrategy.test.d.ts +2 -0
  111. package/dist/strategy/WeightedStrategy.test.d.ts.map +1 -0
  112. package/dist/strategy/WeightedStrategy.test.js +307 -0
  113. package/dist/strategy/WeightedStrategy.test.js.map +1 -0
  114. package/dist/strategy/index.d.ts +5 -0
  115. package/dist/strategy/index.d.ts.map +1 -0
  116. package/dist/strategy/index.js +5 -0
  117. package/dist/strategy/index.js.map +1 -0
  118. package/dist/test/helpers.d.ts +8 -0
  119. package/dist/test/helpers.d.ts.map +1 -0
  120. package/dist/test/helpers.js +33 -0
  121. package/dist/test/helpers.js.map +1 -0
  122. package/dist/utils/ExplorerClient.d.ts +14 -0
  123. package/dist/utils/ExplorerClient.d.ts.map +1 -0
  124. package/dist/utils/ExplorerClient.js +82 -0
  125. package/dist/utils/ExplorerClient.js.map +1 -0
  126. package/dist/utils/balanceUtils.d.ts +13 -0
  127. package/dist/utils/balanceUtils.d.ts.map +1 -0
  128. package/dist/utils/balanceUtils.js +43 -0
  129. package/dist/utils/balanceUtils.js.map +1 -0
  130. package/dist/utils/balanceUtils.test.d.ts +2 -0
  131. package/dist/utils/balanceUtils.test.d.ts.map +1 -0
  132. package/dist/utils/balanceUtils.test.js +54 -0
  133. package/dist/utils/balanceUtils.test.js.map +1 -0
  134. package/dist/utils/bridgeUtils.d.ts +19 -0
  135. package/dist/utils/bridgeUtils.d.ts.map +1 -0
  136. package/dist/utils/bridgeUtils.js +20 -0
  137. package/dist/utils/bridgeUtils.js.map +1 -0
  138. package/dist/utils/bridgeUtils.test.d.ts +2 -0
  139. package/dist/utils/bridgeUtils.test.d.ts.map +1 -0
  140. package/dist/utils/bridgeUtils.test.js +77 -0
  141. package/dist/utils/bridgeUtils.test.js.map +1 -0
  142. package/dist/utils/errors.d.ts +4 -0
  143. package/dist/utils/errors.d.ts.map +1 -0
  144. package/dist/utils/errors.js +6 -0
  145. package/dist/utils/errors.js.map +1 -0
  146. package/dist/utils/files.d.ts +35 -0
  147. package/dist/utils/files.d.ts.map +1 -0
  148. package/dist/utils/files.js +190 -0
  149. package/dist/utils/files.js.map +1 -0
  150. package/dist/utils/generalUtils.d.ts +3 -0
  151. package/dist/utils/generalUtils.d.ts.map +1 -0
  152. package/dist/utils/generalUtils.js +9 -0
  153. package/dist/utils/generalUtils.js.map +1 -0
  154. package/dist/utils/index.d.ts +5 -0
  155. package/dist/utils/index.d.ts.map +1 -0
  156. package/dist/utils/index.js +5 -0
  157. package/dist/utils/index.js.map +1 -0
  158. package/dist/utils/tokenUtils.d.ts +14 -0
  159. package/dist/utils/tokenUtils.d.ts.map +1 -0
  160. package/dist/utils/tokenUtils.js +21 -0
  161. package/dist/utils/tokenUtils.js.map +1 -0
  162. package/package.json +70 -0
  163. package/src/config/RebalancerConfig.test.ts +388 -0
  164. package/src/config/RebalancerConfig.ts +39 -0
  165. package/src/core/Rebalancer.ts +471 -0
  166. package/src/core/RebalancerService.ts +333 -0
  167. package/src/core/WithInflightGuard.test.ts +131 -0
  168. package/src/core/WithInflightGuard.ts +67 -0
  169. package/src/core/WithSemaphore.test.ts +112 -0
  170. package/src/core/WithSemaphore.ts +92 -0
  171. package/src/factories/RebalancerContextFactory.ts +210 -0
  172. package/src/index.ts +68 -0
  173. package/src/interfaces/IMetrics.ts +5 -0
  174. package/src/interfaces/IMonitor.ts +63 -0
  175. package/src/interfaces/IRebalancer.ts +20 -0
  176. package/src/interfaces/IStrategy.ts +13 -0
  177. package/src/metrics/Metrics.ts +558 -0
  178. package/src/metrics/PriceGetter.ts +74 -0
  179. package/src/metrics/scripts/metrics.ts +298 -0
  180. package/src/metrics/types.ts +27 -0
  181. package/src/metrics/utils/metrics.ts +33 -0
  182. package/src/monitor/Monitor.ts +174 -0
  183. package/src/service.ts +154 -0
  184. package/src/strategy/BaseStrategy.ts +210 -0
  185. package/src/strategy/MinAmountStrategy.test.ts +625 -0
  186. package/src/strategy/MinAmountStrategy.ts +170 -0
  187. package/src/strategy/StrategyFactory.test.ts +109 -0
  188. package/src/strategy/StrategyFactory.ts +48 -0
  189. package/src/strategy/WeightedStrategy.test.ts +408 -0
  190. package/src/strategy/WeightedStrategy.ts +93 -0
  191. package/src/strategy/index.ts +4 -0
  192. package/src/test/helpers.ts +46 -0
  193. package/src/utils/ExplorerClient.ts +99 -0
  194. package/src/utils/balanceUtils.test.ts +74 -0
  195. package/src/utils/balanceUtils.ts +69 -0
  196. package/src/utils/bridgeUtils.test.ts +92 -0
  197. package/src/utils/bridgeUtils.ts +42 -0
  198. package/src/utils/errors.ts +5 -0
  199. package/src/utils/files.ts +276 -0
  200. package/src/utils/generalUtils.ts +13 -0
  201. package/src/utils/index.ts +4 -0
  202. package/src/utils/tokenUtils.ts +26 -0
@@ -0,0 +1,298 @@
1
+ import { Logger } from 'pino';
2
+ import { Counter, Gauge, Registry } from 'prom-client';
3
+
4
+ import { ChainName, Token, TokenStandard, WarpCore } from '@hyperlane-xyz/sdk';
5
+ import { Address } from '@hyperlane-xyz/utils';
6
+
7
+ import {
8
+ NativeWalletBalance,
9
+ WarpRouteBalance,
10
+ XERC20Limit,
11
+ } from '../types.js';
12
+
13
+ export const metricsRegister = new Registry();
14
+
15
+ type SupportedTokenStandards = TokenStandard | 'EvmManagedLockbox' | 'xERC20';
16
+
17
+ interface BaseWarpRouteMetricLabels {
18
+ chain_name: ChainName;
19
+ token_address: string;
20
+ token_name: string;
21
+ warp_route_id: string;
22
+ }
23
+
24
+ interface WarpRouteMetricLabels extends BaseWarpRouteMetricLabels {
25
+ wallet_address: string;
26
+ token_standard: SupportedTokenStandards;
27
+ related_chain_names: string;
28
+ }
29
+
30
+ const warpRouteMetricLabels: (keyof WarpRouteMetricLabels)[] = [
31
+ 'chain_name',
32
+ 'token_address',
33
+ 'token_name',
34
+ 'wallet_address',
35
+ 'token_standard',
36
+ 'warp_route_id',
37
+ 'related_chain_names',
38
+ ];
39
+
40
+ const warpRouteTokenBalance = new Gauge({
41
+ name: 'hyperlane_warp_route_token_balance',
42
+ help: 'HypERC20 token balance of a Warp Route',
43
+ registers: [metricsRegister],
44
+ labelNames: warpRouteMetricLabels,
45
+ });
46
+
47
+ const warpRouteCollateralValue = new Gauge({
48
+ name: 'hyperlane_warp_route_collateral_value',
49
+ help: 'Total value of collateral held in a HypERC20Collateral or HypNative contract of a Warp Route',
50
+ registers: [metricsRegister],
51
+ labelNames: warpRouteMetricLabels,
52
+ });
53
+
54
+ interface WarpRouteValueAtRiskMetricLabels extends BaseWarpRouteMetricLabels {
55
+ collateral_chain_name: ChainName;
56
+ collateral_token_standard: SupportedTokenStandards;
57
+ }
58
+
59
+ const warpRouteValueAtRiskMetricLabels: (keyof WarpRouteValueAtRiskMetricLabels)[] =
60
+ [
61
+ 'chain_name',
62
+ 'collateral_chain_name',
63
+ 'token_address',
64
+ 'token_name',
65
+ 'collateral_token_standard',
66
+ 'warp_route_id',
67
+ ];
68
+
69
+ const warpRouteValueAtRisk = new Gauge({
70
+ name: 'hyperlane_warp_route_value_at_risk',
71
+ help: 'Value at risk on chain for a given Warp Route',
72
+ registers: [metricsRegister],
73
+ labelNames: warpRouteValueAtRiskMetricLabels,
74
+ });
75
+
76
+ export const rebalancerExecutionTotal = new Counter({
77
+ name: 'hyperlane_rebalancer_executions_total',
78
+ help: 'Total number of rebalance execution attempts.',
79
+ registers: [metricsRegister],
80
+ labelNames: ['warp_route_id', 'succeeded'],
81
+ });
82
+
83
+ export const rebalancerExecutionAmount = new Counter({
84
+ name: 'hyperlane_rebalancer_execution_amount',
85
+ help: 'Total amount of tokens rebalanced.',
86
+ registers: [metricsRegister],
87
+ labelNames: ['warp_route_id', 'origin', 'destination', 'token'],
88
+ });
89
+
90
+ export const rebalancerPollingErrorsTotal = new Counter({
91
+ name: 'hyperlane_rebalancer_polling_errors_total',
92
+ help: 'Total number of errors during the monitor polling phase.',
93
+ registers: [metricsRegister],
94
+ labelNames: ['warp_route_id'],
95
+ });
96
+
97
+ const walletBalanceGauge = new Gauge({
98
+ // Mirror the rust/main/ethers-prometheus `wallet_balance` gauge metric.
99
+ name: 'hyperlane_wallet_balance',
100
+ help: 'Current balance of a wallet for a token',
101
+ registers: [metricsRegister],
102
+ labelNames: [
103
+ 'chain',
104
+ 'wallet_address',
105
+ 'wallet_name',
106
+ 'token_address',
107
+ 'token_symbol',
108
+ 'token_name',
109
+ ],
110
+ });
111
+
112
+ const xERC20LimitsGauge = new Gauge({
113
+ name: 'hyperlane_xerc20_limits',
114
+ help: 'Current minting and burning limits of xERC20 tokens',
115
+ registers: [metricsRegister],
116
+ labelNames: [
117
+ 'chain_name',
118
+ 'limit_type',
119
+ 'token_name',
120
+ 'bridge_address',
121
+ 'token_address',
122
+ 'bridge_label',
123
+ ],
124
+ });
125
+
126
+ export function updateTokenBalanceMetrics(
127
+ warpCore: WarpCore,
128
+ token: Token,
129
+ balanceInfo: WarpRouteBalance,
130
+ warpRouteId: string,
131
+ logger: Logger,
132
+ ) {
133
+ const allChains = warpCore.getTokenChains().sort();
134
+ const relatedChains = allChains.filter(
135
+ (chainName) => chainName !== token.chainName,
136
+ );
137
+
138
+ const metrics: WarpRouteMetricLabels = {
139
+ chain_name: token.chainName,
140
+ token_address: balanceInfo.tokenAddress,
141
+ token_name: token.name,
142
+ wallet_address:
143
+ // the balance for an EvmHypERC20 token is returned as the total supply of the xERC20 token,
144
+ // therefore we set the wallet address to the token address,
145
+ // we follow the same pattern or synthetic tokens
146
+ token.standard !== TokenStandard.EvmHypXERC20
147
+ ? token.addressOrDenom
148
+ : balanceInfo.tokenAddress,
149
+ token_standard:
150
+ // as we are reporting the total supply for clarity we report the standard as xERC20
151
+ token.standard !== TokenStandard.EvmHypXERC20 ? token.standard : 'xERC20',
152
+ warp_route_id: warpRouteId,
153
+ // TODO: consider deprecating this label given that we have the value at risk metric
154
+ related_chain_names: relatedChains.join(','),
155
+ };
156
+
157
+ warpRouteTokenBalance.labels(metrics).set(balanceInfo.balance);
158
+ logger.info(
159
+ {
160
+ labels: metrics,
161
+ balance: balanceInfo.balance,
162
+ },
163
+ 'Wallet balance updated for token',
164
+ );
165
+
166
+ if (balanceInfo.valueUSD) {
167
+ // TODO: consider deprecating this metric in favor of the value at risk metric
168
+ warpRouteCollateralValue.labels(metrics).set(balanceInfo.valueUSD);
169
+ logger.info(
170
+ {
171
+ labels: metrics,
172
+ valueUSD: balanceInfo.valueUSD,
173
+ },
174
+ 'Wallet value updated for token',
175
+ );
176
+
177
+ for (const chainName of allChains) {
178
+ const labels = {
179
+ chain_name: chainName,
180
+ collateral_chain_name: metrics.chain_name,
181
+ token_address: metrics.token_address,
182
+ token_name: metrics.token_name,
183
+ collateral_token_standard: metrics.token_standard,
184
+ warp_route_id: metrics.warp_route_id,
185
+ };
186
+
187
+ warpRouteValueAtRisk.labels(labels).set(balanceInfo.valueUSD);
188
+ logger.info(
189
+ {
190
+ labels,
191
+ valueUSD: balanceInfo.valueUSD,
192
+ },
193
+ `Value at risk on ${chainName} updated for token`,
194
+ );
195
+ }
196
+ }
197
+ }
198
+ // TODO: This does not need to be a separate function, we can redefine updateTokenBalanceMetrics to be generic
199
+ // TODO: Consider adding some identifier for the managedLockbox contract, could be adding collateralName label for lockboxes, this would help different manages lockboxes that has a different collateral token
200
+ export function updateManagedLockboxBalanceMetrics(
201
+ warpCore: WarpCore,
202
+ chainName: ChainName,
203
+ tokenName: string,
204
+ tokenAddress: string,
205
+ lockBoxAddress: string,
206
+ balanceInfo: WarpRouteBalance,
207
+ warpRouteId: string,
208
+ logger: Logger,
209
+ ) {
210
+ const metrics: WarpRouteMetricLabels = {
211
+ chain_name: chainName,
212
+ token_address: tokenAddress,
213
+ token_name: tokenName,
214
+ wallet_address: lockBoxAddress,
215
+ token_standard: 'EvmManagedLockbox', // TODO: we should eventually a new TokenStandard for this
216
+ warp_route_id: warpRouteId,
217
+ related_chain_names: warpCore
218
+ .getTokenChains()
219
+ .filter((_chainName) => _chainName !== chainName)
220
+ .sort()
221
+ .join(','),
222
+ };
223
+
224
+ warpRouteTokenBalance.labels(metrics).set(balanceInfo.balance);
225
+ logger.info(
226
+ {
227
+ labels: metrics,
228
+ balance: balanceInfo.balance,
229
+ },
230
+ 'ManagedLockbox collateral balance updated',
231
+ );
232
+
233
+ if (balanceInfo.valueUSD) {
234
+ warpRouteCollateralValue.labels(metrics).set(balanceInfo.valueUSD);
235
+ logger.info(
236
+ {
237
+ labels: metrics,
238
+ valueUSD: balanceInfo.valueUSD,
239
+ },
240
+ 'ManagedLockbox value updated for token',
241
+ );
242
+ }
243
+ }
244
+
245
+ export function updateNativeWalletBalanceMetrics(
246
+ balance: NativeWalletBalance,
247
+ logger: Logger,
248
+ ) {
249
+ walletBalanceGauge
250
+ .labels({
251
+ chain: balance.chain,
252
+ wallet_address: balance.walletAddress,
253
+ wallet_name: balance.walletName,
254
+ token_symbol: 'Native',
255
+ token_name: 'Native',
256
+ })
257
+ .set(balance.balance);
258
+ logger.info(
259
+ {
260
+ balanceInfo: balance,
261
+ },
262
+ 'Native wallet balance updated',
263
+ );
264
+ }
265
+
266
+ export function updateXERC20LimitsMetrics(
267
+ token: Token,
268
+ limits: XERC20Limit,
269
+ bridgeAddress: Address,
270
+ bridgeLabel: string,
271
+ xERC20Address: Address,
272
+ logger: Logger,
273
+ ) {
274
+ const labels = {
275
+ chain_name: token.chainName,
276
+ token_name: token.name,
277
+ bridge_address: bridgeAddress,
278
+ token_address: xERC20Address,
279
+ bridge_label: bridgeLabel,
280
+ };
281
+
282
+ for (const [limitType, limit] of Object.entries(limits)) {
283
+ xERC20LimitsGauge
284
+ .labels({
285
+ ...labels,
286
+ limit_type: limitType,
287
+ })
288
+ .set(limit);
289
+ }
290
+
291
+ logger.info(
292
+ {
293
+ ...labels,
294
+ limits,
295
+ },
296
+ 'xERC20 limits updated for bridge on token',
297
+ );
298
+ }
@@ -0,0 +1,27 @@
1
+ import { ChainName } from '@hyperlane-xyz/sdk';
2
+ import { Address } from '@hyperlane-xyz/utils';
3
+
4
+ export interface XERC20Limit {
5
+ mint: number;
6
+ burn: number;
7
+ mintMax: number;
8
+ burnMax: number;
9
+ }
10
+
11
+ export interface XERC20Info {
12
+ limits: XERC20Limit;
13
+ xERC20Address: Address;
14
+ }
15
+
16
+ export interface WarpRouteBalance {
17
+ balance: number;
18
+ valueUSD?: number;
19
+ tokenAddress: Address;
20
+ }
21
+
22
+ export interface NativeWalletBalance {
23
+ chain: ChainName;
24
+ walletAddress: Address;
25
+ walletName: string;
26
+ balance: number;
27
+ }
@@ -0,0 +1,33 @@
1
+ import http from 'http';
2
+ import { Registry } from 'prom-client';
3
+
4
+ import { rootLogger } from '@hyperlane-xyz/utils';
5
+
6
+ const logger = rootLogger.child({ module: 'metrics' });
7
+
8
+ /**
9
+ * Start a simple HTTP server to host metrics. This just takes the registry and dumps the text
10
+ * string to people who request `GET /metrics`.
11
+ *
12
+ * PROMETHEUS_PORT env var is used to determine what port to host on, defaults to 9090.
13
+ */
14
+ export function startMetricsServer(register: Registry): http.Server {
15
+ return http
16
+ .createServer((req, res) => {
17
+ if (req.url != '/metrics') {
18
+ return res.writeHead(404, 'Invalid url').end();
19
+ }
20
+
21
+ if (req.method != 'GET') {
22
+ return res.writeHead(405, 'Invalid method').end();
23
+ }
24
+
25
+ return register
26
+ .metrics()
27
+ .then((metricsStr) => {
28
+ res.writeHead(200, { 'Content-Type': 'text/plain' }).end(metricsStr);
29
+ })
30
+ .catch((err) => logger.error(err));
31
+ })
32
+ .listen(parseInt(process.env['PROMETHEUS_PORT'] || '9090'));
33
+ }
@@ -0,0 +1,174 @@
1
+ import EventEmitter from 'events';
2
+ import { Logger } from 'pino';
3
+
4
+ import type { Token, WarpCore } from '@hyperlane-xyz/sdk';
5
+ import { sleep } from '@hyperlane-xyz/utils';
6
+
7
+ import {
8
+ type IMonitor,
9
+ type MonitorEvent,
10
+ MonitorEventType,
11
+ MonitorPollingError,
12
+ MonitorStartError,
13
+ } from '../interfaces/IMonitor.js';
14
+
15
+ /**
16
+ * Simple monitor implementation that polls warp route collateral balances and emits them as MonitorEvent.
17
+ */
18
+ export class Monitor implements IMonitor {
19
+ private readonly emitter = new EventEmitter();
20
+ private isMonitorRunning = false;
21
+ private resolveStop: (() => void) | null = null;
22
+ private stopPromise: Promise<void> | null = null;
23
+
24
+ /**
25
+ * @param checkFrequency - The frequency to poll balances in ms.
26
+ */
27
+ constructor(
28
+ private readonly checkFrequency: number,
29
+ private readonly warpCore: WarpCore,
30
+ private readonly logger: Logger,
31
+ ) {}
32
+
33
+ // overloads from IMonitor
34
+ on(
35
+ eventName: MonitorEventType.TokenInfo,
36
+ fn: (event: MonitorEvent) => void,
37
+ ): this;
38
+ on(eventName: MonitorEventType.Error, fn: (event: Error) => void): this;
39
+ on(eventName: MonitorEventType.Start, fn: () => void): this;
40
+ on(eventName: string, fn: (...args: any[]) => void): this {
41
+ this.emitter.on(eventName, fn);
42
+ return this;
43
+ }
44
+
45
+ async start() {
46
+ if (this.isMonitorRunning) {
47
+ // Cannot start the same monitor multiple times
48
+ this.emitter.emit(
49
+ MonitorEventType.Error,
50
+ new MonitorStartError('Monitor already running'),
51
+ );
52
+ return;
53
+ }
54
+
55
+ try {
56
+ this.isMonitorRunning = true;
57
+ this.logger.debug(
58
+ { checkFrequency: this.checkFrequency },
59
+ 'Monitor started',
60
+ );
61
+ this.emitter.emit(MonitorEventType.Start);
62
+
63
+ while (this.isMonitorRunning) {
64
+ try {
65
+ this.logger.debug('Polling cycle started');
66
+ const event: MonitorEvent = {
67
+ tokensInfo: [],
68
+ };
69
+
70
+ for (const token of this.warpCore.tokens) {
71
+ this.logger.debug(
72
+ {
73
+ chain: token.chainName,
74
+ tokenSymbol: token.symbol,
75
+ tokenAddress: token.addressOrDenom,
76
+ },
77
+ 'Checking token',
78
+ );
79
+ const bridgedSupply = await this.getTokenBridgedSupply(token);
80
+
81
+ event.tokensInfo.push({
82
+ token,
83
+ bridgedSupply,
84
+ });
85
+ }
86
+
87
+ // Emit the event warp routes info
88
+ this.emitter.emit(MonitorEventType.TokenInfo, event);
89
+ this.logger.debug('Polling cycle completed');
90
+ } catch (error) {
91
+ this.emitter.emit(
92
+ MonitorEventType.Error,
93
+ new MonitorPollingError(
94
+ `Error during monitor execution cycle: ${(error as Error).message}`,
95
+ error as Error,
96
+ ),
97
+ );
98
+ }
99
+
100
+ // Wait for the specified check frequency before the next iteration
101
+ await sleep(this.checkFrequency);
102
+ }
103
+ } catch (error) {
104
+ this.emitter.emit(
105
+ MonitorEventType.Error,
106
+ new MonitorStartError(
107
+ `Error starting monitor: ${(error as Error).message}`,
108
+ error as Error,
109
+ ),
110
+ );
111
+ }
112
+
113
+ // After the loop has been gracefully terminated, we can clean up.
114
+ this.emitter.removeAllListeners();
115
+ this.logger.info('Monitor stopped');
116
+
117
+ // If stop() was called, resolve the promise to signal that we're done.
118
+ if (this.resolveStop) {
119
+ this.resolveStop();
120
+ this.resolveStop = null;
121
+ this.stopPromise = null;
122
+ }
123
+ }
124
+
125
+ private async getTokenBridgedSupply(
126
+ token: Token,
127
+ ): Promise<bigint | undefined> {
128
+ if (!token.isHypToken()) {
129
+ this.logger.warn(
130
+ {
131
+ chain: token.chainName,
132
+ tokenSymbol: token.symbol,
133
+ tokenAddress: token.addressOrDenom,
134
+ },
135
+ 'Cannot get bridged balance for a non-Hyperlane token',
136
+ );
137
+ return;
138
+ }
139
+
140
+ const adapter = token.getHypAdapter(this.warpCore.multiProvider);
141
+ const bridgedSupply = await adapter.getBridgedSupply();
142
+
143
+ if (bridgedSupply === undefined) {
144
+ this.logger.warn(
145
+ {
146
+ chain: token.chainName,
147
+ tokenSymbol: token.symbol,
148
+ tokenAddress: token.addressOrDenom,
149
+ },
150
+ 'Bridged supply not found for token',
151
+ );
152
+ }
153
+
154
+ return bridgedSupply;
155
+ }
156
+
157
+ stop(): Promise<void> {
158
+ if (!this.isMonitorRunning) return Promise.resolve();
159
+
160
+ // If stop is already in progress, return the existing promise
161
+ if (this.stopPromise) return this.stopPromise;
162
+
163
+ this.logger.info('Stopping monitor...');
164
+ // Signal the while loop to terminate after its current iteration
165
+ this.isMonitorRunning = false;
166
+
167
+ // Create a promise that will be resolved by the start() method
168
+ // once the loop and cleanup are complete.
169
+ this.stopPromise = new Promise((resolve) => {
170
+ this.resolveStop = resolve;
171
+ });
172
+ return this.stopPromise;
173
+ }
174
+ }
package/src/service.ts ADDED
@@ -0,0 +1,154 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * Hyperlane Rebalancer Service Entry Point
4
+ *
5
+ * This is the main entry point for running the rebalancer as a standalone service
6
+ * in Kubernetes or other container environments. It reads configuration from
7
+ * environment variables and files, then starts the rebalancer in daemon mode.
8
+ *
9
+ * Environment Variables:
10
+ * - REBALANCER_CONFIG_FILE: Path to the rebalancer configuration YAML file (required)
11
+ * - HYP_KEY: Private key for signing transactions (required)
12
+ * - COINGECKO_API_KEY: API key for CoinGecko price fetching (optional, for metrics)
13
+ * - CHECK_FREQUENCY: Balance check frequency in ms (default: 60000)
14
+ * - WITH_METRICS: Enable Prometheus metrics (default: "true")
15
+ * - MONITOR_ONLY: Run in monitor-only mode without executing transactions (default: "false")
16
+ * - LOG_LEVEL: Logging level (default: "info") - supported by pino
17
+ *
18
+ * Usage:
19
+ * node dist/service.js
20
+ * REBALANCER_CONFIG_FILE=/config/rebalancer.yaml HYP_KEY=0x... node dist/service.js
21
+ */
22
+ import { Wallet } from 'ethers';
23
+ import fs from 'fs';
24
+ import path from 'path';
25
+ import { fileURLToPath } from 'url';
26
+
27
+ import { getRegistry } from '@hyperlane-xyz/registry/fs';
28
+ import { MultiProtocolProvider, MultiProvider } from '@hyperlane-xyz/sdk';
29
+ import { createServiceLogger, rootLogger } from '@hyperlane-xyz/utils';
30
+
31
+ import { RebalancerConfig } from './config/RebalancerConfig.js';
32
+ import { RebalancerService } from './core/RebalancerService.js';
33
+
34
+ function getVersion(): string {
35
+ try {
36
+ const __dirname = fileURLToPath(new URL('.', import.meta.url));
37
+ const packageJson = JSON.parse(
38
+ fs.readFileSync(path.join(__dirname, '../package.json'), 'utf-8'),
39
+ );
40
+ return packageJson.version;
41
+ } catch (error) {
42
+ rootLogger.warn({ error }, 'Could not read version from package.json');
43
+ return 'unknown';
44
+ }
45
+ }
46
+
47
+ async function main(): Promise<void> {
48
+ const VERSION = getVersion();
49
+ // Validate required environment variables
50
+ const configFile = process.env.REBALANCER_CONFIG_FILE;
51
+ if (!configFile) {
52
+ rootLogger.error('REBALANCER_CONFIG_FILE environment variable is required');
53
+ process.exit(1);
54
+ }
55
+
56
+ const privateKey = process.env.HYP_KEY;
57
+ if (!privateKey) {
58
+ rootLogger.error('HYP_KEY environment variable is required');
59
+ process.exit(1);
60
+ }
61
+
62
+ // Parse optional environment variables
63
+ let checkFrequency = 60_000;
64
+ if (process.env.CHECK_FREQUENCY) {
65
+ const parsed = parseInt(process.env.CHECK_FREQUENCY, 10);
66
+ if (isNaN(parsed) || parsed <= 0) {
67
+ rootLogger.error(
68
+ 'CHECK_FREQUENCY must be a positive number (milliseconds)',
69
+ );
70
+ process.exit(1);
71
+ }
72
+ checkFrequency = parsed;
73
+ }
74
+
75
+ const withMetrics = process.env.WITH_METRICS !== 'false';
76
+ const monitorOnly = process.env.MONITOR_ONLY === 'true';
77
+ const coingeckoApiKey = process.env.COINGECKO_API_KEY;
78
+
79
+ // Create logger (uses LOG_LEVEL environment variable for level configuration)
80
+ const logger = await createServiceLogger({
81
+ service: 'rebalancer',
82
+ version: VERSION,
83
+ });
84
+
85
+ logger.info(
86
+ {
87
+ version: VERSION,
88
+ configFile,
89
+ checkFrequency,
90
+ withMetrics,
91
+ monitorOnly,
92
+ },
93
+ 'Starting Hyperlane Rebalancer Service',
94
+ );
95
+
96
+ try {
97
+ // Load rebalancer configuration
98
+ const rebalancerConfig = RebalancerConfig.load(configFile);
99
+ logger.info('✅ Loaded rebalancer configuration');
100
+
101
+ // Initialize registry (uses default registry URIs)
102
+ const registry = getRegistry({
103
+ registryUris: [],
104
+ enableProxy: false,
105
+ logger: rootLogger,
106
+ });
107
+ logger.info('✅ Initialized registry');
108
+
109
+ // Get chain metadata from registry
110
+ const chainMetadata = await registry.getMetadata();
111
+ logger.info(
112
+ `✅ Loaded metadata for ${Object.keys(chainMetadata).length} chains`,
113
+ );
114
+
115
+ // Create MultiProvider with signer
116
+ const multiProvider = new MultiProvider(chainMetadata);
117
+ const signer = new Wallet(privateKey);
118
+ multiProvider.setSharedSigner(signer);
119
+ logger.info('✅ Initialized MultiProvider with signer');
120
+
121
+ // Create MultiProtocolProvider
122
+ const multiProtocolProvider = new MultiProtocolProvider(chainMetadata);
123
+ logger.info('✅ Initialized MultiProtocolProvider');
124
+
125
+ // Create the rebalancer service
126
+ const service = new RebalancerService(
127
+ multiProvider,
128
+ multiProtocolProvider,
129
+ registry,
130
+ rebalancerConfig,
131
+ {
132
+ mode: 'daemon',
133
+ checkFrequency,
134
+ monitorOnly,
135
+ withMetrics,
136
+ coingeckoApiKey,
137
+ logger,
138
+ version: VERSION,
139
+ },
140
+ );
141
+
142
+ // Start the service
143
+ await service.start();
144
+ } catch (error) {
145
+ logger.error({ error }, 'Failed to start rebalancer service');
146
+ process.exit(1);
147
+ }
148
+ }
149
+
150
+ // Run the service
151
+ main().catch((error) => {
152
+ rootLogger.error({ error }, 'Fatal error');
153
+ process.exit(1);
154
+ });