1
1
import {
2
2
Chain ,
3
3
Network ,
4
+ encoding ,
4
5
nativeChainIds ,
5
6
toChainId ,
6
7
} from "@wormhole-foundation/sdk-base" ;
@@ -9,7 +10,9 @@ import {
9
10
ChainAddress ,
10
11
ChainsConfig ,
11
12
Contracts ,
13
+ canonicalAddress ,
12
14
serialize ,
15
+ toUniversal ,
13
16
universalAddress ,
14
17
} from "@wormhole-foundation/sdk-definitions" ;
15
18
import type { EvmChains , EvmPlatformType } from "@wormhole-foundation/sdk-evm" ;
@@ -36,8 +39,7 @@ import {
36
39
} from "./bindings.js" ;
37
40
38
41
export class EvmNttWormholeTranceiver < N extends Network , C extends EvmChains >
39
- implements NttTransceiver < N , C , WormholeNttTransceiver . VAA >
40
- {
42
+ implements NttTransceiver < N , C , WormholeNttTransceiver . VAA > {
41
43
transceiver : NttTransceiverBindings . NttTransceiver ;
42
44
constructor (
43
45
readonly manager : EvmNtt < N , C > ,
@@ -50,17 +52,48 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
50
52
) ;
51
53
}
52
54
55
+ getAddress ( ) : ChainAddress < C > {
56
+ return { chain : this . manager . chain , address : toUniversal ( this . manager . chain , this . address ) } ;
57
+ }
58
+
53
59
encodeFlags ( flags : { skipRelay : boolean } ) : Uint8Array {
54
60
return new Uint8Array ( [ flags . skipRelay ? 1 : 0 ] ) ;
55
61
}
56
62
57
- async * setPeer ( peer : ChainAddress < C > ) {
63
+ async * setPeer < P extends Chain > ( peer : ChainAddress < P > ) : AsyncGenerator < EvmUnsignedTransaction < N , C > > {
58
64
const tx = await this . transceiver . setWormholePeer . populateTransaction (
59
65
toChainId ( peer . chain ) ,
60
66
universalAddress ( peer )
61
67
) ;
62
68
yield this . manager . createUnsignedTx ( tx , "WormholeTransceiver.registerPeer" ) ;
63
69
}
70
+
71
+ async getPeer < C extends Chain > ( chain : C ) : Promise < ChainAddress < C > | null > {
72
+ const peer = await this . transceiver . getWormholePeer ( toChainId ( chain ) ) ;
73
+ const peerAddress = encoding . hex . decode ( peer ) ;
74
+ const zeroAddress = new Uint8Array ( 32 ) ;
75
+ if ( encoding . bytes . equals ( zeroAddress , peerAddress ) ) {
76
+ return null ;
77
+ }
78
+
79
+ return {
80
+ chain : chain ,
81
+ address : toUniversal ( chain , peerAddress ) ,
82
+ } ;
83
+ }
84
+
85
+ async isEvmChain ( chain : Chain ) : Promise < boolean > {
86
+ return await this . transceiver . isWormholeEvmChain ( toChainId ( chain ) ) ;
87
+ }
88
+
89
+ async * setIsEvmChain ( chain : Chain , isEvm : boolean ) {
90
+ const tx = await this . transceiver . setIsWormholeEvmChain . populateTransaction (
91
+ toChainId ( chain ) ,
92
+ isEvm
93
+ ) ;
94
+ yield this . manager . createUnsignedTx ( tx , "WormholeTransceiver.setIsEvmChain" ) ;
95
+ }
96
+
64
97
async * receive ( attestation : WormholeNttTransceiver . VAA ) {
65
98
const tx = await this . transceiver . receiveMessage . populateTransaction (
66
99
serialize ( attestation )
@@ -77,6 +110,17 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
77
110
) ;
78
111
}
79
112
113
+ async * setIsWormholeRelayingEnabled ( destChain : Chain , enabled : boolean ) {
114
+ const tx = await this . transceiver . setIsWormholeRelayingEnabled . populateTransaction (
115
+ toChainId ( destChain ) ,
116
+ enabled
117
+ ) ;
118
+ yield this . manager . createUnsignedTx (
119
+ tx ,
120
+ "WormholeTransceiver.setWormholeRelayingEnabled"
121
+ ) ;
122
+ }
123
+
80
124
async isSpecialRelayingEnabled ( destChain : Chain ) : Promise < boolean > {
81
125
return await this . transceiver . isSpecialRelayingEnabled (
82
126
toChainId ( destChain )
@@ -85,8 +129,7 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
85
129
}
86
130
87
131
export class EvmNtt < N extends Network , C extends EvmChains >
88
- implements Ntt < N , C >
89
- {
132
+ implements Ntt < N , C > {
90
133
tokenAddress : string ;
91
134
readonly chainId : bigint ;
92
135
manager : NttManagerBindings . NttManager ;
@@ -117,14 +160,62 @@ export class EvmNtt<N extends Network, C extends EvmChains>
117
160
this . provider
118
161
) ;
119
162
120
- this . xcvrs = [
121
- // Enable more Transceivers here
122
- new EvmNttWormholeTranceiver (
123
- this ,
124
- contracts . ntt . transceiver . wormhole ! ,
125
- abiBindings !
126
- ) ,
127
- ] ;
163
+ if ( contracts . ntt . transceiver . wormhole != null ) {
164
+ this . xcvrs = [
165
+ // Enable more Transceivers here
166
+ new EvmNttWormholeTranceiver (
167
+ this ,
168
+ contracts . ntt . transceiver . wormhole ,
169
+ abiBindings !
170
+ ) ,
171
+ ] ;
172
+ } else {
173
+ this . xcvrs = [ ] ;
174
+ }
175
+ }
176
+
177
+ async getTransceiver ( ix : number ) : Promise < NttTransceiver < N , C , any > | null > {
178
+ // TODO: should we make an RPC call here, or just trust that the xcvrs are set up correctly?
179
+ return this . xcvrs [ ix ] || null ;
180
+ }
181
+
182
+ async getMode ( ) : Promise < Ntt . Mode > {
183
+ const mode : bigint = await this . manager . getMode ( ) ;
184
+ return mode === 0n ? "locking" : "burning" ;
185
+ }
186
+
187
+ async isPaused ( ) : Promise < boolean > {
188
+ return await this . manager . isPaused ( ) ;
189
+ }
190
+
191
+ async * pause ( ) {
192
+ const tx = await this . manager . pause . populateTransaction ( )
193
+ yield this . createUnsignedTx ( tx , "Ntt.pause" ) ;
194
+ }
195
+
196
+ async * unpause ( ) {
197
+ const tx = await this . manager . unpause . populateTransaction ( )
198
+ yield this . createUnsignedTx ( tx , "Ntt.unpause" ) ;
199
+ }
200
+
201
+ async getOwner ( ) : Promise < AccountAddress < C > > {
202
+ return new EvmAddress ( await this . manager . owner ( ) ) as AccountAddress < C > ;
203
+ }
204
+
205
+ async * setOwner ( owner : AccountAddress < C > ) {
206
+ const canonicalOwner = canonicalAddress ( { chain : this . chain , address : owner } ) ;
207
+ const tx = await this . manager . transferOwnership . populateTransaction ( canonicalOwner ) ;
208
+ yield this . createUnsignedTx ( tx , "Ntt.setOwner" ) ;
209
+ }
210
+
211
+ async * setPauser ( pauser : AccountAddress < C > ) {
212
+ const canonicalPauser = canonicalAddress ( { chain : this . chain , address : pauser } ) ;
213
+ const tx = await this . manager . transferPauserCapability . populateTransaction ( canonicalPauser ) ;
214
+ yield this . createUnsignedTx ( tx , "Ntt.setPauser" ) ;
215
+ }
216
+
217
+ async getThreshold ( ) : Promise < number > {
218
+ return Number ( await this . manager . getThreshold ( ) ) ;
128
219
}
129
220
130
221
async isRelayingAvailable ( destination : Chain ) : Promise < boolean > {
@@ -187,6 +278,21 @@ export class EvmNtt<N extends Network, C extends EvmChains>
187
278
) ;
188
279
}
189
280
281
+ async getPeer < C extends Chain > ( chain : C ) : Promise < Ntt . Peer < C > | null > {
282
+ const peer = await this . manager . getPeer ( toChainId ( chain ) ) ;
283
+ const peerAddress = encoding . hex . decode ( peer . peerAddress ) ;
284
+ const zeroAddress = new Uint8Array ( 32 ) ;
285
+ if ( encoding . bytes . equals ( zeroAddress , peerAddress ) ) {
286
+ return null ;
287
+ }
288
+
289
+ return {
290
+ address : { chain : chain , address : toUniversal ( chain , peerAddress ) } ,
291
+ tokenDecimals : Number ( peer . tokenDecimals ) ,
292
+ inboundLimit : await this . getInboundLimit ( chain ) ,
293
+ } ;
294
+ }
295
+
190
296
static async fromRpc < N extends Network > (
191
297
provider : Provider ,
192
298
config : ChainsConfig < N , EvmPlatformType >
@@ -342,10 +448,39 @@ export class EvmNtt<N extends Network, C extends EvmChains>
342
448
return await this . manager . getCurrentOutboundCapacity ( ) ;
343
449
}
344
450
451
+ async getOutboundLimit ( ) : Promise < bigint > {
452
+ const encoded : EncodedTrimmedAmount = ( await this . manager . getOutboundLimitParams ( ) ) . limit ;
453
+ const trimmedAmount : TrimmedAmount = decodeTrimmedAmount ( encoded ) ;
454
+ const tokenDecimals = await this . getTokenDecimals ( ) ;
455
+
456
+ return untrim ( trimmedAmount , tokenDecimals ) ;
457
+ }
458
+
459
+ async * setOutboundLimit ( limit : bigint ) {
460
+ const tx = await this . manager . setOutboundLimit . populateTransaction ( limit ) ;
461
+ yield this . createUnsignedTx ( tx , "Ntt.setOutboundLimit" ) ;
462
+ }
463
+
345
464
async getCurrentInboundCapacity ( fromChain : Chain ) : Promise < bigint > {
346
465
return await this . manager . getCurrentInboundCapacity ( toChainId ( fromChain ) ) ;
347
466
}
348
467
468
+ async getInboundLimit ( fromChain : Chain ) : Promise < bigint > {
469
+ const encoded : EncodedTrimmedAmount = ( await this . manager . getInboundLimitParams ( toChainId ( fromChain ) ) ) . limit ;
470
+ const trimmedAmount : TrimmedAmount = decodeTrimmedAmount ( encoded ) ;
471
+ const tokenDecimals = await this . getTokenDecimals ( ) ;
472
+
473
+ return untrim ( trimmedAmount , tokenDecimals ) ;
474
+ }
475
+
476
+ async * setInboundLimit ( fromChain : Chain , limit : bigint ) {
477
+ const tx = await this . manager . setInboundLimit . populateTransaction (
478
+ limit ,
479
+ toChainId ( fromChain )
480
+ ) ;
481
+ yield this . createUnsignedTx ( tx , "Ntt.setInboundLimit" ) ;
482
+ }
483
+
349
484
async getRateLimitDuration ( ) : Promise < bigint > {
350
485
return await this . manager . rateLimitDuration ( ) ;
351
486
}
@@ -381,6 +516,40 @@ export class EvmNtt<N extends Network, C extends EvmChains>
381
516
yield this . createUnsignedTx ( tx , "Ntt.completeInboundQueuedTransfer" ) ;
382
517
}
383
518
519
+ async verifyAddresses ( ) : Promise < Partial < Ntt . Contracts > | null > {
520
+ const local : Partial < Ntt . Contracts > = {
521
+ manager : this . managerAddress ,
522
+ token : this . tokenAddress ,
523
+ transceiver : {
524
+ wormhole : this . xcvrs [ 0 ] ?. address ,
525
+ } ,
526
+ // TODO: what about the quoter?
527
+ } ;
528
+
529
+ const remote : Partial < Ntt . Contracts > = {
530
+ manager : this . managerAddress ,
531
+ token : await this . manager . token ( ) ,
532
+ transceiver : {
533
+ wormhole : ( await this . manager . getTransceivers ( ) ) [ 0 ] ! // TODO: make this more generic
534
+ } ,
535
+ } ;
536
+
537
+ const deleteMatching = ( a : any , b : any ) => {
538
+ for ( const k in a ) {
539
+ if ( typeof a [ k ] === "object" ) {
540
+ deleteMatching ( a [ k ] , b [ k ] ) ;
541
+ if ( Object . keys ( a [ k ] ) . length === 0 ) delete a [ k ] ;
542
+ } else if ( a [ k ] === b [ k ] ) {
543
+ delete a [ k ] ;
544
+ }
545
+ }
546
+ }
547
+
548
+ deleteMatching ( remote , local ) ;
549
+
550
+ return Object . keys ( remote ) . length > 0 ? remote : null ;
551
+ }
552
+
384
553
createUnsignedTx (
385
554
txReq : TransactionRequest ,
386
555
description : string ,
@@ -395,3 +564,36 @@ export class EvmNtt<N extends Network, C extends EvmChains>
395
564
) ;
396
565
}
397
566
}
567
+
568
+ type EncodedTrimmedAmount = bigint ; // uint72
569
+
570
+ type TrimmedAmount = {
571
+ amount : bigint ;
572
+ decimals : number ;
573
+ } ;
574
+
575
+ function decodeTrimmedAmount ( encoded : EncodedTrimmedAmount ) : TrimmedAmount {
576
+ const decimals = Number ( encoded & 0xffn ) ;
577
+ const amount = encoded >> 8n ;
578
+ return {
579
+ amount,
580
+ decimals,
581
+ } ;
582
+ }
583
+
584
+ function untrim ( trimmed : TrimmedAmount , toDecimals : number ) : bigint {
585
+ const { amount, decimals : fromDecimals } = trimmed ;
586
+ return scale ( amount , fromDecimals , toDecimals ) ;
587
+ }
588
+
589
+ function scale ( amount : bigint , fromDecimals : number , toDecimals : number ) : bigint {
590
+ if ( fromDecimals == toDecimals ) {
591
+ return amount ;
592
+ }
593
+
594
+ if ( fromDecimals > toDecimals ) {
595
+ return amount / ( 10n ** BigInt ( fromDecimals - toDecimals ) ) ;
596
+ } else {
597
+ return amount * ( 10n ** BigInt ( toDecimals - fromDecimals ) ) ;
598
+ }
599
+ }
0 commit comments