@instadapp/interop-x 0.0.0-dev.92afe89 → 0.0.0-dev.99c6cb0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (98) hide show
  1. package/.env.example +2 -1
  2. package/dist/package.json +74 -0
  3. package/dist/src/abi/erc20.json +350 -0
  4. package/dist/src/abi/gnosisSafe.json +747 -0
  5. package/dist/src/abi/index.js +13 -0
  6. package/dist/src/abi/interopXContract.json +454 -0
  7. package/dist/src/alias.js +10 -0
  8. package/dist/src/api/index.js +36 -0
  9. package/dist/src/config/index.js +31 -0
  10. package/dist/src/constants/addresses.js +20 -0
  11. package/dist/{constants → src/constants}/index.js +1 -0
  12. package/dist/src/constants/tokens.js +137 -0
  13. package/dist/{db → src/db}/index.js +0 -0
  14. package/dist/{db → src/db}/models/index.js +1 -1
  15. package/dist/src/db/models/transaction.js +70 -0
  16. package/dist/{db → src/db}/sequelize.js +2 -1
  17. package/dist/src/gnosis/actions/index.js +9 -0
  18. package/dist/src/gnosis/actions/withdraw/index.js +55 -0
  19. package/dist/src/gnosis/index.js +20 -0
  20. package/dist/src/index.js +119 -0
  21. package/dist/{logger → src/logger}/index.js +0 -0
  22. package/dist/{net → src/net}/index.js +0 -0
  23. package/dist/{net → src/net}/peer/index.js +13 -8
  24. package/dist/{net → src/net}/pool/index.js +34 -11
  25. package/dist/{net → src/net}/protocol/dial/BaseDialProtocol.js +1 -1
  26. package/dist/{net → src/net}/protocol/dial/SignatureDialProtocol.js +21 -14
  27. package/dist/src/net/protocol/dial/TransactionStatusDialProtocol.js +30 -0
  28. package/dist/{net → src/net}/protocol/index.js +54 -4
  29. package/dist/src/tasks/AutoUpdateTask.js +70 -0
  30. package/dist/{tasks → src/tasks}/BaseTask.js +14 -6
  31. package/dist/src/tasks/InteropXContract/ProcessBridgeRequestEvents.js +152 -0
  32. package/dist/src/tasks/InteropXContract/SyncBridgeRequestEvents.js +78 -0
  33. package/dist/src/tasks/InteropXContract/SyncBridgeRequestSentEvents.js +90 -0
  34. package/dist/src/tasks/Transactions/SyncTransactionStatusTask.js +55 -0
  35. package/dist/src/tasks/index.js +38 -0
  36. package/dist/src/typechain/Erc20.js +2 -0
  37. package/dist/src/typechain/GnosisSafe.js +2 -0
  38. package/dist/src/typechain/InteropXContract.js +2 -0
  39. package/dist/src/typechain/common.js +2 -0
  40. package/dist/src/typechain/factories/Erc20__factory.js +367 -0
  41. package/dist/src/typechain/factories/GnosisSafe__factory.js +1174 -0
  42. package/dist/src/typechain/factories/InteropXContract__factory.js +635 -0
  43. package/dist/src/typechain/factories/index.js +12 -0
  44. package/dist/src/typechain/index.js +33 -0
  45. package/dist/{types.js → src/types.js} +0 -0
  46. package/dist/src/utils/index.js +186 -0
  47. package/package.json +30 -15
  48. package/patches/@ethersproject+properties+5.6.0.patch +13 -0
  49. package/src/abi/erc20.json +350 -0
  50. package/src/abi/gnosisSafe.json +747 -0
  51. package/src/abi/index.ts +9 -0
  52. package/src/abi/interopXContract.json +454 -0
  53. package/src/alias.ts +6 -0
  54. package/src/api/index.ts +36 -0
  55. package/src/config/index.ts +18 -2
  56. package/src/constants/addresses.ts +10 -3
  57. package/src/constants/index.ts +1 -0
  58. package/src/constants/tokens.ts +134 -0
  59. package/src/db/index.ts +1 -1
  60. package/src/db/models/index.ts +1 -1
  61. package/src/db/models/transaction.ts +137 -0
  62. package/src/db/sequelize.ts +2 -1
  63. package/src/gnosis/actions/index.ts +5 -0
  64. package/src/gnosis/actions/withdraw/index.ts +77 -0
  65. package/src/gnosis/index.ts +19 -0
  66. package/src/index.ts +118 -7
  67. package/src/net/peer/index.ts +12 -10
  68. package/src/net/pool/index.ts +43 -13
  69. package/src/net/protocol/dial/BaseDialProtocol.ts +1 -1
  70. package/src/net/protocol/dial/SignatureDialProtocol.ts +24 -17
  71. package/src/net/protocol/dial/TransactionStatusDialProtocol.ts +33 -0
  72. package/src/net/protocol/index.ts +70 -4
  73. package/src/tasks/AutoUpdateTask.ts +82 -0
  74. package/src/tasks/BaseTask.ts +16 -7
  75. package/src/tasks/InteropXContract/ProcessBridgeRequestEvents.ts +216 -0
  76. package/src/tasks/InteropXContract/SyncBridgeRequestEvents.ts +115 -0
  77. package/src/tasks/InteropXContract/SyncBridgeRequestSentEvents.ts +121 -0
  78. package/src/tasks/Transactions/SyncTransactionStatusTask.ts +67 -0
  79. package/src/tasks/index.ts +24 -2
  80. package/src/typechain/Erc20.ts +491 -0
  81. package/src/typechain/GnosisSafe.ts +1728 -0
  82. package/src/typechain/InteropXContract.ts +680 -0
  83. package/src/typechain/common.ts +44 -0
  84. package/src/typechain/factories/Erc20__factory.ts +368 -0
  85. package/src/typechain/factories/GnosisSafe__factory.ts +1178 -0
  86. package/src/typechain/factories/InteropXContract__factory.ts +642 -0
  87. package/src/typechain/factories/index.ts +6 -0
  88. package/src/typechain/index.ts +10 -0
  89. package/src/types.ts +2 -2
  90. package/src/utils/index.ts +156 -4
  91. package/tsconfig.json +8 -0
  92. package/dist/config/index.js +0 -17
  93. package/dist/constants/addresses.js +0 -13
  94. package/dist/db/models/execution.js +0 -38
  95. package/dist/index.js +0 -34
  96. package/dist/tasks/index.js +0 -19
  97. package/dist/utils/index.js +0 -89
  98. package/src/db/models/execution.ts +0 -57
@@ -3,14 +3,29 @@
3
3
  */
4
4
  import axios from 'axios'
5
5
  import axiosRetry from "axios-retry";
6
- import { addresses } from '../constants';
7
- import { ChainId } from '../types'
6
+ import { addresses } from '@/constants';
7
+ import { ChainId } from '@/types'
8
+ import { ethers } from 'ethers';
9
+ import { GnosisSafe } from '@/typechain';
10
+ import retry from 'async-retry'
8
11
 
9
12
  export const http = axios.create();
10
13
 
11
14
  axiosRetry(http, { retries: 3, retryDelay: axiosRetry.exponentialDelay });
12
15
 
13
16
 
17
+ export function shortenHash(hash: string, length: number = 4) {
18
+ if (!hash) return;
19
+
20
+ if (hash.length < 12) return hash;
21
+
22
+ const beginningChars = hash.startsWith("0x") ? length + 2 : length;
23
+
24
+ const shortened = hash.substr(0, beginningChars) + "…" + hash.substr(-length);
25
+
26
+ return shortened;
27
+ }
28
+
14
29
  export function short(buffer: Buffer): string {
15
30
  return buffer.toString('hex').slice(0, 8) + '...'
16
31
  }
@@ -70,9 +85,11 @@ export const signGnosisSafeTx = async ({
70
85
  export const getRpcProviderUrl = (chainId: ChainId) => {
71
86
  switch (chainId) {
72
87
  case 1:
73
- return 'https://rpc.instadapp.io/mainnet';
88
+ return 'https://rpc.ankr.com/eth';
74
89
  case 137:
75
- return 'https://rpc.instadapp.io/polygon';
90
+ return 'https://rpc.ankr.com/polygon';
91
+ case 43114:
92
+ return 'https://rpc.ankr.com/avalanche';
76
93
  default:
77
94
  throw new Error(`Unknown chainId: ${chainId}`);
78
95
  }
@@ -113,4 +130,139 @@ export const asyncCallWithTimeout = async <T>(asyncPromise: Promise<T>, timeout:
113
130
  clearTimeout(timeoutHandle);
114
131
  return result;
115
132
  }) as Promise<T>
133
+ }
134
+
135
+
136
+ export const generateInteropTransactionHash = (data: { actionId: string, bridger: string, requestTransactionHash: string, sourceChainId: string | number, targetChainId: string | number }) => {
137
+ return ethers.utils.solidityKeccak256(['string', 'string', 'string', 'string', 'string'], [
138
+ String(data.actionId),
139
+ String(data.bridger),
140
+ String(data.requestTransactionHash),
141
+ String(data.sourceChainId),
142
+ String(data.targetChainId),
143
+ ]);
144
+ }
145
+
146
+
147
+ export class ContractError extends Error {
148
+ method: string;
149
+ address: string;
150
+ args: any[];
151
+ }
152
+
153
+ export function getContract<TContract extends ethers.Contract>(address: string, contractInterface: ethers.ContractInterface | any, signerOrProvider?: ethers.Signer | ethers.providers.Provider) {
154
+ if (!ethers.utils.getAddress(address) || address === ethers.constants.AddressZero) {
155
+ throw Error(`Invalid 'address' parameter '${address}'.`)
156
+ }
157
+
158
+ const contract = new ethers.Contract(
159
+ address,
160
+ contractInterface,
161
+ signerOrProvider
162
+ ) as TContract
163
+
164
+ // Make sure the contract properties is writable
165
+ const desc = Object.getOwnPropertyDescriptor(contract, 'functions');
166
+
167
+ if (!desc || desc.writable !== true) {
168
+ return contract
169
+ }
170
+
171
+ return new Proxy(contract, {
172
+ get(target, prop, receiver) {
173
+ const value = Reflect.get(target, prop, receiver);
174
+
175
+ if (typeof value === 'function' && (contract.functions.hasOwnProperty(prop) || ['queryFilter'].includes(String(prop)))) {
176
+ let isConstant = false;
177
+
178
+ try {
179
+ isConstant = contract.interface.getFunction(String(prop)).constant
180
+ } catch (error) {
181
+ }
182
+
183
+ return async (...args: any[]) => {
184
+ try {
185
+ return await retry(async () => await value.bind(contract)(...args), { retries: isConstant ? 1 : 3 });
186
+ } catch (error) {
187
+ const err = new ContractError(`Error calling "${String(prop)}" on "${address}": ${error.reason || error.message}`)
188
+
189
+ err.method = String(prop)
190
+ err.address = address
191
+ err.args = [...args]
192
+
193
+ throw err
194
+ }
195
+ }
196
+ }
197
+
198
+
199
+ if (typeof value === 'object' && ['populateTransaction', 'estimateGas', 'functions', 'callStatic'].includes(String(prop))) {
200
+ const parentProp = String(prop);
201
+
202
+ return new Proxy(value, {
203
+ get(target, prop, receiver) {
204
+ const value = Reflect.get(target, prop, receiver);
205
+
206
+ if (typeof value === 'function') {
207
+ return async (...args: any[]) => {
208
+ try {
209
+ return await retry(async () => await value.bind(contract)(...args), { retries: parentProp === 'callStatic' ? 3 : 1 });
210
+ } catch (error) {
211
+ const err = new ContractError(`Error calling "${String(prop)}" using "${parentProp}" on "${address}": ${error.reason || error.message}`)
212
+
213
+ err.method = String(prop)
214
+ err.address = address
215
+ err.args = [...args]
216
+
217
+ throw err
218
+ }
219
+ }
220
+ }
221
+ }
222
+ })
223
+ }
224
+
225
+ return value;
226
+ },
227
+ });
228
+ }
229
+
230
+ export const generateGnosisTransaction = async (transactionData: any, safeContract: GnosisSafe) => {
231
+ console.log(transactionData);
232
+
233
+ let isExecuted = await safeContract.dataHashes(
234
+ await safeContract.getTransactionHash(
235
+ transactionData.to,
236
+ transactionData.value,
237
+ transactionData.data,
238
+ transactionData.operation,
239
+ transactionData.safeTxGas,
240
+ transactionData.baseGas,
241
+ transactionData.gasPrice,
242
+ transactionData.gasToken,
243
+ transactionData.refundReceiver,
244
+ transactionData.nonce
245
+ )
246
+ )
247
+
248
+ while (isExecuted == 1) {
249
+ transactionData.safeTxGas = ethers.BigNumber.from(String(transactionData.safeTxGas)).add(1).toString()
250
+
251
+ isExecuted = await safeContract.dataHashes(
252
+ await safeContract.getTransactionHash(
253
+ transactionData.to,
254
+ transactionData.value,
255
+ transactionData.data,
256
+ transactionData.operation,
257
+ transactionData.safeTxGas,
258
+ transactionData.baseGas,
259
+ transactionData.gasPrice,
260
+ transactionData.gasToken,
261
+ transactionData.refundReceiver,
262
+ transactionData.nonce
263
+ )
264
+ )
265
+ }
266
+
267
+ return transactionData
116
268
  }
package/tsconfig.json CHANGED
@@ -16,6 +16,14 @@
16
16
  "noEmit": false,
17
17
  "outDir": "dist",
18
18
  "baseUrl": "src",
19
+ "paths": {
20
+ "@/*": [
21
+ "*"
22
+ ],
23
+ "@": [
24
+ "/"
25
+ ]
26
+ },
19
27
  "typeRoots": [
20
28
  "./node_modules/@types",
21
29
  "./@types"
@@ -1,17 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- const ethers_1 = require("ethers");
4
- const types_1 = require("types");
5
- class Config {
6
- constructor() {
7
- this.events = new types_1.EventBus();
8
- this.maxPeers = 10;
9
- this.privateKey = process.env.PRIVATE_KEY;
10
- this.wallet = new ethers_1.Wallet(this.privateKey);
11
- this.leadNodeAddress = '0x910E413DBF3F6276Fe8213fF656726bDc142E08E';
12
- }
13
- isLeadNode() {
14
- return ethers_1.ethers.utils.getAddress(this.leadNodeAddress) === ethers_1.ethers.utils.getAddress(this.wallet.address);
15
- }
16
- }
17
- exports.default = new Config();
@@ -1,13 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.addresses = void 0;
4
- exports.addresses = {
5
- 1: {
6
- gnosisSafe: '0x5635d2910e51da33d9DC0422c893CF4F28B69A25',
7
- multisend: "0xA238CBeb142c10Ef7Ad8442C6D1f9E89e07e7761",
8
- },
9
- 137: {
10
- gnosisSafe: '0x5635d2910e51da33d9DC0422c893CF4F28B69A25',
11
- multisend: "0xA238CBeb142c10Ef7Ad8442C6D1f9E89e07e7761",
12
- },
13
- };
@@ -1,38 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Execution = void 0;
4
- const sequelize_1 = require("../sequelize");
5
- const sequelize_2 = require("sequelize");
6
- class Execution extends sequelize_2.Model {
7
- }
8
- exports.Execution = Execution;
9
- Execution.init({
10
- id: {
11
- type: sequelize_2.DataTypes.INTEGER,
12
- autoIncrement: true,
13
- primaryKey: true
14
- },
15
- transactionHash: sequelize_2.DataTypes.STRING,
16
- from: sequelize_2.DataTypes.STRING,
17
- executions: sequelize_2.DataTypes.JSON,
18
- chainId: sequelize_2.DataTypes.NUMBER,
19
- gas: sequelize_2.DataTypes.STRING,
20
- maxGasPrice: sequelize_2.DataTypes.STRING,
21
- assets: sequelize_2.DataTypes.JSON,
22
- metadata: sequelize_2.DataTypes.STRING,
23
- vnonce: sequelize_2.DataTypes.STRING,
24
- txHash: sequelize_2.DataTypes.STRING,
25
- blockNumber: sequelize_2.DataTypes.STRING,
26
- status: {
27
- type: sequelize_2.DataTypes.STRING,
28
- defaultValue: 'pending'
29
- },
30
- delayUntil: sequelize_2.DataTypes.DATE,
31
- delayedCount: {
32
- type: sequelize_2.DataTypes.INTEGER,
33
- defaultValue: 0
34
- },
35
- error: sequelize_2.DataTypes.STRING,
36
- createdAt: sequelize_2.DataTypes.DATE,
37
- updatedAt: sequelize_2.DataTypes.DATE,
38
- }, { sequelize: sequelize_1.sequelize, tableName: 'executions' });
package/dist/index.js DELETED
@@ -1,34 +0,0 @@
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 console_1 = require("console");
7
- const dotenv_1 = __importDefault(require("dotenv"));
8
- const tasks_1 = require("tasks");
9
- const logger_1 = __importDefault(require("./logger"));
10
- const net_1 = require("./net");
11
- dotenv_1.default.config();
12
- (0, console_1.assert)(process.env.PRIVATE_KEY, "PRIVATE_KEY is not defined");
13
- const logger = new logger_1.default('Process');
14
- async function main() {
15
- (0, net_1.startPeer)({});
16
- const tasks = new tasks_1.Tasks();
17
- tasks.start();
18
- }
19
- main()
20
- .then(() => {
21
- }).catch(err => {
22
- console.error(err);
23
- });
24
- process.on('SIGINT', () => {
25
- logger.debug('received SIGINT signal. exiting.');
26
- process.exit(0);
27
- });
28
- process.on('SIGTERM', () => {
29
- logger.debug('received SIGTERM signal. exiting.');
30
- process.exit(0);
31
- });
32
- process.on('unhandledRejection', (reason, p) => {
33
- logger.error('unhandled rejection: promise:', p, 'reason:', reason);
34
- });
@@ -1,19 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.Tasks = void 0;
4
- class Tasks {
5
- constructor() {
6
- this.tasks = [];
7
- }
8
- async start() {
9
- for (const task of this.tasks) {
10
- try {
11
- task.start();
12
- }
13
- catch (error) {
14
- console.error(`Error starting task: ${task.constructor.name}`);
15
- }
16
- }
17
- }
18
- }
19
- exports.Tasks = Tasks;
@@ -1,89 +0,0 @@
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.asyncCallWithTimeout = exports.buildSignatureBytes = exports.getRpcProviderUrl = exports.signGnosisSafeTx = exports.short = exports.http = void 0;
7
- /**
8
- * @module util
9
- */
10
- const axios_1 = __importDefault(require("axios"));
11
- const axios_retry_1 = __importDefault(require("axios-retry"));
12
- const constants_1 = require("../constants");
13
- exports.http = axios_1.default.create();
14
- (0, axios_retry_1.default)(exports.http, { retries: 3, retryDelay: axios_retry_1.default.exponentialDelay });
15
- function short(buffer) {
16
- return buffer.toString('hex').slice(0, 8) + '...';
17
- }
18
- exports.short = short;
19
- const signGnosisSafeTx = async ({ to, data = null, value = '0', operation = '1', baseGas = '0', gasPrice = "0", gasToken = "0x0000000000000000000000000000000000000000", refundReceiver = "0x0000000000000000000000000000000000000000", safeTxGas = "79668", nonce = "0", chainId = 137, }, { signer }) => {
20
- const gnosisSafe = constants_1.addresses[chainId].gnosisSafe;
21
- const domain = {
22
- verifyingContract: gnosisSafe,
23
- chainId,
24
- };
25
- const types = {
26
- SafeTx: [
27
- { type: 'address', name: 'to' },
28
- { type: 'uint256', name: 'value' },
29
- { type: 'bytes', name: 'data' },
30
- { type: 'uint8', name: 'operation' },
31
- { type: 'uint256', name: 'safeTxGas' },
32
- { type: 'uint256', name: 'baseGas' },
33
- { type: 'uint256', name: 'gasPrice' },
34
- { type: 'address', name: 'gasToken' },
35
- { type: 'address', name: 'refundReceiver' },
36
- { type: 'uint256', name: 'nonce' },
37
- ],
38
- };
39
- const message = {
40
- baseGas,
41
- data,
42
- gasPrice,
43
- gasToken,
44
- nonce: Number(nonce),
45
- operation,
46
- refundReceiver,
47
- safeAddress: gnosisSafe,
48
- safeTxGas: String(safeTxGas),
49
- to,
50
- value,
51
- };
52
- return await signer._signTypedData(domain, types, message);
53
- };
54
- exports.signGnosisSafeTx = signGnosisSafeTx;
55
- const getRpcProviderUrl = (chainId) => {
56
- switch (chainId) {
57
- case 1:
58
- return 'https://rpc.instadapp.io/mainnet';
59
- case 137:
60
- return 'https://rpc.instadapp.io/polygon';
61
- default:
62
- throw new Error(`Unknown chainId: ${chainId}`);
63
- }
64
- };
65
- exports.getRpcProviderUrl = getRpcProviderUrl;
66
- const buildSignatureBytes = (signatures) => {
67
- signatures.sort((left, right) => left.signer.toLowerCase().localeCompare(right.signer.toLowerCase()));
68
- let signatureBytes = "0x";
69
- for (const sig of signatures) {
70
- signatureBytes += sig.data.slice(2);
71
- }
72
- return signatureBytes;
73
- };
74
- exports.buildSignatureBytes = buildSignatureBytes;
75
- /**
76
- * Call an async function with a maximum time limit (in milliseconds) for the timeout
77
- * Resolved promise for async function call, or an error if time limit reached
78
- */
79
- const asyncCallWithTimeout = async (asyncPromise, timeout) => {
80
- let timeoutHandle;
81
- const timeoutPromise = new Promise((_resolve, reject) => {
82
- timeoutHandle = setTimeout(() => reject(new Error('Async call timeout limit reached')), timeout);
83
- });
84
- return Promise.race([asyncPromise, timeoutPromise]).then(result => {
85
- clearTimeout(timeoutHandle);
86
- return result;
87
- });
88
- };
89
- exports.asyncCallWithTimeout = asyncCallWithTimeout;
@@ -1,57 +0,0 @@
1
- import { sequelize } from '../sequelize'
2
- import { CreationOptional, InferAttributes, InferCreationAttributes, Model, DataTypes } from 'sequelize';
3
-
4
- export class Execution extends Model<InferAttributes<Execution>, InferCreationAttributes<Execution>> {
5
- declare id: CreationOptional<number>;
6
-
7
- declare transactionHash: string;
8
- declare from: string;
9
- declare executions: any[];
10
- declare chainId: number;
11
- declare gas: string;
12
- declare maxGasPrice: string;
13
- declare assets: any[];
14
- declare metadata: string;
15
- declare vnonce: string;
16
- declare txHash: string;
17
-
18
- declare blockNumber: string;
19
- declare status: CreationOptional<string>;
20
- declare delayUntil: CreationOptional<Date>;
21
- declare delayedCount: CreationOptional<number>;
22
-
23
- declare error: CreationOptional<string>;
24
- declare createdAt: CreationOptional<Date>;
25
- declare updatedAt: CreationOptional<Date>;
26
- }
27
-
28
- Execution.init({
29
- id: {
30
- type: DataTypes.INTEGER,
31
- autoIncrement: true,
32
- primaryKey: true
33
- },
34
- transactionHash: DataTypes.STRING,
35
- from: DataTypes.STRING,
36
- executions: DataTypes.JSON,
37
- chainId: DataTypes.NUMBER,
38
- gas: DataTypes.STRING,
39
- maxGasPrice: DataTypes.STRING,
40
- assets: DataTypes.JSON,
41
- metadata: DataTypes.STRING,
42
- vnonce: DataTypes.STRING,
43
- txHash: DataTypes.STRING,
44
- blockNumber: DataTypes.STRING,
45
- status: {
46
- type: DataTypes.STRING,
47
- defaultValue: 'pending'
48
- },
49
- delayUntil: DataTypes.DATE,
50
- delayedCount: {
51
- type: DataTypes.INTEGER,
52
- defaultValue: 0
53
- },
54
- error: DataTypes.STRING,
55
- createdAt: DataTypes.DATE,
56
- updatedAt: DataTypes.DATE,
57
- }, { sequelize, tableName: 'executions' });