Skip to content

Commit 83084e3

Browse files
committed
sdk: add missing admin functionality and queries
1 parent c88f2ed commit 83084e3

File tree

5 files changed

+574
-53
lines changed

5 files changed

+574
-53
lines changed

evm/ts/src/ntt.ts

+215-13
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import {
22
Chain,
33
Network,
4+
encoding,
45
nativeChainIds,
56
toChainId,
67
} from "@wormhole-foundation/sdk-base";
@@ -9,7 +10,9 @@ import {
910
ChainAddress,
1011
ChainsConfig,
1112
Contracts,
13+
canonicalAddress,
1214
serialize,
15+
toUniversal,
1316
universalAddress,
1417
} from "@wormhole-foundation/sdk-definitions";
1518
import type { EvmChains, EvmPlatformType } from "@wormhole-foundation/sdk-evm";
@@ -36,8 +39,7 @@ import {
3639
} from "./bindings.js";
3740

3841
export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
39-
implements NttTransceiver<N, C, WormholeNttTransceiver.VAA>
40-
{
42+
implements NttTransceiver<N, C, WormholeNttTransceiver.VAA> {
4143
transceiver: NttTransceiverBindings.NttTransceiver;
4244
constructor(
4345
readonly manager: EvmNtt<N, C>,
@@ -50,17 +52,48 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
5052
);
5153
}
5254

55+
getAddress(): ChainAddress<C> {
56+
return { chain: this.manager.chain, address: toUniversal(this.manager.chain, this.address) };
57+
}
58+
5359
encodeFlags(flags: { skipRelay: boolean }): Uint8Array {
5460
return new Uint8Array([flags.skipRelay ? 1 : 0]);
5561
}
5662

57-
async *setPeer(peer: ChainAddress<C>) {
63+
async *setPeer<P extends Chain>(peer: ChainAddress<P>): AsyncGenerator<EvmUnsignedTransaction<N, C>> {
5864
const tx = await this.transceiver.setWormholePeer.populateTransaction(
5965
toChainId(peer.chain),
6066
universalAddress(peer)
6167
);
6268
yield this.manager.createUnsignedTx(tx, "WormholeTransceiver.registerPeer");
6369
}
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+
6497
async *receive(attestation: WormholeNttTransceiver.VAA) {
6598
const tx = await this.transceiver.receiveMessage.populateTransaction(
6699
serialize(attestation)
@@ -77,6 +110,17 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
77110
);
78111
}
79112

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+
80124
async isSpecialRelayingEnabled(destChain: Chain): Promise<boolean> {
81125
return await this.transceiver.isSpecialRelayingEnabled(
82126
toChainId(destChain)
@@ -85,8 +129,7 @@ export class EvmNttWormholeTranceiver<N extends Network, C extends EvmChains>
85129
}
86130

87131
export class EvmNtt<N extends Network, C extends EvmChains>
88-
implements Ntt<N, C>
89-
{
132+
implements Ntt<N, C> {
90133
tokenAddress: string;
91134
readonly chainId: bigint;
92135
manager: NttManagerBindings.NttManager;
@@ -117,14 +160,62 @@ export class EvmNtt<N extends Network, C extends EvmChains>
117160
this.provider
118161
);
119162

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());
128219
}
129220

130221
async isRelayingAvailable(destination: Chain): Promise<boolean> {
@@ -187,6 +278,21 @@ export class EvmNtt<N extends Network, C extends EvmChains>
187278
);
188279
}
189280

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+
190296
static async fromRpc<N extends Network>(
191297
provider: Provider,
192298
config: ChainsConfig<N, EvmPlatformType>
@@ -342,10 +448,39 @@ export class EvmNtt<N extends Network, C extends EvmChains>
342448
return await this.manager.getCurrentOutboundCapacity();
343449
}
344450

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+
345464
async getCurrentInboundCapacity(fromChain: Chain): Promise<bigint> {
346465
return await this.manager.getCurrentInboundCapacity(toChainId(fromChain));
347466
}
348467

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+
349484
async getRateLimitDuration(): Promise<bigint> {
350485
return await this.manager.rateLimitDuration();
351486
}
@@ -381,6 +516,40 @@ export class EvmNtt<N extends Network, C extends EvmChains>
381516
yield this.createUnsignedTx(tx, "Ntt.completeInboundQueuedTransfer");
382517
}
383518

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+
384553
createUnsignedTx(
385554
txReq: TransactionRequest,
386555
description: string,
@@ -395,3 +564,36 @@ export class EvmNtt<N extends Network, C extends EvmChains>
395564
);
396565
}
397566
}
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+
}

sdk/__tests__/utils.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ export async function link(chainInfos: Ctx[]) {
127127
chain: hubChain,
128128
emitter: Wormhole.chainAddress(
129129
hubChain,
130-
hub.contracts!.transceiver.wormhole
130+
hub.contracts!.transceiver.wormhole!
131131
).address.toUniversalAddress(),
132132
sequence: 0n,
133133
};
@@ -595,7 +595,7 @@ async function setupPeer(targetCtx: Ctx, peerCtx: Ctx) {
595595
} = peerCtx.contracts!;
596596

597597
const peerManager = Wormhole.chainAddress(peer.chain, manager);
598-
const peerTransceiver = Wormhole.chainAddress(peer.chain, transceiver);
598+
const peerTransceiver = Wormhole.chainAddress(peer.chain, transceiver!);
599599

600600
const tokenDecimals = target.config.nativeTokenDecimals;
601601
const inboundLimit = amount.units(amount.parse("1000", tokenDecimals));
@@ -625,7 +625,7 @@ async function setupPeer(targetCtx: Ctx, peerCtx: Ctx) {
625625
) {
626626
const nativeSigner = (signer as NativeSigner).unwrap();
627627
const xcvr = WormholeTransceiver__factory.connect(
628-
targetCtx.contracts!.transceiver.wormhole,
628+
targetCtx.contracts!.transceiver.wormhole!,
629629
nativeSigner.signer
630630
);
631631
const peerChainId = toChainId(peer.chain);

0 commit comments

Comments
 (0)