@instadapp/interop-x 0.0.0-dev.c696e38 → 0.0.0-dev.cb34b2e

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 (92) hide show
  1. package/dist/package.json +9 -2
  2. package/dist/src/abi/aaveV2Resolver.json +832 -0
  3. package/dist/src/abi/balanceResolver.json +211 -0
  4. package/dist/src/abi/index.js +6 -0
  5. package/dist/src/abi/instList.json +232 -0
  6. package/dist/src/api/index.js +7 -0
  7. package/dist/src/constants/addresses.js +7 -1
  8. package/dist/src/constants/capPerChain.js +8 -0
  9. package/dist/src/constants/index.js +2 -0
  10. package/dist/src/constants/tokens.js +44 -44
  11. package/dist/src/constants/wrappedNativeToken.js +8 -0
  12. package/dist/src/crons/index.js +3 -0
  13. package/dist/src/crons/prices.js +16 -0
  14. package/dist/src/db/models/transaction.js +4 -0
  15. package/dist/src/errors/index.js +30 -0
  16. package/dist/src/gnosis/actions/aaveV2/source.js +2 -1
  17. package/dist/src/gnosis/actions/aaveV2/target.js +5 -3
  18. package/dist/src/index.js +2 -1
  19. package/dist/src/providers/index.js +17 -0
  20. package/dist/src/providers/retry-provider.js +45 -0
  21. package/dist/src/services/Prices.js +74 -0
  22. package/dist/src/services/index.js +8 -0
  23. package/dist/src/tasks/InteropX/{ProcessSubmitSubmitEvents.js → ProcessSubmitEvents.js} +106 -14
  24. package/dist/src/tasks/InteropX/ProcessValidateEvents.js +30 -10
  25. package/dist/src/tasks/InteropX/SyncLogExecuteEvents.js +113 -0
  26. package/dist/src/tasks/InteropX/SyncLogSubmitEvents.js +2 -1
  27. package/dist/src/tasks/InteropX/SyncLogValidateEvents.js +5 -4
  28. package/dist/src/tasks/index.js +7 -5
  29. package/dist/src/typechain/AaveV2Resolver.js +2 -0
  30. package/dist/src/typechain/BalanceResolver.js +2 -0
  31. package/dist/src/typechain/InstList.js +2 -0
  32. package/dist/src/typechain/factories/AaveV2Resolver__factory.js +1191 -0
  33. package/dist/src/typechain/factories/BalanceResolver__factory.js +228 -0
  34. package/dist/src/typechain/factories/InstList__factory.js +249 -0
  35. package/dist/src/typechain/factories/index.js +7 -1
  36. package/dist/src/typechain/index.js +7 -1
  37. package/dist/src/utils/async.js +18 -0
  38. package/dist/src/utils/dsa.js +24 -0
  39. package/dist/src/utils/formatting.js +67 -0
  40. package/dist/src/utils/gnosis.js +87 -0
  41. package/dist/src/utils/http.js +10 -0
  42. package/dist/src/utils/index.js +22 -220
  43. package/dist/src/utils/interop.js +16 -0
  44. package/dist/src/utils/tokens.js +22 -0
  45. package/dist/src/utils/validate.js +111 -0
  46. package/dist/src/utils/web3.js +93 -0
  47. package/package.json +9 -2
  48. package/src/abi/aaveV2Resolver.json +832 -0
  49. package/src/abi/balanceResolver.json +211 -0
  50. package/src/abi/index.ts +6 -0
  51. package/src/abi/instList.json +232 -0
  52. package/src/api/index.ts +8 -0
  53. package/src/constants/addresses.ts +18 -2
  54. package/src/constants/capPerChain.ts +5 -0
  55. package/src/constants/index.ts +2 -0
  56. package/src/constants/tokens.ts +44 -44
  57. package/src/constants/wrappedNativeToken.ts +5 -0
  58. package/src/crons/index.ts +1 -0
  59. package/src/crons/prices.ts +12 -0
  60. package/src/db/models/transaction.ts +21 -0
  61. package/src/errors/index.ts +26 -0
  62. package/src/gnosis/actions/aaveV2/source.ts +2 -1
  63. package/src/gnosis/actions/aaveV2/target.ts +13 -3
  64. package/src/index.ts +1 -0
  65. package/src/providers/index.ts +1 -0
  66. package/src/providers/retry-provider.ts +51 -0
  67. package/src/services/Prices.ts +89 -0
  68. package/src/services/index.ts +1 -0
  69. package/src/tasks/InteropX/{ProcessSubmitSubmitEvents.ts → ProcessSubmitEvents.ts} +133 -20
  70. package/src/tasks/InteropX/ProcessValidateEvents.ts +42 -19
  71. package/src/tasks/InteropX/SyncLogExecuteEvents.ts +161 -0
  72. package/src/tasks/InteropX/SyncLogSubmitEvents.ts +5 -6
  73. package/src/tasks/InteropX/SyncLogValidateEvents.ts +8 -9
  74. package/src/tasks/index.ts +8 -5
  75. package/src/typechain/AaveV2Resolver.ts +1017 -0
  76. package/src/typechain/BalanceResolver.ts +266 -0
  77. package/src/typechain/InstList.ts +402 -0
  78. package/src/typechain/factories/AaveV2Resolver__factory.ts +1198 -0
  79. package/src/typechain/factories/BalanceResolver__factory.ts +235 -0
  80. package/src/typechain/factories/InstList__factory.ts +253 -0
  81. package/src/typechain/factories/index.ts +3 -0
  82. package/src/typechain/index.ts +6 -0
  83. package/src/utils/async.ts +22 -0
  84. package/src/utils/dsa.ts +30 -0
  85. package/src/utils/formatting.ts +68 -0
  86. package/src/utils/gnosis.ts +166 -0
  87. package/src/utils/http.ts +6 -0
  88. package/src/utils/index.ts +9 -365
  89. package/src/utils/interop.ts +28 -0
  90. package/src/utils/tokens.ts +21 -0
  91. package/src/utils/validate.ts +174 -0
  92. package/src/utils/web3.ts +132 -0
@@ -0,0 +1,174 @@
1
+ import abi from "@/abi";
2
+ import { addresses, capPerChain, tokens } from "@/constants";
3
+ import { InvalidChaindIdError, LowLiquidityError } from "@/errors";
4
+ import { ethers } from "ethers";
5
+ import { getContract, getRpcProviderUrl } from "./web3";
6
+ import BigNumber from "bignumber.js";
7
+ import { chainIdToName, formatUsd } from "./formatting";
8
+ import { isNativeWrappedToken } from "./tokens";
9
+ import { Prices } from "@/services";
10
+ import { JsonRpcRetryProvider } from "@/providers";
11
+
12
+ export const validateChains = ({ sourceChainId, targetChainId }) => {
13
+ if (!sourceChainId) {
14
+ throw new InvalidChaindIdError("Source chain id is required");
15
+ }
16
+
17
+ if (!targetChainId) {
18
+ throw new InvalidChaindIdError("Target chain id is required");
19
+ }
20
+
21
+ if (sourceChainId == targetChainId) {
22
+ throw new InvalidChaindIdError("Source and target chain cannot be the same");
23
+ }
24
+
25
+ if (!addresses[sourceChainId]) {
26
+ throw new InvalidChaindIdError("Invalid source chain id");
27
+ }
28
+
29
+ if (!addresses[targetChainId]) {
30
+ throw new InvalidChaindIdError("Invalid target chain id");
31
+ }
32
+ };
33
+
34
+
35
+ export const validateSourceLiquidity = async ({
36
+ position,
37
+ sourceChainId,
38
+ isSupply = false,
39
+ sourceProvider = null as unknown as JsonRpcRetryProvider,
40
+ }) => {
41
+ sourceProvider = sourceProvider || new JsonRpcRetryProvider(getRpcProviderUrl(sourceChainId));
42
+
43
+
44
+ const sourceBalanceResolverContract = getContract(
45
+ addresses[sourceChainId].balanceResolver,
46
+ abi.balanceResolver,
47
+ sourceProvider,
48
+ )
49
+
50
+ const srcBalance = await sourceBalanceResolverContract.methods
51
+ .checkLiquidity(position, addresses[sourceChainId].interopX, isSupply, false)
52
+ .call();
53
+
54
+ if (!srcBalance.isOk) {
55
+
56
+ let liquidityRequired = ""
57
+
58
+ srcBalance.withdraw.forEach(a => {
59
+ const token = tokens[sourceChainId].find(t => t.address.toLowerCase() === a.token.toLowerCase());
60
+
61
+ if (token && (new BigNumber(a.liquidityShort).isZero()) == false) {
62
+ liquidityRequired += `- Need ${new BigNumber(a.liquidityShort).dividedBy(10 ** token.decimals).toFormat()} ${token.symbol}\n`
63
+ }
64
+ })
65
+
66
+ throw new LowLiquidityError(`Liquidity low on ${chainIdToName(sourceChainId)}: \n` + liquidityRequired);
67
+ }
68
+ };
69
+
70
+ export const validateTargetLiquidity = async ({
71
+ position,
72
+ targetChainId,
73
+ targetProvider = null as unknown as JsonRpcRetryProvider,
74
+ }) => {
75
+ targetProvider = targetProvider || new JsonRpcRetryProvider(getRpcProviderUrl(targetChainId));
76
+
77
+
78
+ const sourceBalanceResolverContract = getContract(
79
+ addresses[targetChainId].balanceResolver,
80
+ abi.balanceResolver,
81
+ targetProvider,
82
+ )
83
+
84
+ const targetBalance = await sourceBalanceResolverContract.methods
85
+ .checkLiquidity(position, addresses[targetChainId].interopX, true, true)
86
+ .call();
87
+
88
+ if (!targetBalance.isOk) {
89
+
90
+ let liquidityRequired = ""
91
+
92
+ targetBalance.supply.forEach(a => {
93
+ const token = tokens[targetChainId].find(t => t.address.toLowerCase() === a.token.toLowerCase());
94
+
95
+
96
+ if (token) {
97
+ liquidityRequired += `- Need ${new BigNumber(a.liquidityShort).dividedBy(10 ** token.decimals).toFormat()} ${token.symbol}\n`
98
+ }
99
+ })
100
+
101
+ throw new LowLiquidityError(`Liquidity low on ${chainIdToName(targetChainId)}: \n` + liquidityRequired);
102
+ }
103
+ };
104
+
105
+ const mapTokenAddressToMainnetToken = (
106
+ address: string,
107
+ targetChainId: number|string
108
+ ) => {
109
+ targetChainId = String(targetChainId);
110
+ const token = tokens[targetChainId].find(
111
+ (t) => t.address.toLowerCase() === address.toLowerCase()
112
+ );
113
+
114
+ if (targetChainId === "1") {
115
+ return token;
116
+ }
117
+
118
+ if (!token) {
119
+ return null;
120
+ }
121
+
122
+ return tokens["1"].find((t) => t.symbol === token.symbol);
123
+ };
124
+
125
+ export const validateLiquidityCap = async (
126
+ position: any,
127
+ sourceChainId: number|string,
128
+ targetChainId: number|string
129
+ ) => {
130
+ const cap = capPerChain[targetChainId] || 1_000_000;
131
+ const mainnetPrices = Prices.getMainnetPrices();
132
+ const nativeTokenPrice = Prices.getNetworkTokenPrices()[targetChainId];
133
+
134
+ let total = new BigNumber(0);
135
+
136
+ const supplyPosition = position[0];
137
+
138
+ for (const [srcTokenAddress, tokenAddress, amount] of supplyPosition) {
139
+ let tokenPrice;
140
+ let token;
141
+
142
+ if (isNativeWrappedToken(srcTokenAddress, sourceChainId)) {
143
+ tokenPrice = nativeTokenPrice;
144
+ token = tokens[targetChainId].find(t => t.address === '0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE');
145
+ } else {
146
+ const mainnetToken = mapTokenAddressToMainnetToken(
147
+ tokenAddress,
148
+ targetChainId
149
+ );
150
+ if (!mainnetToken) {
151
+ throw new Error(`Token ${tokenAddress} is not valid.`);
152
+ }
153
+
154
+ tokenPrice = mainnetPrices[mainnetToken.address];
155
+
156
+ if (!tokenPrice) {
157
+ throw new Error(`Token ${tokenAddress} is not valid!`);
158
+ }
159
+
160
+ token = mainnetToken;
161
+ }
162
+
163
+
164
+ total = total.plus((tokenPrice * amount) / 10 ** token.decimals);
165
+ }
166
+
167
+ if (total.isGreaterThan(cap)) {
168
+ throw new Error(
169
+ `Total amount of tokens in the transaction is greater than the maximum limit of estimated ${formatUsd(
170
+ cap
171
+ )}`
172
+ );
173
+ }
174
+ };
@@ -0,0 +1,132 @@
1
+ import { UnsupportedChaindIdError } from "@/errors";
2
+ import { ChainId } from "@/types";
3
+ import retry from "async-retry";
4
+ import { ethers } from "ethers";
5
+
6
+ export const getRpcProviderUrl = (chainId: ChainId) => {
7
+ switch (chainId) {
8
+ case 1:
9
+ return "https://rpc.ankr.com/eth";
10
+ case 137:
11
+ return "https://rpc.ankr.com/polygon";
12
+ case 43114:
13
+ return "https://rpc.ankr.com/avalanche";
14
+ default:
15
+ throw new UnsupportedChaindIdError(chainId);
16
+ }
17
+ };
18
+
19
+
20
+ export class ContractError extends Error {
21
+ method: string;
22
+ address: string;
23
+ args: any[];
24
+ }
25
+
26
+ export function getContract<TContract extends ethers.Contract>(
27
+ address: string,
28
+ contractInterface: ethers.ContractInterface | any,
29
+ signerOrProvider: ethers.Signer | ethers.providers.Provider
30
+ ) {
31
+ if (
32
+ !ethers.utils.getAddress(address) ||
33
+ address === ethers.constants.AddressZero
34
+ ) {
35
+ throw Error(`Invalid 'address' parameter '${address}'.`);
36
+ }
37
+
38
+ const contract = new ethers.Contract(
39
+ address,
40
+ contractInterface,
41
+ signerOrProvider
42
+ ) as TContract;
43
+
44
+ // Make sure the contract properties is writable
45
+ const desc = Object.getOwnPropertyDescriptor(contract, "functions");
46
+
47
+ if (!desc || desc.writable !== true) {
48
+ return contract;
49
+ }
50
+
51
+ return new Proxy(contract, {
52
+ get(target, prop, receiver) {
53
+ const value = Reflect.get(target, prop, receiver);
54
+
55
+ if (
56
+ typeof value === "function" &&
57
+ (contract.functions.hasOwnProperty(prop) ||
58
+ ["queryFilter"].includes(String(prop)))
59
+ ) {
60
+ let isConstant = false;
61
+
62
+ try {
63
+ isConstant = contract.interface.getFunction(String(prop)).constant;
64
+ } catch (error) { }
65
+
66
+ return async (...args: any[]) => {
67
+ try {
68
+ return await retry(
69
+ async () => await value.bind(contract)(...args),
70
+ { retries: isConstant ? 1 : 3 }
71
+ );
72
+ } catch (error) {
73
+ const err = new ContractError(
74
+ `Error calling "${String(prop)}" on "${address}": ${error.reason || error.message
75
+ }`
76
+ );
77
+
78
+ err.method = String(prop);
79
+ err.address = address;
80
+ err.args = [...args];
81
+
82
+ throw err;
83
+ }
84
+ };
85
+ }
86
+
87
+ if (
88
+ typeof value === "object" &&
89
+ [
90
+ "populateTransaction",
91
+ "estimateGas",
92
+ "functions",
93
+ "callStatic",
94
+ ].includes(String(prop))
95
+ ) {
96
+ const parentProp = String(prop);
97
+
98
+ return new Proxy(value, {
99
+ get(target, prop, receiver) {
100
+ const value = Reflect.get(target, prop, receiver);
101
+
102
+ if (typeof value === "function") {
103
+ return async (...args: any[]) => {
104
+ try {
105
+ return await retry(
106
+ async () => await value.bind(contract)(...args),
107
+ { retries: parentProp === "callStatic" ? 3 : 1 }
108
+ );
109
+ } catch (error) {
110
+ const err = new ContractError(
111
+ `Error calling "${String(
112
+ prop
113
+ )}" using "${parentProp}" on "${address}": ${error.reason || error.message
114
+ }`
115
+ );
116
+
117
+ err.method = String(prop);
118
+ err.address = address;
119
+ err.args = [...args];
120
+
121
+ throw err;
122
+ }
123
+ };
124
+ }
125
+ },
126
+ });
127
+ }
128
+
129
+ return value;
130
+ },
131
+ });
132
+ }