Skip to content

Commit 13707de

Browse files
committed
add utils for creating order request
1 parent b55c9fb commit 13707de

File tree

4 files changed

+228
-150
lines changed

4 files changed

+228
-150
lines changed

solana/ts/src/protocol/tokenRouter.ts

+105-29
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,19 @@ import {
99
VersionedTransaction,
1010
} from "@solana/web3.js";
1111
import {
12+
FastMarketOrder,
1213
FastTransfer,
14+
Message,
1315
TokenRouter,
1416
} from "@wormhole-foundation/example-liquidity-layer-definitions";
15-
import { Chain, Network, Platform, toChainId } from "@wormhole-foundation/sdk-base";
17+
import { Chain, ChainId, Network, Platform, toChainId } from "@wormhole-foundation/sdk-base";
1618
import {
1719
AccountAddress,
1820
ChainAddress,
1921
ChainsConfig,
2022
CircleBridge,
2123
Contracts,
24+
UniversalAddress,
2225
UnsignedTransaction,
2326
} from "@wormhole-foundation/sdk-definitions";
2427
import {
@@ -73,6 +76,10 @@ export class SolanaTokenRouter<N extends Network, C extends SolanaChains>
7376
});
7477
}
7578

79+
private usdcMint(wallet: PublicKey) {
80+
return splToken.getAssociatedTokenAddressSync(this.mint, wallet);
81+
}
82+
7683
async *initialize(
7784
owner: AnySolanaAddress,
7885
ownerAssistant: AnySolanaAddress,
@@ -89,47 +96,79 @@ export class SolanaTokenRouter<N extends Network, C extends SolanaChains>
8996
yield this.createUnsignedTx({ transaction }, "TokenRouter.Initialize");
9097
}
9198

92-
async *prepareMarketOrder(
99+
private makeFastMarketOrder(
93100
sender: AnySolanaAddress,
94-
amount: bigint,
95-
redeemer: ChainAddress<Chain>,
96-
minAmountOut?: bigint,
97-
redeemerMessage?: Uint8Array,
98-
preparedOrder?: Keypair,
99-
) {
101+
order: Partial<FastMarketOrder>,
102+
): FastMarketOrder {
103+
const senderAddress = new SolanaAddress(sender).toUniversalAddress();
104+
const o: FastMarketOrder = {
105+
// TODO: from auction params? take as args?
106+
maxFee: order.maxFee ?? 0n,
107+
initAuctionFee: order.initAuctionFee ?? 0n,
108+
deadline: order.deadline ?? 0,
109+
// TODO: specify which params we need, not just partial
110+
amountIn: order.amountIn!,
111+
minAmountOut: order.minAmountOut!,
112+
targetChain: order.targetChain!,
113+
redeemer: order.redeemer!,
114+
// TODO: which of these can we assume? any
115+
sender: order.sender ?? senderAddress,
116+
refundAddress: order.refundAddress ?? senderAddress,
117+
redeemerMessage: order.redeemerMessage ?? new Uint8Array(),
118+
};
119+
120+
return o;
121+
}
122+
123+
private async _prepareMarketOrderIxs(
124+
sender: AnySolanaAddress,
125+
order: TokenRouter.OrderRequest,
126+
prepareTo?: Keypair,
127+
): Promise<[TransactionInstruction[], Keypair[]]> {
100128
const payer = new SolanaAddress(sender).unwrap();
101129

102-
// assume sender token is the usdc mint address
103-
const senderToken = splToken.getAssociatedTokenAddressSync(this.mint, payer);
130+
// TODO: assumes sender token is the usdc mint address
131+
const senderToken = this.usdcMint(payer);
104132

105133
// Where we'll write the prepared order
106-
preparedOrder = preparedOrder ?? Keypair.generate();
134+
prepareTo = prepareTo ?? Keypair.generate();
107135

108136
const [approveIx, prepareIx] = await this.prepareMarketOrderIx(
109137
{
110138
payer,
111139
senderToken,
112-
preparedOrder: preparedOrder.publicKey,
140+
preparedOrder: prepareTo.publicKey,
113141
},
114142
{
115-
amountIn: amount,
116-
minAmountOut: minAmountOut !== undefined ? minAmountOut : null,
117-
targetChain: toChainId(redeemer.chain),
118-
redeemer: Array.from(redeemer.address.toUniversalAddress().toUint8Array()),
119-
redeemerMessage: redeemerMessage ? Buffer.from(redeemerMessage) : Buffer.from(""),
143+
amountIn: order.amountIn,
144+
minAmountOut: order.minAmountOut !== undefined ? order.minAmountOut : null,
145+
targetChain: toChainId(order.targetChain),
146+
redeemer: Array.from(order.redeemer.toUint8Array()),
147+
redeemerMessage: order.redeemerMessage
148+
? Buffer.from(order.redeemerMessage)
149+
: Buffer.from(""),
120150
},
121151
);
122152

123-
// TODO: fix prepareMarketOrderIx to not return null at all
153+
// TODO: fix prepareMarketOrderIx to not return null at all?
124154
const ixs = [];
125155
if (approveIx) ixs.push(approveIx);
126156
ixs.push(prepareIx);
127157

158+
return [ixs, [prepareTo]];
159+
}
160+
161+
async *prepareMarketOrder(
162+
sender: AnySolanaAddress,
163+
order: TokenRouter.OrderRequest,
164+
prepareTo?: Keypair,
165+
) {
166+
const payer = new SolanaAddress(sender).unwrap();
167+
168+
const [ixs, signers] = await this._prepareMarketOrderIxs(sender, order, prepareTo);
169+
128170
const transaction = this.createTx(payer, ixs);
129-
yield this.createUnsignedTx(
130-
{ transaction, signers: [preparedOrder] },
131-
"TokenRouter.PrepareMarketOrder",
132-
);
171+
yield this.createUnsignedTx({ transaction, signers }, "TokenRouter.PrepareMarketOrder");
133172
}
134173

135174
async *closePreparedOrder(sender: AnySolanaAddress, order: AnySolanaAddress) {
@@ -139,20 +178,57 @@ export class SolanaTokenRouter<N extends Network, C extends SolanaChains>
139178
const ix = await this.closePreparedOrderIx({
140179
preparedOrder,
141180
preparedBy: payer,
181+
orderSender: payer,
142182
});
143183

144184
const transaction = this.createTx(payer, [ix]);
185+
145186
yield this.createUnsignedTx({ transaction }, "TokenRouter.ClosePreparedOrder");
146187
}
147188

148-
placeMarketOrder(
149-
amount: bigint,
150-
redeemer: ChainAddress<Chain>,
151-
redeemerMessage: Uint8Array,
152-
minAmountOut?: bigint | undefined,
153-
refundAddress?: AccountAddress<C> | undefined,
189+
async *placeMarketOrder(
190+
sender: AnySolanaAddress,
191+
order: TokenRouter.OrderRequest | AnySolanaAddress,
192+
prepareTo?: Keypair,
154193
): AsyncGenerator<UnsignedTransaction<N, C>, any, unknown> {
155-
throw new Error("Method not implemented.");
194+
const payer = new SolanaAddress(sender).unwrap();
195+
196+
let ixs: TransactionInstruction[] = [];
197+
let signers: Keypair[] = [];
198+
let preparedOrder: PublicKey;
199+
let targetChain: ChainId | undefined;
200+
201+
if (TokenRouter.isOrderRequest(order)) {
202+
prepareTo = prepareTo ?? Keypair.generate();
203+
204+
const combined = false; // TODO how to choose?
205+
if (combined) {
206+
const [ixs, signers] = await this._prepareMarketOrderIxs(sender, order, prepareTo);
207+
ixs.push(...ixs);
208+
signers.push(...signers);
209+
} else {
210+
yield* this.prepareMarketOrder(sender, order, prepareTo);
211+
}
212+
213+
preparedOrder = prepareTo.publicKey;
214+
targetChain = toChainId(order.targetChain);
215+
} else {
216+
preparedOrder = new SolanaAddress(order).unwrap();
217+
}
218+
219+
const ix = await this.placeMarketOrderCctpIx(
220+
{
221+
payer,
222+
preparedOrder,
223+
preparedBy: payer,
224+
},
225+
// TODO: add cctpDomain fn
226+
{ targetChain }, //,destinationDomain: 3 },
227+
);
228+
ixs.push(ix);
229+
230+
const transaction = this.createTx(payer, ixs);
231+
yield this.createUnsignedTx({ transaction, signers }, "TokenRouter.PlaceMarketOrder");
156232
}
157233

158234
placeFastMarketOrder<RC extends Chain>(

solana/ts/src/testing/mock.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@ export async function createLiquidityLayerVaa(
6464

6565
const vaa = MOCK_GUARDIANS.addSignatures(published, [0]);
6666

67-
// @ts-ignore -- lie
67+
// @ts-ignore -- TODO: this is lie, need to define discriminator
6868
return vaa;
6969
}
7070

0 commit comments

Comments
 (0)