@instadapp/interop-x 0.0.0-dev.73da8c9 → 0.0.0-dev.73e5298
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.
- package/dist/package.json +18 -16
- package/dist/src/abi/index.js +2 -4
- package/dist/src/abi/interopXContract.json +454 -0
- package/dist/src/alias.js +10 -0
- package/dist/src/api/index.js +6 -3
- package/dist/src/config/index.js +11 -1
- package/dist/src/constants/addresses.js +7 -7
- package/dist/src/constants/index.js +0 -1
- package/dist/src/constants/tokens.js +62 -39
- package/dist/src/db/models/transaction.js +27 -11
- package/dist/src/gnosis/actions/index.js +9 -0
- package/dist/src/gnosis/actions/withdraw/index.js +115 -0
- package/dist/src/gnosis/index.js +20 -0
- package/dist/src/index.js +75 -24
- package/dist/src/net/peer/index.js +2 -1
- package/dist/src/net/pool/index.js +18 -2
- package/dist/src/net/protocol/dial/SignatureDialProtocol.js +8 -2
- package/dist/src/net/protocol/dial/TransactionStatusDialProtocol.js +30 -0
- package/dist/src/net/protocol/index.js +51 -1
- package/dist/src/tasks/AutoUpdateTask.js +70 -0
- package/dist/src/tasks/BaseTask.js +11 -3
- package/dist/src/tasks/InteropXContract/ProcessBridgeRequestEvents.js +158 -0
- package/dist/src/tasks/InteropXContract/SyncBridgeCommittedEvents.js +93 -0
- package/dist/src/tasks/InteropXContract/SyncBridgeRequestEvents.js +78 -0
- package/dist/src/tasks/InteropXContract/SyncBridgeRequestSentEvents.js +90 -0
- package/dist/src/tasks/Transactions/SyncTransactionStatusTask.js +57 -0
- package/dist/src/tasks/index.js +18 -8
- package/dist/src/typechain/{InteropBridgeToken.js → InteropXContract.js} +0 -0
- package/dist/src/typechain/factories/InteropXContract__factory.js +635 -0
- package/dist/src/typechain/factories/index.js +3 -5
- package/dist/src/typechain/index.js +3 -5
- package/dist/src/utils/index.js +62 -47
- package/package.json +18 -16
- package/src/abi/index.ts +2 -4
- package/src/abi/interopXContract.json +454 -0
- package/src/alias.ts +6 -0
- package/src/api/index.ts +5 -2
- package/src/config/index.ts +11 -1
- package/src/constants/addresses.ts +8 -8
- package/src/constants/index.ts +0 -1
- package/src/constants/tokens.ts +63 -40
- package/src/db/models/transaction.ts +76 -27
- package/src/gnosis/actions/index.ts +5 -0
- package/src/gnosis/actions/withdraw/index.ts +155 -0
- package/src/gnosis/index.ts +19 -0
- package/src/index.ts +96 -26
- package/src/net/peer/index.ts +2 -1
- package/src/net/pool/index.ts +25 -5
- package/src/net/protocol/dial/SignatureDialProtocol.ts +10 -4
- package/src/net/protocol/dial/TransactionStatusDialProtocol.ts +33 -0
- package/src/net/protocol/index.ts +67 -1
- package/src/tasks/AutoUpdateTask.ts +82 -0
- package/src/tasks/BaseTask.ts +13 -3
- package/src/tasks/InteropXContract/ProcessBridgeRequestEvents.ts +226 -0
- package/src/tasks/InteropXContract/SyncBridgeCommittedEvents.ts +125 -0
- package/src/tasks/InteropXContract/SyncBridgeRequestEvents.ts +115 -0
- package/src/tasks/InteropXContract/SyncBridgeRequestSentEvents.ts +121 -0
- package/src/tasks/Transactions/SyncTransactionStatusTask.ts +69 -0
- package/src/tasks/index.ts +27 -9
- package/src/typechain/InteropXContract.ts +680 -0
- package/src/typechain/factories/InteropXContract__factory.ts +642 -0
- package/src/typechain/factories/index.ts +1 -2
- package/src/typechain/index.ts +2 -4
- package/src/utils/index.ts +103 -66
- package/tsconfig.json +7 -2
- package/dist/src/abi/interopBridgeToken.json +0 -286
- package/dist/src/abi/interopXGateway.json +0 -184
- package/dist/src/constants/itokens.js +0 -13
- package/dist/src/tasks/InteropXGateway/ProcessDepositEvents.js +0 -158
- package/dist/src/tasks/InteropXGateway/SyncDepositEvents.js +0 -75
- package/dist/src/typechain/InteropXGateway.js +0 -2
- package/dist/src/typechain/factories/InteropBridgeToken__factory.js +0 -459
- package/dist/src/typechain/factories/InteropXGateway__factory.js +0 -265
- package/src/abi/interopBridgeToken.json +0 -286
- package/src/abi/interopXGateway.json +0 -184
- package/src/constants/itokens.ts +0 -10
- package/src/tasks/InteropXGateway/ProcessDepositEvents.ts +0 -253
- package/src/tasks/InteropXGateway/SyncDepositEvents.ts +0 -126
- package/src/typechain/InteropBridgeToken.ts +0 -686
- package/src/typechain/InteropXGateway.ts +0 -407
- package/src/typechain/factories/InteropBridgeToken__factory.ts +0 -466
- package/src/typechain/factories/InteropXGateway__factory.ts +0 -272
package/src/net/pool/index.ts
CHANGED
@@ -1,6 +1,9 @@
|
|
1
1
|
import { Event } from "@/types";
|
2
2
|
import config from "@/config";
|
3
3
|
import Logger from "@/logger";
|
4
|
+
import { getAddress } from "ethers/lib/utils";
|
5
|
+
import { shortenHash } from "@/utils";
|
6
|
+
import chalk from "chalk";
|
4
7
|
|
5
8
|
|
6
9
|
const logger = new Logger('PeerPool')
|
@@ -83,10 +86,10 @@ export class PeerPool {
|
|
83
86
|
const newPeer = !this.pool.get(peer.id);
|
84
87
|
this.pool.set(peer.id, peer)
|
85
88
|
peer.pooled = true
|
86
|
-
|
87
|
-
if(newPeer) {
|
89
|
+
|
90
|
+
if (newPeer) {
|
88
91
|
config.events.emit(Event.POOL_PEER_ADDED, peer)
|
89
|
-
logger.info(`Peer ${peer.id} with address ${peer.publicAddress} added to pool`)
|
92
|
+
logger.info(`Peer ${chalk.bold(shortenHash(peer.id, 16))} with address ${chalk.bold(shortenHash(peer.publicAddress))} added to pool`)
|
90
93
|
}
|
91
94
|
}
|
92
95
|
}
|
@@ -101,7 +104,7 @@ export class PeerPool {
|
|
101
104
|
if (this.pool.delete(peer.id)) {
|
102
105
|
peer.pooled = false
|
103
106
|
config.events.emit(Event.POOL_PEER_REMOVED, peer)
|
104
|
-
logger.info(`Peer ${peer.id} with address ${peer.publicAddress} removed from pool`)
|
107
|
+
logger.info(`Peer ${chalk.bold(shortenHash(peer.id, 16))} with address ${chalk.bold(shortenHash(peer.publicAddress))} removed from pool`)
|
105
108
|
}
|
106
109
|
}
|
107
110
|
}
|
@@ -110,7 +113,7 @@ export class PeerPool {
|
|
110
113
|
this.cleanup()
|
111
114
|
|
112
115
|
return this.peers.filter((p) => {
|
113
|
-
if(!p.pooled) return false;
|
116
|
+
if (!p.pooled) return false;
|
114
117
|
|
115
118
|
const now = new Date()
|
116
119
|
|
@@ -122,6 +125,23 @@ export class PeerPool {
|
|
122
125
|
return this.activePeers.map((p) => p.id)
|
123
126
|
}
|
124
127
|
|
128
|
+
getPeer(id: string){
|
129
|
+
return this.pool.get(id);
|
130
|
+
}
|
131
|
+
|
132
|
+
isLeadNode(id: string) {
|
133
|
+
const peer = this.pool.get(id);
|
134
|
+
|
135
|
+
if (!peer) {
|
136
|
+
return false;
|
137
|
+
}
|
138
|
+
|
139
|
+
return getAddress(peer.publicAddress) === getAddress(config.leadNodeAddress)
|
140
|
+
}
|
141
|
+
|
142
|
+
getLeadPeer() {
|
143
|
+
return this.peers.find((p) => this.isLeadNode(p.id))
|
144
|
+
}
|
125
145
|
|
126
146
|
cleanup() {
|
127
147
|
// let compDate = Date.now() - this.PEERS_CLEANUP_TIME_LIMIT * 60
|
@@ -2,12 +2,13 @@ import { BaseDialProtocol } from "./BaseDialProtocol";
|
|
2
2
|
import wait from "waait";
|
3
3
|
import config from "@/config";
|
4
4
|
import { Transaction } from "@/db";
|
5
|
-
import {
|
5
|
+
import { signGnosisSafeTx } from "@/utils";
|
6
6
|
import { addresses } from "@/constants";
|
7
7
|
import { ChainId } from "@/types";
|
8
|
+
import { buildGnosisAction } from "@/gnosis";
|
8
9
|
|
9
10
|
export interface ISignatureRequest {
|
10
|
-
type: 'source' | 'target'
|
11
|
+
type: 'source' | 'target',
|
11
12
|
transactionHash: string
|
12
13
|
safeTxGas: string
|
13
14
|
safeNonce: string
|
@@ -25,6 +26,10 @@ export class SignatureDialProtocol extends BaseDialProtocol<ISignatureRequest, I
|
|
25
26
|
}
|
26
27
|
|
27
28
|
async response(data: ISignatureRequest): Promise<ISignatureResponse> {
|
29
|
+
console.log({
|
30
|
+
tag: 'SignatureDialProtocol',
|
31
|
+
data
|
32
|
+
})
|
28
33
|
const signer = config.wallet;
|
29
34
|
|
30
35
|
let transaction: Transaction | null;
|
@@ -46,13 +51,14 @@ export class SignatureDialProtocol extends BaseDialProtocol<ISignatureRequest, I
|
|
46
51
|
error: 'Event not found'
|
47
52
|
};
|
48
53
|
}
|
54
|
+
const { data: gnosisData } = await buildGnosisAction(transaction, data.type);
|
49
55
|
|
50
56
|
const signedData = await signGnosisSafeTx({
|
51
|
-
//TODO: chain id depends on event type
|
52
57
|
to: addresses[transaction.sourceChainId].multisend,
|
53
|
-
data:
|
58
|
+
data: gnosisData,
|
54
59
|
chainId: transaction.sourceChainId as ChainId,
|
55
60
|
safeTxGas: data.safeTxGas,
|
61
|
+
nonce: data.safeNonce,
|
56
62
|
}, { signer });
|
57
63
|
|
58
64
|
return {
|
@@ -0,0 +1,33 @@
|
|
1
|
+
import { BaseDialProtocol } from "./BaseDialProtocol";
|
2
|
+
import { Transaction } from "@/db";
|
3
|
+
|
4
|
+
export class TransactionStatusDialProtocol extends BaseDialProtocol<string, Pick<Transaction, 'transactionHash' | 'sourceStatus' | 'sourceTransactionHash' | 'sourceErrors' | 'sourceLogs' | 'targetStatus' | 'targetTransactionHash' | 'targetErrors' | 'targetLogs' | 'status'> | null> {
|
5
|
+
protected timeout = 30000;
|
6
|
+
|
7
|
+
constructor(libp2p) {
|
8
|
+
super(libp2p, '/interop-x/transaction-status')
|
9
|
+
}
|
10
|
+
|
11
|
+
async response(transactionHash: string){
|
12
|
+
const transaction = await Transaction.findOne({ where: { transactionHash } })
|
13
|
+
|
14
|
+
if(! transaction){
|
15
|
+
return null
|
16
|
+
}
|
17
|
+
return {
|
18
|
+
transactionHash: transaction.transactionHash,
|
19
|
+
|
20
|
+
sourceStatus: transaction.sourceStatus,
|
21
|
+
sourceTransactionHash: transaction.sourceTransactionHash,
|
22
|
+
sourceErrors: transaction.sourceErrors,
|
23
|
+
sourceLogs: transaction.sourceLogs,
|
24
|
+
|
25
|
+
targetStatus: transaction.targetStatus,
|
26
|
+
targetTransactionHash: transaction.targetTransactionHash,
|
27
|
+
targetErrors: transaction.targetErrors,
|
28
|
+
targetLogs: transaction.targetLogs,
|
29
|
+
|
30
|
+
status: transaction.status,
|
31
|
+
}
|
32
|
+
}
|
33
|
+
}
|
@@ -5,6 +5,8 @@ import { SignatureDialProtocol, ISignatureRequest, ISignatureResponse } from "./
|
|
5
5
|
import { IPeerInfo, peerPool } from "..";
|
6
6
|
import config from "@/config";
|
7
7
|
import { Event } from "@/types";
|
8
|
+
import { Transaction } from "@/db";
|
9
|
+
import { TransactionStatusDialProtocol } from "./dial/TransactionStatusDialProtocol";
|
8
10
|
|
9
11
|
export interface ProtocolOptions {
|
10
12
|
/* Handshake timeout in ms (default: 8000) */
|
@@ -33,7 +35,12 @@ interface PeerInfoEvent extends BaseMessageEvent {
|
|
33
35
|
data: Omit<IPeerInfo, 'id' | 'updated' | 'idle' | 'pooled'>
|
34
36
|
}
|
35
37
|
|
38
|
+
interface TransactionStatusEvent extends BaseMessageEvent {
|
39
|
+
data: Pick<Transaction, 'transactionHash' | 'sourceStatus' | 'sourceTransactionHash' | 'sourceErrors' | 'sourceLogs' | 'targetStatus' | 'targetTransactionHash' | 'targetErrors' | 'targetLogs' | 'status'>
|
40
|
+
}
|
41
|
+
|
36
42
|
declare interface Protocol {
|
43
|
+
on(event: 'TransactionStatus', listener: (payload: TransactionStatusEvent) => void): this;
|
37
44
|
on(event: 'PeerInfo', listener: (payload: PeerInfoEvent) => void): this;
|
38
45
|
on(event: string, listener: (payload: BaseMessageEvent) => void): this;
|
39
46
|
}
|
@@ -44,7 +51,7 @@ class Protocol extends EventEmitter {
|
|
44
51
|
private protocolMessages: Message[] = [
|
45
52
|
{
|
46
53
|
name: 'PeerInfo',
|
47
|
-
code:
|
54
|
+
code: 0x01,
|
48
55
|
encode: (info: Pick<IPeerInfo, 'publicAddress'>) => [
|
49
56
|
Buffer.from(info.publicAddress),
|
50
57
|
],
|
@@ -52,8 +59,49 @@ class Protocol extends EventEmitter {
|
|
52
59
|
publicAddress: publicAddress.toString(),
|
53
60
|
}),
|
54
61
|
},
|
62
|
+
{
|
63
|
+
name: 'TransactionStatus',
|
64
|
+
code: 0x02,
|
65
|
+
encode: (transaction: Transaction) => [
|
66
|
+
Buffer.from(transaction.transactionHash),
|
67
|
+
|
68
|
+
Buffer.from(transaction.sourceStatus),
|
69
|
+
Buffer.from(transaction.sourceTransactionHash || ''),
|
70
|
+
transaction.sourceErrors ? transaction.sourceErrors.map((e) => Buffer.from(e)) : [],
|
71
|
+
transaction.sourceLogs ? transaction.sourceLogs.map((e) => [Buffer.from(e.type), Buffer.from(e.message)]) : [],
|
72
|
+
|
73
|
+
Buffer.from(transaction.targetStatus),
|
74
|
+
Buffer.from(transaction.targetTransactionHash || ''),
|
75
|
+
transaction.targetErrors ? transaction.targetErrors.map((e) => Buffer.from(e)) : [],
|
76
|
+
transaction.targetLogs ? transaction.targetLogs.map((e) => [Buffer.from(e.type), Buffer.from(e.message)]) : [],
|
77
|
+
|
78
|
+
Buffer.from(transaction.status),
|
79
|
+
],
|
80
|
+
decode: ([transactionHash, sourceStatus, sourceTransactionHash, sourceErrors, sourceLogs, targetStatus, targetTransactionHash, targetErrors, targetLogs, status]: [Buffer, Buffer, Buffer, Buffer[],[Buffer,Buffer][], Buffer, Buffer, Buffer[],[Buffer,Buffer][], Buffer]) => ({
|
81
|
+
transactionHash: transactionHash.toString(),
|
82
|
+
|
83
|
+
sourceStatus: sourceStatus.toString(),
|
84
|
+
sourceTransactionHash: sourceTransactionHash.toString() || null,
|
85
|
+
sourceErrors: sourceErrors.map((e) => e.toString()),
|
86
|
+
sourceLogs: sourceLogs.map(e => ({
|
87
|
+
type: e[0].toString(),
|
88
|
+
message: e[1].toString(),
|
89
|
+
})),
|
90
|
+
|
91
|
+
targetStatus: targetStatus.toString(),
|
92
|
+
targetTransactionHash: targetTransactionHash.toString() || null,
|
93
|
+
targetErrors: targetErrors.map((e) => e.toString()),
|
94
|
+
targetLogs: targetLogs.map(e => ({
|
95
|
+
type: e[0].toString(),
|
96
|
+
message: e[1].toString(),
|
97
|
+
})),
|
98
|
+
|
99
|
+
status: status.toString(),
|
100
|
+
}),
|
101
|
+
},
|
55
102
|
];
|
56
103
|
private signature: SignatureDialProtocol;
|
104
|
+
private transactionStatus: TransactionStatusDialProtocol;
|
57
105
|
|
58
106
|
|
59
107
|
start({ libp2p, topic = null, }) {
|
@@ -73,6 +121,7 @@ class Protocol extends EventEmitter {
|
|
73
121
|
})
|
74
122
|
|
75
123
|
this.signature = new SignatureDialProtocol(this.libp2p);
|
124
|
+
this.transactionStatus = new TransactionStatusDialProtocol(this.libp2p);
|
76
125
|
}
|
77
126
|
|
78
127
|
|
@@ -121,6 +170,14 @@ class Protocol extends EventEmitter {
|
|
121
170
|
this.libp2p.pubsub.publish(this.topic, encoded)
|
122
171
|
}
|
123
172
|
|
173
|
+
public sendTransaction(transaction: Transaction) {
|
174
|
+
const message = this.protocolMessages.find((m) => m.name === 'TransactionStatus')!
|
175
|
+
|
176
|
+
const encoded = rlp.encode([message.code, message.encode(transaction)]);
|
177
|
+
|
178
|
+
this.libp2p.pubsub.publish(this.topic, encoded)
|
179
|
+
}
|
180
|
+
|
124
181
|
async requestSignatures(data: ISignatureRequest, peerIds?: string[]) {
|
125
182
|
try {
|
126
183
|
peerIds = peerIds || peerPool.activePeerIds;
|
@@ -133,6 +190,15 @@ class Protocol extends EventEmitter {
|
|
133
190
|
return []
|
134
191
|
}
|
135
192
|
}
|
193
|
+
|
194
|
+
async requestTransactionStatus(transactionHash: string, peerId: string) {
|
195
|
+
try {
|
196
|
+
return await this.transactionStatus.send(transactionHash, peerId);
|
197
|
+
} catch (error) {
|
198
|
+
console.log(error);
|
199
|
+
return null
|
200
|
+
}
|
201
|
+
}
|
136
202
|
}
|
137
203
|
|
138
204
|
export const protocol = new Protocol();
|
@@ -0,0 +1,82 @@
|
|
1
|
+
import { BaseTask } from "./BaseTask";
|
2
|
+
import Logger from '@/logger';
|
3
|
+
import spawnAsync from 'await-spawn';
|
4
|
+
import { spawn } from 'child_process'
|
5
|
+
import config from "@/config";
|
6
|
+
import wait from "waait";
|
7
|
+
import packageJson from "../../package.json";
|
8
|
+
|
9
|
+
const currentVersion = packageJson.version;
|
10
|
+
const tag = config.staging ? 'dev' : 'latest';
|
11
|
+
|
12
|
+
class AutoUpdateTask extends BaseTask {
|
13
|
+
pollIntervalMs: number = 60 * 10 * 1000
|
14
|
+
|
15
|
+
constructor() {
|
16
|
+
super({
|
17
|
+
logger: new Logger("AutoUpdateTask"),
|
18
|
+
})
|
19
|
+
}
|
20
|
+
|
21
|
+
prePollHandler(): boolean {
|
22
|
+
return config.autoUpdate && !config.isLeadNode();
|
23
|
+
}
|
24
|
+
|
25
|
+
async getInstalledVersion() {
|
26
|
+
try {
|
27
|
+
const stdout = await spawnAsync('npm', ['-g', 'ls', '--depth=0', '--json'])
|
28
|
+
return JSON.parse(stdout.toString()).dependencies[packageJson.name].version
|
29
|
+
} catch (error) {
|
30
|
+
this.logger.error(error)
|
31
|
+
return currentVersion
|
32
|
+
}
|
33
|
+
}
|
34
|
+
|
35
|
+
async getLatestVersion() {
|
36
|
+
try {
|
37
|
+
const stdout = await spawnAsync('npm', ['view', `${packageJson.name}@${tag}`, 'version'])
|
38
|
+
return stdout.toString().trim()
|
39
|
+
} catch (error) {
|
40
|
+
this.logger.error(error)
|
41
|
+
return currentVersion
|
42
|
+
}
|
43
|
+
}
|
44
|
+
|
45
|
+
async pollHandler() {
|
46
|
+
const version = await this.getLatestVersion()
|
47
|
+
|
48
|
+
if (version === currentVersion) {
|
49
|
+
return;
|
50
|
+
}
|
51
|
+
|
52
|
+
this.logger.warn(`New version ${version} available.`)
|
53
|
+
|
54
|
+
this.logger.info('Updating...')
|
55
|
+
|
56
|
+
await spawnAsync('npm', ['-g', 'install', `@instadapp/interop-x@${tag}`, '-f']);
|
57
|
+
|
58
|
+
await wait(5000)
|
59
|
+
|
60
|
+
if (version !== await this.getInstalledVersion()) {
|
61
|
+
this.logger.warn(`failed to install ${version}, retrying in 5 minutes`)
|
62
|
+
return;
|
63
|
+
}
|
64
|
+
|
65
|
+
this.logger.warn(`Installed version ${version}`)
|
66
|
+
this.logger.warn(`Restarting...`)
|
67
|
+
|
68
|
+
|
69
|
+
// TODO: its restarting in the bg, but it should be in the fg
|
70
|
+
const subprocess = spawn(process.argv[0], process.argv.slice(1), {
|
71
|
+
cwd: process.cwd(),
|
72
|
+
stdio: "inherit",
|
73
|
+
// shell: process.env.SHELL,
|
74
|
+
});
|
75
|
+
|
76
|
+
subprocess.unref();
|
77
|
+
|
78
|
+
process.exit()
|
79
|
+
}
|
80
|
+
}
|
81
|
+
|
82
|
+
export default AutoUpdateTask;
|
package/src/tasks/BaseTask.ts
CHANGED
@@ -19,6 +19,7 @@ export class BaseTask extends EventEmitter implements IBaseTask {
|
|
19
19
|
started: boolean = false
|
20
20
|
pollIntervalMs: number = 10 * 1000
|
21
21
|
leadNodeOnly: boolean = false
|
22
|
+
exceptLeadNode: boolean = false
|
22
23
|
|
23
24
|
public constructor({ logger }: { logger?: Logger }) {
|
24
25
|
super()
|
@@ -45,11 +46,20 @@ export class BaseTask extends EventEmitter implements IBaseTask {
|
|
45
46
|
}
|
46
47
|
|
47
48
|
prePollHandler(): boolean {
|
48
|
-
if
|
49
|
-
|
49
|
+
if(config.isMaintenanceMode()){
|
50
|
+
this.logger.warn('Maintenance mode is enabled. Skipping task.')
|
51
|
+
return false
|
50
52
|
}
|
51
53
|
|
52
|
-
|
54
|
+
if (this.exceptLeadNode) {
|
55
|
+
return !config.isLeadNode();
|
56
|
+
}
|
57
|
+
|
58
|
+
if (this.leadNodeOnly) {
|
59
|
+
return config.isLeadNode()
|
60
|
+
}
|
61
|
+
|
62
|
+
return true
|
53
63
|
}
|
54
64
|
|
55
65
|
async pollHandler() {
|
@@ -0,0 +1,226 @@
|
|
1
|
+
import { BaseTask } from "../BaseTask";
|
2
|
+
import Logger from '@/logger';
|
3
|
+
import { ethers, Wallet } from "ethers";
|
4
|
+
import abi from "@/abi";
|
5
|
+
import { Transaction } from "@/db";
|
6
|
+
import { buildSignatureBytes, generateGnosisTransaction, generateInteropTransactionHash, getContract, getRpcProviderUrl, LiquidityError, Signature } from "@/utils";
|
7
|
+
import { addresses } from "@/constants";
|
8
|
+
import { Op } from "sequelize";
|
9
|
+
import { ChainId } from "@/types";
|
10
|
+
import config from "@/config";
|
11
|
+
import { GnosisSafe, InteropXContract } from "@/typechain";
|
12
|
+
import wait from "waait";
|
13
|
+
import { buildGnosisAction } from "@/gnosis";
|
14
|
+
import { peerPool, protocol } from "@/net";
|
15
|
+
import { LogDescription } from "ethers/lib/utils";
|
16
|
+
|
17
|
+
class ProccessBridgeRequestEvents extends BaseTask {
|
18
|
+
contractAddress: string;
|
19
|
+
safeContractAddress: string;
|
20
|
+
provider: ethers.providers.JsonRpcProvider;
|
21
|
+
contract: InteropXContract;
|
22
|
+
safeContract: GnosisSafe;
|
23
|
+
chainId: ChainId;
|
24
|
+
sourceWallet: Wallet;
|
25
|
+
|
26
|
+
leadNodeOnly: boolean = true;
|
27
|
+
|
28
|
+
constructor({ chainId }: { chainId: ChainId }) {
|
29
|
+
super({
|
30
|
+
logger: new Logger("InteropXContract::ProccessBridgeRequestEvents"),
|
31
|
+
})
|
32
|
+
this.chainId = chainId;
|
33
|
+
}
|
34
|
+
|
35
|
+
async pollHandler() {
|
36
|
+
const blockNumber = await this.provider.getBlockNumber()
|
37
|
+
|
38
|
+
const transaction = await Transaction.findOne({
|
39
|
+
where: {
|
40
|
+
status: 'pending',
|
41
|
+
sourceStatus: 'uninitialised',
|
42
|
+
createdAt: {
|
43
|
+
[Op.gte]: new Date(Date.now() - 12 * 60 * 60 * 1000),
|
44
|
+
},
|
45
|
+
sourceDelayUntil: {
|
46
|
+
[Op.or]: {
|
47
|
+
[Op.is]: null,
|
48
|
+
[Op.lt]: new Date(),
|
49
|
+
}
|
50
|
+
},
|
51
|
+
requestBlockNumber: {
|
52
|
+
[Op.lt]: blockNumber - 12,
|
53
|
+
},
|
54
|
+
sourceChainId: this.chainId,
|
55
|
+
}
|
56
|
+
})
|
57
|
+
|
58
|
+
if (!transaction) {
|
59
|
+
return;
|
60
|
+
}
|
61
|
+
|
62
|
+
console.log(`Processing transaction ${transaction.transactionHash}`);
|
63
|
+
|
64
|
+
transaction.targetStatus = 'pending';
|
65
|
+
await transaction.save();
|
66
|
+
|
67
|
+
const ownersThreshold = await this.safeContract.getThreshold();
|
68
|
+
await wait(10000);
|
69
|
+
|
70
|
+
|
71
|
+
let data, logs = [];
|
72
|
+
|
73
|
+
try {
|
74
|
+
({ data, logs } = await buildGnosisAction(transaction, 'source'));
|
75
|
+
|
76
|
+
} catch (error) {
|
77
|
+
|
78
|
+
if(error instanceof LiquidityError){
|
79
|
+
await transaction.save();
|
80
|
+
transaction.sourceDelayUntil = new Date(Date.now() + 60 * 5 * 1000);
|
81
|
+
transaction.sourceStatus = 'uninitialised'
|
82
|
+
|
83
|
+
await transaction.save();
|
84
|
+
|
85
|
+
throw error
|
86
|
+
return;
|
87
|
+
}
|
88
|
+
|
89
|
+
transaction.sourceStatus = 'failed';
|
90
|
+
transaction.sourceErrors = [error.message];
|
91
|
+
transaction.targetStatus = 'failed';
|
92
|
+
|
93
|
+
transaction.status = 'failed'
|
94
|
+
await transaction.save();
|
95
|
+
protocol.sendTransaction(transaction)
|
96
|
+
return;
|
97
|
+
}
|
98
|
+
|
99
|
+
let gnosisTx = await generateGnosisTransaction({
|
100
|
+
baseGas: "0",
|
101
|
+
data,
|
102
|
+
gasPrice: "0",
|
103
|
+
gasToken: "0x0000000000000000000000000000000000000000",
|
104
|
+
nonce: '0',
|
105
|
+
operation: "1",
|
106
|
+
refundReceiver: "0x0000000000000000000000000000000000000000",
|
107
|
+
safeAddress: this.safeContractAddress,
|
108
|
+
safeTxGas: "79668",
|
109
|
+
to: addresses[transaction.sourceChainId].multisend,
|
110
|
+
value: "0",
|
111
|
+
}, this.safeContract);
|
112
|
+
|
113
|
+
const owners = await this.safeContract.getOwners().then(owners => owners.map(owner => owner.toLowerCase()));
|
114
|
+
|
115
|
+
const ownerPeerIds = peerPool.activePeers.filter(peer => owners.includes(peer.publicAddress.toLowerCase())).map(peer => peer.id)
|
116
|
+
|
117
|
+
console.log(`Collecting signatures for execution ${transaction.transactionHash}`)
|
118
|
+
|
119
|
+
console.log(ownerPeerIds);
|
120
|
+
|
121
|
+
const signatures = await protocol.requestSignatures({
|
122
|
+
type: 'source',
|
123
|
+
transactionHash: transaction.transactionHash,
|
124
|
+
safeTxGas: gnosisTx.safeTxGas,
|
125
|
+
safeNonce: gnosisTx.nonce
|
126
|
+
}, ownerPeerIds)
|
127
|
+
|
128
|
+
|
129
|
+
const validSignatures = signatures.filter(s => !!s.data && s.data !== '0x') as Signature[];
|
130
|
+
|
131
|
+
console.log({ signatures, validSignatures, ownersThreshold: ownersThreshold.toString() });
|
132
|
+
|
133
|
+
if (validSignatures.length === 0 || ownersThreshold.gt(validSignatures.length)) {
|
134
|
+
await transaction.save();
|
135
|
+
transaction.sourceDelayUntil = new Date(Date.now() + 30 * 1000);
|
136
|
+
transaction.sourceStatus = 'uninitialised'
|
137
|
+
|
138
|
+
await transaction.save();
|
139
|
+
const errorMessage = signatures.find(s => !!s.error)?.error;
|
140
|
+
throw new Error(`Not enough signatures` + (errorMessage ? `: ${errorMessage}` : ''));
|
141
|
+
}
|
142
|
+
|
143
|
+
console.log(`Executing transaction for execution ${transaction.transactionHash}`)
|
144
|
+
|
145
|
+
const { data: txData } = await this.safeContract.populateTransaction.execTransaction(
|
146
|
+
gnosisTx.to,
|
147
|
+
gnosisTx.value,
|
148
|
+
gnosisTx.data,
|
149
|
+
gnosisTx.operation,
|
150
|
+
gnosisTx.safeTxGas,
|
151
|
+
gnosisTx.baseGas,
|
152
|
+
gnosisTx.gasPrice,
|
153
|
+
gnosisTx.gasToken,
|
154
|
+
gnosisTx.refundReceiver,
|
155
|
+
buildSignatureBytes(validSignatures)
|
156
|
+
);
|
157
|
+
|
158
|
+
const txSent = await this.sourceWallet.sendTransaction({
|
159
|
+
from: this.sourceWallet.address,
|
160
|
+
gasPrice: ethers.BigNumber.from(120 * 10 ** 9),
|
161
|
+
to: this.safeContractAddress,
|
162
|
+
data: txData,
|
163
|
+
})
|
164
|
+
|
165
|
+
console.log(txSent);
|
166
|
+
|
167
|
+
|
168
|
+
const receipt = await txSent.wait();
|
169
|
+
|
170
|
+
const parsedLogs: LogDescription[] = [];
|
171
|
+
|
172
|
+
receipt.logs.forEach((log) => {
|
173
|
+
try {
|
174
|
+
parsedLogs.push(this.safeContract.interface.parseLog(log));
|
175
|
+
} catch (e) { }
|
176
|
+
});
|
177
|
+
|
178
|
+
if (parsedLogs.find(e => e.name === 'ExecutionSuccess')) {
|
179
|
+
console.log('ExecutionSuccess')
|
180
|
+
transaction.sourceStatus = 'success'
|
181
|
+
if (txSent.blockNumber)
|
182
|
+
transaction.sourceBlockNumber = txSent.blockNumber;
|
183
|
+
transaction.sourceTransactionHash = txSent.hash
|
184
|
+
transaction.sourceLogs = logs;
|
185
|
+
await transaction.save();
|
186
|
+
} else {
|
187
|
+
console.log('ExecutionFailure')
|
188
|
+
transaction.sourceStatus = 'failed'
|
189
|
+
if (txSent.blockNumber)
|
190
|
+
transaction.sourceBlockNumber = txSent.blockNumber;
|
191
|
+
transaction.sourceTransactionHash = txSent.hash
|
192
|
+
transaction.sourceTransactionHash = txSent.hash
|
193
|
+
transaction.status = 'failed'
|
194
|
+
await transaction.save();
|
195
|
+
}
|
196
|
+
|
197
|
+
protocol.sendTransaction(transaction)
|
198
|
+
}
|
199
|
+
|
200
|
+
async start(): Promise<void> {
|
201
|
+
this.contractAddress = addresses[this.chainId].interopXContract;
|
202
|
+
|
203
|
+
this.provider = new ethers.providers.JsonRpcProvider(
|
204
|
+
getRpcProviderUrl(this.chainId)
|
205
|
+
);
|
206
|
+
|
207
|
+
this.sourceWallet = new ethers.Wallet(config.privateKey!, this.provider);
|
208
|
+
|
209
|
+
this.contract = getContract<InteropXContract>(
|
210
|
+
this.contractAddress,
|
211
|
+
abi.interopXContract,
|
212
|
+
this.sourceWallet
|
213
|
+
);
|
214
|
+
|
215
|
+
this.safeContractAddress = addresses[this.chainId].gnosisSafe;
|
216
|
+
this.safeContract = getContract<GnosisSafe>(
|
217
|
+
this.safeContractAddress,
|
218
|
+
abi.gnosisSafe,
|
219
|
+
this.sourceWallet
|
220
|
+
);
|
221
|
+
|
222
|
+
await super.start()
|
223
|
+
}
|
224
|
+
}
|
225
|
+
|
226
|
+
export default ProccessBridgeRequestEvents;
|