@zcomb/programs-sdk 1.11.0 → 1.11.1

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/amm/utils.js CHANGED
@@ -1,181 +1 @@
1
- "use strict";
2
- /*
3
- * Utility functions for the AMM program.
4
- * PDA derivation, state parsing, price calculations, and account fetching.
5
- */
6
- Object.defineProperty(exports, "__esModule", { value: true });
7
- exports.derivePoolPDA = derivePoolPDA;
8
- exports.deriveReservePDA = deriveReservePDA;
9
- exports.deriveFeeVaultPDA = deriveFeeVaultPDA;
10
- exports.parsePoolState = parsePoolState;
11
- exports.fetchPoolAccount = fetchPoolAccount;
12
- exports.calculateSpotPrice = calculateSpotPrice;
13
- exports.computeSwapOutput = computeSwapOutput;
14
- exports.computeSwapInput = computeSwapInput;
15
- exports.calculatePriceImpact = calculatePriceImpact;
16
- exports.createSwapQuote = createSwapQuote;
17
- exports.calculateTwap = calculateTwap;
18
- exports.isOracleInWarmup = isOracleInWarmup;
19
- const anchor_1 = require("@coral-xyz/anchor");
20
- const web3_js_1 = require("@solana/web3.js");
21
- const constants_1 = require("./constants");
22
- const types_1 = require("./types");
23
- /* PDA Derivation */
24
- function derivePoolPDA(admin, mintA, mintB, programId = constants_1.PROGRAM_ID) {
25
- return web3_js_1.PublicKey.findProgramAddressSync([constants_1.POOL_SEED, admin.toBuffer(), mintA.toBuffer(), mintB.toBuffer()], programId);
26
- }
27
- function deriveReservePDA(pool, mint, programId = constants_1.PROGRAM_ID) {
28
- return web3_js_1.PublicKey.findProgramAddressSync([constants_1.RESERVE_SEED, pool.toBuffer(), mint.toBuffer()], programId);
29
- }
30
- function deriveFeeVaultPDA(pool, programId = constants_1.PROGRAM_ID) {
31
- return web3_js_1.PublicKey.findProgramAddressSync([constants_1.FEE_VAULT_SEED, pool.toBuffer()], programId);
32
- }
33
- /* Parsers */
34
- function parsePoolState(state) {
35
- if ("trading" in state)
36
- return types_1.PoolState.Trading;
37
- if ("finalized" in state)
38
- return types_1.PoolState.Finalized;
39
- throw new Error("Unknown pool state");
40
- }
41
- /* Fetch */
42
- async function fetchPoolAccount(program, poolPda) {
43
- return program.account.poolAccount.fetch(poolPda);
44
- }
45
- /* Math Utilities */
46
- const PRICE_SCALE_BN = new anchor_1.BN(constants_1.PRICE_SCALE.toString());
47
- function calculateSpotPrice(reserveA, reserveB, decimalsA, decimalsB) {
48
- if (reserveB.isZero())
49
- return new anchor_1.BN(0);
50
- const decimalDiff = decimalsB - decimalsA;
51
- if (decimalDiff >= 0) {
52
- const multiplier = new anchor_1.BN(10).pow(new anchor_1.BN(decimalDiff));
53
- return reserveA.mul(multiplier).mul(PRICE_SCALE_BN).div(reserveB);
54
- }
55
- else {
56
- const divisor = new anchor_1.BN(10).pow(new anchor_1.BN(-decimalDiff));
57
- return reserveA.mul(PRICE_SCALE_BN).div(reserveB).div(divisor);
58
- }
59
- }
60
- /**
61
- * Compute swap output using constant product formula.
62
- * Fee is ALWAYS collected in token A:
63
- * - A -> B: fee on input (before swap)
64
- * - B -> A: fee on output (after swap)
65
- */
66
- function computeSwapOutput(inputAmount, reserveIn, reserveOut, feeBps, swapAToB) {
67
- const input = typeof inputAmount === "number" ? new anchor_1.BN(inputAmount) : inputAmount;
68
- if (reserveIn.isZero() || reserveOut.isZero()) {
69
- return { outputAmount: new anchor_1.BN(0), feeAmount: new anchor_1.BN(0) };
70
- }
71
- if (swapAToB) {
72
- // A -> B: fee on input (token A)
73
- const feeAmount = input.mul(new anchor_1.BN(feeBps)).div(new anchor_1.BN(10000));
74
- const inputAfterFee = input.sub(feeAmount);
75
- // Constant product: output = reserveOut * inputAfterFee / (reserveIn + inputAfterFee)
76
- const numerator = reserveOut.mul(inputAfterFee);
77
- const denominator = reserveIn.add(inputAfterFee);
78
- const outputAmount = numerator.div(denominator);
79
- return { outputAmount, feeAmount };
80
- }
81
- else {
82
- // B -> A: fee on output (token A)
83
- // First compute gross output without fee
84
- const numerator = reserveOut.mul(input);
85
- const denominator = reserveIn.add(input);
86
- const grossOutput = numerator.div(denominator);
87
- // Then deduct fee from output
88
- const feeAmount = grossOutput.mul(new anchor_1.BN(feeBps)).div(new anchor_1.BN(10000));
89
- const outputAmount = grossOutput.sub(feeAmount);
90
- return { outputAmount, feeAmount };
91
- }
92
- }
93
- /**
94
- * Compute swap input needed to get a specific output.
95
- * Fee is ALWAYS collected in token A:
96
- * - A -> B: fee on input
97
- * - B -> A: fee on output
98
- */
99
- function computeSwapInput(outputAmount, reserveIn, reserveOut, feeBps, swapAToB) {
100
- const output = typeof outputAmount === "number" ? new anchor_1.BN(outputAmount) : outputAmount;
101
- if (reserveIn.isZero() || reserveOut.isZero() || output.gte(reserveOut)) {
102
- return { inputAmount: new anchor_1.BN(0), feeAmount: new anchor_1.BN(0) };
103
- }
104
- if (swapAToB) {
105
- // A -> B: fee on input
106
- // We need: output = (reserveOut * inputAfterFee) / (reserveIn + inputAfterFee)
107
- // Solve for inputAfterFee: inputAfterFee = (reserveIn * output) / (reserveOut - output)
108
- const numerator = reserveIn.mul(output);
109
- const denominator = reserveOut.sub(output);
110
- const inputAfterFee = numerator.div(denominator).add(new anchor_1.BN(1)); // Round up
111
- // inputAmount = inputAfterFee * 10000 / (10000 - feeBps)
112
- const inputAmount = inputAfterFee.mul(new anchor_1.BN(10000)).div(new anchor_1.BN(10000 - feeBps)).add(new anchor_1.BN(1));
113
- const feeAmount = inputAmount.sub(inputAfterFee);
114
- return { inputAmount, feeAmount };
115
- }
116
- else {
117
- // B -> A: fee on output
118
- // User wants `output` after fee, so grossOutput = output * 10000 / (10000 - feeBps)
119
- const grossOutput = output.mul(new anchor_1.BN(10000)).div(new anchor_1.BN(10000 - feeBps)).add(new anchor_1.BN(1));
120
- const feeAmount = grossOutput.sub(output);
121
- // Now compute input needed for grossOutput
122
- // grossOutput = (reserveOut * input) / (reserveIn + input)
123
- // input = (reserveIn * grossOutput) / (reserveOut - grossOutput)
124
- const numerator = reserveIn.mul(grossOutput);
125
- const denominator = reserveOut.sub(grossOutput);
126
- if (denominator.lte(new anchor_1.BN(0))) {
127
- return { inputAmount: new anchor_1.BN(0), feeAmount: new anchor_1.BN(0) };
128
- }
129
- const inputAmount = numerator.div(denominator).add(new anchor_1.BN(1)); // Round up
130
- return { inputAmount, feeAmount };
131
- }
132
- }
133
- function calculatePriceImpact(inputAmount, outputAmount, reserveIn, reserveOut, decimalsIn, decimalsOut) {
134
- if (reserveIn.isZero() || reserveOut.isZero() || inputAmount.isZero()) {
135
- return 0;
136
- }
137
- const spotPrice = calculateSpotPrice(reserveIn, reserveOut, decimalsIn, decimalsOut);
138
- const executionPrice = calculateSpotPrice(inputAmount, outputAmount, decimalsIn, decimalsOut);
139
- if (spotPrice.isZero())
140
- return 0;
141
- const impact = executionPrice.sub(spotPrice).mul(new anchor_1.BN(10000)).div(spotPrice);
142
- return Math.abs(impact.toNumber()) / 100;
143
- }
144
- function createSwapQuote(inputAmount, reserveIn, reserveOut, feeBps, decimalsIn, decimalsOut, swapAToB, slippagePercent = 0.5) {
145
- const input = typeof inputAmount === "number" ? new anchor_1.BN(inputAmount) : inputAmount;
146
- const { outputAmount, feeAmount } = computeSwapOutput(input, reserveIn, reserveOut, feeBps, swapAToB);
147
- const slippageBps = Math.floor(slippagePercent * 100);
148
- const minOutputAmount = outputAmount.mul(new anchor_1.BN(10000 - slippageBps)).div(new anchor_1.BN(10000));
149
- const spotPriceBefore = calculateSpotPrice(reserveIn, reserveOut, decimalsIn, decimalsOut);
150
- const newReserveIn = reserveIn.add(input);
151
- const newReserveOut = reserveOut.sub(outputAmount);
152
- const spotPriceAfter = calculateSpotPrice(newReserveIn, newReserveOut, decimalsIn, decimalsOut);
153
- const priceImpact = calculatePriceImpact(input, outputAmount, reserveIn, reserveOut, decimalsIn, decimalsOut);
154
- return {
155
- inputAmount: input,
156
- outputAmount,
157
- minOutputAmount,
158
- feeAmount,
159
- priceImpact,
160
- spotPriceBefore,
161
- spotPriceAfter,
162
- };
163
- }
164
- /* TWAP Utilities */
165
- function calculateTwap(oracle) {
166
- const warmupEnd = oracle.createdAtUnixTime.add(new anchor_1.BN(oracle.warmupDuration));
167
- if (oracle.lastUpdateUnixTime.lte(warmupEnd)) {
168
- return oracle.startingObservation;
169
- }
170
- const elapsed = oracle.lastUpdateUnixTime.sub(warmupEnd);
171
- if (elapsed.isZero() || oracle.cumulativeObservations.isZero()) {
172
- return null;
173
- }
174
- return oracle.cumulativeObservations.div(elapsed);
175
- }
176
- function isOracleInWarmup(oracle, currentTime) {
177
- const now = currentTime ?? new anchor_1.BN(Math.floor(Date.now() / 1000));
178
- const warmupEnd = oracle.createdAtUnixTime.add(new anchor_1.BN(oracle.warmupDuration));
179
- return now.lt(warmupEnd);
180
- }
181
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/amm/utils.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Utility functions for the AMM program.\n * PDA derivation, state parsing, price calculations, and account fetching.\n */\n\nimport { Program, BN } from \"@coral-xyz/anchor\";\nimport { PublicKey } from \"@solana/web3.js\";\nimport { POOL_SEED, RESERVE_SEED, FEE_VAULT_SEED, PROGRAM_ID, PRICE_SCALE } from \"./constants\";\nimport { Amm, PoolState, PoolAccount, TwapOracle, SwapQuote } from \"./types\";\n\n/* PDA Derivation */\n\nexport function derivePoolPDA(\n  admin: PublicKey,\n  mintA: PublicKey,\n  mintB: PublicKey,\n  programId: PublicKey = PROGRAM_ID\n): [PublicKey, number] {\n  return PublicKey.findProgramAddressSync(\n    [POOL_SEED, admin.toBuffer(), mintA.toBuffer(), mintB.toBuffer()],\n    programId\n  );\n}\n\nexport function deriveReservePDA(\n  pool: PublicKey,\n  mint: PublicKey,\n  programId: PublicKey = PROGRAM_ID\n): [PublicKey, number] {\n  return PublicKey.findProgramAddressSync(\n    [RESERVE_SEED, pool.toBuffer(), mint.toBuffer()],\n    programId\n  );\n}\n\nexport function deriveFeeVaultPDA(\n  pool: PublicKey,\n  programId: PublicKey = PROGRAM_ID\n): [PublicKey, number] {\n  return PublicKey.findProgramAddressSync(\n    [FEE_VAULT_SEED, pool.toBuffer()],\n    programId\n  );\n}\n\n/* Parsers */\n\nexport function parsePoolState(state: any): PoolState {\n  if (\"trading\" in state) return PoolState.Trading;\n  if (\"finalized\" in state) return PoolState.Finalized;\n  throw new Error(\"Unknown pool state\");\n}\n\n/* Fetch */\n\nexport async function fetchPoolAccount(\n  program: Program<Amm>,\n  poolPda: PublicKey\n): Promise<PoolAccount> {\n  return program.account.poolAccount.fetch(poolPda);\n}\n\n/* Math Utilities */\n\nconst PRICE_SCALE_BN = new BN(PRICE_SCALE.toString());\n\nexport function calculateSpotPrice(\n  reserveA: BN,\n  reserveB: BN,\n  decimalsA: number,\n  decimalsB: number,\n): BN {\n  if (reserveB.isZero()) return new BN(0);\n\n  const decimalDiff = decimalsB - decimalsA;\n\n  if (decimalDiff >= 0) {\n    const multiplier = new BN(10).pow(new BN(decimalDiff));\n    return reserveA.mul(multiplier).mul(PRICE_SCALE_BN).div(reserveB);\n  } else {\n    const divisor = new BN(10).pow(new BN(-decimalDiff));\n    return reserveA.mul(PRICE_SCALE_BN).div(reserveB).div(divisor);\n  }\n}\n\n/**\n * Compute swap output using constant product formula.\n * Fee is ALWAYS collected in token A:\n * - A -> B: fee on input (before swap)\n * - B -> A: fee on output (after swap)\n */\nexport function computeSwapOutput(\n  inputAmount: BN | number,\n  reserveIn: BN,\n  reserveOut: BN,\n  feeBps: number,\n  swapAToB: boolean\n): { outputAmount: BN; feeAmount: BN } {\n  const input = typeof inputAmount === \"number\" ? new BN(inputAmount) : inputAmount;\n\n  if (reserveIn.isZero() || reserveOut.isZero()) {\n    return { outputAmount: new BN(0), feeAmount: new BN(0) };\n  }\n\n  if (swapAToB) {\n    // A -> B: fee on input (token A)\n    const feeAmount = input.mul(new BN(feeBps)).div(new BN(10000));\n    const inputAfterFee = input.sub(feeAmount);\n\n    // Constant product: output = reserveOut * inputAfterFee / (reserveIn + inputAfterFee)\n    const numerator = reserveOut.mul(inputAfterFee);\n    const denominator = reserveIn.add(inputAfterFee);\n    const outputAmount = numerator.div(denominator);\n\n    return { outputAmount, feeAmount };\n  } else {\n    // B -> A: fee on output (token A)\n    // First compute gross output without fee\n    const numerator = reserveOut.mul(input);\n    const denominator = reserveIn.add(input);\n    const grossOutput = numerator.div(denominator);\n\n    // Then deduct fee from output\n    const feeAmount = grossOutput.mul(new BN(feeBps)).div(new BN(10000));\n    const outputAmount = grossOutput.sub(feeAmount);\n\n    return { outputAmount, feeAmount };\n  }\n}\n\n/**\n * Compute swap input needed to get a specific output.\n * Fee is ALWAYS collected in token A:\n * - A -> B: fee on input\n * - B -> A: fee on output\n */\nexport function computeSwapInput(\n  outputAmount: BN | number,\n  reserveIn: BN,\n  reserveOut: BN,\n  feeBps: number,\n  swapAToB: boolean\n): { inputAmount: BN; feeAmount: BN } {\n  const output = typeof outputAmount === \"number\" ? new BN(outputAmount) : outputAmount;\n\n  if (reserveIn.isZero() || reserveOut.isZero() || output.gte(reserveOut)) {\n    return { inputAmount: new BN(0), feeAmount: new BN(0) };\n  }\n\n  if (swapAToB) {\n    // A -> B: fee on input\n    // We need: output = (reserveOut * inputAfterFee) / (reserveIn + inputAfterFee)\n    // Solve for inputAfterFee: inputAfterFee = (reserveIn * output) / (reserveOut - output)\n    const numerator = reserveIn.mul(output);\n    const denominator = reserveOut.sub(output);\n    const inputAfterFee = numerator.div(denominator).add(new BN(1)); // Round up\n\n    // inputAmount = inputAfterFee * 10000 / (10000 - feeBps)\n    const inputAmount = inputAfterFee.mul(new BN(10000)).div(new BN(10000 - feeBps)).add(new BN(1));\n    const feeAmount = inputAmount.sub(inputAfterFee);\n\n    return { inputAmount, feeAmount };\n  } else {\n    // B -> A: fee on output\n    // User wants `output` after fee, so grossOutput = output * 10000 / (10000 - feeBps)\n    const grossOutput = output.mul(new BN(10000)).div(new BN(10000 - feeBps)).add(new BN(1));\n    const feeAmount = grossOutput.sub(output);\n\n    // Now compute input needed for grossOutput\n    // grossOutput = (reserveOut * input) / (reserveIn + input)\n    // input = (reserveIn * grossOutput) / (reserveOut - grossOutput)\n    const numerator = reserveIn.mul(grossOutput);\n    const denominator = reserveOut.sub(grossOutput);\n\n    if (denominator.lte(new BN(0))) {\n      return { inputAmount: new BN(0), feeAmount: new BN(0) };\n    }\n\n    const inputAmount = numerator.div(denominator).add(new BN(1)); // Round up\n\n    return { inputAmount, feeAmount };\n  }\n}\n\nexport function calculatePriceImpact(\n  inputAmount: BN,\n  outputAmount: BN,\n  reserveIn: BN,\n  reserveOut: BN,\n  decimalsIn: number,\n  decimalsOut: number,\n): number {\n  if (reserveIn.isZero() || reserveOut.isZero() || inputAmount.isZero()) {\n    return 0;\n  }\n\n  const spotPrice = calculateSpotPrice(reserveIn, reserveOut, decimalsIn, decimalsOut);\n  const executionPrice = calculateSpotPrice(inputAmount, outputAmount, decimalsIn, decimalsOut);\n\n  if (spotPrice.isZero()) return 0;\n\n  const impact = executionPrice.sub(spotPrice).mul(new BN(10000)).div(spotPrice);\n  return Math.abs(impact.toNumber()) / 100;\n}\n\nexport function createSwapQuote(\n  inputAmount: BN | number,\n  reserveIn: BN,\n  reserveOut: BN,\n  feeBps: number,\n  decimalsIn: number,\n  decimalsOut: number,\n  swapAToB: boolean,\n  slippagePercent: number = 0.5,\n): SwapQuote {\n  const input = typeof inputAmount === \"number\" ? new BN(inputAmount) : inputAmount;\n\n  const { outputAmount, feeAmount } = computeSwapOutput(input, reserveIn, reserveOut, feeBps, swapAToB);\n\n  const slippageBps = Math.floor(slippagePercent * 100);\n  const minOutputAmount = outputAmount.mul(new BN(10000 - slippageBps)).div(new BN(10000));\n\n  const spotPriceBefore = calculateSpotPrice(reserveIn, reserveOut, decimalsIn, decimalsOut);\n  const newReserveIn = reserveIn.add(input);\n  const newReserveOut = reserveOut.sub(outputAmount);\n  const spotPriceAfter = calculateSpotPrice(newReserveIn, newReserveOut, decimalsIn, decimalsOut);\n\n  const priceImpact = calculatePriceImpact(input, outputAmount, reserveIn, reserveOut, decimalsIn, decimalsOut);\n\n  return {\n    inputAmount: input,\n    outputAmount,\n    minOutputAmount,\n    feeAmount,\n    priceImpact,\n    spotPriceBefore,\n    spotPriceAfter,\n  };\n}\n\n/* TWAP Utilities */\n\nexport function calculateTwap(oracle: TwapOracle): BN | null {\n  const warmupEnd = oracle.createdAtUnixTime.add(new BN(oracle.warmupDuration));\n\n  if (oracle.lastUpdateUnixTime.lte(warmupEnd)) {\n    return oracle.startingObservation;\n  }\n\n  const elapsed = oracle.lastUpdateUnixTime.sub(warmupEnd);\n\n  if (elapsed.isZero() || oracle.cumulativeObservations.isZero()) {\n    return null;\n  }\n\n  return oracle.cumulativeObservations.div(elapsed);\n}\n\nexport function isOracleInWarmup(oracle: TwapOracle, currentTime?: BN): boolean {\n  const now = currentTime ?? new BN(Math.floor(Date.now() / 1000));\n  const warmupEnd = oracle.createdAtUnixTime.add(new BN(oracle.warmupDuration));\n  return now.lt(warmupEnd);\n}\n"]}
1
+ "use strict";Object.defineProperty(exports,"__esModule",{value:!0}),exports.derivePoolPDA=derivePoolPDA,exports.deriveReservePDA=deriveReservePDA,exports.deriveFeeVaultPDA=deriveFeeVaultPDA,exports.parsePoolState=parsePoolState,exports.fetchPoolAccount=fetchPoolAccount,exports.calculateSpotPrice=calculateSpotPrice,exports.computeSwapOutput=computeSwapOutput,exports.computeSwapInput=computeSwapInput,exports.calculatePriceImpact=calculatePriceImpact,exports.createSwapQuote=createSwapQuote,exports.calculateTwap=calculateTwap,exports.isOracleInWarmup=isOracleInWarmup;const anchor_1=require("@coral-xyz/anchor"),web3_js_1=require("@solana/web3.js"),constants_1=require("./constants"),types_1=require("./types");function derivePoolPDA(e,t,n,o=constants_1.PROGRAM_ID){return web3_js_1.PublicKey.findProgramAddressSync([constants_1.POOL_SEED,e.toBuffer(),t.toBuffer(),n.toBuffer()],o)}function deriveReservePDA(e,t,n=constants_1.PROGRAM_ID){return web3_js_1.PublicKey.findProgramAddressSync([constants_1.RESERVE_SEED,e.toBuffer(),t.toBuffer()],n)}function deriveFeeVaultPDA(e,t=constants_1.PROGRAM_ID){return web3_js_1.PublicKey.findProgramAddressSync([constants_1.FEE_VAULT_SEED,e.toBuffer()],t)}function parsePoolState(e){if("trading"in e)return types_1.PoolState.Trading;if("finalized"in e)return types_1.PoolState.Finalized;throw new Error("Unknown pool state")}async function fetchPoolAccount(e,t){return e.account.poolAccount.fetch(t)}const PRICE_SCALE_BN=new anchor_1.BN(constants_1.PRICE_SCALE.toString());function calculateSpotPrice(e,t,n,o){if(t.isZero())return new anchor_1.BN(0);const r=o-n;if(r>=0){const n=new anchor_1.BN(10).pow(new anchor_1.BN(r));return e.mul(n).mul(PRICE_SCALE_BN).div(t)}{const n=new anchor_1.BN(10).pow(new anchor_1.BN(-r));return e.mul(PRICE_SCALE_BN).div(t).div(n)}}function computeSwapOutput(e,t,n,o,r){const u="number"==typeof e?new anchor_1.BN(e):e;if(t.isZero()||n.isZero())return{outputAmount:new anchor_1.BN(0),feeAmount:new anchor_1.BN(0)};if(r){const e=u.mul(new anchor_1.BN(o)).div(new anchor_1.BN(1e4)),r=u.sub(e),c=n.mul(r),a=t.add(r);return{outputAmount:c.div(a),feeAmount:e}}{const e=n.mul(u),r=t.add(u),c=e.div(r),a=c.mul(new anchor_1.BN(o)).div(new anchor_1.BN(1e4));return{outputAmount:c.sub(a),feeAmount:a}}}function computeSwapInput(e,t,n,o,r){const u="number"==typeof e?new anchor_1.BN(e):e;if(t.isZero()||n.isZero()||u.gte(n))return{inputAmount:new anchor_1.BN(0),feeAmount:new anchor_1.BN(0)};if(r){const e=t.mul(u),r=n.sub(u),c=e.div(r).add(new anchor_1.BN(1)),a=c.mul(new anchor_1.BN(1e4)).div(new anchor_1.BN(1e4-o)).add(new anchor_1.BN(1)),i=a.sub(c);return{inputAmount:a,feeAmount:i}}{const e=u.mul(new anchor_1.BN(1e4)).div(new anchor_1.BN(1e4-o)).add(new anchor_1.BN(1)),r=e.sub(u),c=t.mul(e),a=n.sub(e);if(a.lte(new anchor_1.BN(0)))return{inputAmount:new anchor_1.BN(0),feeAmount:new anchor_1.BN(0)};return{inputAmount:c.div(a).add(new anchor_1.BN(1)),feeAmount:r}}}function calculatePriceImpact(e,t,n,o,r,u){if(n.isZero()||o.isZero()||e.isZero())return 0;const c=calculateSpotPrice(n,o,r,u),a=calculateSpotPrice(e,t,r,u);if(c.isZero())return 0;const i=a.sub(c).mul(new anchor_1.BN(1e4)).div(c);return Math.abs(i.toNumber())/100}function createSwapQuote(e,t,n,o,r,u,c,a=.5){const i="number"==typeof e?new anchor_1.BN(e):e,{outputAmount:s,feeAmount:p}=computeSwapOutput(i,t,n,o,c),l=Math.floor(100*a),_=s.mul(new anchor_1.BN(1e4-l)).div(new anchor_1.BN(1e4)),m=calculateSpotPrice(t,n,r,u),f=calculateSpotPrice(t.add(i),n.sub(s),r,u);return{inputAmount:i,outputAmount:s,minOutputAmount:_,feeAmount:p,priceImpact:calculatePriceImpact(i,s,t,n,r,u),spotPriceBefore:m,spotPriceAfter:f}}function calculateTwap(e){const t=e.createdAtUnixTime.add(new anchor_1.BN(e.warmupDuration));if(e.lastUpdateUnixTime.lte(t))return e.startingObservation;const n=e.lastUpdateUnixTime.sub(t);return n.isZero()||e.cumulativeObservations.isZero()?null:e.cumulativeObservations.div(n)}function isOracleInWarmup(e,t){const n=t??new anchor_1.BN(Math.floor(Date.now()/1e3)),o=e.createdAtUnixTime.add(new anchor_1.BN(e.warmupDuration));return n.lt(o)}