@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
@@ -6,6 +6,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
6
6
  const abi_1 = __importDefault(require("@/abi"));
7
7
  const config_1 = __importDefault(require("@/config"));
8
8
  const constants_1 = require("@/constants");
9
+ const providers_1 = require("@/providers");
9
10
  const utils_1 = require("@/utils");
10
11
  const ethers_1 = require("ethers");
11
12
  const ethers_multisend_1 = require("ethers-multisend");
@@ -13,7 +14,7 @@ async function default_1(transaction) {
13
14
  const transactions = [];
14
15
  const logs = [];
15
16
  const { position, actionId, actionIdHashHash, sourceSender, sourceDsaId, targetDsaId, sourceChainId, targetChainId, vnonce, metadata, } = transaction.submitEvent;
16
- const sourceChainProvider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(sourceChainId));
17
+ const sourceChainProvider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(sourceChainId));
17
18
  const sourceWallet = new ethers_1.ethers.Wallet(config_1.default.privateKey, sourceChainProvider);
18
19
  const dsaAddress = constants_1.addresses[sourceChainId].dsaAddress;
19
20
  const sourceUserAddress = Number(sourceDsaId) == 0 ? sourceSender : dsaAddress;
@@ -6,6 +6,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
6
6
  const abi_1 = __importDefault(require("@/abi"));
7
7
  const config_1 = __importDefault(require("@/config"));
8
8
  const constants_1 = require("@/constants");
9
+ const providers_1 = require("@/providers");
9
10
  const utils_1 = require("@/utils");
10
11
  const ethers_1 = require("ethers");
11
12
  const ethers_multisend_1 = require("ethers-multisend");
@@ -13,10 +14,11 @@ async function default_1(transaction) {
13
14
  const transactions = [];
14
15
  const logs = [];
15
16
  const { sourceSpells, position, actionId, sourceSender, sourceDsaId, targetDsaId, sourceChainId, targetChainId, vnonce, metadata, } = transaction.validateEvent;
16
- const targetChainProvider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(targetChainId));
17
+ const targetChainProvider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(targetChainId));
17
18
  const targetWallet = new ethers_1.ethers.Wallet(config_1.default.privateKey, targetChainProvider);
19
+ const targetInstListContract = (0, utils_1.getContract)(constants_1.addresses[targetChainId].instList, abi_1.default.instList, targetChainProvider);
20
+ const targetDsaAddress = await targetInstListContract.accountAddr(targetDsaId);
18
21
  const dsaAddress = constants_1.addresses[targetChainId].dsaAddress;
19
- const sourceUserAddress = Number(sourceDsaId) == 0 ? sourceSender : dsaAddress;
20
22
  const interopAddress = constants_1.addresses[targetChainId].interopX;
21
23
  const contract = (0, utils_1.getContract)(interopAddress, abi_1.default.interopX, targetWallet);
22
24
  const targetSpells = [];
@@ -34,7 +36,7 @@ async function default_1(transaction) {
34
36
  let spellDataBasicWithdraw = {
35
37
  connector: "BASIC-A",
36
38
  method: "withdraw",
37
- args: [supplyToken.targetToken, supplyToken.amount, sourceUserAddress, "0", "0"],
39
+ args: [supplyToken.targetToken, supplyToken.amount, targetDsaAddress, "0", "0"],
38
40
  };
39
41
  commonSpells.push({
40
42
  connector: spellDataBasicWithdraw.connector,
package/dist/src/index.js CHANGED
@@ -13,7 +13,7 @@ const package_json_1 = __importDefault(require("../package.json"));
13
13
  dotenv_1.default.config();
14
14
  const logger_1 = __importDefault(require("@/logger"));
15
15
  const logger = new logger_1.default('Process');
16
- const GIT_SHORT_HASH = 'c696e38';
16
+ const GIT_SHORT_HASH = 'cb34b2e';
17
17
  const printUsage = () => {
18
18
  console.log();
19
19
  console.log(`Interop X Node (v${package_json_1.default.version} - rev.${GIT_SHORT_HASH})`);
@@ -69,6 +69,7 @@ const net_1 = require("@/net");
69
69
  const api_1 = require("@/api");
70
70
  const db_1 = require("./db");
71
71
  const utils_1 = require("./utils");
72
+ require("./crons");
72
73
  async function main() {
73
74
  (0, net_1.startPeer)({});
74
75
  const tasks = new tasks_1.Tasks();
@@ -0,0 +1,17 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./retry-provider"), exports);
@@ -0,0 +1,45 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.JsonRpcRetryProvider = exports.retryOperation = exports.promiseTimeout = void 0;
7
+ const ethers_1 = require("ethers");
8
+ const waait_1 = __importDefault(require("waait"));
9
+ const bluebird_1 = __importDefault(require("bluebird"));
10
+ function promiseTimeout(ms, promise) {
11
+ return bluebird_1.default.resolve(promise).timeout(ms);
12
+ }
13
+ exports.promiseTimeout = promiseTimeout;
14
+ function retryOperation(retriesLeft, operation, options) {
15
+ return new Promise((resolve, reject) => {
16
+ const { timeouts } = options;
17
+ // Find the timeout for this specific iteration
18
+ const timeout = timeouts[timeouts.length - retriesLeft];
19
+ // Wrap the original operation in a timeout
20
+ const execution = promiseTimeout(timeout, operation());
21
+ // If the promise is successful, resolve it and bubble the result up
22
+ return execution.then(resolve).catch((reason) => {
23
+ // If there are any retries left, we call the same retryOperation function again,
24
+ // but decrementing the number of retries left by 1
25
+ if (retriesLeft - 1 > 0) {
26
+ // Delay the new attempt slightly
27
+ return (0, waait_1.default)(options.delay || 50)
28
+ .then(retryOperation.bind(null, retriesLeft - 1, operation, options))
29
+ .then(resolve)
30
+ .catch(reject);
31
+ }
32
+ // Reject (and bubble the result up) if there are no more retries
33
+ return reject(reason);
34
+ });
35
+ });
36
+ }
37
+ exports.retryOperation = retryOperation;
38
+ class JsonRpcRetryProvider extends ethers_1.ethers.providers.JsonRpcProvider {
39
+ perform(method, params) {
40
+ const timeouts = [5000, 10000];
41
+ const operation = () => super.perform(method, params);
42
+ return retryOperation(2, operation, { timeouts, delay: 50 });
43
+ }
44
+ }
45
+ exports.JsonRpcRetryProvider = JsonRpcRetryProvider;
@@ -0,0 +1,74 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const web3_utils_1 = __importDefault(require("web3-utils"));
7
+ const path_1 = __importDefault(require("path"));
8
+ const fs_1 = __importDefault(require("fs"));
9
+ const constants_1 = require("@/constants");
10
+ const expand_home_dir_1 = __importDefault(require("expand-home-dir"));
11
+ const config_1 = __importDefault(require("@/config"));
12
+ const utils_1 = require("@/utils");
13
+ const basePath = (0, expand_home_dir_1.default)(`~/.interop-x/data/${config_1.default.publicAddress}/${config_1.default.staging ? 'staging' : ''}`);
14
+ const mainnetPricesFilePath = path_1.default.resolve(basePath, "./mainnetPrices.json");
15
+ const networkTokenPricesFilePath = path_1.default.resolve(basePath, "./networkTokenPrices.json");
16
+ class Prices {
17
+ static async fetch() {
18
+ if (!fs_1.default.existsSync(mainnetPricesFilePath)) {
19
+ fs_1.default.writeFileSync(mainnetPricesFilePath, JSON.stringify({}));
20
+ }
21
+ if (!fs_1.default.existsSync(networkTokenPricesFilePath)) {
22
+ fs_1.default.writeFileSync(networkTokenPricesFilePath, JSON.stringify({}));
23
+ }
24
+ this.mainnetPrices = JSON.parse(fs_1.default.readFileSync(mainnetPricesFilePath, "utf8"));
25
+ this.networkTokenPrices = JSON.parse(fs_1.default.readFileSync(networkTokenPricesFilePath, "utf8"));
26
+ try {
27
+ const path = constants_1.tokens["1"]
28
+ .filter((a) => a.symbol !== "ETH")
29
+ .map((token) => token.address)
30
+ .join(",");
31
+ const [response, ethResponse, avaxResponse, polygonResponse] = await Promise.all([
32
+ utils_1.http.get("https://api.coingecko.com/api/v3/simple/token_price/ethereum", {
33
+ params: { contract_addresses: path, vs_currencies: "usd" },
34
+ }),
35
+ utils_1.http.get("https://api.coingecko.com/api/v3/simple/price", {
36
+ params: { ids: "ethereum", vs_currencies: "usd" },
37
+ }),
38
+ utils_1.http.get("https://api.coingecko.com/api/v3/simple/price", {
39
+ params: { ids: "avalanche-2", vs_currencies: "usd" },
40
+ }),
41
+ utils_1.http.get("https://api.coingecko.com/api/v3/simple/price", {
42
+ params: { ids: "matic-network", vs_currencies: "usd" },
43
+ }),
44
+ ]);
45
+ this.networkTokenPrices = {
46
+ "1": ethResponse.data.ethereum.usd.toString(),
47
+ "137": polygonResponse.data["matic-network"].usd.toString(),
48
+ "43114": avaxResponse.data["avalanche-2"].usd.toString(),
49
+ };
50
+ const data = response.data;
51
+ for (const key in data) {
52
+ const _key = web3_utils_1.default.toChecksumAddress(key);
53
+ if (!data[key].usd)
54
+ continue;
55
+ this.mainnetPrices[_key] = data[key].usd.toString();
56
+ }
57
+ this.mainnetPrices["0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE"] =
58
+ ethResponse.data.ethereum.usd.toString();
59
+ }
60
+ catch (error) { }
61
+ fs_1.default.writeFileSync(mainnetPricesFilePath, JSON.stringify(this.mainnetPrices));
62
+ fs_1.default.writeFileSync(networkTokenPricesFilePath, JSON.stringify(this.networkTokenPrices));
63
+ return this.mainnetPrices;
64
+ }
65
+ static getMainnetPrices() {
66
+ return this.mainnetPrices;
67
+ }
68
+ static getNetworkTokenPrices() {
69
+ return this.networkTokenPrices;
70
+ }
71
+ }
72
+ Prices.mainnetPrices = {};
73
+ Prices.networkTokenPrices = {};
74
+ exports.default = Prices;
@@ -0,0 +1,8 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.Prices = void 0;
7
+ var Prices_1 = require("./Prices");
8
+ Object.defineProperty(exports, "Prices", { enumerable: true, get: function () { return __importDefault(Prices_1).default; } });
@@ -16,10 +16,14 @@ const sequelize_1 = require("sequelize");
16
16
  const gnosis_1 = require("@/gnosis");
17
17
  const net_1 = require("@/net");
18
18
  const waait_1 = __importDefault(require("waait"));
19
- class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
19
+ const errors_1 = require("@/errors");
20
+ const bignumber_js_1 = require("bignumber.js");
21
+ const dedent_1 = __importDefault(require("dedent"));
22
+ const providers_1 = require("@/providers");
23
+ class ProcessSubmitEvents extends BaseTask_1.BaseTask {
20
24
  constructor({ chainId }) {
21
25
  super({
22
- logger: new logger_1.default("InteropX::ProcessSubmitSubmitEvents"),
26
+ logger: new logger_1.default("InteropX::ProcessSubmitEvents"),
23
27
  });
24
28
  this.leadNodeOnly = true;
25
29
  this.blockConfirmationsCount = 12;
@@ -54,6 +58,68 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
54
58
  this.logger.debug(`Processing transaction ${transaction.transactionHash}`);
55
59
  transaction.sourceStatus = "proccessing";
56
60
  await transaction.save();
61
+ const { submitEvent: { position, sourceChainId, targetChainId } } = transaction;
62
+ try {
63
+ (0, utils_1.validateChains)({ sourceChainId, targetChainId });
64
+ }
65
+ catch (error) {
66
+ transaction.sourceErrors = [error.message];
67
+ transaction.sourceStatus = "failed";
68
+ transaction.targetStatus = "failed";
69
+ transaction.status = "failed";
70
+ await transaction.save();
71
+ return;
72
+ }
73
+ const walletBalance = (await this.sourceWallet.getBalance()).toString();
74
+ const currentGasPrice = (await this.sourceWallet.getGasPrice()).toString();
75
+ const minBalanceRequired = new bignumber_js_1.BigNumber(currentGasPrice)
76
+ .multipliedBy(4000000)
77
+ .multipliedBy(2); // 2x balance
78
+ if (new bignumber_js_1.BigNumber(walletBalance).isLessThan(minBalanceRequired)) {
79
+ console.log((0, dedent_1.default) `Not enough balance in wallet ${(0, utils_1.getExplorerUrl)(sourceChainId, '/address/' + this.sourceWallet.address)} on ${(0, utils_1.chainIdToName)(sourceChainId)}
80
+ Balance: ${ethers_1.ethers.utils.parseEther(walletBalance)} ${(0, utils_1.getChainIdNativeSymbol)(sourceChainId)}
81
+ Required Balance: ${ethers_1.ethers.utils.parseEther(minBalanceRequired.toString())}
82
+ Require more: ${ethers_1.ethers.utils.parseEther(minBalanceRequired.minus(walletBalance).toString())}`);
83
+ transaction.sourceDelayUntil = (0, moment_1.default)().add({ minutes: 5 }).toDate();
84
+ transaction.sourceStatus = "pending";
85
+ await transaction.save();
86
+ return;
87
+ }
88
+ try {
89
+ await (0, utils_1.validateLiquidityCap)(position, sourceChainId, targetChainId);
90
+ }
91
+ catch (error) {
92
+ transaction.sourceErrors = [error.message];
93
+ transaction.sourceStatus = "failed";
94
+ transaction.targetStatus = "failed";
95
+ transaction.status = "failed";
96
+ await transaction.save();
97
+ return;
98
+ }
99
+ try {
100
+ await (0, utils_1.validateSourceLiquidity)({
101
+ position,
102
+ sourceChainId,
103
+ sourceProvider: this.sourceProvider,
104
+ });
105
+ }
106
+ catch (error) {
107
+ if (error instanceof errors_1.UnsupportedChaindIdError) {
108
+ console.log(`Dropping transaction ${transaction.transactionHash}`);
109
+ transaction.sourceErrors = [error.message];
110
+ transaction.sourceStatus = "failed";
111
+ transaction.targetStatus = "failed";
112
+ transaction.status = "failed";
113
+ await transaction.save();
114
+ }
115
+ else {
116
+ transaction.sourceStatus = "pending";
117
+ transaction.sourceDelayUntil = (0, moment_1.default)().add({ minutes: 5 }).toDate();
118
+ await transaction.save();
119
+ console.log("[validateSourceLiquidity][Warning]", error.message);
120
+ return;
121
+ }
122
+ }
57
123
  const ownersThreshold = await this.sourceGnosisContract.getThreshold();
58
124
  await (0, waait_1.default)(10000);
59
125
  this.logger.debug(`Build gnosis action for ${transaction.transactionHash}`);
@@ -62,7 +128,7 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
62
128
  ({ data, logs } = await (0, gnosis_1.buildGnosisAction)(transaction, "source"));
63
129
  }
64
130
  catch (error) {
65
- if (error instanceof utils_1.LiquidityError) {
131
+ if (error instanceof errors_1.LiquidityError) {
66
132
  await transaction.save();
67
133
  transaction.sourceDelayUntil = new Date(Date.now() + 60 * 5 * 1000);
68
134
  transaction.sourceStatus = "pending";
@@ -92,14 +158,26 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
92
158
  to: constants_1.addresses[transaction.sourceChainId].multisend,
93
159
  value: "0",
94
160
  }, this.sourceGnosisContract);
95
- const owners = await this.sourceGnosisContract
96
- .getOwners()
97
- .then((owners) => owners.map((owner) => owner.toLowerCase()));
98
- const ownerPeerIds = net_1.peerPool.activePeers
99
- .filter((peer) => owners.includes(peer.publicAddress.toLowerCase()))
100
- .map((peer) => peer.id);
101
- console.log(`Collecting signatures for execution ${transaction.transactionHash}`);
161
+ async function getGnosisOwnerPeerIds({ gnosisContract }) {
162
+ const owners = await gnosisContract
163
+ .getOwners()
164
+ .then((owners) => owners.map((owner) => owner.toLowerCase()));
165
+ return net_1.peerPool.activePeers
166
+ .filter((peer) => owners.includes(peer.publicAddress.toLowerCase()))
167
+ .map((peer) => peer.id);
168
+ }
169
+ const ownerPeerIds = await getGnosisOwnerPeerIds({
170
+ gnosisContract: this.sourceGnosisContract,
171
+ });
172
+ console.log(`Collecting signatures for execution ${transaction.transactionHash} `);
102
173
  console.log(ownerPeerIds);
174
+ const message = (0, utils_1.generateGnosisSignatureMessage)({
175
+ to: constants_1.addresses[data.chainId].multisend,
176
+ data,
177
+ chainId: this.chainId,
178
+ safeTxGas: gnosisTx.safeTxGas,
179
+ nonce: gnosisTx.safeNonce,
180
+ });
103
181
  const signatures = await net_1.protocol.requestSignatures({
104
182
  type: "source",
105
183
  transactionHash: transaction.transactionHash,
@@ -107,7 +185,21 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
107
185
  safeNonce: gnosisTx.nonce,
108
186
  chainId: this.chainId,
109
187
  }, ownerPeerIds);
110
- const validSignatures = signatures.filter((s) => !!s.data && s.data !== "0x");
188
+ const validSignatures = signatures
189
+ .filter((s) => !!s.data && s.data !== "0x")
190
+ .filter((s) => {
191
+ try {
192
+ const address = (0, utils_1.getGnosisSignatureAddress)({
193
+ message,
194
+ signature: s.data,
195
+ chainId: this.chainId,
196
+ });
197
+ return (address === null || address === void 0 ? void 0 : address.toLowerCase()) === s.signer.toLowerCase();
198
+ }
199
+ catch (error) {
200
+ return false;
201
+ }
202
+ });
111
203
  console.log({
112
204
  signatures,
113
205
  validSignatures,
@@ -120,7 +212,7 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
120
212
  transaction.sourceStatus = "pending";
121
213
  await transaction.save();
122
214
  const errorMessage = (_a = signatures.find((s) => !!s.error)) === null || _a === void 0 ? void 0 : _a.error;
123
- throw new Error(`Not enough signatures` + (errorMessage ? `: ${errorMessage}` : ""));
215
+ throw new Error(`Not enough signatures` + (errorMessage ? `: ${errorMessage} ` : ""));
124
216
  }
125
217
  console.log(`Executing transaction for execution ${transaction.transactionHash}`);
126
218
  const { data: txData } = await this.sourceGnosisContract.populateTransaction.execTransaction(gnosisTx.to, gnosisTx.value, gnosisTx.data, gnosisTx.operation, gnosisTx.safeTxGas, gnosisTx.baseGas, gnosisTx.gasPrice, gnosisTx.gasToken, gnosisTx.refundReceiver, (0, utils_1.buildSignatureBytes)(validSignatures));
@@ -170,10 +262,10 @@ class ProcessSubmitSubmitEvents extends BaseTask_1.BaseTask {
170
262
  }
171
263
  async start() {
172
264
  this.blockConfirmationsCount = constants_1.blockConfirmations[this.chainId] + 1;
173
- this.sourceProvider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
265
+ this.sourceProvider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
174
266
  this.sourceWallet = new ethers_1.ethers.Wallet(config_1.default.privateKey, this.sourceProvider);
175
267
  this.sourceGnosisContract = (0, utils_1.getContract)(constants_1.addresses[this.chainId].gnosisSafe, abi_1.default.gnosisSafe, this.sourceWallet);
176
268
  await super.start();
177
269
  }
178
270
  }
179
- exports.default = ProcessSubmitSubmitEvents;
271
+ exports.default = ProcessSubmitEvents;
@@ -16,6 +16,8 @@ const sequelize_1 = require("sequelize");
16
16
  const gnosis_1 = require("@/gnosis");
17
17
  const net_1 = require("@/net");
18
18
  const waait_1 = __importDefault(require("waait"));
19
+ const errors_1 = require("@/errors");
20
+ const providers_1 = require("@/providers");
19
21
  class ProcessValidateEvents extends BaseTask_1.BaseTask {
20
22
  constructor({ chainId }) {
21
23
  super({
@@ -57,7 +59,7 @@ class ProcessValidateEvents extends BaseTask_1.BaseTask {
57
59
  transaction.targetStatus = "proccessing";
58
60
  await transaction.save();
59
61
  const { sourceChainId, targetChainId } = transaction.validateEvent;
60
- const targetProvider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(targetChainId));
62
+ const targetProvider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(targetChainId));
61
63
  const targetWallet = new ethers_1.ethers.Wallet(config_1.default.privateKey, targetProvider);
62
64
  const targetGnosisContract = (0, utils_1.getContract)(constants_1.addresses[targetChainId].gnosisSafe, abi_1.default.gnosisSafe, targetWallet);
63
65
  const ownersThreshold = await targetGnosisContract.getThreshold();
@@ -68,7 +70,7 @@ class ProcessValidateEvents extends BaseTask_1.BaseTask {
68
70
  ({ data, logs } = await (0, gnosis_1.buildGnosisAction)(transaction, "target"));
69
71
  }
70
72
  catch (error) {
71
- if (error instanceof utils_1.LiquidityError) {
73
+ if (error instanceof errors_1.LiquidityError) {
72
74
  await transaction.save();
73
75
  transaction.targetDelayUntil = new Date(Date.now() + 60 * 5 * 1000);
74
76
  transaction.targetStatus = "pending";
@@ -97,14 +99,18 @@ class ProcessValidateEvents extends BaseTask_1.BaseTask {
97
99
  to: constants_1.addresses[transaction.targetChainId].multisend,
98
100
  value: "0",
99
101
  }, targetGnosisContract);
100
- const owners = await targetGnosisContract
101
- .getOwners()
102
- .then((owners) => owners.map((owner) => owner.toLowerCase()));
103
- const ownerPeerIds = net_1.peerPool.activePeers
104
- .filter((peer) => owners.includes(peer.publicAddress.toLowerCase()))
105
- .map((peer) => peer.id);
102
+ const ownerPeerIds = await (0, utils_1.getGnosisOwnerPeerIds)({
103
+ gnosisContract: targetGnosisContract,
104
+ });
106
105
  console.log(`Collecting signatures for execution ${transaction.transactionHash}`);
107
106
  console.log(ownerPeerIds);
107
+ const message = (0, utils_1.generateGnosisSignatureMessage)({
108
+ to: constants_1.addresses[data.chainId].multisend,
109
+ data,
110
+ chainId: this.chainId,
111
+ safeTxGas: gnosisTx.safeTxGas,
112
+ nonce: gnosisTx.safeNonce,
113
+ });
108
114
  const signatures = await net_1.protocol.requestSignatures({
109
115
  type: "target",
110
116
  transactionHash: transaction.transactionHash,
@@ -112,7 +118,21 @@ class ProcessValidateEvents extends BaseTask_1.BaseTask {
112
118
  safeNonce: gnosisTx.nonce,
113
119
  chainId: targetChainId,
114
120
  }, ownerPeerIds);
115
- const validSignatures = signatures.filter((s) => !!s.data && s.data !== "0x");
121
+ const validSignatures = signatures
122
+ .filter((s) => !!s.data && s.data !== "0x")
123
+ .filter((s) => {
124
+ try {
125
+ const address = (0, utils_1.getGnosisSignatureAddress)({
126
+ message,
127
+ signature: s.data,
128
+ chainId: this.chainId,
129
+ });
130
+ return (address === null || address === void 0 ? void 0 : address.toLowerCase()) === s.signer.toLowerCase();
131
+ }
132
+ catch (error) {
133
+ return false;
134
+ }
135
+ });
116
136
  console.log({
117
137
  signatures,
118
138
  validSignatures,
@@ -176,7 +196,7 @@ class ProcessValidateEvents extends BaseTask_1.BaseTask {
176
196
  }
177
197
  async start() {
178
198
  this.blockConfirmationsCount = constants_1.blockConfirmations[this.chainId] + 1;
179
- this.sourceProvider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
199
+ this.sourceProvider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
180
200
  await super.start();
181
201
  }
182
202
  }
@@ -0,0 +1,113 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ const BaseTask_1 = require("../BaseTask");
7
+ const logger_1 = __importDefault(require("@/logger"));
8
+ const ethers_1 = require("ethers");
9
+ const abi_1 = __importDefault(require("@/abi"));
10
+ const db_1 = require("@/db");
11
+ const utils_1 = require("@/utils");
12
+ const constants_1 = require("@/constants");
13
+ const config_1 = __importDefault(require("@/config"));
14
+ const sequelize_1 = require("sequelize");
15
+ const providers_1 = require("@/providers");
16
+ class SyncLogExecuteEvents extends BaseTask_1.BaseTask {
17
+ constructor({ targetChainId }) {
18
+ super({
19
+ logger: new logger_1.default("InteropX::SyncLogExecuteEvents"),
20
+ });
21
+ this.targetChainId = targetChainId;
22
+ }
23
+ async pollHandler() {
24
+ const currentBlock = await this.provider.getBlockNumber();
25
+ const events = await this.contract.queryFilter(this.contract.filters.LogExecute(), currentBlock - 2000, currentBlock);
26
+ let processedEvents = 0;
27
+ for (const event of events) {
28
+ try {
29
+ if (!event.args) {
30
+ continue;
31
+ }
32
+ const { sourceSpells, targetSpells, position, actionId, actionIdHash, sourceSender, sourceDsaId, targetDsaId, sourceChainId, targetChainId, vnonce, metadata, } = event.args;
33
+ const uniqueIdentifier = {
34
+ actionId,
35
+ vnonce: vnonce.toString(),
36
+ sourceSender: sourceSender.toString(),
37
+ sourceChainId: sourceChainId.toNumber(),
38
+ targetChainId: targetChainId.toNumber(),
39
+ sourceDsaId: sourceDsaId.toString(),
40
+ targetDsaId: targetDsaId.toString(),
41
+ };
42
+ let transactionHash = (0, utils_1.generateInteropTransactionHash)(uniqueIdentifier);
43
+ const transaction = await db_1.Transaction.findOne({
44
+ where: {
45
+ transactionHash,
46
+ executeEvent: { [sequelize_1.Op.eq]: null },
47
+ },
48
+ });
49
+ if (!transaction) {
50
+ continue;
51
+ }
52
+ if (transaction.targetStatus != "success") {
53
+ transaction.targetStatus = "success";
54
+ }
55
+ if (transaction.status != "success") {
56
+ transaction.status = "success";
57
+ }
58
+ if (!transaction.targetCreatedAt) {
59
+ transaction.targetCreatedAt = new Date();
60
+ }
61
+ transaction.targetTransactionHash = event.transactionHash;
62
+ transaction.targetBlockNumber = event.blockNumber;
63
+ transaction.targetLogs = [];
64
+ transaction.executeEvent = {
65
+ actionId,
66
+ actionIdHashHash: actionIdHash,
67
+ actionIdHash,
68
+ vnonce: vnonce.toString(),
69
+ sourceSpells: sourceSpells.map(({ connector, data }) => ({
70
+ connector,
71
+ data,
72
+ })),
73
+ targetSpells: targetSpells.map(({ connector, data }) => ({
74
+ connector,
75
+ data,
76
+ })),
77
+ position: {
78
+ withdraw: position.withdraw.map((v) => ({
79
+ sourceToken: v.sourceToken,
80
+ targetToken: v.targetToken,
81
+ amount: v.amount.toString(),
82
+ })),
83
+ supply: position.supply.map((v) => ({
84
+ sourceToken: v.sourceToken,
85
+ targetToken: v.targetToken,
86
+ amount: v.amount.toString(),
87
+ })),
88
+ },
89
+ sourceChainId: sourceChainId.toNumber(),
90
+ targetChainId: targetChainId.toNumber(),
91
+ sourceSender,
92
+ sourceDsaId: sourceDsaId.toString(),
93
+ targetDsaId: targetDsaId.toString(),
94
+ metadata,
95
+ };
96
+ await transaction.save();
97
+ this.logger.info(`New InteropX tranaction: ${transactionHash} `);
98
+ }
99
+ catch (error) {
100
+ this.logger.error(error);
101
+ }
102
+ }
103
+ if (processedEvents > 0)
104
+ this.logger.info(`${processedEvents} events processed`);
105
+ }
106
+ async start() {
107
+ this.contractAddress = constants_1.addresses[this.targetChainId].interopX;
108
+ this.provider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(this.targetChainId));
109
+ this.contract = (0, utils_1.getContract)(this.contractAddress, abi_1.default.interopX, new ethers_1.ethers.Wallet(config_1.default.privateKey, this.provider));
110
+ await super.start();
111
+ }
112
+ }
113
+ exports.default = SyncLogExecuteEvents;
@@ -11,6 +11,7 @@ const db_1 = require("@/db");
11
11
  const utils_1 = require("@/utils");
12
12
  const constants_1 = require("@/constants");
13
13
  const config_1 = __importDefault(require("@/config"));
14
+ const providers_1 = require("@/providers");
14
15
  class SyncLogSubmitEvents extends BaseTask_1.BaseTask {
15
16
  constructor({ chainId }) {
16
17
  super({
@@ -79,7 +80,7 @@ class SyncLogSubmitEvents extends BaseTask_1.BaseTask {
79
80
  }
80
81
  async start() {
81
82
  this.contractAddress = constants_1.addresses[this.chainId].interopX;
82
- this.provider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
83
+ this.provider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
83
84
  this.contract = (0, utils_1.getContract)(this.contractAddress, abi_1.default.interopX, new ethers_1.ethers.Wallet(config_1.default.privateKey, this.provider));
84
85
  await super.start();
85
86
  }
@@ -12,6 +12,7 @@ const utils_1 = require("@/utils");
12
12
  const constants_1 = require("@/constants");
13
13
  const config_1 = __importDefault(require("@/config"));
14
14
  const sequelize_1 = require("sequelize");
15
+ const providers_1 = require("@/providers");
15
16
  class SyncLogValidateEvents extends BaseTask_1.BaseTask {
16
17
  constructor({ chainId }) {
17
18
  super({
@@ -57,7 +58,7 @@ class SyncLogValidateEvents extends BaseTask_1.BaseTask {
57
58
  transaction.sourceTransactionHash = event.transactionHash;
58
59
  transaction.sourceBlockNumber = event.blockNumber;
59
60
  transaction.sourceLogs = [];
60
- (transaction.validateEvent = {
61
+ transaction.validateEvent = {
61
62
  actionId,
62
63
  actionIdHashHash: actionIdHash,
63
64
  actionIdHash,
@@ -84,8 +85,8 @@ class SyncLogValidateEvents extends BaseTask_1.BaseTask {
84
85
  sourceDsaId: sourceDsaId.toString(),
85
86
  targetDsaId: targetDsaId.toString(),
86
87
  metadata,
87
- }),
88
- await transaction.save();
88
+ };
89
+ await transaction.save();
89
90
  this.logger.info(`New InteropX tranaction: ${transactionHash} `);
90
91
  }
91
92
  catch (error) {
@@ -97,7 +98,7 @@ class SyncLogValidateEvents extends BaseTask_1.BaseTask {
97
98
  }
98
99
  async start() {
99
100
  this.contractAddress = constants_1.addresses[this.chainId].interopX;
100
- this.provider = new ethers_1.ethers.providers.JsonRpcProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
101
+ this.provider = new providers_1.JsonRpcRetryProvider((0, utils_1.getRpcProviderUrl)(this.chainId));
101
102
  this.contract = (0, utils_1.getContract)(this.contractAddress, abi_1.default.interopX, new ethers_1.ethers.Wallet(config_1.default.privateKey, this.provider));
102
103
  await super.start();
103
104
  }