Skip to content

Commit 3a120b8

Browse files
committed
feat: add twap demo to send-usd example app
1 parent c38a341 commit 3a120b8

File tree

9 files changed

+424
-137
lines changed

9 files changed

+424
-137
lines changed

package-lock.json

+6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

price_feeds/solana/send_usd/app/package-lock.json

+114-74
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

price_feeds/solana/send_usd/app/package.json

+4-4
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
{
22
"name": "send_usd",
3-
"version": "0.1.0",
3+
"version": "0.2.0",
44
"private": true,
55
"dependencies": {
6-
"@pythnetwork/price-service-client": "^1.8.2",
7-
"@pythnetwork/pyth-solana-receiver": "^0.7.0",
6+
"@pythnetwork/hermes-client": "^2.0.0",
7+
"@pythnetwork/pyth-solana-receiver": "^0.10.0",
88
"@solana/wallet-adapter-base": "^0.9.23",
99
"@solana/wallet-adapter-react": "^0.15.35",
1010
"@solana/wallet-adapter-react-ui": "^0.9.35",
@@ -49,4 +49,4 @@
4949
"last 1 safari version"
5050
]
5151
}
52-
}
52+
}

price_feeds/solana/send_usd/app/src/App.tsx

+127-40
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ import { Connection, PublicKey } from "@solana/web3.js";
2020
import * as buffer from "buffer";
2121
import { AnchorProvider, BN, Program, Wallet } from "@coral-xyz/anchor";
2222
import { SendUSDApp, IDL } from "./idl/send_usd_app";
23-
import { PriceServiceConnection } from "@pythnetwork/price-service-client";
23+
import { HermesClient } from "@pythnetwork/hermes-client";
2424
import { useState } from "react";
2525
window.Buffer = buffer.Buffer;
2626

@@ -43,17 +43,13 @@ async function postPriceUpdate(
4343
if (!(wallet && destination && amount)) {
4444
return;
4545
} else {
46-
const priceServiceConnection = new PriceServiceConnection(HERMES_URL, {
47-
priceFeedRequestConfig: { binary: true },
48-
});
46+
const hermesClient = new HermesClient(HERMES_URL);
4947
const pythSolanaReceiver = new PythSolanaReceiver({
5048
connection,
5149
wallet: wallet as Wallet,
5250
});
5351

54-
const priceUpdateData = await priceServiceConnection.getLatestVaas([
55-
SOL_PRICE_FEED_ID,
56-
]);
52+
const priceUpdateData = await hermesClient.getLatestPriceUpdates([SOL_PRICE_FEED_ID], { encoding: "base64" });
5753

5854
const sendUsdApp = new Program<SendUSDApp>(
5955
IDL as SendUSDApp,
@@ -64,7 +60,7 @@ async function postPriceUpdate(
6460
const transactionBuilder = pythSolanaReceiver.newTransactionBuilder({
6561
closeUpdateAccounts: true,
6662
});
67-
await transactionBuilder.addPostPriceUpdates([priceUpdateData[0]]);
63+
await transactionBuilder.addPostPriceUpdates(priceUpdateData.binary.data);
6864

6965
await transactionBuilder.addPriceConsumerInstructions(
7066
async (
@@ -94,26 +90,115 @@ async function postPriceUpdate(
9490
}
9591
}
9692

97-
function Button(props: {
93+
async function postTwapPriceUpdate(
94+
connection: Connection,
95+
wallet: AnchorWallet | undefined,
96+
destination: PublicKey | undefined,
97+
amount: number | undefined,
98+
twapWindowSeconds: number
99+
) {
100+
if (!(wallet && destination && amount)) {
101+
return;
102+
} else {
103+
const hermesClient = new HermesClient(HERMES_URL);
104+
const pythSolanaReceiver = new PythSolanaReceiver({
105+
connection,
106+
wallet: wallet as Wallet,
107+
});
108+
109+
const twapUpdateData = await hermesClient.getLatestTwaps([SOL_PRICE_FEED_ID], twapWindowSeconds, { encoding: "base64" });
110+
111+
const sendUsdApp = new Program<SendUSDApp>(
112+
IDL as SendUSDApp,
113+
SEND_USD_PROGRAM_ID,
114+
new AnchorProvider(connection, wallet, AnchorProvider.defaultOptions())
115+
);
116+
117+
const transactionBuilder = pythSolanaReceiver.newTransactionBuilder({
118+
closeUpdateAccounts: true,
119+
});
120+
await transactionBuilder.addPostTwapUpdates(twapUpdateData.binary.data);
121+
122+
await transactionBuilder.addTwapConsumerInstructions(
123+
async (
124+
getTwapUpdateAccount: (priceFeedId: string) => PublicKey
125+
): Promise<InstructionWithEphemeralSigners[]> => {
126+
return [
127+
{
128+
instruction: await sendUsdApp.methods
129+
.sendUsingTwap(new BN(amount), new BN(twapWindowSeconds))
130+
.accounts({
131+
destination,
132+
twapUpdate: getTwapUpdateAccount(SOL_PRICE_FEED_ID),
133+
})
134+
.instruction(),
135+
signers: [],
136+
},
137+
];
138+
}
139+
);
140+
141+
await pythSolanaReceiver.provider.sendAll(
142+
await transactionBuilder.buildVersionedTransactions({
143+
computeUnitPriceMicroLamports: 50000,
144+
}),
145+
{ skipPreflight: true }
146+
);
147+
}
148+
}
149+
150+
function Buttons(props: {
98151
destination: PublicKey | undefined;
99152
amount: number | undefined;
100153
}) {
101154
const connectionContext = useConnection();
102155
const wallet = useAnchorWallet();
103-
156+
const [twapWindowSeconds, setTwapWindowSeconds] = useState<number>(300);
104157
return (
105-
<button
106-
onClick={async () => {
107-
await postPriceUpdate(
108-
connectionContext.connection,
109-
wallet,
110-
props.destination,
111-
props.amount
112-
);
113-
}}
114-
>
115-
Send
116-
</button>
158+
<>
159+
<div style={{ display: "flex", marginBottom: "20px" }}>
160+
<button
161+
onClick={async () => {
162+
await postPriceUpdate(
163+
connectionContext.connection,
164+
wallet,
165+
props.destination,
166+
props.amount
167+
);
168+
}}
169+
className="wallet-adapter-button wallet-adapter-button-trigger"
170+
style={{ flex: "1", marginRight: "20px", height: "48px", fontSize: "16px" }}
171+
>
172+
Send using Spot Price
173+
</button>
174+
<div style={{ flex: "1", display: "flex", flexDirection: "column" }}>
175+
<button
176+
onClick={async () => {
177+
await postTwapPriceUpdate(
178+
connectionContext.connection,
179+
wallet,
180+
props.destination,
181+
props.amount,
182+
twapWindowSeconds
183+
);
184+
}}
185+
className="wallet-adapter-button wallet-adapter-button-trigger"
186+
style={{ height: "48px", fontSize: "16px", marginBottom: "10px" }}
187+
>
188+
Send using TWAP Price
189+
</button>
190+
<p style={{ fontSize: "16px", margin: "5px 0" }}>TWAP Window (seconds): {twapWindowSeconds}</p>
191+
<input
192+
type="range"
193+
min="0"
194+
max="599"
195+
value={twapWindowSeconds}
196+
onChange={(e) => setTwapWindowSeconds(parseInt(e.target.value))}
197+
style={{ width: "100%" }}
198+
/>
199+
</div>
200+
</div>
201+
</>
117202
);
118203
}
119204

@@ -146,24 +231,26 @@ function App() {
146231
<WalletMultiButton />
147232
<WalletDisconnectButton />
148233
<p>Click to send the amount of USD in SOL</p>
149-
<p style={{ fontSize: "16px" }}>
150-
Destination (paste a Solana public key)
151-
</p>
152-
<input
153-
type="text"
154-
value={destination ? destination.toString() : ""}
155-
onChange={handleSetDestination}
156-
style={{ width: "100%", height: "40px", fontSize: "16px" }}
157-
/>
158-
<p style={{ fontSize: "16px" }}>Amount (USD)</p>
159-
<input
160-
type="text"
161-
value={amount ? amount.toString() : ""}
162-
onChange={handleSetAmount}
163-
style={{ width: "100%", height: "40px", fontSize: "16px" }}
164-
/>
165-
166-
<Button destination={destination} amount={amount} />
234+
<div style={{ width: "50%", margin: "0 auto" }}>
235+
<p style={{ fontSize: "16px" }}>
236+
Destination (paste a Solana public key)
237+
</p>
238+
<input
239+
type="text"
240+
value={destination ? destination.toString() : ""}
241+
onChange={handleSetDestination}
242+
style={{ width: "100%", height: "40px", fontSize: "16px", marginBottom: "20px" }}
243+
/>
244+
<p style={{ fontSize: "16px" }}>Amount (USD)</p>
245+
<input
246+
type="text"
247+
value={amount ? amount.toString() : ""}
248+
onChange={handleSetAmount}
249+
style={{ width: "100%", height: "40px", fontSize: "16px", marginBottom: "20px" }}
250+
/>
251+
252+
<Buttons destination={destination} amount={amount} />
253+
</div>
167254
</header>
168255
</div>
169256
</WalletModalProvider>

price_feeds/solana/send_usd/app/src/idl/send_usd_app.json

+42-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,9 @@
1414
"name": "destination",
1515
"isMut": true,
1616
"isSigner": false,
17-
"docs": ["CHECK : Just a destination"]
17+
"docs": [
18+
"CHECK : Just a destination"
19+
]
1820
},
1921
{
2022
"name": "priceUpdate",
@@ -33,6 +35,44 @@
3335
"type": "u64"
3436
}
3537
]
38+
},
39+
{
40+
"name": "sendUsingTwap",
41+
"accounts": [
42+
{
43+
"name": "payer",
44+
"isMut": true,
45+
"isSigner": true
46+
},
47+
{
48+
"name": "destination",
49+
"isMut": true,
50+
"isSigner": false,
51+
"docs": [
52+
"CHECK : Just a destination"
53+
]
54+
},
55+
{
56+
"name": "twapUpdate",
57+
"isMut": false,
58+
"isSigner": false
59+
},
60+
{
61+
"name": "systemProgram",
62+
"isMut": false,
63+
"isSigner": false
64+
}
65+
],
66+
"args": [
67+
{
68+
"name": "amountInUsd",
69+
"type": "u64"
70+
},
71+
{
72+
"name": "twapWindowSeconds",
73+
"type": "u64"
74+
}
75+
]
3676
}
3777
]
38-
}
78+
}

price_feeds/solana/send_usd/app/src/idl/send_usd_app.ts

+74-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
export type SendUSDApp = {
2-
version: "0.1.0";
2+
version: "0.2.0";
33
name: "send_usd";
44
instructions: [
55
{
@@ -33,12 +33,48 @@ export type SendUSDApp = {
3333
type: "u64";
3434
}
3535
];
36+
},
37+
{
38+
name: "sendUsingTwap";
39+
accounts: [
40+
{
41+
name: "payer";
42+
isMut: true;
43+
isSigner: true;
44+
},
45+
{
46+
name: "destination";
47+
isMut: true;
48+
isSigner: false;
49+
docs: ["CHECK : Just a destination"];
50+
},
51+
{
52+
name: "twapUpdate";
53+
isMut: false;
54+
isSigner: false;
55+
},
56+
{
57+
name: "systemProgram";
58+
isMut: false;
59+
isSigner: false;
60+
}
61+
];
62+
args: [
63+
{
64+
name: "amountInUsd";
65+
type: "u64";
66+
},
67+
{
68+
"name": "twapWindowSeconds",
69+
"type": "u64"
70+
}
71+
];
3672
}
3773
];
3874
};
3975

4076
export const IDL: SendUSDApp = {
41-
version: "0.1.0",
77+
version: "0.2.0",
4278
name: "send_usd",
4379
instructions: [
4480
{
@@ -70,6 +106,42 @@ export const IDL: SendUSDApp = {
70106
{
71107
name: "amountInUsd",
72108
type: "u64",
109+
}
110+
],
111+
},
112+
{
113+
name: "sendUsingTwap",
114+
accounts: [
115+
{
116+
name: "payer",
117+
isMut: true,
118+
isSigner: true,
119+
},
120+
{
121+
name: "destination",
122+
isMut: true,
123+
isSigner: false,
124+
docs: ["CHECK : Just a destination"],
125+
},
126+
{
127+
name: "twapUpdate",
128+
isMut: false,
129+
isSigner: false,
130+
},
131+
{
132+
name: "systemProgram",
133+
isMut: false,
134+
isSigner: false,
135+
},
136+
],
137+
args: [
138+
{
139+
name: "amountInUsd",
140+
type: "u64",
141+
},
142+
{
143+
name: "twapWindowSeconds",
144+
type: "u64",
73145
},
74146
],
75147
},

0 commit comments

Comments
 (0)