@meteora-ag/dlmm 1.6.0-rc.17 → 1.6.0-rc.19

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/index.js CHANGED
@@ -11779,1594 +11779,1861 @@ function getTokenProgramId(lbPairState) {
11779
11779
  };
11780
11780
  }
11781
11781
 
11782
- // src/dlmm/helpers/index.ts
11783
- function chunks(array, size) {
11784
- return Array.apply(0, new Array(Math.ceil(array.length / size))).map(
11785
- (_, index) => array.slice(index * size, (index + 1) * size)
11782
+ // src/dlmm/helpers/rebalance/rebalancePosition.ts
11783
+
11784
+
11785
+
11786
+ function buildBitFlagAndNegateStrategyParameters(x0, y0, deltaX, deltaY) {
11787
+ let bitFlag = 0;
11788
+ if (x0.isNeg()) {
11789
+ bitFlag |= 1;
11790
+ x0 = x0.neg();
11791
+ }
11792
+ if (y0.isNeg()) {
11793
+ bitFlag |= 2;
11794
+ y0 = y0.neg();
11795
+ }
11796
+ if (deltaX.isNeg()) {
11797
+ bitFlag |= 4;
11798
+ deltaX = deltaX.neg();
11799
+ }
11800
+ if (deltaY.isNeg()) {
11801
+ bitFlag |= 8;
11802
+ deltaY = deltaY.neg();
11803
+ }
11804
+ return {
11805
+ bitFlag,
11806
+ x0,
11807
+ y0,
11808
+ deltaX,
11809
+ deltaY
11810
+ };
11811
+ }
11812
+ function toRebalancePositionBinData(positionData) {
11813
+ return positionData.positionBinData.map(
11814
+ ({
11815
+ binId,
11816
+ price,
11817
+ pricePerToken,
11818
+ positionXAmount,
11819
+ positionYAmount,
11820
+ positionFeeXAmount,
11821
+ positionFeeYAmount,
11822
+ positionRewardAmount
11823
+ }) => {
11824
+ return {
11825
+ binId,
11826
+ price,
11827
+ pricePerToken,
11828
+ amountX: new (0, _bnjs2.default)(positionXAmount),
11829
+ amountY: new (0, _bnjs2.default)(positionYAmount),
11830
+ claimableRewardAmount: positionRewardAmount.map(
11831
+ (amount) => new (0, _bnjs2.default)(amount)
11832
+ ),
11833
+ claimableFeeXAmount: new (0, _bnjs2.default)(positionFeeXAmount),
11834
+ claimableFeeYAmount: new (0, _bnjs2.default)(positionFeeYAmount)
11835
+ };
11836
+ }
11786
11837
  );
11787
11838
  }
11788
- function range(min, max, mapfn) {
11789
- const length = max - min + 1;
11790
- return Array.from({ length }, (_, i) => mapfn(min + i));
11839
+ function getDepositBinIds(activeId, deposits) {
11840
+ const uniqueBinId = /* @__PURE__ */ new Set();
11841
+ for (const { minDeltaId, maxDeltaId } of deposits) {
11842
+ const minBinId = activeId.add(minDeltaId);
11843
+ const maxBinId = activeId.add(maxDeltaId);
11844
+ for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
11845
+ uniqueBinId.add(binId);
11846
+ }
11847
+ }
11848
+ const binIds = Array.from(uniqueBinId);
11849
+ binIds.sort((a, b) => a - b);
11850
+ return binIds;
11791
11851
  }
11792
- async function chunkedFetchMultiplePoolAccount(program, pks, chunkSize = 100) {
11793
- const accounts = (await Promise.all(
11794
- chunks(pks, chunkSize).map(
11795
- (chunk) => program.account.lbPair.fetchMultiple(chunk)
11796
- )
11797
- )).flat();
11798
- return accounts.filter(Boolean);
11852
+ function findMinMaxBinIdWithLiquidity(rebalancePositionBinData) {
11853
+ let minBinId = null;
11854
+ let maxBinId = null;
11855
+ for (const binData of rebalancePositionBinData) {
11856
+ if (binData.amountX.isZero() && binData.amountY.isZero() && binData.claimableFeeXAmount.isZero() && binData.claimableFeeYAmount.isZero() && binData.claimableRewardAmount.every((amount) => amount.isZero())) {
11857
+ continue;
11858
+ }
11859
+ if (minBinId == null || binData.binId < minBinId) {
11860
+ minBinId = binData.binId;
11861
+ }
11862
+ if (maxBinId == null || binData.binId > maxBinId) {
11863
+ maxBinId = binData.binId;
11864
+ }
11865
+ }
11866
+ return [minBinId, maxBinId];
11799
11867
  }
11800
- async function chunkedFetchMultipleBinArrayBitmapExtensionAccount(program, pks, chunkSize = 100) {
11801
- const accounts = (await Promise.all(
11802
- chunks(pks, chunkSize).map(
11803
- (chunk) => program.account.binArrayBitmapExtension.fetchMultiple(chunk)
11804
- )
11805
- )).flat();
11806
- return accounts;
11868
+ function onlyDepositToBidSide(maxDeltaId, favorXInActiveBin) {
11869
+ if (favorXInActiveBin) {
11870
+ return maxDeltaId.lt(new (0, _bnjs2.default)(0));
11871
+ }
11872
+ return maxDeltaId.lte(new (0, _bnjs2.default)(0));
11807
11873
  }
11808
- function getOutAmount(bin, inAmount, swapForY) {
11809
- return swapForY ? mulShr(inAmount, bin.price, SCALE_OFFSET, 1 /* Down */) : shlDiv(inAmount, bin.price, SCALE_OFFSET, 1 /* Down */);
11874
+ function onlyDepositToAskSide(minDeltaId, favorXInActiveBin) {
11875
+ if (favorXInActiveBin) {
11876
+ return minDeltaId.gte(new (0, _bnjs2.default)(0));
11877
+ }
11878
+ return minDeltaId.gt(new (0, _bnjs2.default)(0));
11810
11879
  }
11811
- async function getTokenDecimals(conn, mint) {
11812
- const token = await _spltoken.getMint.call(void 0, conn, mint);
11813
- return await token.decimals;
11880
+ function getAmountInBinsBidSide(activeId, minDeltaId, maxDeltaId, deltaY, y0) {
11881
+ const amountInBins = [];
11882
+ const minBinId = activeId.add(minDeltaId);
11883
+ const maxBinId = activeId.add(maxDeltaId);
11884
+ for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
11885
+ const deltaBin = activeId.toNumber() - binId;
11886
+ const totalDeltaY = deltaY.mul(new (0, _bnjs2.default)(deltaBin));
11887
+ const amountY = y0.add(totalDeltaY);
11888
+ amountInBins.push({
11889
+ binId: new (0, _bnjs2.default)(binId),
11890
+ amountX: new (0, _bnjs2.default)(0),
11891
+ amountY
11892
+ });
11893
+ }
11894
+ return amountInBins;
11814
11895
  }
11815
- var getOrCreateATAInstruction = async (connection, tokenMint, owner, programId, payer = owner, allowOwnerOffCurve = true) => {
11816
- programId = _nullishCoalesce(programId, () => ( _spltoken.TOKEN_PROGRAM_ID));
11817
- const toAccount = _spltoken.getAssociatedTokenAddressSync.call(void 0,
11818
- tokenMint,
11819
- owner,
11820
- allowOwnerOffCurve,
11821
- programId,
11822
- _spltoken.ASSOCIATED_TOKEN_PROGRAM_ID
11823
- );
11824
- try {
11825
- await _spltoken.getAccount.call(void 0, connection, toAccount, connection.commitment, programId);
11826
- return { ataPubKey: toAccount, ix: void 0 };
11827
- } catch (e) {
11828
- if (e instanceof _spltoken.TokenAccountNotFoundError || e instanceof _spltoken.TokenInvalidAccountOwnerError) {
11829
- const ix = _spltoken.createAssociatedTokenAccountIdempotentInstruction.call(void 0,
11830
- payer,
11831
- toAccount,
11832
- owner,
11833
- tokenMint,
11834
- programId,
11835
- _spltoken.ASSOCIATED_TOKEN_PROGRAM_ID
11836
- );
11837
- return { ataPubKey: toAccount, ix };
11838
- } else {
11839
- console.error("Error::getOrCreateATAInstruction", e);
11840
- throw e;
11841
- }
11896
+ function getAmountInBinsAskSide(activeId, binStep, minDeltaId, maxDeltaId, deltaX, x0) {
11897
+ const binCount = maxDeltaId.sub(minDeltaId).add(new (0, _bnjs2.default)(1));
11898
+ const minBinId = activeId.add(minDeltaId);
11899
+ const maxBinId = activeId.add(maxDeltaId);
11900
+ const amountInBins = new Array(binCount.toNumber());
11901
+ const base = getQPriceBaseFactor(binStep);
11902
+ let inverseBasePrice = pow(base, maxBinId.neg());
11903
+ for (let binId = maxBinId.toNumber(); binId >= minBinId.toNumber(); binId--) {
11904
+ const delta = binId - activeId.toNumber();
11905
+ const totalDeltaX = deltaX.mul(new (0, _bnjs2.default)(delta));
11906
+ const amountX = x0.add(totalDeltaX).mul(inverseBasePrice).shrn(SCALE_OFFSET);
11907
+ const idx = binId - minBinId.toNumber();
11908
+ amountInBins[idx] = {
11909
+ binId: new (0, _bnjs2.default)(binId),
11910
+ amountX,
11911
+ amountY: new (0, _bnjs2.default)(0)
11912
+ };
11913
+ inverseBasePrice = inverseBasePrice.mul(base).shrn(SCALE_OFFSET);
11842
11914
  }
11843
- };
11844
- async function getTokenBalance(conn, tokenAccount) {
11845
- const acc = await _spltoken.getAccount.call(void 0, conn, tokenAccount);
11846
- return acc.amount;
11915
+ return amountInBins;
11847
11916
  }
11848
- var parseLogs = (eventParser, logs) => {
11849
- if (!logs.length)
11850
- throw new Error("No logs found");
11851
- for (const event of _optionalChain([eventParser, 'optionalAccess', _41 => _41.parseLogs, 'call', _42 => _42(logs)])) {
11852
- return event.data;
11917
+ function toAmountIntoBins(activeId, minDeltaId, maxDeltaId, deltaX, deltaY, x0, y0, binStep, favorXInActiveBin) {
11918
+ if (onlyDepositToBidSide(maxDeltaId, favorXInActiveBin)) {
11919
+ return getAmountInBinsBidSide(activeId, minDeltaId, maxDeltaId, deltaY, y0);
11853
11920
  }
11854
- throw new Error("No events found");
11855
- };
11856
- var wrapSOLInstruction = (from, to, amount) => {
11857
- return [
11858
- _web3js.SystemProgram.transfer({
11859
- fromPubkey: from,
11860
- toPubkey: to,
11861
- lamports: amount
11862
- }),
11863
- new (0, _web3js.TransactionInstruction)({
11864
- keys: [
11865
- {
11866
- pubkey: to,
11867
- isSigner: false,
11868
- isWritable: true
11869
- }
11870
- ],
11871
- data: Buffer.from(new Uint8Array([17])),
11872
- programId: _spltoken.TOKEN_PROGRAM_ID
11873
- })
11874
- ];
11875
- };
11876
- var unwrapSOLInstruction = async (owner, allowOwnerOffCurve = true) => {
11877
- const wSolATAAccount = _spltoken.getAssociatedTokenAddressSync.call(void 0,
11878
- _spltoken.NATIVE_MINT,
11879
- owner,
11880
- allowOwnerOffCurve
11881
- );
11882
- if (wSolATAAccount) {
11883
- const closedWrappedSolInstruction = _spltoken.createCloseAccountInstruction.call(void 0,
11884
- wSolATAAccount,
11885
- owner,
11886
- owner,
11887
- [],
11888
- _spltoken.TOKEN_PROGRAM_ID
11921
+ if (onlyDepositToAskSide(minDeltaId, favorXInActiveBin)) {
11922
+ return getAmountInBinsAskSide(
11923
+ activeId,
11924
+ binStep,
11925
+ minDeltaId,
11926
+ maxDeltaId,
11927
+ deltaX,
11928
+ x0
11889
11929
  );
11890
- return closedWrappedSolInstruction;
11891
- }
11892
- return null;
11893
- };
11894
- async function chunkedGetMultipleAccountInfos(connection, pks, chunkSize = 100) {
11895
- const accountInfos = (await Promise.all(
11896
- chunks(pks, chunkSize).map(
11897
- (chunk) => connection.getMultipleAccountsInfo(chunk)
11898
- )
11899
- )).flat();
11900
- return accountInfos;
11901
- }
11902
- var getEstimatedComputeUnitUsageWithBuffer = async (connection, instructions, feePayer, buffer) => {
11903
- if (!buffer) {
11904
- buffer = 0.1;
11905
11930
  }
11906
- buffer = Math.max(0, buffer);
11907
- buffer = Math.min(1, buffer);
11908
- const estimatedComputeUnitUsage = await getSimulationComputeUnits(
11909
- connection,
11910
- instructions,
11911
- feePayer,
11912
- []
11913
- );
11914
- let extraComputeUnitBuffer = estimatedComputeUnitUsage * buffer;
11915
- if (extraComputeUnitBuffer > MAX_CU_BUFFER) {
11916
- extraComputeUnitBuffer = MAX_CU_BUFFER;
11917
- } else if (extraComputeUnitBuffer < MIN_CU_BUFFER) {
11918
- extraComputeUnitBuffer = MIN_CU_BUFFER;
11919
- }
11920
- return estimatedComputeUnitUsage + extraComputeUnitBuffer;
11921
- };
11922
- var getEstimatedComputeUnitIxWithBuffer = async (connection, instructions, feePayer, buffer) => {
11923
- const units = await getEstimatedComputeUnitUsageWithBuffer(
11924
- connection,
11925
- instructions,
11926
- feePayer,
11927
- buffer
11928
- ).catch((error) => {
11929
- console.error("Error::getEstimatedComputeUnitUsageWithBuffer", error);
11930
- return 14e5;
11931
- });
11932
- return _web3js.ComputeBudgetProgram.setComputeUnitLimit({ units });
11933
- };
11934
- function createProgram(connection, opt) {
11935
- const cluster = _optionalChain([opt, 'optionalAccess', _43 => _43.cluster]) || "mainnet-beta";
11936
- const provider = new (0, _anchor.AnchorProvider)(
11937
- connection,
11938
- {},
11939
- _anchor.AnchorProvider.defaultOptions()
11931
+ const [bidSideEndDeltaId, askSideStartDeltaId] = favorXInActiveBin ? [-1, 0] : [0, 1];
11932
+ const amountInBinsBidSide = getAmountInBinsBidSide(
11933
+ activeId,
11934
+ minDeltaId,
11935
+ new (0, _bnjs2.default)(bidSideEndDeltaId),
11936
+ deltaY,
11937
+ y0
11940
11938
  );
11941
- return new (0, _anchor.Program)(
11942
- { ...dlmm_default, address: LBCLMM_PROGRAM_IDS[cluster] },
11943
- provider
11939
+ const amountInBinsAskSide = getAmountInBinsAskSide(
11940
+ activeId,
11941
+ binStep,
11942
+ new (0, _bnjs2.default)(askSideStartDeltaId),
11943
+ maxDeltaId,
11944
+ deltaX,
11945
+ x0
11944
11946
  );
11947
+ return amountInBinsBidSide.concat(amountInBinsAskSide);
11945
11948
  }
11946
- function decodeAccount(program, accountName, buffer) {
11947
- return program.coder.accounts.decode(accountName, buffer);
11949
+ function getLiquidity(x, y, price) {
11950
+ const px = price.mul(x);
11951
+ const shly = y.shln(SCALE_OFFSET);
11952
+ return px.add(shly);
11948
11953
  }
11949
- function getAccountDiscriminator(accountName) {
11950
- return _optionalChain([dlmm_default, 'access', _44 => _44.accounts, 'access', _45 => _45.find, 'call', _46 => _46(
11951
- (acc) => acc.name.toLowerCase() === accountName.toLowerCase()
11952
- ), 'optionalAccess', _47 => _47.discriminator]);
11954
+ function computeCompositionFee(binStep, sParameters3, vParameters3, outAmountX, inAmountX, outAmountY, inAmountY) {
11955
+ if (outAmountX.gt(inAmountX)) {
11956
+ const delta = inAmountY.sub(outAmountY);
11957
+ const totalFeeRate = getTotalFee(
11958
+ binStep.toNumber(),
11959
+ sParameters3,
11960
+ vParameters3
11961
+ );
11962
+ const feeAmount = delta.mul(totalFeeRate);
11963
+ return feeAmount.mul(FEE_PRECISION.add(totalFeeRate)).div(FEE_PRECISION.pow(new (0, _bnjs2.default)(2)));
11964
+ }
11965
+ return new (0, _bnjs2.default)(0);
11953
11966
  }
11954
- function capSlippagePercentage(slippage) {
11955
- if (slippage > 100) {
11956
- slippage = 100;
11967
+ function simulateDepositBin(binId, binStep, amountX, amountY, bin) {
11968
+ if (!bin) {
11969
+ return {
11970
+ amountXIntoBin: amountX,
11971
+ amountYIntoBin: amountY
11972
+ };
11957
11973
  }
11958
- if (slippage < 0) {
11959
- slippage = 0;
11974
+ const price = getQPriceFromId(binId, binStep);
11975
+ const inLiquidity = getLiquidity(amountX, amountY, price);
11976
+ const binLiquidity = getLiquidity(bin.amountX, bin.amountY, price);
11977
+ if (bin.liquiditySupply.isZero()) {
11978
+ return {
11979
+ amountXIntoBin: amountX,
11980
+ amountYIntoBin: amountY
11981
+ };
11960
11982
  }
11961
- return slippage;
11962
- }
11963
-
11964
- // src/dlmm/helpers/accountFilters.ts
11965
- var _bytes = require('@coral-xyz/anchor/dist/cjs/utils/bytes');
11966
- var presetParameter2BinStepFilter = (binStep) => {
11967
- return {
11968
- memcmp: {
11969
- bytes: _bytes.bs58.encode(binStep.toArrayLike(Buffer, "le", 2)),
11970
- offset: 8
11971
- }
11972
- };
11973
- };
11974
- var presetParameter2BaseFactorFilter = (baseFactor) => {
11975
- return {
11976
- memcmp: {
11977
- bytes: _bytes.bs58.encode(baseFactor.toArrayLike(Buffer, "le", 2)),
11978
- offset: 8 + 2
11979
- }
11980
- };
11981
- };
11982
- var presetParameter2BaseFeePowerFactor = (baseFeePowerFactor) => {
11983
- return {
11984
- memcmp: {
11985
- bytes: _bytes.bs58.encode(baseFeePowerFactor.toArrayLike(Buffer, "le", 1)),
11986
- offset: 8 + 22
11987
- }
11988
- };
11989
- };
11990
- var binArrayLbPairFilter = (lbPair) => {
11991
- return {
11992
- memcmp: {
11993
- bytes: lbPair.toBase58(),
11994
- offset: 8 + 16
11995
- }
11996
- };
11997
- };
11998
- var positionOwnerFilter = (owner) => {
11999
- return {
12000
- memcmp: {
12001
- bytes: owner.toBase58(),
12002
- offset: 8 + 32
12003
- }
12004
- };
12005
- };
12006
- var positionLbPairFilter = (lbPair) => {
12007
- return {
12008
- memcmp: {
12009
- bytes: _bytes.bs58.encode(lbPair.toBuffer()),
12010
- offset: 8
12011
- }
12012
- };
12013
- };
12014
- var positionV2Filter = () => {
12015
- return {
12016
- memcmp: {
12017
- bytes: _bytes.bs58.encode(Buffer.from(getAccountDiscriminator("positionV2"))),
12018
- offset: 0
12019
- }
12020
- };
12021
- };
12022
-
12023
- // src/dlmm/helpers/positions/index.ts
12024
-
12025
-
12026
- // src/dlmm/helpers/positions/wrapper.ts
12027
-
12028
- function combineBaseAndExtendedPositionBinData(base, extended) {
12029
- const combinedLiquidityShares = base.liquidityShares;
12030
- const combinedRewardInfos = base.rewardInfos;
12031
- const combinedFeeInfos = base.feeInfos;
12032
- for (const binData of extended) {
12033
- combinedLiquidityShares.push(binData.liquidityShare);
12034
- combinedRewardInfos.push(binData.rewardInfo);
12035
- combinedFeeInfos.push(binData.feeInfo);
11983
+ const liquidityShare = inLiquidity.mul(bin.liquiditySupply).div(binLiquidity);
11984
+ const updatedBinXAmount = bin.amountX.add(amountX);
11985
+ const updatedBinYAmount = bin.amountY.add(amountY);
11986
+ const updatedBinSupply = bin.liquiditySupply.add(liquidityShare);
11987
+ let amountXIntoBin = liquidityShare.mul(
11988
+ updatedBinXAmount.div(updatedBinSupply)
11989
+ );
11990
+ let amountYIntoBin = liquidityShare.mul(
11991
+ updatedBinYAmount.div(updatedBinSupply)
11992
+ );
11993
+ if (amountXIntoBin.gt(amountX)) {
12036
11994
  }
12037
11995
  return {
12038
- liquidityShares: combinedLiquidityShares,
12039
- rewardInfos: combinedRewardInfos,
12040
- feeInfos: combinedFeeInfos
11996
+ amountXIntoBin,
11997
+ amountYIntoBin
12041
11998
  };
12042
11999
  }
12043
- function wrapPosition(program, key, account) {
12044
- const disc = account.data.subarray(0, 8);
12045
- if (disc.equals(Buffer.from(getAccountDiscriminator("positionV2")))) {
12046
- const state = decodeAccount(
12000
+ var RebalancePosition = class {
12001
+
12002
+
12003
+
12004
+
12005
+
12006
+
12007
+
12008
+
12009
+
12010
+
12011
+ constructor(positionAddress, positionData, lbPair, activeBin, shouldClaimFee, shouldClaimReward, currentTimestamp) {
12012
+ this.address = positionAddress;
12013
+ this.rebalancePositionBinData = toRebalancePositionBinData(positionData);
12014
+ this.lowerBinId = new (0, _bnjs2.default)(positionData.lowerBinId);
12015
+ this.upperBinId = new (0, _bnjs2.default)(positionData.upperBinId);
12016
+ this.lbPair = lbPair;
12017
+ this.shouldClaimFee = shouldClaimFee;
12018
+ this.shouldClaimReward = shouldClaimReward;
12019
+ this.owner = positionData.owner;
12020
+ this.activeBin = activeBin;
12021
+ this.currentTimestamp = currentTimestamp;
12022
+ }
12023
+ static async create(params) {
12024
+ const {
12047
12025
  program,
12048
- "positionV2",
12049
- account.data
12026
+ positionAddress,
12027
+ pairAddress,
12028
+ positionData,
12029
+ shouldClaimFee,
12030
+ shouldClaimReward
12031
+ } = params;
12032
+ const [lbPairAccount, clockAccount] = await program.provider.connection.getMultipleAccountsInfo([
12033
+ pairAddress,
12034
+ _web3js.SYSVAR_CLOCK_PUBKEY
12035
+ ]);
12036
+ const lbPair = decodeAccount(program, "lbPair", lbPairAccount.data);
12037
+ const clock = ClockLayout.decode(clockAccount.data);
12038
+ const activeBinArrayIdx = binIdToBinArrayIndex(new (0, _bnjs2.default)(lbPair.activeId));
12039
+ const [activeBinArrayPubkey] = deriveBinArray(
12040
+ pairAddress,
12041
+ activeBinArrayIdx,
12042
+ program.programId
12050
12043
  );
12051
- const extended = decodeExtendedPosition(
12052
- state,
12053
- program,
12054
- account.data.subarray(8 + POSITION_MIN_SIZE)
12044
+ const activeBinArrayState = await program.account.binArray.fetch(
12045
+ activeBinArrayPubkey
12055
12046
  );
12056
- const combinedPositionBinData = combineBaseAndExtendedPositionBinData(
12057
- state,
12058
- extended
12047
+ const [lowerBinId, upperBinId] = getBinArrayLowerUpperBinId(activeBinArrayIdx);
12048
+ const idx = getBinIdIndexInBinArray(
12049
+ new (0, _bnjs2.default)(lbPair.activeId),
12050
+ lowerBinId,
12051
+ upperBinId
12052
+ );
12053
+ const activeBin = activeBinArrayState[idx.toNumber()];
12054
+ return new RebalancePosition(
12055
+ positionAddress,
12056
+ positionData,
12057
+ lbPair,
12058
+ activeBin,
12059
+ shouldClaimFee,
12060
+ shouldClaimReward,
12061
+ clock.unixTimestamp
12059
12062
  );
12060
- return new PositionV2Wrapper(key, state, extended, combinedPositionBinData);
12061
- } else {
12062
- throw new Error("Unknown position account");
12063
12063
  }
12064
- }
12065
- var PositionV2Wrapper = class {
12066
- constructor(positionAddress, inner, extended, combinedPositionBinData) {
12067
- this.positionAddress = positionAddress;
12068
- this.inner = inner;
12069
- this.extended = extended;
12070
- this.combinedPositionBinData = combinedPositionBinData;
12071
- }
12072
- address() {
12073
- return this.positionAddress;
12074
- }
12075
- totalClaimedRewards() {
12076
- return this.inner.totalClaimedRewards;
12077
- }
12078
- feeOwner() {
12079
- return this.inner.feeOwner;
12080
- }
12081
- lockReleasePoint() {
12082
- return this.inner.lockReleasePoint;
12083
- }
12084
- operator() {
12085
- return this.inner.operator;
12086
- }
12087
- totalClaimedFeeYAmount() {
12088
- return this.inner.totalClaimedFeeYAmount;
12089
- }
12090
- totalClaimedFeeXAmount() {
12091
- return this.inner.totalClaimedFeeXAmount;
12092
- }
12093
- lbPair() {
12094
- return this.inner.lbPair;
12095
- }
12096
- lowerBinId() {
12097
- return new (0, _bnjs2.default)(this.inner.lowerBinId);
12098
- }
12099
- upperBinId() {
12100
- return new (0, _bnjs2.default)(this.inner.upperBinId);
12101
- }
12102
- liquidityShares() {
12103
- return this.combinedPositionBinData.liquidityShares;
12104
- }
12105
- rewardInfos() {
12106
- return this.combinedPositionBinData.rewardInfos;
12107
- }
12108
- feeInfos() {
12109
- return this.combinedPositionBinData.feeInfos;
12110
- }
12111
- lastUpdatedAt() {
12112
- return this.inner.lastUpdatedAt;
12113
- }
12114
- getBinArrayIndexesCoverage() {
12115
- const isExtended = this.extended.length > 0;
12116
- if (isExtended) {
12117
- return getBinArrayIndexesCoverage(this.lowerBinId(), this.upperBinId());
12064
+ _simulateDeposit(binStep, tokenXDecimal, tokenYDecimal, deposits, simulatedWithdrawResult) {
12065
+ const { liquidityAndFeeXWithdrawn, liquidityAndFeeYWithdrawn } = simulatedWithdrawResult;
12066
+ const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12067
+ const depositBinIds = getDepositBinIds(activeId, deposits);
12068
+ if (depositBinIds.length > 0) {
12069
+ const depositMinBinId = depositBinIds[0];
12070
+ const depositMaxBinId = depositBinIds[depositBinIds.length - 1];
12071
+ this._simulateResize(
12072
+ new (0, _bnjs2.default)(depositMinBinId),
12073
+ new (0, _bnjs2.default)(depositMaxBinId),
12074
+ binStep,
12075
+ tokenXDecimal,
12076
+ tokenYDecimal
12077
+ );
12078
+ }
12079
+ let totalAmountXDeposited = new (0, _bnjs2.default)(0);
12080
+ let totalAmountYDeposited = new (0, _bnjs2.default)(0);
12081
+ const addLiquidityParam = [];
12082
+ for (const {
12083
+ x0,
12084
+ y0,
12085
+ favorXInActiveBin,
12086
+ deltaX,
12087
+ deltaY,
12088
+ minDeltaId,
12089
+ maxDeltaId
12090
+ } of deposits) {
12091
+ const params = buildBitFlagAndNegateStrategyParameters(
12092
+ x0,
12093
+ y0,
12094
+ deltaX,
12095
+ deltaY
12096
+ );
12097
+ addLiquidityParam.push({
12098
+ minDeltaId: minDeltaId.toNumber(),
12099
+ maxDeltaId: maxDeltaId.toNumber(),
12100
+ x0: params.x0,
12101
+ y0: params.y0,
12102
+ deltaX: params.deltaX,
12103
+ deltaY: params.deltaY,
12104
+ bitFlag: params.bitFlag,
12105
+ padding: Array(16).fill(0),
12106
+ favorXInActiveId: favorXInActiveBin
12107
+ });
12108
+ const amountIntoBins = toAmountIntoBins(
12109
+ activeId,
12110
+ minDeltaId,
12111
+ maxDeltaId,
12112
+ deltaX,
12113
+ deltaY,
12114
+ x0,
12115
+ y0,
12116
+ binStep,
12117
+ favorXInActiveBin
12118
+ );
12119
+ for (const { binId, amountX, amountY } of amountIntoBins) {
12120
+ totalAmountXDeposited = totalAmountXDeposited.add(amountX);
12121
+ totalAmountYDeposited = totalAmountYDeposited.add(amountY);
12122
+ const idx = this.rebalancePositionBinData.findIndex(
12123
+ (data) => data.binId == binId.toNumber()
12124
+ );
12125
+ if (binId.eq(activeId)) {
12126
+ const vParameters3 = Object.assign({}, this.lbPair.vParameters);
12127
+ const sParameters3 = Object.assign({}, this.lbPair.parameters);
12128
+ DLMM.updateReference(
12129
+ activeId.toNumber(),
12130
+ vParameters3,
12131
+ sParameters3,
12132
+ this.currentTimestamp.toNumber()
12133
+ );
12134
+ DLMM.updateVolatilityAccumulator(
12135
+ vParameters3,
12136
+ sParameters3,
12137
+ activeId.toNumber()
12138
+ );
12139
+ const { amountXIntoBin, amountYIntoBin } = simulateDepositBin(
12140
+ binId,
12141
+ binStep,
12142
+ amountX,
12143
+ amountY,
12144
+ this.activeBin
12145
+ );
12146
+ const feeY = computeCompositionFee(
12147
+ binStep,
12148
+ sParameters3,
12149
+ vParameters3,
12150
+ amountXIntoBin,
12151
+ amountX,
12152
+ amountYIntoBin,
12153
+ amountY
12154
+ );
12155
+ const feeX = computeCompositionFee(
12156
+ binStep,
12157
+ sParameters3,
12158
+ vParameters3,
12159
+ amountYIntoBin,
12160
+ amountY,
12161
+ amountXIntoBin,
12162
+ amountX
12163
+ );
12164
+ const amountXIntoBinExcludeFee = amountXIntoBin.sub(feeX);
12165
+ const amountYIntoBinExcludeFee = amountYIntoBin.sub(feeY);
12166
+ this.rebalancePositionBinData[idx].amountX = this.rebalancePositionBinData[idx].amountX.add(
12167
+ amountXIntoBinExcludeFee
12168
+ );
12169
+ this.rebalancePositionBinData[idx].amountY = this.rebalancePositionBinData[idx].amountY.add(
12170
+ amountYIntoBinExcludeFee
12171
+ );
12172
+ } else {
12173
+ this.rebalancePositionBinData[idx].amountX = this.rebalancePositionBinData[idx].amountX.add(amountX);
12174
+ this.rebalancePositionBinData[idx].amountY = this.rebalancePositionBinData[idx].amountY.add(amountY);
12175
+ }
12176
+ }
12177
+ }
12178
+ let actualTotalAmountXDeposited = totalAmountXDeposited;
12179
+ let actualTotalAmountYDeposited = totalAmountYDeposited;
12180
+ let actualLiquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn;
12181
+ let actualLiquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn;
12182
+ if (actualTotalAmountXDeposited.gt(actualLiquidityAndFeeXWithdrawn)) {
12183
+ actualTotalAmountXDeposited = actualTotalAmountXDeposited.sub(
12184
+ actualLiquidityAndFeeXWithdrawn
12185
+ );
12186
+ actualLiquidityAndFeeXWithdrawn = new (0, _bnjs2.default)(0);
12118
12187
  } else {
12119
- const lowerBinArrayIndex = binIdToBinArrayIndex(this.lowerBinId());
12120
- const upperBinArrayIndex = lowerBinArrayIndex.add(new (0, _bnjs2.default)(1));
12121
- return [lowerBinArrayIndex, upperBinArrayIndex];
12188
+ actualLiquidityAndFeeXWithdrawn = actualLiquidityAndFeeXWithdrawn.sub(
12189
+ actualTotalAmountXDeposited
12190
+ );
12191
+ actualTotalAmountXDeposited = new (0, _bnjs2.default)(0);
12122
12192
  }
12193
+ if (actualTotalAmountYDeposited.gt(actualLiquidityAndFeeYWithdrawn)) {
12194
+ actualTotalAmountYDeposited = actualTotalAmountYDeposited.sub(
12195
+ actualLiquidityAndFeeYWithdrawn
12196
+ );
12197
+ actualLiquidityAndFeeYWithdrawn = new (0, _bnjs2.default)(0);
12198
+ } else {
12199
+ actualLiquidityAndFeeYWithdrawn = actualLiquidityAndFeeYWithdrawn.sub(
12200
+ actualTotalAmountYDeposited
12201
+ );
12202
+ actualTotalAmountYDeposited = new (0, _bnjs2.default)(0);
12203
+ }
12204
+ return {
12205
+ result: {
12206
+ totalAmountXDeposited,
12207
+ totalAmountYDeposited,
12208
+ actualLiquidityAndFeeXWithdrawn,
12209
+ actualLiquidityAndFeeYWithdrawn,
12210
+ actualTotalAmountXDeposited,
12211
+ actualTotalAmountYDeposited
12212
+ },
12213
+ depositParams: addLiquidityParam
12214
+ };
12123
12215
  }
12124
- getBinArrayKeysCoverage(programId) {
12125
- return this.getBinArrayIndexesCoverage().map(
12126
- (index) => deriveBinArray(this.lbPair(), index, programId)[0]
12216
+ _simulateResize(depositMinBinId, depositMaxBinId, binStep, tokenXDecimal, tokenYDecimal) {
12217
+ const tokenXMultiplier = new (0, _decimaljs2.default)(10 ** tokenXDecimal.toNumber());
12218
+ const tokenYMultiplier = new (0, _decimaljs2.default)(10 ** tokenYDecimal.toNumber());
12219
+ const [minBinId, maxBinId] = findMinMaxBinIdWithLiquidity(
12220
+ this.rebalancePositionBinData
12127
12221
  );
12128
- }
12129
- version() {
12130
- return 1 /* V2 */;
12131
- }
12132
- owner() {
12133
- return this.inner.owner;
12134
- }
12135
- width() {
12136
- return this.upperBinId().sub(this.lowerBinId()).add(new (0, _bnjs2.default)(1));
12137
- }
12138
- };
12139
-
12140
- // src/dlmm/helpers/positions/index.ts
12141
- function getBinArrayIndexesCoverage(lowerBinId, upperBinId) {
12142
- const lowerBinArrayIndex = binIdToBinArrayIndex(lowerBinId);
12143
- const upperBinArrayIndex = binIdToBinArrayIndex(upperBinId);
12144
- const binArrayIndexes = [];
12145
- for (let i = lowerBinArrayIndex.toNumber(); i <= upperBinArrayIndex.toNumber(); i++) {
12146
- binArrayIndexes.push(new (0, _bnjs2.default)(i));
12147
- }
12148
- return binArrayIndexes;
12149
- }
12150
- function getBinArrayKeysCoverage2(lowerBinId, upperBinId, lbPair, programId) {
12151
- const binArrayIndexes = getBinArrayIndexesCoverage(lowerBinId, upperBinId);
12152
- return binArrayIndexes.map((index) => {
12153
- return deriveBinArray(lbPair, index, programId)[0];
12154
- });
12155
- }
12156
- function getBinArrayAccountMetasCoverage(lowerBinId, upperBinId, lbPair, programId) {
12157
- return getBinArrayKeysCoverage2(lowerBinId, upperBinId, lbPair, programId).map(
12158
- (key) => {
12159
- return {
12160
- pubkey: key,
12161
- isSigner: false,
12162
- isWritable: true
12163
- };
12164
- }
12165
- );
12166
- }
12167
- function getPositionLowerUpperBinIdWithLiquidity(position) {
12168
- const binWithLiquidity = position.positionBinData.filter(
12169
- (b) => !new (0, _bnjs2.default)(b.binLiquidity).isZero() || !new (0, _bnjs2.default)(b.positionFeeXAmount.toString()).isZero() || !new (0, _bnjs2.default)(b.positionFeeYAmount.toString()).isZero() || !new (0, _bnjs2.default)(b.positionRewardAmount[0].toString()).isZero() || !new (0, _bnjs2.default)(b.positionRewardAmount[1].toString()).isZero()
12170
- );
12171
- return binWithLiquidity.length > 0 ? {
12172
- lowerBinId: new (0, _bnjs2.default)(binWithLiquidity[0].binId),
12173
- upperBinId: new (0, _bnjs2.default)(binWithLiquidity[binWithLiquidity.length - 1].binId)
12174
- } : null;
12175
- }
12176
- function isPositionNoFee(position) {
12177
- return position.feeX.isZero() && position.feeY.isZero();
12178
- }
12179
- function isPositionNoReward(position) {
12180
- return position.rewardOne.isZero() && position.rewardTwo.isZero();
12181
- }
12182
- function chunkBinRange(minBinId, maxBinId) {
12183
- const chunkedBinRange = [];
12184
- let startBinId = minBinId;
12185
- while (startBinId <= maxBinId) {
12186
- const endBinId = Math.min(
12187
- startBinId + DEFAULT_BIN_PER_POSITION.toNumber() - 1,
12188
- maxBinId
12222
+ const newMinBinId = new (0, _bnjs2.default)(
12223
+ Math.min(depositMinBinId.toNumber(), _nullishCoalesce(minBinId, () => ( Number.MAX_SAFE_INTEGER)))
12189
12224
  );
12190
- chunkedBinRange.push({
12191
- lowerBinId: startBinId,
12192
- upperBinId: endBinId
12193
- });
12194
- startBinId += DEFAULT_BIN_PER_POSITION.toNumber();
12195
- }
12196
- return chunkedBinRange;
12197
- }
12198
- async function getPositionExpandRentExemption(currentMinBinId, currentMaxBinId, connection, binCountToExpand) {
12199
- const currentPositionWidth = currentMaxBinId.sub(currentMinBinId).addn(1);
12200
- const positionWidthAfterExpand = currentPositionWidth.add(binCountToExpand);
12201
- if (positionWidthAfterExpand.lte(DEFAULT_BIN_PER_POSITION)) {
12202
- return 0;
12203
- } else {
12204
- const binCountInExpandedBytes = positionWidthAfterExpand.sub(
12205
- DEFAULT_BIN_PER_POSITION
12225
+ const newMaxBinId = new (0, _bnjs2.default)(
12226
+ Math.max(depositMaxBinId.toNumber(), _nullishCoalesce(maxBinId, () => ( Number.MIN_SAFE_INTEGER)))
12206
12227
  );
12207
- const expandSize = binCountInExpandedBytes.toNumber() * POSITION_BIN_DATA_SIZE;
12208
- const [minimumLamports, rentExemptionLamports] = await Promise.all([
12209
- connection.getMinimumBalanceForRentExemption(0),
12210
- connection.getMinimumBalanceForRentExemption(expandSize)
12211
- ]);
12212
- return rentExemptionLamports - minimumLamports;
12228
+ if (newMinBinId.lt(this.lowerBinId)) {
12229
+ const binCountToExpand = this.lowerBinId.sub(depositMinBinId);
12230
+ for (let i = 1; i <= binCountToExpand.toNumber(); i++) {
12231
+ const binId = this.lowerBinId.subn(i);
12232
+ const price = getPriceOfBinByBinId(
12233
+ binId.toNumber(),
12234
+ binStep.toNumber()
12235
+ );
12236
+ const adjustedPrice = price.mul(tokenXMultiplier).div(tokenYMultiplier);
12237
+ this.rebalancePositionBinData.unshift({
12238
+ binId: binId.toNumber(),
12239
+ price: adjustedPrice.toString(),
12240
+ pricePerToken: adjustedPrice.toString(),
12241
+ amountX: new (0, _bnjs2.default)(0),
12242
+ amountY: new (0, _bnjs2.default)(0),
12243
+ claimableRewardAmount: [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)],
12244
+ claimableFeeXAmount: new (0, _bnjs2.default)(0),
12245
+ claimableFeeYAmount: new (0, _bnjs2.default)(0)
12246
+ });
12247
+ }
12248
+ } else {
12249
+ const binCountToShrink = newMinBinId.sub(this.lowerBinId);
12250
+ for (let i = 1; i <= binCountToShrink.toNumber(); i++) {
12251
+ this.rebalancePositionBinData.shift();
12252
+ }
12253
+ }
12254
+ if (newMaxBinId.gt(this.upperBinId)) {
12255
+ const binCountToExpand = newMaxBinId.sub(this.upperBinId);
12256
+ for (let i = 1; i <= binCountToExpand.toNumber(); i++) {
12257
+ const binId = this.upperBinId.addn(i);
12258
+ const price = getPriceOfBinByBinId(
12259
+ binId.toNumber(),
12260
+ binStep.toNumber()
12261
+ );
12262
+ const adjustedPrice = price.mul(tokenXMultiplier).div(tokenYMultiplier);
12263
+ this.rebalancePositionBinData.push({
12264
+ binId: binId.toNumber(),
12265
+ price: adjustedPrice.toString(),
12266
+ pricePerToken: adjustedPrice.toString(),
12267
+ amountX: new (0, _bnjs2.default)(0),
12268
+ amountY: new (0, _bnjs2.default)(0),
12269
+ claimableRewardAmount: [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)],
12270
+ claimableFeeXAmount: new (0, _bnjs2.default)(0),
12271
+ claimableFeeYAmount: new (0, _bnjs2.default)(0)
12272
+ });
12273
+ }
12274
+ } else {
12275
+ const binCountToShrink = this.upperBinId.sub(newMaxBinId);
12276
+ for (let i = 1; i <= binCountToShrink.toNumber(); i++) {
12277
+ this.rebalancePositionBinData.pop();
12278
+ }
12279
+ }
12280
+ this.lowerBinId = newMinBinId;
12281
+ this.upperBinId = newMaxBinId;
12213
12282
  }
12214
- }
12215
- function getExtendedPositionBinCount(minBinId, maxBinId) {
12216
- const width = maxBinId.sub(minBinId).addn(1);
12217
- const extended = width.sub(DEFAULT_BIN_PER_POSITION);
12218
- return extended.lte(new (0, _bnjs2.default)(0)) ? new (0, _bnjs2.default)(0) : extended;
12219
- }
12220
- function decodeExtendedPosition(base, program, bytes) {
12221
- const width = base.upperBinId - base.lowerBinId + 1;
12222
- const extendedWidth = width - DEFAULT_BIN_PER_POSITION.toNumber();
12223
- const extendedPosition = [];
12224
- for (let i = 0; i < extendedWidth; i++) {
12225
- const offset = i * POSITION_BIN_DATA_SIZE;
12226
- const data = bytes.subarray(offset, offset + POSITION_BIN_DATA_SIZE);
12227
- const decodedPositionBinData = program.coder.types.decode(
12228
- // TODO: Find a type safe way
12229
- "positionBinData",
12230
- data
12283
+ _simulateWithdraw(withdraws) {
12284
+ let liquidityAndFeeXWithdrawn = new (0, _bnjs2.default)(0);
12285
+ let liquidityAndFeeYWithdrawn = new (0, _bnjs2.default)(0);
12286
+ let rewardsAmountClaimed = [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)];
12287
+ const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12288
+ for (const { minBinId, maxBinId, bps } of withdraws) {
12289
+ const fromBinId = _nullishCoalesce(minBinId, () => ( activeId));
12290
+ const toBinId = _nullishCoalesce(maxBinId, () => ( activeId));
12291
+ const binIds = binRangeToBinIdArray(fromBinId, toBinId).filter(
12292
+ (binId) => binId.gte(this.lowerBinId) && binId.lte(this.upperBinId)
12293
+ );
12294
+ for (const binId of binIds) {
12295
+ const idx = this.rebalancePositionBinData.findIndex(
12296
+ (b) => b.binId === binId.toNumber()
12297
+ );
12298
+ const binData = this.rebalancePositionBinData[idx];
12299
+ const amountXWithdrawn = binData.amountX.mul(bps).divn(BASIS_POINT_MAX);
12300
+ const amountYWithdrawn = binData.amountY.mul(bps).divn(BASIS_POINT_MAX);
12301
+ liquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn.add(amountXWithdrawn);
12302
+ liquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn.add(amountYWithdrawn);
12303
+ binData.amountX = binData.amountX.sub(amountXWithdrawn);
12304
+ binData.amountY = binData.amountY.sub(amountYWithdrawn);
12305
+ if (this.shouldClaimFee) {
12306
+ liquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn.add(
12307
+ binData.claimableFeeXAmount
12308
+ );
12309
+ liquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn.add(
12310
+ binData.claimableFeeYAmount
12311
+ );
12312
+ binData.claimableFeeXAmount = new (0, _bnjs2.default)(0);
12313
+ binData.claimableFeeYAmount = new (0, _bnjs2.default)(0);
12314
+ }
12315
+ if (this.shouldClaimReward) {
12316
+ for (const [idx2, amount] of binData.claimableRewardAmount.entries()) {
12317
+ rewardsAmountClaimed[idx2] = rewardsAmountClaimed[idx2].add(amount);
12318
+ binData.claimableRewardAmount[idx2] = new (0, _bnjs2.default)(0);
12319
+ }
12320
+ }
12321
+ this.rebalancePositionBinData[idx] = binData;
12322
+ }
12323
+ }
12324
+ const withdrawParams = withdraws.map(
12325
+ ({ minBinId, maxBinId, bps }) => {
12326
+ return {
12327
+ minBinId: minBinId ? minBinId.toNumber() : null,
12328
+ maxBinId: maxBinId ? maxBinId.toNumber() : null,
12329
+ bps: bps.toNumber(),
12330
+ padding: Array(16).fill(0)
12331
+ };
12332
+ }
12231
12333
  );
12232
- extendedPosition.push(decodedPositionBinData);
12334
+ return {
12335
+ result: {
12336
+ liquidityAndFeeXWithdrawn,
12337
+ liquidityAndFeeYWithdrawn,
12338
+ rewardsAmountClaimed
12339
+ },
12340
+ withdrawParams
12341
+ };
12233
12342
  }
12234
- return extendedPosition;
12235
- }
12236
-
12237
- // src/dlmm/helpers/rebalance/rebalancePosition.ts
12238
-
12239
-
12240
-
12241
- function buildBitFlagAndNegateStrategyParameters(x0, y0, deltaX, deltaY) {
12242
- let bitFlag = 0;
12243
- if (x0.isNeg()) {
12244
- bitFlag |= 1;
12245
- x0 = x0.neg();
12343
+ async simulateRebalance(connection, binStep, tokenXDecimal, tokenYDecimal, withdraws, deposits) {
12344
+ if (withdraws.length == 0 && deposits.length == 0) {
12345
+ throw "No rebalance action";
12346
+ }
12347
+ const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12348
+ withdraws = validateAndSortRebalanceWithdraw(withdraws, activeId);
12349
+ deposits = validateAndSortRebalanceDeposit(deposits);
12350
+ const beforeWidth = getPositionWidthWithMinWidth(
12351
+ this.lowerBinId.toNumber(),
12352
+ this.upperBinId.toNumber()
12353
+ );
12354
+ const { withdrawParams, result: withdrawResult } = this._simulateWithdraw(withdraws);
12355
+ const { depositParams, result: depositResult } = this._simulateDeposit(
12356
+ binStep,
12357
+ tokenXDecimal,
12358
+ tokenYDecimal,
12359
+ deposits,
12360
+ withdrawResult
12361
+ );
12362
+ const afterWidth = getPositionWidthWithMinWidth(
12363
+ this.lowerBinId.toNumber(),
12364
+ this.upperBinId.toNumber()
12365
+ );
12366
+ const widthDelta = afterWidth - beforeWidth;
12367
+ let rentalCostLamports = new (0, _bnjs2.default)(0);
12368
+ if (widthDelta != 0) {
12369
+ const sizeChanges = Math.abs(widthDelta) * POSITION_BIN_DATA_SIZE;
12370
+ const [minimumLamports, rentExemptionLamports] = await Promise.all([
12371
+ connection.getMinimumBalanceForRentExemption(0),
12372
+ connection.getMinimumBalanceForRentExemption(sizeChanges)
12373
+ ]);
12374
+ const lamportChanges = new (0, _bnjs2.default)(rentExemptionLamports).sub(
12375
+ new (0, _bnjs2.default)(minimumLamports)
12376
+ );
12377
+ if (widthDelta > 0) {
12378
+ rentalCostLamports = rentalCostLamports.add(lamportChanges);
12379
+ } else {
12380
+ rentalCostLamports = rentalCostLamports.sub(lamportChanges);
12381
+ }
12382
+ }
12383
+ return {
12384
+ amountXDeposited: depositResult.totalAmountXDeposited,
12385
+ amountYDeposited: depositResult.totalAmountYDeposited,
12386
+ actualAmountXDeposited: depositResult.actualTotalAmountXDeposited,
12387
+ actualAmountYDeposited: depositResult.actualTotalAmountYDeposited,
12388
+ actualAmountXWithdrawn: depositResult.actualLiquidityAndFeeXWithdrawn,
12389
+ actualAmountYWithdrawn: depositResult.actualLiquidityAndFeeYWithdrawn,
12390
+ rewardAmountsClaimed: withdrawResult.rewardsAmountClaimed,
12391
+ withdrawParams,
12392
+ depositParams,
12393
+ rentalCostLamports
12394
+ };
12246
12395
  }
12247
- if (y0.isNeg()) {
12248
- bitFlag |= 2;
12249
- y0 = y0.neg();
12396
+ totalAmounts() {
12397
+ let totalAmountX = new (0, _bnjs2.default)(0);
12398
+ let totalAmountY = new (0, _bnjs2.default)(0);
12399
+ for (const binData of this.rebalancePositionBinData) {
12400
+ totalAmountX = totalAmountX.add(binData.amountX);
12401
+ totalAmountY = totalAmountY.add(binData.amountY);
12402
+ }
12403
+ return [totalAmountX, totalAmountY];
12250
12404
  }
12251
- if (deltaX.isNeg()) {
12252
- bitFlag |= 4;
12253
- deltaX = deltaX.neg();
12405
+ totalFeeAmounts() {
12406
+ let totalFeeXAmount = new (0, _bnjs2.default)(0);
12407
+ let totalFeeYAmount = new (0, _bnjs2.default)(0);
12408
+ for (const binData of this.rebalancePositionBinData) {
12409
+ totalFeeXAmount = totalFeeXAmount.add(binData.claimableFeeXAmount);
12410
+ totalFeeYAmount = totalFeeYAmount.add(binData.claimableFeeYAmount);
12411
+ }
12412
+ return [totalFeeXAmount, totalFeeYAmount];
12254
12413
  }
12255
- if (deltaY.isNeg()) {
12256
- bitFlag |= 8;
12257
- deltaY = deltaY.neg();
12414
+ totalRewardAmounts() {
12415
+ let totalRewardAmounts = [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)];
12416
+ for (const binData of this.rebalancePositionBinData) {
12417
+ totalRewardAmounts[0] = totalRewardAmounts[0].add(
12418
+ binData.claimableRewardAmount[0]
12419
+ );
12420
+ totalRewardAmounts[1] = totalRewardAmounts[1].add(
12421
+ binData.claimableRewardAmount[1]
12422
+ );
12423
+ }
12424
+ return totalRewardAmounts;
12258
12425
  }
12259
- return {
12260
- bitFlag,
12261
- x0,
12262
- y0,
12263
- deltaX,
12264
- deltaY
12265
- };
12426
+ };
12427
+ function getPositionWidthWithMinWidth(lowerBinId, upperBinId) {
12428
+ const width = upperBinId - lowerBinId + 1;
12429
+ return Math.max(width, DEFAULT_BIN_PER_POSITION.toNumber());
12266
12430
  }
12267
- function toRebalancePositionBinData(positionData) {
12268
- return positionData.positionBinData.map(
12269
- ({
12270
- binId,
12271
- price,
12272
- pricePerToken,
12273
- positionXAmount,
12274
- positionYAmount,
12275
- positionFeeXAmount,
12276
- positionFeeYAmount,
12277
- positionRewardAmount
12278
- }) => {
12279
- return {
12280
- binId,
12281
- price,
12282
- pricePerToken,
12283
- amountX: new (0, _bnjs2.default)(positionXAmount),
12284
- amountY: new (0, _bnjs2.default)(positionYAmount),
12285
- claimableRewardAmount: positionRewardAmount.map(
12286
- (amount) => new (0, _bnjs2.default)(amount)
12287
- ),
12288
- claimableFeeXAmount: new (0, _bnjs2.default)(positionFeeXAmount),
12289
- claimableFeeYAmount: new (0, _bnjs2.default)(positionFeeYAmount)
12290
- };
12291
- }
12431
+ function validateAndSortRebalanceDeposit(deposits) {
12432
+ const sortedDeposits = deposits.sort(
12433
+ (a, b) => a.minDeltaId.sub(b.minDeltaId).toNumber()
12292
12434
  );
12435
+ for (const deposit of deposits) {
12436
+ if (deposit.minDeltaId.gte(deposit.maxDeltaId)) {
12437
+ throw "Invalid minDeltaId or maxDeltaId";
12438
+ }
12439
+ }
12440
+ for (let i = 1; i < sortedDeposits.length; i++) {
12441
+ const prevDeposit = sortedDeposits[i - 1];
12442
+ const currDeposit = sortedDeposits[i];
12443
+ if (prevDeposit.maxDeltaId.gte(currDeposit.minDeltaId)) {
12444
+ throw "Overlap deposit bin range";
12445
+ }
12446
+ }
12447
+ return sortedDeposits;
12293
12448
  }
12294
- function getDepositBinIds(activeId, deposits) {
12295
- const uniqueBinId = /* @__PURE__ */ new Set();
12296
- for (const { minDeltaId, maxDeltaId } of deposits) {
12297
- const minBinId = activeId.add(minDeltaId);
12298
- const maxBinId = activeId.add(maxDeltaId);
12299
- for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
12300
- uniqueBinId.add(binId);
12449
+ function validateAndSortRebalanceWithdraw(withdraws, activeId) {
12450
+ const filledWithdraws = [];
12451
+ for (const { minBinId, maxBinId, bps } of withdraws) {
12452
+ if (bps.toNumber() < 0 || bps.toNumber() > BASIS_POINT_MAX) {
12453
+ throw "Invalid bps";
12454
+ }
12455
+ const filledMinBinId = _nullishCoalesce(minBinId, () => ( activeId));
12456
+ const filledMaxBinId = _nullishCoalesce(maxBinId, () => ( activeId));
12457
+ if (filledMinBinId.gt(filledMaxBinId)) {
12458
+ throw "Invalid minBinId or maxBinId";
12301
12459
  }
12460
+ filledWithdraws.push({
12461
+ minBinId: filledMinBinId,
12462
+ maxBinId: filledMaxBinId,
12463
+ bps
12464
+ });
12302
12465
  }
12303
- const binIds = Array.from(uniqueBinId);
12304
- binIds.sort((a, b) => a - b);
12305
- return binIds;
12466
+ filledWithdraws.sort((a, b) => {
12467
+ return a.minBinId.sub(b.minBinId).toNumber();
12468
+ });
12469
+ for (let i = 1; i < filledWithdraws.length; i++) {
12470
+ const prev = filledWithdraws[i - 1];
12471
+ const curr = filledWithdraws[i];
12472
+ if (curr.minBinId.lte(prev.maxBinId)) {
12473
+ throw "Overlap withdraw bin range";
12474
+ }
12475
+ }
12476
+ return filledWithdraws;
12306
12477
  }
12307
- function findMinMaxBinIdWithLiquidity(rebalancePositionBinData) {
12308
- let minBinId = null;
12309
- let maxBinId = null;
12310
- for (const binData of rebalancePositionBinData) {
12311
- if (binData.amountX.isZero() && binData.amountY.isZero() && binData.claimableFeeXAmount.isZero() && binData.claimableFeeYAmount.isZero() && binData.claimableRewardAmount.every((amount) => amount.isZero())) {
12312
- continue;
12478
+ function binRangeToBinIdArray(minBinId, maxBinId) {
12479
+ const binIdArray = [];
12480
+ const fromBinId = minBinId.toNumber();
12481
+ const toBinId = maxBinId.toNumber();
12482
+ for (let binId = fromBinId; binId <= toBinId; binId++) {
12483
+ binIdArray.push(new (0, _bnjs2.default)(binId));
12484
+ }
12485
+ return binIdArray;
12486
+ }
12487
+ function getRebalanceBinArrayIndexesAndBitmapCoverage(adds, removes, activeId, pairAddress, programId) {
12488
+ let indexMap = /* @__PURE__ */ new Map();
12489
+ removes.forEach((value) => {
12490
+ let minBinId = value.minBinId;
12491
+ if (minBinId == null) {
12492
+ minBinId = activeId;
12313
12493
  }
12314
- if (minBinId == null || binData.binId < minBinId) {
12315
- minBinId = binData.binId;
12494
+ let maxBinId = value.maxBinId;
12495
+ if (maxBinId == null) {
12496
+ maxBinId = activeId;
12316
12497
  }
12317
- if (maxBinId == null || binData.binId > maxBinId) {
12318
- maxBinId = binData.binId;
12498
+ let binArrayIndex = binIdToBinArrayIndex(new (0, _bnjs2.default)(minBinId));
12499
+ const upperBinId = new (0, _bnjs2.default)(maxBinId);
12500
+ while (true) {
12501
+ indexMap.set(binArrayIndex.toNumber(), true);
12502
+ const [binArrayLowerBinId, binArrayUpperBinId] = getBinArrayLowerUpperBinId(binArrayIndex);
12503
+ if (upperBinId.gte(binArrayLowerBinId) && upperBinId.lte(binArrayUpperBinId)) {
12504
+ break;
12505
+ } else {
12506
+ binArrayIndex = binArrayIndex.add(new (0, _bnjs2.default)(1));
12507
+ }
12508
+ }
12509
+ });
12510
+ adds.forEach((value) => {
12511
+ const minBinId = activeId + value.minDeltaId;
12512
+ const maxBinId = activeId + value.maxDeltaId;
12513
+ let binArrayIndex = binIdToBinArrayIndex(new (0, _bnjs2.default)(minBinId));
12514
+ const upperBinId = new (0, _bnjs2.default)(maxBinId);
12515
+ while (true) {
12516
+ indexMap.set(binArrayIndex.toNumber(), true);
12517
+ const [binArrayLowerBinId, binArrayUpperBinId] = getBinArrayLowerUpperBinId(binArrayIndex);
12518
+ if (upperBinId.gte(binArrayLowerBinId) && upperBinId.lte(binArrayUpperBinId)) {
12519
+ break;
12520
+ } else {
12521
+ binArrayIndex = binArrayIndex.add(new (0, _bnjs2.default)(1));
12522
+ }
12319
12523
  }
12320
- }
12321
- return [minBinId, maxBinId];
12322
- }
12323
- function onlyDepositToBidSide(maxDeltaId, favorXInActiveBin) {
12324
- if (favorXInActiveBin) {
12325
- return maxDeltaId.lt(new (0, _bnjs2.default)(0));
12326
- }
12327
- return maxDeltaId.lte(new (0, _bnjs2.default)(0));
12524
+ });
12525
+ const binArrayIndexes = Array.from(indexMap.keys()).map((idx) => new (0, _bnjs2.default)(idx));
12526
+ const requireBitmapExtension = binArrayIndexes.some(
12527
+ (index) => isOverflowDefaultBinArrayBitmap(new (0, _bnjs2.default)(index))
12528
+ );
12529
+ return {
12530
+ binArrayIndexes,
12531
+ binArrayBitmap: requireBitmapExtension ? deriveBinArrayBitmapExtension(pairAddress, programId)[0] : programId
12532
+ };
12328
12533
  }
12329
- function onlyDepositToAskSide(minDeltaId, favorXInActiveBin) {
12330
- if (favorXInActiveBin) {
12331
- return minDeltaId.gte(new (0, _bnjs2.default)(0));
12534
+
12535
+ // src/dlmm/helpers/rebalance/liquidity_strategy/index.ts
12536
+
12537
+
12538
+ // src/dlmm/helpers/rebalance/liquidity_strategy/bidAsk.ts
12539
+
12540
+ function findBaseDeltaY(amountY, minDeltaId, maxDeltaId) {
12541
+ if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0))) {
12542
+ return new (0, _bnjs2.default)(0);
12332
12543
  }
12333
- return minDeltaId.gt(new (0, _bnjs2.default)(0));
12334
- }
12335
- function getAmountInBinsBidSide(activeId, minDeltaId, maxDeltaId, deltaY, y0) {
12336
- const amountInBins = [];
12337
- const minBinId = activeId.add(minDeltaId);
12338
- const maxBinId = activeId.add(maxDeltaId);
12339
- for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
12340
- const deltaBin = activeId.toNumber() - binId;
12341
- const totalDeltaY = deltaY.mul(new (0, _bnjs2.default)(deltaBin));
12342
- const amountY = y0.add(totalDeltaY);
12343
- amountInBins.push({
12344
- binId: new (0, _bnjs2.default)(binId),
12345
- amountX: new (0, _bnjs2.default)(0),
12346
- amountY
12347
- });
12544
+ if (minDeltaId.eq(maxDeltaId)) {
12545
+ return amountY;
12348
12546
  }
12349
- return amountInBins;
12547
+ const m1 = minDeltaId.neg().subn(1);
12548
+ const m2 = maxDeltaId.neg();
12549
+ const b = m2.neg().mul(m1.sub(m2).addn(1));
12550
+ const c = m1.mul(m1.addn(1)).divn(2);
12551
+ const d = m2.mul(m2.subn(1)).divn(2);
12552
+ const a = b.add(c.sub(d));
12553
+ return amountY.div(a);
12350
12554
  }
12351
- function getAmountInBinsAskSide(activeId, binStep, minDeltaId, maxDeltaId, deltaX, x0) {
12352
- const binCount = maxDeltaId.sub(minDeltaId).add(new (0, _bnjs2.default)(1));
12353
- const minBinId = activeId.add(minDeltaId);
12354
- const maxBinId = activeId.add(maxDeltaId);
12355
- const amountInBins = new Array(binCount.toNumber());
12356
- const base = getQPriceBaseFactor(binStep);
12357
- let inverseBasePrice = pow(base, maxBinId.neg());
12358
- for (let binId = maxBinId.toNumber(); binId >= minBinId.toNumber(); binId--) {
12359
- const delta = binId - activeId.toNumber();
12360
- const totalDeltaX = deltaX.mul(new (0, _bnjs2.default)(delta));
12361
- const amountX = x0.add(totalDeltaX).mul(inverseBasePrice).shrn(SCALE_OFFSET);
12362
- const idx = binId - minBinId.toNumber();
12363
- amountInBins[idx] = {
12364
- binId: new (0, _bnjs2.default)(binId),
12365
- amountX,
12366
- amountY: new (0, _bnjs2.default)(0)
12555
+ function findY0AndDeltaY(amountY, minDeltaId, maxDeltaId, activeId) {
12556
+ if (minDeltaId.gt(maxDeltaId) || amountY.isZero()) {
12557
+ return {
12558
+ base: new (0, _bnjs2.default)(0),
12559
+ delta: new (0, _bnjs2.default)(0)
12367
12560
  };
12368
- inverseBasePrice = inverseBasePrice.mul(base).shrn(SCALE_OFFSET);
12369
- }
12370
- return amountInBins;
12371
- }
12372
- function toAmountIntoBins(activeId, minDeltaId, maxDeltaId, deltaX, deltaY, x0, y0, binStep, favorXInActiveBin) {
12373
- if (onlyDepositToBidSide(maxDeltaId, favorXInActiveBin)) {
12374
- return getAmountInBinsBidSide(activeId, minDeltaId, maxDeltaId, deltaY, y0);
12375
12561
  }
12376
- if (onlyDepositToAskSide(minDeltaId, favorXInActiveBin)) {
12377
- return getAmountInBinsAskSide(
12562
+ let baseDeltaY = findBaseDeltaY(amountY, minDeltaId, maxDeltaId);
12563
+ const y0 = baseDeltaY.neg().mul(maxDeltaId).add(baseDeltaY);
12564
+ while (true) {
12565
+ const amountInBins = getAmountInBinsBidSide(
12378
12566
  activeId,
12379
- binStep,
12380
12567
  minDeltaId,
12381
12568
  maxDeltaId,
12382
- deltaX,
12383
- x0
12569
+ baseDeltaY,
12570
+ y0
12384
12571
  );
12572
+ const totalAmountY = amountInBins.reduce((acc, { amountY: amountY2 }) => {
12573
+ return acc.add(amountY2);
12574
+ }, new (0, _bnjs2.default)(0));
12575
+ if (totalAmountY.gt(amountY)) {
12576
+ baseDeltaY = baseDeltaY.sub(new (0, _bnjs2.default)(1));
12577
+ } else {
12578
+ return {
12579
+ base: y0,
12580
+ delta: baseDeltaY
12581
+ };
12582
+ }
12385
12583
  }
12386
- const [bidSideEndDeltaId, askSideStartDeltaId] = favorXInActiveBin ? [-1, 0] : [0, 1];
12387
- const amountInBinsBidSide = getAmountInBinsBidSide(
12388
- activeId,
12389
- minDeltaId,
12390
- new (0, _bnjs2.default)(bidSideEndDeltaId),
12391
- deltaY,
12392
- y0
12393
- );
12394
- const amountInBinsAskSide = getAmountInBinsAskSide(
12395
- activeId,
12396
- binStep,
12397
- new (0, _bnjs2.default)(askSideStartDeltaId),
12398
- maxDeltaId,
12399
- deltaX,
12400
- x0
12401
- );
12402
- return amountInBinsBidSide.concat(amountInBinsAskSide);
12403
- }
12404
- function getLiquidity(x, y, price) {
12405
- const px = price.mul(x);
12406
- const shly = y.shln(SCALE_OFFSET);
12407
- return px.add(shly);
12408
12584
  }
12409
- function computeCompositionFee(binStep, sParameters3, vParameters3, outAmountX, inAmountX, outAmountY, inAmountY) {
12410
- if (outAmountX.gt(inAmountX)) {
12411
- const delta = inAmountY.sub(outAmountY);
12412
- const totalFeeRate = getTotalFee(
12413
- binStep.toNumber(),
12414
- sParameters3,
12415
- vParameters3
12416
- );
12417
- const feeAmount = delta.mul(totalFeeRate);
12418
- return feeAmount.mul(FEE_PRECISION.add(totalFeeRate)).div(FEE_PRECISION.pow(new (0, _bnjs2.default)(2)));
12585
+ function findBaseDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12586
+ if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0))) {
12587
+ return new (0, _bnjs2.default)(0);
12419
12588
  }
12420
- return new (0, _bnjs2.default)(0);
12421
- }
12422
- function simulateDepositBin(binId, binStep, amountX, amountY, bin) {
12423
- if (!bin) {
12424
- return {
12425
- amountXIntoBin: amountX,
12426
- amountYIntoBin: amountY
12427
- };
12589
+ let b = new (0, _bnjs2.default)(0);
12590
+ let c = new (0, _bnjs2.default)(0);
12591
+ let m1 = minDeltaId;
12592
+ let m2 = maxDeltaId.addn(1);
12593
+ for (let m = m1.toNumber(); m <= m2.toNumber(); m++) {
12594
+ const binId = activeId.addn(m);
12595
+ const pm = getQPriceFromId(binId.neg(), binStep);
12596
+ const bDelta = m1.mul(pm);
12597
+ b = b.add(bDelta);
12598
+ const cDelta = new (0, _bnjs2.default)(m).mul(pm);
12599
+ c = c.add(cDelta);
12428
12600
  }
12429
- const price = getQPriceFromId(binId, binStep);
12430
- const inLiquidity = getLiquidity(amountX, amountY, price);
12431
- const binLiquidity = getLiquidity(bin.amountX, bin.amountY, price);
12432
- if (bin.liquiditySupply.isZero()) {
12601
+ return amountX.shln(SCALE_OFFSET).div(c.sub(b));
12602
+ }
12603
+ function findX0AndDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12604
+ if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
12433
12605
  return {
12434
- amountXIntoBin: amountX,
12435
- amountYIntoBin: amountY
12606
+ base: new (0, _bnjs2.default)(0),
12607
+ delta: new (0, _bnjs2.default)(0)
12436
12608
  };
12437
12609
  }
12438
- const liquidityShare = inLiquidity.mul(bin.liquiditySupply).div(binLiquidity);
12439
- const updatedBinXAmount = bin.amountX.add(amountX);
12440
- const updatedBinYAmount = bin.amountY.add(amountY);
12441
- const updatedBinSupply = bin.liquiditySupply.add(liquidityShare);
12442
- let amountXIntoBin = liquidityShare.mul(
12443
- updatedBinXAmount.div(updatedBinSupply)
12444
- );
12445
- let amountYIntoBin = liquidityShare.mul(
12446
- updatedBinYAmount.div(updatedBinSupply)
12610
+ let baseDeltaX = findBaseDeltaX(
12611
+ amountX,
12612
+ minDeltaId,
12613
+ maxDeltaId,
12614
+ binStep,
12615
+ activeId
12447
12616
  );
12448
- if (amountXIntoBin.gt(amountX)) {
12449
- }
12450
- return {
12451
- amountXIntoBin,
12452
- amountYIntoBin
12453
- };
12454
- }
12455
- var RebalancePosition = class {
12456
-
12457
-
12458
-
12459
-
12460
-
12461
-
12462
-
12463
-
12464
-
12465
-
12466
- constructor(positionAddress, positionData, lbPair, activeBin, shouldClaimFee, shouldClaimReward, currentTimestamp) {
12467
- this.address = positionAddress;
12468
- this.rebalancePositionBinData = toRebalancePositionBinData(positionData);
12469
- this.lowerBinId = new (0, _bnjs2.default)(positionData.lowerBinId);
12470
- this.upperBinId = new (0, _bnjs2.default)(positionData.upperBinId);
12471
- this.lbPair = lbPair;
12472
- this.shouldClaimFee = shouldClaimFee;
12473
- this.shouldClaimReward = shouldClaimReward;
12474
- this.owner = positionData.owner;
12475
- this.activeBin = activeBin;
12476
- this.currentTimestamp = currentTimestamp;
12477
- }
12478
- static async create(params) {
12479
- const {
12480
- program,
12481
- positionAddress,
12482
- pairAddress,
12483
- positionData,
12484
- shouldClaimFee,
12485
- shouldClaimReward
12486
- } = params;
12487
- const [lbPairAccount, clockAccount] = await program.provider.connection.getMultipleAccountsInfo([
12488
- pairAddress,
12489
- _web3js.SYSVAR_CLOCK_PUBKEY
12490
- ]);
12491
- const lbPair = decodeAccount(program, "lbPair", lbPairAccount.data);
12492
- const clock = ClockLayout.decode(clockAccount.data);
12493
- const activeBinArrayIdx = binIdToBinArrayIndex(new (0, _bnjs2.default)(lbPair.activeId));
12494
- const [activeBinArrayPubkey] = deriveBinArray(
12495
- pairAddress,
12496
- activeBinArrayIdx,
12497
- program.programId
12498
- );
12499
- const activeBinArrayState = await program.account.binArray.fetch(
12500
- activeBinArrayPubkey
12501
- );
12502
- const [lowerBinId, upperBinId] = getBinArrayLowerUpperBinId(activeBinArrayIdx);
12503
- const idx = getBinIdIndexInBinArray(
12504
- new (0, _bnjs2.default)(lbPair.activeId),
12505
- lowerBinId,
12506
- upperBinId
12507
- );
12508
- const activeBin = activeBinArrayState[idx.toNumber()];
12509
- return new RebalancePosition(
12510
- positionAddress,
12511
- positionData,
12512
- lbPair,
12513
- activeBin,
12514
- shouldClaimFee,
12515
- shouldClaimReward,
12516
- clock.unixTimestamp
12617
+ const x0 = minDeltaId.neg().mul(baseDeltaX).add(baseDeltaX);
12618
+ while (true) {
12619
+ const amountInBins = getAmountInBinsAskSide(
12620
+ activeId,
12621
+ binStep,
12622
+ minDeltaId,
12623
+ maxDeltaId,
12624
+ baseDeltaX,
12625
+ x0
12517
12626
  );
12518
- }
12519
- _simulateDeposit(binStep, tokenXDecimal, tokenYDecimal, deposits, simulatedWithdrawResult) {
12520
- const { liquidityAndFeeXWithdrawn, liquidityAndFeeYWithdrawn } = simulatedWithdrawResult;
12521
- const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12522
- const depositBinIds = getDepositBinIds(activeId, deposits);
12523
- if (depositBinIds.length > 0) {
12524
- const depositMinBinId = depositBinIds[0];
12525
- const depositMaxBinId = depositBinIds[depositBinIds.length - 1];
12526
- this._simulateResize(
12527
- new (0, _bnjs2.default)(depositMinBinId),
12528
- new (0, _bnjs2.default)(depositMaxBinId),
12529
- binStep,
12530
- tokenXDecimal,
12531
- tokenYDecimal
12532
- );
12627
+ const totalAmountX = amountInBins.reduce((acc, { amountX: amountX2 }) => {
12628
+ return acc.add(amountX2);
12629
+ }, new (0, _bnjs2.default)(0));
12630
+ if (totalAmountX.gt(amountX)) {
12631
+ baseDeltaX = baseDeltaX.sub(new (0, _bnjs2.default)(1));
12632
+ } else {
12633
+ return {
12634
+ base: x0,
12635
+ delta: baseDeltaX
12636
+ };
12533
12637
  }
12534
- let totalAmountXDeposited = new (0, _bnjs2.default)(0);
12535
- let totalAmountYDeposited = new (0, _bnjs2.default)(0);
12536
- const addLiquidityParam = [];
12537
- for (const {
12538
- x0,
12539
- y0,
12540
- favorXInActiveBin,
12638
+ }
12639
+ }
12640
+ var BidAskStrategyParameterBuilder = class {
12641
+ findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12642
+ return findX0AndDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId);
12643
+ }
12644
+ findYParameters(amountY, minDeltaId, maxDeltaId, activeId) {
12645
+ return findY0AndDeltaY(amountY, minDeltaId, maxDeltaId, activeId);
12646
+ }
12647
+ suggestBalancedXParametersFromY(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountY) {
12648
+ const deltaX = amountY.div(
12649
+ maxDeltaId.addn(1).mul(maxDeltaId.addn(2)).divn(2)
12650
+ );
12651
+ const x0 = minDeltaId.neg().mul(deltaX).add(deltaX);
12652
+ const totalAmountX = toAmountIntoBins(
12653
+ activeId,
12654
+ minDeltaId,
12655
+ maxDeltaId,
12541
12656
  deltaX,
12657
+ new (0, _bnjs2.default)(0),
12658
+ x0,
12659
+ new (0, _bnjs2.default)(0),
12660
+ binStep,
12661
+ favorXInActiveBin
12662
+ ).reduce((acc, bin) => {
12663
+ return acc.add(bin.amountX);
12664
+ }, new (0, _bnjs2.default)(0));
12665
+ return {
12666
+ base: x0,
12667
+ delta: deltaX,
12668
+ amountX: totalAmountX
12669
+ };
12670
+ }
12671
+ suggestBalancedYParametersFromX(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountXInQuoteValue) {
12672
+ const m1 = minDeltaId.neg().subn(1);
12673
+ const m2 = maxDeltaId.neg();
12674
+ const a1 = m2.neg().mul(m1.sub(m2).addn(1));
12675
+ const a2 = m1.mul(m1.addn(1)).divn(2);
12676
+ const a3 = m2.mul(m2.subn(1)).divn(2);
12677
+ const a = a1.add(a2.sub(a3));
12678
+ const deltaY = amountXInQuoteValue.div(a);
12679
+ const y0 = deltaY.neg().mul(m2).add(deltaY);
12680
+ const amountY = toAmountIntoBins(
12681
+ activeId,
12682
+ minDeltaId,
12683
+ maxDeltaId,
12684
+ new (0, _bnjs2.default)(0),
12542
12685
  deltaY,
12686
+ new (0, _bnjs2.default)(0),
12687
+ y0,
12688
+ binStep,
12689
+ favorXInActiveBin
12690
+ ).reduce((acc, bin) => {
12691
+ return acc.add(bin.amountY);
12692
+ }, new (0, _bnjs2.default)(0));
12693
+ return {
12694
+ base: y0,
12695
+ delta: deltaY,
12696
+ amountY
12697
+ };
12698
+ }
12699
+ };
12700
+
12701
+ // src/dlmm/helpers/rebalance/liquidity_strategy/curve.ts
12702
+
12703
+ function findBaseY0(amountY, minDeltaId, maxDeltaId) {
12704
+ if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0))) {
12705
+ return new (0, _bnjs2.default)(0);
12706
+ }
12707
+ if (minDeltaId.eq(maxDeltaId)) {
12708
+ return amountY;
12709
+ }
12710
+ const m1 = minDeltaId.neg();
12711
+ const m2 = maxDeltaId.neg();
12712
+ const b = m1.sub(m2).addn(1);
12713
+ const c = m1.mul(m1.addn(1)).divn(2);
12714
+ const d = m2.mul(m2.subn(1)).divn(2);
12715
+ const a = b.sub(c.sub(d).div(m1.addn(1)));
12716
+ return amountY.div(a);
12717
+ }
12718
+ function findY0AndDeltaY2(amountY, minDeltaId, maxDeltaId, activeId) {
12719
+ if (minDeltaId.gt(maxDeltaId) || amountY.isZero()) {
12720
+ return {
12721
+ base: new (0, _bnjs2.default)(0),
12722
+ delta: new (0, _bnjs2.default)(0)
12723
+ };
12724
+ }
12725
+ let baseY0 = findBaseY0(amountY, minDeltaId, maxDeltaId);
12726
+ while (true) {
12727
+ const deltaY = baseY0.neg().div(minDeltaId.neg().addn(1));
12728
+ const amountInBins = getAmountInBinsBidSide(
12729
+ activeId,
12543
12730
  minDeltaId,
12544
- maxDeltaId
12545
- } of deposits) {
12546
- const params = buildBitFlagAndNegateStrategyParameters(
12547
- x0,
12548
- y0,
12549
- deltaX,
12550
- deltaY
12551
- );
12552
- addLiquidityParam.push({
12553
- minDeltaId: minDeltaId.toNumber(),
12554
- maxDeltaId: maxDeltaId.toNumber(),
12555
- x0: params.x0,
12556
- y0: params.y0,
12557
- deltaX: params.deltaX,
12558
- deltaY: params.deltaY,
12559
- bitFlag: params.bitFlag,
12560
- padding: Array(16).fill(0),
12561
- favorXInActiveId: favorXInActiveBin
12562
- });
12563
- const amountIntoBins = toAmountIntoBins(
12564
- activeId,
12565
- minDeltaId,
12566
- maxDeltaId,
12567
- deltaX,
12568
- deltaY,
12569
- x0,
12570
- y0,
12571
- binStep,
12572
- favorXInActiveBin
12573
- );
12574
- for (const { binId, amountX, amountY } of amountIntoBins) {
12575
- totalAmountXDeposited = totalAmountXDeposited.add(amountX);
12576
- totalAmountYDeposited = totalAmountYDeposited.add(amountY);
12577
- const idx = this.rebalancePositionBinData.findIndex(
12578
- (data) => data.binId == binId.toNumber()
12579
- );
12580
- if (binId.eq(activeId)) {
12581
- const vParameters3 = Object.assign({}, this.lbPair.vParameters);
12582
- const sParameters3 = Object.assign({}, this.lbPair.parameters);
12583
- DLMM.updateReference(
12584
- activeId.toNumber(),
12585
- vParameters3,
12586
- sParameters3,
12587
- this.currentTimestamp.toNumber()
12588
- );
12589
- DLMM.updateVolatilityAccumulator(
12590
- vParameters3,
12591
- sParameters3,
12592
- activeId.toNumber()
12593
- );
12594
- const { amountXIntoBin, amountYIntoBin } = simulateDepositBin(
12595
- binId,
12596
- binStep,
12597
- amountX,
12598
- amountY,
12599
- this.activeBin
12600
- );
12601
- const feeY = computeCompositionFee(
12602
- binStep,
12603
- sParameters3,
12604
- vParameters3,
12605
- amountXIntoBin,
12606
- amountX,
12607
- amountYIntoBin,
12608
- amountY
12609
- );
12610
- const feeX = computeCompositionFee(
12611
- binStep,
12612
- sParameters3,
12613
- vParameters3,
12614
- amountYIntoBin,
12615
- amountY,
12616
- amountXIntoBin,
12617
- amountX
12618
- );
12619
- const amountXIntoBinExcludeFee = amountXIntoBin.sub(feeX);
12620
- const amountYIntoBinExcludeFee = amountYIntoBin.sub(feeY);
12621
- this.rebalancePositionBinData[idx].amountX = this.rebalancePositionBinData[idx].amountX.add(
12622
- amountXIntoBinExcludeFee
12623
- );
12624
- this.rebalancePositionBinData[idx].amountY = this.rebalancePositionBinData[idx].amountY.add(
12625
- amountYIntoBinExcludeFee
12626
- );
12627
- } else {
12628
- this.rebalancePositionBinData[idx].amountX = this.rebalancePositionBinData[idx].amountX.add(amountX);
12629
- this.rebalancePositionBinData[idx].amountY = this.rebalancePositionBinData[idx].amountY.add(amountY);
12630
- }
12631
- }
12632
- }
12633
- let actualTotalAmountXDeposited = totalAmountXDeposited;
12634
- let actualTotalAmountYDeposited = totalAmountYDeposited;
12635
- let actualLiquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn;
12636
- let actualLiquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn;
12637
- if (actualTotalAmountXDeposited.gt(actualLiquidityAndFeeXWithdrawn)) {
12638
- actualTotalAmountXDeposited = actualTotalAmountXDeposited.sub(
12639
- actualLiquidityAndFeeXWithdrawn
12640
- );
12641
- actualLiquidityAndFeeXWithdrawn = new (0, _bnjs2.default)(0);
12642
- } else {
12643
- actualLiquidityAndFeeXWithdrawn = actualLiquidityAndFeeXWithdrawn.sub(
12644
- actualTotalAmountXDeposited
12645
- );
12646
- actualTotalAmountXDeposited = new (0, _bnjs2.default)(0);
12647
- }
12648
- if (actualTotalAmountYDeposited.gt(actualLiquidityAndFeeYWithdrawn)) {
12649
- actualTotalAmountYDeposited = actualTotalAmountYDeposited.sub(
12650
- actualLiquidityAndFeeYWithdrawn
12651
- );
12652
- actualLiquidityAndFeeYWithdrawn = new (0, _bnjs2.default)(0);
12731
+ maxDeltaId,
12732
+ deltaY,
12733
+ baseY0
12734
+ );
12735
+ const totalAmountY = amountInBins.reduce((acc, { amountY: amountY2 }) => {
12736
+ return acc.add(amountY2);
12737
+ }, new (0, _bnjs2.default)(0));
12738
+ if (totalAmountY.gt(amountY)) {
12739
+ baseY0 = baseY0.sub(new (0, _bnjs2.default)(1));
12653
12740
  } else {
12654
- actualLiquidityAndFeeYWithdrawn = actualLiquidityAndFeeYWithdrawn.sub(
12655
- actualTotalAmountYDeposited
12656
- );
12657
- actualTotalAmountYDeposited = new (0, _bnjs2.default)(0);
12741
+ return {
12742
+ base: baseY0,
12743
+ delta: deltaY
12744
+ };
12658
12745
  }
12746
+ }
12747
+ }
12748
+ function findBaseX0(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12749
+ if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0))) {
12750
+ return new (0, _bnjs2.default)(0);
12751
+ }
12752
+ let b = new (0, _bnjs2.default)(0);
12753
+ let c = new (0, _bnjs2.default)(0);
12754
+ let m1 = minDeltaId;
12755
+ let m2 = maxDeltaId;
12756
+ for (let m = m1.toNumber(); m <= m2.toNumber(); m++) {
12757
+ const binId = activeId.addn(m);
12758
+ const pm = getQPriceFromId(binId.neg(), binStep);
12759
+ b = b.add(pm);
12760
+ const cDelta = new (0, _bnjs2.default)(m).mul(pm).div(m2);
12761
+ c = c.add(cDelta);
12762
+ }
12763
+ return amountX.shln(SCALE_OFFSET).div(b.sub(c));
12764
+ }
12765
+ function findX0AndDeltaX2(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12766
+ if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
12659
12767
  return {
12660
- result: {
12661
- totalAmountXDeposited,
12662
- totalAmountYDeposited,
12663
- actualLiquidityAndFeeXWithdrawn,
12664
- actualLiquidityAndFeeYWithdrawn,
12665
- actualTotalAmountXDeposited,
12666
- actualTotalAmountYDeposited
12667
- },
12668
- depositParams: addLiquidityParam
12768
+ base: new (0, _bnjs2.default)(0),
12769
+ delta: new (0, _bnjs2.default)(0)
12669
12770
  };
12670
12771
  }
12671
- _simulateResize(depositMinBinId, depositMaxBinId, binStep, tokenXDecimal, tokenYDecimal) {
12672
- const tokenXMultiplier = new (0, _decimaljs2.default)(10 ** tokenXDecimal.toNumber());
12673
- const tokenYMultiplier = new (0, _decimaljs2.default)(10 ** tokenYDecimal.toNumber());
12674
- const [minBinId, maxBinId] = findMinMaxBinIdWithLiquidity(
12675
- this.rebalancePositionBinData
12676
- );
12677
- const newMinBinId = new (0, _bnjs2.default)(
12678
- Math.min(depositMinBinId.toNumber(), _nullishCoalesce(minBinId, () => ( Number.MAX_SAFE_INTEGER)))
12679
- );
12680
- const newMaxBinId = new (0, _bnjs2.default)(
12681
- Math.max(depositMaxBinId.toNumber(), _nullishCoalesce(maxBinId, () => ( Number.MIN_SAFE_INTEGER)))
12772
+ let baseX0 = findBaseX0(amountX, minDeltaId, maxDeltaId, binStep, activeId);
12773
+ const deltaX = baseX0.neg().div(maxDeltaId);
12774
+ while (true) {
12775
+ const amountInBins = getAmountInBinsAskSide(
12776
+ activeId,
12777
+ binStep,
12778
+ minDeltaId,
12779
+ maxDeltaId,
12780
+ deltaX,
12781
+ baseX0
12682
12782
  );
12683
- if (newMinBinId.lt(this.lowerBinId)) {
12684
- const binCountToExpand = this.lowerBinId.sub(depositMinBinId);
12685
- for (let i = 1; i <= binCountToExpand.toNumber(); i++) {
12686
- const binId = this.lowerBinId.subn(i);
12687
- const price = getPriceOfBinByBinId(
12688
- binId.toNumber(),
12689
- binStep.toNumber()
12690
- );
12691
- const adjustedPrice = price.mul(tokenXMultiplier).div(tokenYMultiplier);
12692
- this.rebalancePositionBinData.unshift({
12693
- binId: binId.toNumber(),
12694
- price: adjustedPrice.toString(),
12695
- pricePerToken: adjustedPrice.toString(),
12696
- amountX: new (0, _bnjs2.default)(0),
12697
- amountY: new (0, _bnjs2.default)(0),
12698
- claimableRewardAmount: [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)],
12699
- claimableFeeXAmount: new (0, _bnjs2.default)(0),
12700
- claimableFeeYAmount: new (0, _bnjs2.default)(0)
12701
- });
12702
- }
12703
- } else {
12704
- const binCountToShrink = newMinBinId.sub(this.lowerBinId);
12705
- for (let i = 1; i <= binCountToShrink.toNumber(); i++) {
12706
- this.rebalancePositionBinData.shift();
12707
- }
12708
- }
12709
- if (newMaxBinId.gt(this.upperBinId)) {
12710
- const binCountToExpand = newMaxBinId.sub(this.upperBinId);
12711
- for (let i = 1; i <= binCountToExpand.toNumber(); i++) {
12712
- const binId = this.upperBinId.addn(i);
12713
- const price = getPriceOfBinByBinId(
12714
- binId.toNumber(),
12715
- binStep.toNumber()
12716
- );
12717
- const adjustedPrice = price.mul(tokenXMultiplier).div(tokenYMultiplier);
12718
- this.rebalancePositionBinData.push({
12719
- binId: binId.toNumber(),
12720
- price: adjustedPrice.toString(),
12721
- pricePerToken: adjustedPrice.toString(),
12722
- amountX: new (0, _bnjs2.default)(0),
12723
- amountY: new (0, _bnjs2.default)(0),
12724
- claimableRewardAmount: [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)],
12725
- claimableFeeXAmount: new (0, _bnjs2.default)(0),
12726
- claimableFeeYAmount: new (0, _bnjs2.default)(0)
12727
- });
12728
- }
12783
+ const totalAmountX = amountInBins.reduce((acc, { amountX: amountX2 }) => {
12784
+ return acc.add(amountX2);
12785
+ }, new (0, _bnjs2.default)(0));
12786
+ if (totalAmountX.gt(amountX)) {
12787
+ baseX0 = baseX0.sub(new (0, _bnjs2.default)(1));
12729
12788
  } else {
12730
- const binCountToShrink = this.upperBinId.sub(newMaxBinId);
12731
- for (let i = 1; i <= binCountToShrink.toNumber(); i++) {
12732
- this.rebalancePositionBinData.pop();
12733
- }
12789
+ return {
12790
+ base: baseX0,
12791
+ delta: deltaX
12792
+ };
12734
12793
  }
12735
- this.lowerBinId = newMinBinId;
12736
- this.upperBinId = newMaxBinId;
12737
12794
  }
12738
- _simulateWithdraw(withdraws) {
12739
- let liquidityAndFeeXWithdrawn = new (0, _bnjs2.default)(0);
12740
- let liquidityAndFeeYWithdrawn = new (0, _bnjs2.default)(0);
12741
- let rewardsAmountClaimed = [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)];
12742
- const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12743
- for (const { minBinId, maxBinId, bps } of withdraws) {
12744
- const fromBinId = _nullishCoalesce(minBinId, () => ( activeId));
12745
- const toBinId = _nullishCoalesce(maxBinId, () => ( activeId));
12746
- const binIds = binRangeToBinIdArray(fromBinId, toBinId).filter(
12747
- (binId) => binId.gte(this.lowerBinId) && binId.lte(this.upperBinId)
12748
- );
12749
- for (const binId of binIds) {
12750
- const idx = this.rebalancePositionBinData.findIndex(
12751
- (b) => b.binId === binId.toNumber()
12752
- );
12753
- const binData = this.rebalancePositionBinData[idx];
12754
- const amountXWithdrawn = binData.amountX.mul(bps).divn(BASIS_POINT_MAX);
12755
- const amountYWithdrawn = binData.amountY.mul(bps).divn(BASIS_POINT_MAX);
12756
- liquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn.add(amountXWithdrawn);
12757
- liquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn.add(amountYWithdrawn);
12758
- binData.amountX = binData.amountX.sub(amountXWithdrawn);
12759
- binData.amountY = binData.amountY.sub(amountYWithdrawn);
12760
- if (this.shouldClaimFee) {
12761
- liquidityAndFeeXWithdrawn = liquidityAndFeeXWithdrawn.add(
12762
- binData.claimableFeeXAmount
12763
- );
12764
- liquidityAndFeeYWithdrawn = liquidityAndFeeYWithdrawn.add(
12765
- binData.claimableFeeYAmount
12766
- );
12767
- binData.claimableFeeXAmount = new (0, _bnjs2.default)(0);
12768
- binData.claimableFeeYAmount = new (0, _bnjs2.default)(0);
12769
- }
12770
- if (this.shouldClaimReward) {
12771
- for (const [idx2, amount] of binData.claimableRewardAmount.entries()) {
12772
- rewardsAmountClaimed[idx2] = rewardsAmountClaimed[idx2].add(amount);
12773
- binData.claimableRewardAmount[idx2] = new (0, _bnjs2.default)(0);
12774
- }
12775
- }
12776
- this.rebalancePositionBinData[idx] = binData;
12777
- }
12778
- }
12779
- const withdrawParams = withdraws.map(
12780
- ({ minBinId, maxBinId, bps }) => {
12781
- return {
12782
- minBinId: minBinId ? minBinId.toNumber() : null,
12783
- maxBinId: maxBinId ? maxBinId.toNumber() : null,
12784
- bps: bps.toNumber(),
12785
- padding: Array(16).fill(0)
12786
- };
12787
- }
12788
- );
12795
+ }
12796
+ var CurveStrategyParameterBuilder = class {
12797
+ findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12798
+ return findX0AndDeltaX2(amountX, minDeltaId, maxDeltaId, binStep, activeId);
12799
+ }
12800
+ findYParameters(amountY, minDeltaId, maxDeltaId, activeId) {
12801
+ return findY0AndDeltaY2(amountY, minDeltaId, maxDeltaId, activeId);
12802
+ }
12803
+ suggestBalancedXParametersFromY(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountY) {
12804
+ const x0 = amountY.muln(2).div(maxDeltaId.addn(1));
12805
+ const deltaX = x0.neg().div(maxDeltaId);
12806
+ const totalAmountX = toAmountIntoBins(
12807
+ activeId,
12808
+ minDeltaId,
12809
+ maxDeltaId,
12810
+ deltaX,
12811
+ new (0, _bnjs2.default)(0),
12812
+ x0,
12813
+ new (0, _bnjs2.default)(0),
12814
+ binStep,
12815
+ favorXInActiveBin
12816
+ ).reduce((acc, bin) => {
12817
+ return acc.add(bin.amountX);
12818
+ }, new (0, _bnjs2.default)(0));
12789
12819
  return {
12790
- result: {
12791
- liquidityAndFeeXWithdrawn,
12792
- liquidityAndFeeYWithdrawn,
12793
- rewardsAmountClaimed
12794
- },
12795
- withdrawParams
12820
+ base: x0,
12821
+ delta: deltaX,
12822
+ amountX: totalAmountX
12796
12823
  };
12797
12824
  }
12798
- async simulateRebalance(connection, binStep, tokenXDecimal, tokenYDecimal, withdraws, deposits) {
12799
- if (withdraws.length == 0 && deposits.length == 0) {
12800
- throw "No rebalance action";
12801
- }
12802
- const activeId = new (0, _bnjs2.default)(this.lbPair.activeId);
12803
- withdraws = validateAndSortRebalanceWithdraw(withdraws, activeId);
12804
- deposits = validateAndSortRebalanceDeposit(deposits);
12805
- const beforeWidth = getPositionWidthWithMinWidth(
12806
- this.lowerBinId.toNumber(),
12807
- this.upperBinId.toNumber()
12808
- );
12809
- const { withdrawParams, result: withdrawResult } = this._simulateWithdraw(withdraws);
12810
- const { depositParams, result: depositResult } = this._simulateDeposit(
12825
+ suggestBalancedYParametersFromX(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountXInQuoteValue) {
12826
+ const m1 = minDeltaId.neg();
12827
+ const m2 = maxDeltaId.neg();
12828
+ const a1 = m1.sub(m2).addn(1);
12829
+ const a2 = m1.mul(m1.addn(1)).divn(2);
12830
+ const a3 = m2.mul(m2.subn(1)).divn(2);
12831
+ const a = m1.sub(a3.sub(a2)).div(m1);
12832
+ const y0 = amountXInQuoteValue.div(a);
12833
+ const deltaY = y0.neg().div(m1);
12834
+ const amountY = toAmountIntoBins(
12835
+ activeId,
12836
+ minDeltaId,
12837
+ maxDeltaId,
12838
+ new (0, _bnjs2.default)(0),
12839
+ deltaY,
12840
+ new (0, _bnjs2.default)(0),
12841
+ y0,
12811
12842
  binStep,
12812
- tokenXDecimal,
12813
- tokenYDecimal,
12814
- deposits,
12815
- withdrawResult
12816
- );
12817
- const afterWidth = getPositionWidthWithMinWidth(
12818
- this.lowerBinId.toNumber(),
12819
- this.upperBinId.toNumber()
12843
+ favorXInActiveBin
12844
+ ).reduce((acc, bin) => {
12845
+ return acc.add(bin.amountY);
12846
+ }, new (0, _bnjs2.default)(0));
12847
+ return {
12848
+ base: y0,
12849
+ delta: deltaY,
12850
+ amountY
12851
+ };
12852
+ }
12853
+ };
12854
+
12855
+ // src/dlmm/helpers/rebalance/liquidity_strategy/spot.ts
12856
+
12857
+ function findY0(amountY, minDeltaId, maxDeltaId) {
12858
+ if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0)) || amountY.isZero()) {
12859
+ return new (0, _bnjs2.default)(0);
12860
+ }
12861
+ const m1 = minDeltaId.neg();
12862
+ const m2 = maxDeltaId.neg();
12863
+ const delta = m1.sub(m2).addn(1);
12864
+ return amountY.div(delta);
12865
+ }
12866
+ function findBaseX02(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12867
+ let totalWeight = new (0, _bnjs2.default)(0);
12868
+ const minBinId = activeId.add(minDeltaId);
12869
+ const maxBinId = activeId.add(maxDeltaId);
12870
+ let baseFactor = getQPriceBaseFactor(binStep);
12871
+ let basePrice = getQPriceFromId(maxBinId.neg(), binStep);
12872
+ for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
12873
+ totalWeight = totalWeight.add(basePrice);
12874
+ basePrice = basePrice.mul(baseFactor).shrn(SCALE_OFFSET);
12875
+ }
12876
+ return amountX.shln(SCALE_OFFSET).div(totalWeight);
12877
+ }
12878
+ function findX0(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12879
+ if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
12880
+ return new (0, _bnjs2.default)(0);
12881
+ }
12882
+ let x0 = findBaseX02(amountX, minDeltaId, maxDeltaId, binStep, activeId);
12883
+ while (true) {
12884
+ const amountInBins = getAmountInBinsAskSide(
12885
+ activeId,
12886
+ binStep,
12887
+ minDeltaId,
12888
+ maxDeltaId,
12889
+ new (0, _bnjs2.default)(0),
12890
+ x0
12820
12891
  );
12821
- const widthDelta = afterWidth - beforeWidth;
12822
- let rentalCostLamports = new (0, _bnjs2.default)(0);
12823
- if (widthDelta != 0) {
12824
- const sizeChanges = Math.abs(widthDelta) * POSITION_BIN_DATA_SIZE;
12825
- const [minimumLamports, rentExemptionLamports] = await Promise.all([
12826
- connection.getMinimumBalanceForRentExemption(0),
12827
- connection.getMinimumBalanceForRentExemption(sizeChanges)
12828
- ]);
12829
- const lamportChanges = new (0, _bnjs2.default)(rentExemptionLamports).sub(
12830
- new (0, _bnjs2.default)(minimumLamports)
12831
- );
12832
- if (widthDelta > 0) {
12833
- rentalCostLamports = rentalCostLamports.add(lamportChanges);
12834
- } else {
12835
- rentalCostLamports = rentalCostLamports.sub(lamportChanges);
12836
- }
12892
+ const totalAmountX = amountInBins.reduce((acc, bin) => {
12893
+ return acc.add(bin.amountX);
12894
+ }, new (0, _bnjs2.default)(0));
12895
+ if (totalAmountX.lt(amountX)) {
12896
+ x0 = x0.add(new (0, _bnjs2.default)(1));
12897
+ } else {
12898
+ x0 = x0.sub(new (0, _bnjs2.default)(1));
12899
+ return x0;
12837
12900
  }
12901
+ }
12902
+ }
12903
+ var SpotStrategyParameterBuilder = class {
12904
+ findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
12838
12905
  return {
12839
- amountXDeposited: depositResult.totalAmountXDeposited,
12840
- amountYDeposited: depositResult.totalAmountYDeposited,
12841
- actualAmountXDeposited: depositResult.actualTotalAmountXDeposited,
12842
- actualAmountYDeposited: depositResult.actualTotalAmountYDeposited,
12843
- actualAmountXWithdrawn: depositResult.actualLiquidityAndFeeXWithdrawn,
12844
- actualAmountYWithdrawn: depositResult.actualLiquidityAndFeeYWithdrawn,
12845
- rewardAmountsClaimed: withdrawResult.rewardsAmountClaimed,
12846
- withdrawParams,
12847
- depositParams,
12848
- rentalCostLamports
12906
+ base: findX0(amountX, minDeltaId, maxDeltaId, binStep, activeId),
12907
+ delta: new (0, _bnjs2.default)(0)
12849
12908
  };
12850
12909
  }
12851
- totalAmounts() {
12852
- let totalAmountX = new (0, _bnjs2.default)(0);
12853
- let totalAmountY = new (0, _bnjs2.default)(0);
12854
- for (const binData of this.rebalancePositionBinData) {
12855
- totalAmountX = totalAmountX.add(binData.amountX);
12856
- totalAmountY = totalAmountY.add(binData.amountY);
12857
- }
12858
- return [totalAmountX, totalAmountY];
12910
+ findYParameters(amountY, minDeltaId, maxDeltaId, _activeId) {
12911
+ return {
12912
+ base: findY0(amountY, minDeltaId, maxDeltaId),
12913
+ delta: new (0, _bnjs2.default)(0)
12914
+ };
12859
12915
  }
12860
- totalFeeAmounts() {
12861
- let totalFeeXAmount = new (0, _bnjs2.default)(0);
12862
- let totalFeeYAmount = new (0, _bnjs2.default)(0);
12863
- for (const binData of this.rebalancePositionBinData) {
12864
- totalFeeXAmount = totalFeeXAmount.add(binData.claimableFeeXAmount);
12865
- totalFeeYAmount = totalFeeYAmount.add(binData.claimableFeeYAmount);
12866
- }
12867
- return [totalFeeXAmount, totalFeeYAmount];
12916
+ suggestBalancedXParametersFromY(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountY) {
12917
+ const x0 = amountY.div(maxDeltaId.addn(1));
12918
+ const totalAmountX = toAmountIntoBins(
12919
+ activeId,
12920
+ minDeltaId,
12921
+ maxDeltaId,
12922
+ new (0, _bnjs2.default)(0),
12923
+ new (0, _bnjs2.default)(0),
12924
+ x0,
12925
+ new (0, _bnjs2.default)(0),
12926
+ binStep,
12927
+ favorXInActiveBin
12928
+ ).reduce((acc, bin) => {
12929
+ return acc.add(bin.amountX);
12930
+ }, new (0, _bnjs2.default)(0));
12931
+ return {
12932
+ base: new (0, _bnjs2.default)(x0.toString()),
12933
+ delta: new (0, _bnjs2.default)(0),
12934
+ amountX: totalAmountX
12935
+ };
12868
12936
  }
12869
- totalRewardAmounts() {
12870
- let totalRewardAmounts = [new (0, _bnjs2.default)(0), new (0, _bnjs2.default)(0)];
12871
- for (const binData of this.rebalancePositionBinData) {
12872
- totalRewardAmounts[0] = totalRewardAmounts[0].add(
12873
- binData.claimableRewardAmount[0]
12874
- );
12875
- totalRewardAmounts[1] = totalRewardAmounts[1].add(
12876
- binData.claimableRewardAmount[1]
12877
- );
12878
- }
12879
- return totalRewardAmounts;
12937
+ suggestBalancedYParametersFromX(activeId, binStep, favorXInActiveBin, minDeltaId, maxDeltaId, amountXInQuoteValue) {
12938
+ const y0 = amountXInQuoteValue.div(maxDeltaId.sub(minDeltaId).addn(1));
12939
+ const amountY = toAmountIntoBins(
12940
+ activeId,
12941
+ minDeltaId,
12942
+ maxDeltaId,
12943
+ new (0, _bnjs2.default)(0),
12944
+ new (0, _bnjs2.default)(0),
12945
+ new (0, _bnjs2.default)(0),
12946
+ y0,
12947
+ binStep,
12948
+ favorXInActiveBin
12949
+ ).reduce((acc, bin) => {
12950
+ return acc.add(bin.amountY);
12951
+ }, new (0, _bnjs2.default)(0));
12952
+ return {
12953
+ base: y0,
12954
+ delta: new (0, _bnjs2.default)(0),
12955
+ amountY
12956
+ };
12880
12957
  }
12881
12958
  };
12882
- function getPositionWidthWithMinWidth(lowerBinId, upperBinId) {
12883
- const width = upperBinId - lowerBinId + 1;
12884
- return Math.max(width, DEFAULT_BIN_PER_POSITION.toNumber());
12885
- }
12886
- function validateAndSortRebalanceDeposit(deposits) {
12887
- const sortedDeposits = deposits.sort(
12888
- (a, b) => a.minDeltaId.sub(b.minDeltaId).toNumber()
12889
- );
12890
- for (const deposit of deposits) {
12891
- if (deposit.minDeltaId.gte(deposit.maxDeltaId)) {
12892
- throw "Invalid minDeltaId or maxDeltaId";
12893
- }
12959
+
12960
+ // src/dlmm/helpers/rebalance/liquidity_strategy/index.ts
12961
+
12962
+ function getLiquidityStrategyParameterBuilder(strategyType) {
12963
+ switch (strategyType) {
12964
+ case 0 /* Spot */:
12965
+ return new SpotStrategyParameterBuilder();
12966
+ case 1 /* Curve */:
12967
+ return new CurveStrategyParameterBuilder();
12968
+ case 2 /* BidAsk */:
12969
+ return new BidAskStrategyParameterBuilder();
12970
+ default:
12971
+ throw new Error("Strategy not supported");
12894
12972
  }
12895
- for (let i = 1; i < sortedDeposits.length; i++) {
12896
- const prevDeposit = sortedDeposits[i - 1];
12897
- const currDeposit = sortedDeposits[i];
12898
- if (prevDeposit.maxDeltaId.gte(currDeposit.minDeltaId)) {
12899
- throw "Overlap deposit bin range";
12900
- }
12973
+ }
12974
+ function suggestBalancedXParametersFromY(y0, deltaY, minDeltaId, maxDeltaId, activeId, binStep, favorXInActiveBin, builder) {
12975
+ const endDeltaIdBidSide = favorXInActiveBin ? new (0, _bnjs2.default)(-1) : new (0, _bnjs2.default)(0);
12976
+ if (maxDeltaId.lte(endDeltaIdBidSide)) {
12977
+ return {
12978
+ base: new (0, _bnjs2.default)(0),
12979
+ delta: new (0, _bnjs2.default)(0),
12980
+ amountX: new (0, _bnjs2.default)(0)
12981
+ };
12901
12982
  }
12902
- return sortedDeposits;
12983
+ const minYDeltaId = minDeltaId;
12984
+ const maxYDeltaId = endDeltaIdBidSide;
12985
+ const totalAmountY = toAmountIntoBins(
12986
+ activeId,
12987
+ minYDeltaId,
12988
+ maxYDeltaId,
12989
+ new (0, _bnjs2.default)(0),
12990
+ deltaY,
12991
+ new (0, _bnjs2.default)(0),
12992
+ y0,
12993
+ binStep,
12994
+ favorXInActiveBin
12995
+ ).reduce((acc, bin) => {
12996
+ return acc.add(bin.amountY);
12997
+ }, new (0, _bnjs2.default)(0));
12998
+ const minXDeltaId = maxYDeltaId.addn(1);
12999
+ const maxXDeltaId = maxDeltaId;
13000
+ return builder.suggestBalancedXParametersFromY(
13001
+ activeId,
13002
+ binStep,
13003
+ favorXInActiveBin,
13004
+ minXDeltaId,
13005
+ maxXDeltaId,
13006
+ totalAmountY
13007
+ );
12903
13008
  }
12904
- function validateAndSortRebalanceWithdraw(withdraws, activeId) {
12905
- const filledWithdraws = [];
12906
- for (const { minBinId, maxBinId, bps } of withdraws) {
12907
- if (bps.toNumber() < 0 || bps.toNumber() > BASIS_POINT_MAX) {
12908
- throw "Invalid bps";
12909
- }
12910
- const filledMinBinId = _nullishCoalesce(minBinId, () => ( activeId));
12911
- const filledMaxBinId = _nullishCoalesce(maxBinId, () => ( activeId));
12912
- if (filledMinBinId.gt(filledMaxBinId)) {
12913
- throw "Invalid minBinId or maxBinId";
13009
+ function getAutoFillAmountByRebalancedPosition(rebalancePosition, strategyType) {
13010
+ let liquidityInBidSide = new (0, _bnjs2.default)(0);
13011
+ let liquidityInAskSide = new (0, _bnjs2.default)(0);
13012
+ const builder = getLiquidityStrategyParameterBuilder(strategyType);
13013
+ const { lbPair } = rebalancePosition;
13014
+ let favorXInActiveBin = false;
13015
+ let activeIdIndex = -1;
13016
+ for (const [
13017
+ idx,
13018
+ binData
13019
+ ] of rebalancePosition.rebalancePositionBinData.entries()) {
13020
+ const liquidityBid = binData.amountY;
13021
+ const liquidityAsk = new (0, _decimaljs2.default)(binData.price).mul(new (0, _decimaljs2.default)(binData.amountX.toString())).floor().toString();
13022
+ liquidityInBidSide = liquidityInBidSide.add(liquidityBid);
13023
+ liquidityInAskSide = liquidityInAskSide.add(new (0, _bnjs2.default)(liquidityAsk));
13024
+ if (binData.binId == lbPair.activeId) {
13025
+ favorXInActiveBin = binData.amountX.gt(binData.amountY);
13026
+ activeIdIndex = idx;
12914
13027
  }
12915
- filledWithdraws.push({
12916
- minBinId: filledMinBinId,
12917
- maxBinId: filledMaxBinId,
12918
- bps
12919
- });
12920
13028
  }
12921
- filledWithdraws.sort((a, b) => {
12922
- return a.minBinId.sub(b.minBinId).toNumber();
12923
- });
12924
- for (let i = 1; i < filledWithdraws.length; i++) {
12925
- const prev = filledWithdraws[i - 1];
12926
- const curr = filledWithdraws[i];
12927
- if (curr.minBinId.lte(prev.maxBinId)) {
12928
- throw "Overlap withdraw bin range";
13029
+ if (liquidityInAskSide.gt(liquidityInBidSide)) {
13030
+ const minBinId = rebalancePosition.rebalancePositionBinData[0].binId;
13031
+ let maxBinId;
13032
+ if (activeIdIndex == -1) {
13033
+ maxBinId = rebalancePosition.rebalancePositionBinData[rebalancePosition.rebalancePositionBinData.length - 1].binId;
13034
+ } else {
13035
+ maxBinId = rebalancePosition.rebalancePositionBinData[favorXInActiveBin ? activeIdIndex - 1 : activeIdIndex].binId;
13036
+ }
13037
+ const minDeltaId = minBinId - lbPair.activeId;
13038
+ const maxDeltaId = maxBinId - lbPair.activeId;
13039
+ const { amountY } = builder.suggestBalancedYParametersFromX(
13040
+ new (0, _bnjs2.default)(lbPair.activeId),
13041
+ new (0, _bnjs2.default)(lbPair.binStep),
13042
+ favorXInActiveBin,
13043
+ new (0, _bnjs2.default)(minDeltaId),
13044
+ new (0, _bnjs2.default)(maxDeltaId),
13045
+ liquidityInAskSide
13046
+ );
13047
+ return {
13048
+ amount: amountY,
13049
+ isBidSide: true
13050
+ };
13051
+ } else if (liquidityInAskSide.lt(liquidityInBidSide)) {
13052
+ const maxBinId = rebalancePosition.rebalancePositionBinData[rebalancePosition.rebalancePositionBinData.length - 1].binId;
13053
+ let minBinId;
13054
+ if (activeIdIndex == -1) {
13055
+ minBinId = rebalancePosition.rebalancePositionBinData[0].binId;
13056
+ } else {
13057
+ minBinId = rebalancePosition.rebalancePositionBinData[favorXInActiveBin ? activeIdIndex - 1 : activeIdIndex].binId;
12929
13058
  }
13059
+ const minDeltaId = lbPair.activeId - minBinId;
13060
+ const maxDeltaId = lbPair.activeId - maxBinId;
13061
+ const { amountX } = builder.suggestBalancedXParametersFromY(
13062
+ new (0, _bnjs2.default)(lbPair.activeId),
13063
+ new (0, _bnjs2.default)(lbPair.binStep),
13064
+ favorXInActiveBin,
13065
+ new (0, _bnjs2.default)(minDeltaId),
13066
+ new (0, _bnjs2.default)(maxDeltaId),
13067
+ liquidityInBidSide
13068
+ );
13069
+ return {
13070
+ amount: amountX,
13071
+ isBidSide: false
13072
+ };
13073
+ } else {
13074
+ return {
13075
+ amount: new (0, _bnjs2.default)(0),
13076
+ isBidSide: false
13077
+ };
12930
13078
  }
12931
- return filledWithdraws;
12932
13079
  }
12933
- function binRangeToBinIdArray(minBinId, maxBinId) {
12934
- const binIdArray = [];
12935
- const fromBinId = minBinId.toNumber();
12936
- const toBinId = maxBinId.toNumber();
12937
- for (let binId = fromBinId; binId <= toBinId; binId++) {
12938
- binIdArray.push(new (0, _bnjs2.default)(binId));
13080
+ function suggestBalancedYParametersFromX(x0, deltaX, minDeltaId, maxDeltaId, activeId, binStep, favorXInActiveBin, builder) {
13081
+ const startDeltaIdAskSide = favorXInActiveBin ? new (0, _bnjs2.default)(0) : new (0, _bnjs2.default)(1);
13082
+ if (minDeltaId.gte(startDeltaIdAskSide)) {
13083
+ return {
13084
+ base: new (0, _bnjs2.default)(0),
13085
+ delta: new (0, _bnjs2.default)(0),
13086
+ amountY: new (0, _bnjs2.default)(0)
13087
+ };
12939
13088
  }
12940
- return binIdArray;
13089
+ const minXDeltaId = startDeltaIdAskSide;
13090
+ const maxXDeltaId = maxDeltaId;
13091
+ const amountXInBins = toAmountIntoBins(
13092
+ activeId,
13093
+ minXDeltaId,
13094
+ maxXDeltaId,
13095
+ deltaX,
13096
+ new (0, _bnjs2.default)(0),
13097
+ x0,
13098
+ new (0, _bnjs2.default)(0),
13099
+ binStep,
13100
+ favorXInActiveBin
13101
+ );
13102
+ const totalAmountXInQuote = amountXInBins.reduce((acc, bin) => {
13103
+ const price = getPriceOfBinByBinId(
13104
+ bin.binId.toNumber(),
13105
+ binStep.toNumber()
13106
+ );
13107
+ return acc.add(price.mul(new (0, _decimaljs2.default)(bin.amountX.toString())));
13108
+ }, new (0, _decimaljs2.default)(0));
13109
+ const totalAmountXInQuoteBN = new (0, _bnjs2.default)(totalAmountXInQuote.floor().toString());
13110
+ const minYDeltaId = minDeltaId;
13111
+ const maxYDeltaId = startDeltaIdAskSide.subn(1);
13112
+ return builder.suggestBalancedYParametersFromX(
13113
+ activeId,
13114
+ binStep,
13115
+ favorXInActiveBin,
13116
+ minYDeltaId,
13117
+ maxYDeltaId,
13118
+ totalAmountXInQuoteBN
13119
+ );
12941
13120
  }
12942
- function getRebalanceBinArrayIndexesAndBitmapCoverage(adds, removes, activeId, pairAddress, programId) {
12943
- let indexMap = /* @__PURE__ */ new Map();
12944
- removes.forEach((value) => {
12945
- let minBinId = value.minBinId;
12946
- if (minBinId == null) {
12947
- minBinId = activeId;
12948
- }
12949
- let maxBinId = value.maxBinId;
12950
- if (maxBinId == null) {
12951
- maxBinId = activeId;
12952
- }
12953
- let binArrayIndex = binIdToBinArrayIndex(new (0, _bnjs2.default)(minBinId));
12954
- const upperBinId = new (0, _bnjs2.default)(maxBinId);
12955
- while (true) {
12956
- indexMap.set(binArrayIndex.toNumber(), true);
12957
- const [binArrayLowerBinId, binArrayUpperBinId] = getBinArrayLowerUpperBinId(binArrayIndex);
12958
- if (upperBinId.gte(binArrayLowerBinId) && upperBinId.lte(binArrayUpperBinId)) {
12959
- break;
12960
- } else {
12961
- binArrayIndex = binArrayIndex.add(new (0, _bnjs2.default)(1));
12962
- }
12963
- }
12964
- });
12965
- adds.forEach((value) => {
12966
- const minBinId = activeId + value.minDeltaId;
12967
- const maxBinId = activeId + value.maxDeltaId;
12968
- let binArrayIndex = binIdToBinArrayIndex(new (0, _bnjs2.default)(minBinId));
12969
- const upperBinId = new (0, _bnjs2.default)(maxBinId);
12970
- while (true) {
12971
- indexMap.set(binArrayIndex.toNumber(), true);
12972
- const [binArrayLowerBinId, binArrayUpperBinId] = getBinArrayLowerUpperBinId(binArrayIndex);
12973
- if (upperBinId.gte(binArrayLowerBinId) && upperBinId.lte(binArrayUpperBinId)) {
12974
- break;
12975
- } else {
12976
- binArrayIndex = binArrayIndex.add(new (0, _bnjs2.default)(1));
12977
- }
12978
- }
12979
- });
12980
- const binArrayIndexes = Array.from(indexMap.keys()).map((idx) => new (0, _bnjs2.default)(idx));
12981
- const requireBitmapExtension = binArrayIndexes.some(
12982
- (index) => isOverflowDefaultBinArrayBitmap(new (0, _bnjs2.default)(index))
13121
+ function buildLiquidityStrategyParameters(amountX, amountY, minDeltaId, maxDeltaId, binStep, favorXInActiveId, activeId, strategyParameterBuilder) {
13122
+ if (minDeltaId.gt(maxDeltaId)) {
13123
+ return {
13124
+ x0: new (0, _bnjs2.default)(0),
13125
+ y0: new (0, _bnjs2.default)(0),
13126
+ deltaX: new (0, _bnjs2.default)(0),
13127
+ deltaY: new (0, _bnjs2.default)(0)
13128
+ };
13129
+ }
13130
+ const depositOnlyY = maxDeltaId.lt(new (0, _bnjs2.default)(0)) || maxDeltaId.isZero() && !favorXInActiveId;
13131
+ const depositOnlyX = minDeltaId.gt(new (0, _bnjs2.default)(0)) || minDeltaId.isZero() && favorXInActiveId;
13132
+ if (depositOnlyY) {
13133
+ const { base, delta } = strategyParameterBuilder.findYParameters(
13134
+ amountY,
13135
+ minDeltaId,
13136
+ maxDeltaId,
13137
+ activeId
13138
+ );
13139
+ return {
13140
+ x0: new (0, _bnjs2.default)(0),
13141
+ deltaX: new (0, _bnjs2.default)(0),
13142
+ y0: base,
13143
+ deltaY: delta
13144
+ };
13145
+ }
13146
+ if (depositOnlyX) {
13147
+ const { base, delta } = strategyParameterBuilder.findXParameters(
13148
+ amountX,
13149
+ minDeltaId,
13150
+ maxDeltaId,
13151
+ binStep,
13152
+ activeId
13153
+ );
13154
+ return {
13155
+ x0: base,
13156
+ deltaX: delta,
13157
+ y0: new (0, _bnjs2.default)(0),
13158
+ deltaY: new (0, _bnjs2.default)(0)
13159
+ };
13160
+ }
13161
+ const maxDeltaIdBidSide = favorXInActiveId ? new (0, _bnjs2.default)(-1) : new (0, _bnjs2.default)(0);
13162
+ const minDeltaIdAskSide = favorXInActiveId ? new (0, _bnjs2.default)(0) : new (0, _bnjs2.default)(1);
13163
+ const { base: y0, delta: deltaY } = strategyParameterBuilder.findYParameters(
13164
+ amountY,
13165
+ minDeltaId,
13166
+ maxDeltaIdBidSide,
13167
+ activeId
13168
+ );
13169
+ const { base: x0, delta: deltaX } = strategyParameterBuilder.findXParameters(
13170
+ amountX,
13171
+ minDeltaIdAskSide,
13172
+ maxDeltaId,
13173
+ binStep,
13174
+ activeId
12983
13175
  );
12984
13176
  return {
12985
- binArrayIndexes,
12986
- binArrayBitmap: requireBitmapExtension ? deriveBinArrayBitmapExtension(pairAddress, programId)[0] : programId
13177
+ x0,
13178
+ deltaX,
13179
+ y0,
13180
+ deltaY
12987
13181
  };
12988
13182
  }
12989
13183
 
12990
- // src/dlmm/helpers/rebalance/liquidity_strategy/index.ts
12991
-
12992
-
12993
- // src/dlmm/helpers/rebalance/liquidity_strategy/bidAsk.ts
12994
-
12995
- function findMinY0(amountY, minDeltaId, maxDeltaId) {
12996
- const binCount = maxDeltaId.sub(minDeltaId).addn(1);
12997
- const totalWeight = binCount.mul(binCount.addn(1)).divn(2);
12998
- return amountY.div(totalWeight);
13184
+ // src/dlmm/helpers/index.ts
13185
+ function chunks(array, size) {
13186
+ return Array.apply(0, new Array(Math.ceil(array.length / size))).map(
13187
+ (_, index) => array.slice(index * size, (index + 1) * size)
13188
+ );
12999
13189
  }
13000
- function findBaseDeltaY(amountY, minDeltaId, maxDeltaId, minY0) {
13001
- if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0))) {
13002
- return new (0, _bnjs2.default)(0);
13003
- }
13004
- if (minDeltaId.eq(maxDeltaId)) {
13005
- return amountY;
13006
- }
13007
- const m1 = minDeltaId.neg();
13008
- const m2 = maxDeltaId.neg();
13009
- const b = m2.neg().mul(m1.sub(m2).addn(1));
13010
- const c = m1.mul(m1.addn(1)).divn(2);
13011
- const d = m2.mul(m2.subn(1)).divn(2);
13012
- const a = b.add(c.sub(d));
13013
- const e = minY0;
13014
- return amountY.sub(e).div(a);
13190
+ function range(min, max, mapfn) {
13191
+ const length = max - min + 1;
13192
+ return Array.from({ length }, (_, i) => mapfn(min + i));
13015
13193
  }
13016
- function findY0AndDeltaY(amountY, minDeltaId, maxDeltaId, activeId) {
13017
- if (minDeltaId.gt(maxDeltaId) || amountY.isZero()) {
13018
- return {
13019
- base: new (0, _bnjs2.default)(0),
13020
- delta: new (0, _bnjs2.default)(0)
13021
- };
13022
- }
13023
- const minY0 = findMinY0(amountY, minDeltaId, maxDeltaId);
13024
- let baseDeltaY = findBaseDeltaY(amountY, minDeltaId, maxDeltaId, minY0);
13025
- const y0 = baseDeltaY.neg().mul(maxDeltaId).add(minY0);
13026
- while (true) {
13027
- const amountInBins = getAmountInBinsBidSide(
13028
- activeId,
13029
- minDeltaId,
13030
- maxDeltaId,
13031
- baseDeltaY,
13032
- y0
13033
- );
13034
- const totalAmountY = amountInBins.reduce((acc, { amountY: amountY2 }) => {
13035
- return acc.add(amountY2);
13036
- }, new (0, _bnjs2.default)(0));
13037
- if (totalAmountY.gt(amountY)) {
13038
- baseDeltaY = baseDeltaY.sub(new (0, _bnjs2.default)(1));
13194
+ async function chunkedFetchMultiplePoolAccount(program, pks, chunkSize = 100) {
13195
+ const accounts = (await Promise.all(
13196
+ chunks(pks, chunkSize).map(
13197
+ (chunk) => program.account.lbPair.fetchMultiple(chunk)
13198
+ )
13199
+ )).flat();
13200
+ return accounts.filter(Boolean);
13201
+ }
13202
+ async function chunkedFetchMultipleBinArrayBitmapExtensionAccount(program, pks, chunkSize = 100) {
13203
+ const accounts = (await Promise.all(
13204
+ chunks(pks, chunkSize).map(
13205
+ (chunk) => program.account.binArrayBitmapExtension.fetchMultiple(chunk)
13206
+ )
13207
+ )).flat();
13208
+ return accounts;
13209
+ }
13210
+ function getOutAmount(bin, inAmount, swapForY) {
13211
+ return swapForY ? mulShr(inAmount, bin.price, SCALE_OFFSET, 1 /* Down */) : shlDiv(inAmount, bin.price, SCALE_OFFSET, 1 /* Down */);
13212
+ }
13213
+ async function getTokenDecimals(conn, mint) {
13214
+ const token = await _spltoken.getMint.call(void 0, conn, mint);
13215
+ return await token.decimals;
13216
+ }
13217
+ var getOrCreateATAInstruction = async (connection, tokenMint, owner, programId, payer = owner, allowOwnerOffCurve = true) => {
13218
+ programId = _nullishCoalesce(programId, () => ( _spltoken.TOKEN_PROGRAM_ID));
13219
+ const toAccount = _spltoken.getAssociatedTokenAddressSync.call(void 0,
13220
+ tokenMint,
13221
+ owner,
13222
+ allowOwnerOffCurve,
13223
+ programId,
13224
+ _spltoken.ASSOCIATED_TOKEN_PROGRAM_ID
13225
+ );
13226
+ try {
13227
+ await _spltoken.getAccount.call(void 0, connection, toAccount, connection.commitment, programId);
13228
+ return { ataPubKey: toAccount, ix: void 0 };
13229
+ } catch (e) {
13230
+ if (e instanceof _spltoken.TokenAccountNotFoundError || e instanceof _spltoken.TokenInvalidAccountOwnerError) {
13231
+ const ix = _spltoken.createAssociatedTokenAccountIdempotentInstruction.call(void 0,
13232
+ payer,
13233
+ toAccount,
13234
+ owner,
13235
+ tokenMint,
13236
+ programId,
13237
+ _spltoken.ASSOCIATED_TOKEN_PROGRAM_ID
13238
+ );
13239
+ return { ataPubKey: toAccount, ix };
13039
13240
  } else {
13040
- return {
13041
- base: y0,
13042
- delta: baseDeltaY
13043
- };
13241
+ console.error("Error::getOrCreateATAInstruction", e);
13242
+ throw e;
13044
13243
  }
13045
13244
  }
13245
+ };
13246
+ async function getTokenBalance(conn, tokenAccount) {
13247
+ const acc = await _spltoken.getAccount.call(void 0, conn, tokenAccount);
13248
+ return acc.amount;
13046
13249
  }
13047
- function findMinX0(amountX, minDeltaId, maxDeltaId, activeId, binStep) {
13048
- const minBinId = activeId.add(minDeltaId);
13049
- const maxBinId = activeId.add(maxDeltaId);
13050
- let totalWeight = new (0, _bnjs2.default)(0);
13051
- for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
13052
- const binDelta = binId - minBinId.toNumber() + 1;
13053
- const binPrice = getQPriceFromId(new (0, _bnjs2.default)(binId), binStep);
13054
- const weight = new (0, _bnjs2.default)(binDelta).mul(binPrice);
13055
- totalWeight = totalWeight.add(weight);
13056
- }
13057
- return amountX.shln(SCALE_OFFSET).div(totalWeight);
13058
- }
13059
- function findBaseDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId, minX0) {
13060
- if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0))) {
13061
- return new (0, _bnjs2.default)(0);
13250
+ var parseLogs = (eventParser, logs) => {
13251
+ if (!logs.length)
13252
+ throw new Error("No logs found");
13253
+ for (const event of _optionalChain([eventParser, 'optionalAccess', _41 => _41.parseLogs, 'call', _42 => _42(logs)])) {
13254
+ return event.data;
13062
13255
  }
13063
- let b = new (0, _bnjs2.default)(0);
13064
- let c = new (0, _bnjs2.default)(0);
13065
- let m1 = minDeltaId;
13066
- let m2 = maxDeltaId;
13067
- console.log(m1.toNumber(), m2.toNumber());
13068
- for (let m = m1.toNumber(); m <= m2.toNumber(); m++) {
13069
- const binId = activeId.addn(m);
13070
- const pm = getQPriceFromId(binId.neg(), binStep);
13071
- const bDelta = m1.mul(pm);
13072
- b = b.add(bDelta);
13073
- const cDelta = new (0, _bnjs2.default)(m).mul(pm);
13074
- c = c.add(cDelta);
13256
+ throw new Error("No events found");
13257
+ };
13258
+ var wrapSOLInstruction = (from, to, amount) => {
13259
+ return [
13260
+ _web3js.SystemProgram.transfer({
13261
+ fromPubkey: from,
13262
+ toPubkey: to,
13263
+ lamports: amount
13264
+ }),
13265
+ new (0, _web3js.TransactionInstruction)({
13266
+ keys: [
13267
+ {
13268
+ pubkey: to,
13269
+ isSigner: false,
13270
+ isWritable: true
13271
+ }
13272
+ ],
13273
+ data: Buffer.from(new Uint8Array([17])),
13274
+ programId: _spltoken.TOKEN_PROGRAM_ID
13275
+ })
13276
+ ];
13277
+ };
13278
+ var unwrapSOLInstruction = async (owner, allowOwnerOffCurve = true) => {
13279
+ const wSolATAAccount = _spltoken.getAssociatedTokenAddressSync.call(void 0,
13280
+ _spltoken.NATIVE_MINT,
13281
+ owner,
13282
+ allowOwnerOffCurve
13283
+ );
13284
+ if (wSolATAAccount) {
13285
+ const closedWrappedSolInstruction = _spltoken.createCloseAccountInstruction.call(void 0,
13286
+ wSolATAAccount,
13287
+ owner,
13288
+ owner,
13289
+ [],
13290
+ _spltoken.TOKEN_PROGRAM_ID
13291
+ );
13292
+ return closedWrappedSolInstruction;
13075
13293
  }
13076
- console.log(amountX.sub(minX0).shln(SCALE_OFFSET).div(c.sub(b)).toString());
13077
- return amountX.sub(minX0).shln(SCALE_OFFSET).div(c.sub(b));
13294
+ return null;
13295
+ };
13296
+ async function chunkedGetMultipleAccountInfos(connection, pks, chunkSize = 100) {
13297
+ const accountInfos = (await Promise.all(
13298
+ chunks(pks, chunkSize).map(
13299
+ (chunk) => connection.getMultipleAccountsInfo(chunk)
13300
+ )
13301
+ )).flat();
13302
+ return accountInfos;
13078
13303
  }
13079
- function findX0AndDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13080
- if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
13081
- return {
13082
- base: new (0, _bnjs2.default)(0),
13083
- delta: new (0, _bnjs2.default)(0)
13084
- };
13304
+ var getEstimatedComputeUnitUsageWithBuffer = async (connection, instructions, feePayer, buffer) => {
13305
+ if (!buffer) {
13306
+ buffer = 0.1;
13085
13307
  }
13086
- const minX0 = findMinX0(amountX, minDeltaId, maxDeltaId, activeId, binStep);
13087
- let baseDeltaX = findBaseDeltaX(
13088
- amountX,
13089
- minDeltaId,
13090
- maxDeltaId,
13091
- binStep,
13092
- activeId,
13093
- minX0
13308
+ buffer = Math.max(0, buffer);
13309
+ buffer = Math.min(1, buffer);
13310
+ const estimatedComputeUnitUsage = await getSimulationComputeUnits(
13311
+ connection,
13312
+ instructions,
13313
+ feePayer,
13314
+ []
13094
13315
  );
13095
- console.log("\u{1F680} ~ baseDeltaX:", baseDeltaX.toString());
13096
- const x0 = minDeltaId.neg().mul(baseDeltaX).add(minX0);
13097
- console.log("\u{1F680} ~ x0:", x0.toString());
13098
- while (true) {
13099
- const amountInBins = getAmountInBinsAskSide(
13100
- activeId,
13101
- binStep,
13102
- minDeltaId,
13103
- maxDeltaId,
13104
- baseDeltaX,
13105
- x0
13106
- );
13107
- const totalAmountX = amountInBins.reduce((acc, { amountX: amountX2 }) => {
13108
- return acc.add(amountX2);
13109
- }, new (0, _bnjs2.default)(0));
13110
- if (totalAmountX.gt(amountX)) {
13111
- baseDeltaX = baseDeltaX.sub(new (0, _bnjs2.default)(1));
13112
- } else {
13113
- return {
13114
- base: x0,
13115
- delta: baseDeltaX
13116
- };
13117
- }
13316
+ let extraComputeUnitBuffer = estimatedComputeUnitUsage * buffer;
13317
+ if (extraComputeUnitBuffer > MAX_CU_BUFFER) {
13318
+ extraComputeUnitBuffer = MAX_CU_BUFFER;
13319
+ } else if (extraComputeUnitBuffer < MIN_CU_BUFFER) {
13320
+ extraComputeUnitBuffer = MIN_CU_BUFFER;
13118
13321
  }
13322
+ return estimatedComputeUnitUsage + extraComputeUnitBuffer;
13323
+ };
13324
+ var getEstimatedComputeUnitIxWithBuffer = async (connection, instructions, feePayer, buffer) => {
13325
+ const units = await getEstimatedComputeUnitUsageWithBuffer(
13326
+ connection,
13327
+ instructions,
13328
+ feePayer,
13329
+ buffer
13330
+ ).catch((error) => {
13331
+ console.error("Error::getEstimatedComputeUnitUsageWithBuffer", error);
13332
+ return 14e5;
13333
+ });
13334
+ return _web3js.ComputeBudgetProgram.setComputeUnitLimit({ units });
13335
+ };
13336
+ function createProgram(connection, opt) {
13337
+ const cluster = _optionalChain([opt, 'optionalAccess', _43 => _43.cluster]) || "mainnet-beta";
13338
+ const provider = new (0, _anchor.AnchorProvider)(
13339
+ connection,
13340
+ {},
13341
+ _anchor.AnchorProvider.defaultOptions()
13342
+ );
13343
+ return new (0, _anchor.Program)(
13344
+ { ...dlmm_default, address: LBCLMM_PROGRAM_IDS[cluster] },
13345
+ provider
13346
+ );
13119
13347
  }
13120
- var BidAskStrategyParameterBuilder = class {
13121
- findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13122
- return findX0AndDeltaX(amountX, minDeltaId, maxDeltaId, binStep, activeId);
13348
+ function decodeAccount(program, accountName, buffer) {
13349
+ return program.coder.accounts.decode(accountName, buffer);
13350
+ }
13351
+ function getAccountDiscriminator(accountName) {
13352
+ return _optionalChain([dlmm_default, 'access', _44 => _44.accounts, 'access', _45 => _45.find, 'call', _46 => _46(
13353
+ (acc) => acc.name.toLowerCase() === accountName.toLowerCase()
13354
+ ), 'optionalAccess', _47 => _47.discriminator]);
13355
+ }
13356
+ function capSlippagePercentage(slippage) {
13357
+ if (slippage > 100) {
13358
+ slippage = 100;
13123
13359
  }
13124
- findYParameters(amountY, minDeltaId, maxDeltaId, activeId) {
13125
- return findY0AndDeltaY(amountY, minDeltaId, maxDeltaId, activeId);
13360
+ if (slippage < 0) {
13361
+ slippage = 0;
13126
13362
  }
13363
+ return slippage;
13364
+ }
13365
+
13366
+ // src/dlmm/helpers/accountFilters.ts
13367
+ var _bytes = require('@coral-xyz/anchor/dist/cjs/utils/bytes');
13368
+ var presetParameter2BinStepFilter = (binStep) => {
13369
+ return {
13370
+ memcmp: {
13371
+ bytes: _bytes.bs58.encode(binStep.toArrayLike(Buffer, "le", 2)),
13372
+ offset: 8
13373
+ }
13374
+ };
13375
+ };
13376
+ var presetParameter2BaseFactorFilter = (baseFactor) => {
13377
+ return {
13378
+ memcmp: {
13379
+ bytes: _bytes.bs58.encode(baseFactor.toArrayLike(Buffer, "le", 2)),
13380
+ offset: 8 + 2
13381
+ }
13382
+ };
13383
+ };
13384
+ var presetParameter2BaseFeePowerFactor = (baseFeePowerFactor) => {
13385
+ return {
13386
+ memcmp: {
13387
+ bytes: _bytes.bs58.encode(baseFeePowerFactor.toArrayLike(Buffer, "le", 1)),
13388
+ offset: 8 + 22
13389
+ }
13390
+ };
13391
+ };
13392
+ var binArrayLbPairFilter = (lbPair) => {
13393
+ return {
13394
+ memcmp: {
13395
+ bytes: lbPair.toBase58(),
13396
+ offset: 8 + 16
13397
+ }
13398
+ };
13399
+ };
13400
+ var positionOwnerFilter = (owner) => {
13401
+ return {
13402
+ memcmp: {
13403
+ bytes: owner.toBase58(),
13404
+ offset: 8 + 32
13405
+ }
13406
+ };
13407
+ };
13408
+ var positionLbPairFilter = (lbPair) => {
13409
+ return {
13410
+ memcmp: {
13411
+ bytes: _bytes.bs58.encode(lbPair.toBuffer()),
13412
+ offset: 8
13413
+ }
13414
+ };
13415
+ };
13416
+ var positionV2Filter = () => {
13417
+ return {
13418
+ memcmp: {
13419
+ bytes: _bytes.bs58.encode(Buffer.from(getAccountDiscriminator("positionV2"))),
13420
+ offset: 0
13421
+ }
13422
+ };
13127
13423
  };
13128
13424
 
13129
- // src/dlmm/helpers/rebalance/liquidity_strategy/curve.ts
13425
+ // src/dlmm/helpers/positions/index.ts
13130
13426
 
13131
- function findBaseY0(amountY, minDeltaId, maxDeltaId) {
13132
- if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0))) {
13133
- return new (0, _bnjs2.default)(0);
13134
- }
13135
- if (minDeltaId.eq(maxDeltaId)) {
13136
- return amountY;
13427
+
13428
+ // src/dlmm/helpers/positions/wrapper.ts
13429
+
13430
+ function combineBaseAndExtendedPositionBinData(base, extended) {
13431
+ const combinedLiquidityShares = base.liquidityShares;
13432
+ const combinedRewardInfos = base.rewardInfos;
13433
+ const combinedFeeInfos = base.feeInfos;
13434
+ for (const binData of extended) {
13435
+ combinedLiquidityShares.push(binData.liquidityShare);
13436
+ combinedRewardInfos.push(binData.rewardInfo);
13437
+ combinedFeeInfos.push(binData.feeInfo);
13137
13438
  }
13138
- const m1 = minDeltaId.neg();
13139
- const m2 = maxDeltaId.neg();
13140
- const b = m1.sub(m2).addn(1);
13141
- const c = m1.mul(m1.addn(1)).divn(2);
13142
- const d = m2.mul(m2.subn(1)).divn(2);
13143
- const a = b.sub(c.sub(d).div(m1.addn(1)));
13144
- return amountY.div(a);
13439
+ return {
13440
+ liquidityShares: combinedLiquidityShares,
13441
+ rewardInfos: combinedRewardInfos,
13442
+ feeInfos: combinedFeeInfos
13443
+ };
13145
13444
  }
13146
- function findY0AndDeltaY2(amountY, minDeltaId, maxDeltaId, activeId) {
13147
- if (minDeltaId.gt(maxDeltaId) || amountY.isZero()) {
13148
- return {
13149
- base: new (0, _bnjs2.default)(0),
13150
- delta: new (0, _bnjs2.default)(0)
13151
- };
13152
- }
13153
- let baseY0 = findBaseY0(amountY, minDeltaId, maxDeltaId);
13154
- while (true) {
13155
- const deltaY = baseY0.neg().div(minDeltaId.neg().addn(1));
13156
- const amountInBins = getAmountInBinsBidSide(
13157
- activeId,
13158
- minDeltaId,
13159
- maxDeltaId,
13160
- deltaY,
13161
- baseY0
13445
+ function wrapPosition(program, key, account) {
13446
+ const disc = account.data.subarray(0, 8);
13447
+ if (disc.equals(Buffer.from(getAccountDiscriminator("positionV2")))) {
13448
+ const state = decodeAccount(
13449
+ program,
13450
+ "positionV2",
13451
+ account.data
13162
13452
  );
13163
- const totalAmountY = amountInBins.reduce((acc, { amountY: amountY2 }) => {
13164
- return acc.add(amountY2);
13165
- }, new (0, _bnjs2.default)(0));
13166
- if (totalAmountY.gt(amountY)) {
13167
- baseY0 = baseY0.sub(new (0, _bnjs2.default)(1));
13168
- } else {
13169
- return {
13170
- base: baseY0,
13171
- delta: deltaY
13172
- };
13173
- }
13453
+ const extended = decodeExtendedPosition(
13454
+ state,
13455
+ program,
13456
+ account.data.subarray(8 + POSITION_MIN_SIZE)
13457
+ );
13458
+ const combinedPositionBinData = combineBaseAndExtendedPositionBinData(
13459
+ state,
13460
+ extended
13461
+ );
13462
+ return new PositionV2Wrapper(key, state, extended, combinedPositionBinData);
13463
+ } else {
13464
+ throw new Error("Unknown position account");
13174
13465
  }
13175
13466
  }
13176
- function findBaseX0(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13177
- if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0))) {
13178
- return new (0, _bnjs2.default)(0);
13467
+ var PositionV2Wrapper = class {
13468
+ constructor(positionAddress, inner, extended, combinedPositionBinData) {
13469
+ this.positionAddress = positionAddress;
13470
+ this.inner = inner;
13471
+ this.extended = extended;
13472
+ this.combinedPositionBinData = combinedPositionBinData;
13179
13473
  }
13180
- let b = new (0, _bnjs2.default)(0);
13181
- let c = new (0, _bnjs2.default)(0);
13182
- let m1 = minDeltaId;
13183
- let m2 = maxDeltaId;
13184
- for (let m = m1.toNumber(); m <= m2.toNumber(); m++) {
13185
- const binId = activeId.addn(m);
13186
- const pm = getQPriceFromId(binId.neg(), binStep);
13187
- b = b.add(pm);
13188
- const cDelta = new (0, _bnjs2.default)(m).mul(pm).div(m2);
13189
- c = c.add(cDelta);
13474
+ address() {
13475
+ return this.positionAddress;
13190
13476
  }
13191
- return amountX.shln(SCALE_OFFSET).div(b.sub(c));
13192
- }
13193
- function findX0AndDeltaX2(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13194
- if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
13195
- return {
13196
- base: new (0, _bnjs2.default)(0),
13197
- delta: new (0, _bnjs2.default)(0)
13198
- };
13477
+ totalClaimedRewards() {
13478
+ return this.inner.totalClaimedRewards;
13199
13479
  }
13200
- let baseX0 = findBaseX0(amountX, minDeltaId, maxDeltaId, binStep, activeId);
13201
- const deltaX = baseX0.neg().div(maxDeltaId);
13202
- while (true) {
13203
- const amountInBins = getAmountInBinsAskSide(
13204
- activeId,
13205
- binStep,
13206
- minDeltaId,
13207
- maxDeltaId,
13208
- deltaX,
13209
- baseX0
13210
- );
13211
- const totalAmountX = amountInBins.reduce((acc, { amountX: amountX2 }) => {
13212
- return acc.add(amountX2);
13213
- }, new (0, _bnjs2.default)(0));
13214
- if (totalAmountX.gt(amountX)) {
13215
- baseX0 = baseX0.sub(new (0, _bnjs2.default)(1));
13216
- } else {
13217
- return {
13218
- base: baseX0,
13219
- delta: deltaX
13220
- };
13221
- }
13480
+ feeOwner() {
13481
+ return this.inner.feeOwner;
13222
13482
  }
13223
- }
13224
- var CurveStrategyParameterBuilder = class {
13225
- findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13226
- return findX0AndDeltaX2(amountX, minDeltaId, maxDeltaId, binStep, activeId);
13483
+ lockReleasePoint() {
13484
+ return this.inner.lockReleasePoint;
13227
13485
  }
13228
- findYParameters(amountY, minDeltaId, maxDeltaId, activeId) {
13229
- return findY0AndDeltaY2(amountY, minDeltaId, maxDeltaId, activeId);
13486
+ operator() {
13487
+ return this.inner.operator;
13230
13488
  }
13231
- };
13232
-
13233
- // src/dlmm/helpers/rebalance/liquidity_strategy/spot.ts
13234
-
13235
- function findY0(amountY, minDeltaId, maxDeltaId) {
13236
- if (minDeltaId.gt(maxDeltaId) || amountY.lte(new (0, _bnjs2.default)(0)) || amountY.isZero()) {
13237
- return new (0, _bnjs2.default)(0);
13489
+ totalClaimedFeeYAmount() {
13490
+ return this.inner.totalClaimedFeeYAmount;
13238
13491
  }
13239
- const m1 = minDeltaId.neg();
13240
- const m2 = maxDeltaId.neg();
13241
- const delta = m1.sub(m2).addn(1);
13242
- return amountY.div(delta);
13243
- }
13244
- function findBaseX02(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13245
- let totalWeight = new (0, _bnjs2.default)(0);
13246
- const minBinId = activeId.add(minDeltaId);
13247
- const maxBinId = activeId.add(maxDeltaId);
13248
- let baseFactor = getQPriceBaseFactor(binStep);
13249
- let basePrice = getQPriceFromId(maxBinId.neg(), binStep);
13250
- for (let binId = minBinId.toNumber(); binId <= maxBinId.toNumber(); binId++) {
13251
- totalWeight = totalWeight.add(basePrice);
13252
- basePrice = basePrice.mul(baseFactor).shrn(SCALE_OFFSET);
13492
+ totalClaimedFeeXAmount() {
13493
+ return this.inner.totalClaimedFeeXAmount;
13494
+ }
13495
+ lbPair() {
13496
+ return this.inner.lbPair;
13497
+ }
13498
+ lowerBinId() {
13499
+ return new (0, _bnjs2.default)(this.inner.lowerBinId);
13253
13500
  }
13254
- return amountX.shln(SCALE_OFFSET).div(totalWeight);
13255
- }
13256
- function findX0(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13257
- if (minDeltaId.gt(maxDeltaId) || amountX.lte(new (0, _bnjs2.default)(0)) || amountX.isZero()) {
13258
- return new (0, _bnjs2.default)(0);
13501
+ upperBinId() {
13502
+ return new (0, _bnjs2.default)(this.inner.upperBinId);
13259
13503
  }
13260
- let x0 = findBaseX02(amountX, minDeltaId, maxDeltaId, binStep, activeId);
13261
- while (true) {
13262
- const amountInBins = getAmountInBinsAskSide(
13263
- activeId,
13264
- binStep,
13265
- minDeltaId,
13266
- maxDeltaId,
13267
- new (0, _bnjs2.default)(0),
13268
- x0
13269
- );
13270
- const totalAmountX = amountInBins.reduce((acc, bin) => {
13271
- return acc.add(bin.amountX);
13272
- }, new (0, _bnjs2.default)(0));
13273
- if (totalAmountX.lt(amountX)) {
13274
- x0 = x0.add(new (0, _bnjs2.default)(1));
13504
+ liquidityShares() {
13505
+ return this.combinedPositionBinData.liquidityShares;
13506
+ }
13507
+ rewardInfos() {
13508
+ return this.combinedPositionBinData.rewardInfos;
13509
+ }
13510
+ feeInfos() {
13511
+ return this.combinedPositionBinData.feeInfos;
13512
+ }
13513
+ lastUpdatedAt() {
13514
+ return this.inner.lastUpdatedAt;
13515
+ }
13516
+ getBinArrayIndexesCoverage() {
13517
+ const isExtended = this.extended.length > 0;
13518
+ if (isExtended) {
13519
+ return getBinArrayIndexesCoverage(this.lowerBinId(), this.upperBinId());
13275
13520
  } else {
13276
- x0 = x0.sub(new (0, _bnjs2.default)(1));
13277
- return x0;
13521
+ const lowerBinArrayIndex = binIdToBinArrayIndex(this.lowerBinId());
13522
+ const upperBinArrayIndex = lowerBinArrayIndex.add(new (0, _bnjs2.default)(1));
13523
+ return [lowerBinArrayIndex, upperBinArrayIndex];
13278
13524
  }
13279
13525
  }
13280
- }
13281
- var SpotStrategyParameterBuilder = class {
13282
- findXParameters(amountX, minDeltaId, maxDeltaId, binStep, activeId) {
13283
- return {
13284
- base: findX0(amountX, minDeltaId, maxDeltaId, binStep, activeId),
13285
- delta: new (0, _bnjs2.default)(0)
13286
- };
13526
+ getBinArrayKeysCoverage(programId) {
13527
+ return this.getBinArrayIndexesCoverage().map(
13528
+ (index) => deriveBinArray(this.lbPair(), index, programId)[0]
13529
+ );
13287
13530
  }
13288
- findYParameters(amountY, minDeltaId, maxDeltaId, _activeId) {
13289
- return {
13290
- base: findY0(amountY, minDeltaId, maxDeltaId),
13291
- delta: new (0, _bnjs2.default)(0)
13292
- };
13531
+ version() {
13532
+ return 1 /* V2 */;
13533
+ }
13534
+ owner() {
13535
+ return this.inner.owner;
13536
+ }
13537
+ width() {
13538
+ return this.upperBinId().sub(this.lowerBinId()).add(new (0, _bnjs2.default)(1));
13293
13539
  }
13294
13540
  };
13295
13541
 
13296
- // src/dlmm/helpers/rebalance/liquidity_strategy/index.ts
13297
- function getLiquidityStrategyParameterBuilder(strategyType) {
13298
- switch (strategyType) {
13299
- case 0 /* Spot */:
13300
- return new SpotStrategyParameterBuilder();
13301
- case 1 /* Curve */:
13302
- return new CurveStrategyParameterBuilder();
13303
- case 2 /* BidAsk */:
13304
- return new BidAskStrategyParameterBuilder();
13305
- default:
13306
- throw new Error("Strategy not supported");
13542
+ // src/dlmm/helpers/positions/index.ts
13543
+ function getBinArrayIndexesCoverage(lowerBinId, upperBinId) {
13544
+ const lowerBinArrayIndex = binIdToBinArrayIndex(lowerBinId);
13545
+ const upperBinArrayIndex = binIdToBinArrayIndex(upperBinId);
13546
+ const binArrayIndexes = [];
13547
+ for (let i = lowerBinArrayIndex.toNumber(); i <= upperBinArrayIndex.toNumber(); i++) {
13548
+ binArrayIndexes.push(new (0, _bnjs2.default)(i));
13307
13549
  }
13550
+ return binArrayIndexes;
13308
13551
  }
13309
- function buildLiquidityStrategyParameters(amountX, amountY, minDeltaId, maxDeltaId, binStep, favorXInActiveId, activeId, strategyParameterBuilder) {
13310
- if (minDeltaId.gt(maxDeltaId)) {
13311
- return {
13312
- x0: new (0, _bnjs2.default)(0),
13313
- y0: new (0, _bnjs2.default)(0),
13314
- deltaX: new (0, _bnjs2.default)(0),
13315
- deltaY: new (0, _bnjs2.default)(0)
13316
- };
13552
+ function getBinArrayKeysCoverage2(lowerBinId, upperBinId, lbPair, programId) {
13553
+ const binArrayIndexes = getBinArrayIndexesCoverage(lowerBinId, upperBinId);
13554
+ return binArrayIndexes.map((index) => {
13555
+ return deriveBinArray(lbPair, index, programId)[0];
13556
+ });
13557
+ }
13558
+ function getBinArrayAccountMetasCoverage(lowerBinId, upperBinId, lbPair, programId) {
13559
+ return getBinArrayKeysCoverage2(lowerBinId, upperBinId, lbPair, programId).map(
13560
+ (key) => {
13561
+ return {
13562
+ pubkey: key,
13563
+ isSigner: false,
13564
+ isWritable: true
13565
+ };
13566
+ }
13567
+ );
13568
+ }
13569
+ function getPositionLowerUpperBinIdWithLiquidity(position) {
13570
+ const binWithLiquidity = position.positionBinData.filter(
13571
+ (b) => !new (0, _bnjs2.default)(b.binLiquidity).isZero() || !new (0, _bnjs2.default)(b.positionFeeXAmount.toString()).isZero() || !new (0, _bnjs2.default)(b.positionFeeYAmount.toString()).isZero() || !new (0, _bnjs2.default)(b.positionRewardAmount[0].toString()).isZero() || !new (0, _bnjs2.default)(b.positionRewardAmount[1].toString()).isZero()
13572
+ );
13573
+ return binWithLiquidity.length > 0 ? {
13574
+ lowerBinId: new (0, _bnjs2.default)(binWithLiquidity[0].binId),
13575
+ upperBinId: new (0, _bnjs2.default)(binWithLiquidity[binWithLiquidity.length - 1].binId)
13576
+ } : null;
13577
+ }
13578
+ function isPositionNoFee(position) {
13579
+ return position.feeX.isZero() && position.feeY.isZero();
13580
+ }
13581
+ function isPositionNoReward(position) {
13582
+ return position.rewardOne.isZero() && position.rewardTwo.isZero();
13583
+ }
13584
+ function chunkBinRange(minBinId, maxBinId) {
13585
+ const chunkedBinRange = [];
13586
+ let startBinId = minBinId;
13587
+ while (startBinId <= maxBinId) {
13588
+ const endBinId = Math.min(
13589
+ startBinId + DEFAULT_BIN_PER_POSITION.toNumber() - 1,
13590
+ maxBinId
13591
+ );
13592
+ chunkedBinRange.push({
13593
+ lowerBinId: startBinId,
13594
+ upperBinId: endBinId
13595
+ });
13596
+ startBinId += DEFAULT_BIN_PER_POSITION.toNumber();
13317
13597
  }
13318
- const depositOnlyY = maxDeltaId.lt(new (0, _bnjs2.default)(0)) || maxDeltaId.isZero() && !favorXInActiveId;
13319
- const depositOnlyX = minDeltaId.gt(new (0, _bnjs2.default)(0)) || minDeltaId.isZero() && favorXInActiveId;
13320
- if (depositOnlyY) {
13321
- const { base, delta } = strategyParameterBuilder.findYParameters(
13322
- amountY,
13323
- minDeltaId,
13324
- maxDeltaId,
13325
- activeId
13598
+ return chunkedBinRange;
13599
+ }
13600
+ async function getPositionExpandRentExemption(currentMinBinId, currentMaxBinId, connection, binCountToExpand) {
13601
+ const currentPositionWidth = currentMaxBinId.sub(currentMinBinId).addn(1);
13602
+ const positionWidthAfterExpand = currentPositionWidth.add(binCountToExpand);
13603
+ if (positionWidthAfterExpand.lte(DEFAULT_BIN_PER_POSITION)) {
13604
+ return 0;
13605
+ } else {
13606
+ const binCountInExpandedBytes = positionWidthAfterExpand.sub(
13607
+ DEFAULT_BIN_PER_POSITION
13326
13608
  );
13327
- return {
13328
- x0: new (0, _bnjs2.default)(0),
13329
- deltaX: new (0, _bnjs2.default)(0),
13330
- y0: base,
13331
- deltaY: delta
13332
- };
13609
+ const expandSize = binCountInExpandedBytes.toNumber() * POSITION_BIN_DATA_SIZE;
13610
+ const [minimumLamports, rentExemptionLamports] = await Promise.all([
13611
+ connection.getMinimumBalanceForRentExemption(0),
13612
+ connection.getMinimumBalanceForRentExemption(expandSize)
13613
+ ]);
13614
+ return rentExemptionLamports - minimumLamports;
13333
13615
  }
13334
- if (depositOnlyX) {
13335
- const { base, delta } = strategyParameterBuilder.findXParameters(
13336
- amountX,
13337
- minDeltaId,
13338
- maxDeltaId,
13339
- binStep,
13340
- activeId
13616
+ }
13617
+ function getExtendedPositionBinCount(minBinId, maxBinId) {
13618
+ const width = maxBinId.sub(minBinId).addn(1);
13619
+ const extended = width.sub(DEFAULT_BIN_PER_POSITION);
13620
+ return extended.lte(new (0, _bnjs2.default)(0)) ? new (0, _bnjs2.default)(0) : extended;
13621
+ }
13622
+ function decodeExtendedPosition(base, program, bytes) {
13623
+ const width = base.upperBinId - base.lowerBinId + 1;
13624
+ const extendedWidth = width - DEFAULT_BIN_PER_POSITION.toNumber();
13625
+ const extendedPosition = [];
13626
+ for (let i = 0; i < extendedWidth; i++) {
13627
+ const offset = i * POSITION_BIN_DATA_SIZE;
13628
+ const data = bytes.subarray(offset, offset + POSITION_BIN_DATA_SIZE);
13629
+ const decodedPositionBinData = program.coder.types.decode(
13630
+ // TODO: Find a type safe way
13631
+ "positionBinData",
13632
+ data
13341
13633
  );
13342
- return {
13343
- x0: base,
13344
- deltaX: delta,
13345
- y0: new (0, _bnjs2.default)(0),
13346
- deltaY: new (0, _bnjs2.default)(0)
13347
- };
13634
+ extendedPosition.push(decodedPositionBinData);
13348
13635
  }
13349
- const maxDeltaIdBidSide = favorXInActiveId ? new (0, _bnjs2.default)(-1) : new (0, _bnjs2.default)(0);
13350
- const minDeltaIdAskSide = favorXInActiveId ? new (0, _bnjs2.default)(0) : new (0, _bnjs2.default)(1);
13351
- const { base: y0, delta: deltaY } = strategyParameterBuilder.findYParameters(
13352
- amountY,
13353
- minDeltaId,
13354
- maxDeltaIdBidSide,
13355
- activeId
13356
- );
13357
- const { base: x0, delta: deltaX } = strategyParameterBuilder.findXParameters(
13358
- amountX,
13359
- minDeltaIdAskSide,
13360
- maxDeltaId,
13361
- binStep,
13362
- activeId
13363
- );
13364
- return {
13365
- x0,
13366
- deltaX,
13367
- y0,
13368
- deltaY
13369
- };
13636
+ return extendedPosition;
13370
13637
  }
13371
13638
 
13372
13639
  // src/dlmm/helpers/rebalance/strategy/balanced.ts
@@ -18158,7 +18425,6 @@ var DLMM = class {
18158
18425
  shouldClaimReward: true,
18159
18426
  pairAddress: this.pubkey
18160
18427
  });
18161
- console.log("\u{1F680} ~ DLMM ~ rebalancePosition:", rebalancePosition);
18162
18428
  const rebalanceStrategyBuilder = new BalancedStrategyBuilder(
18163
18429
  new (0, _anchor.BN)(rebalancePosition.lbPair.activeId),
18164
18430
  new (0, _anchor.BN)(rebalancePosition.lbPair.binStep),
@@ -18184,10 +18450,65 @@ var DLMM = class {
18184
18450
  withdraws,
18185
18451
  deposits
18186
18452
  );
18187
- console.log("\u{1F680} ~ DLMM ~ simulationResult:", simulationResult);
18453
+ const binArrayQuoteResult = await this.quoteBinArrayAccountsRentalCost(
18454
+ simulationResult.depositParams,
18455
+ simulationResult.withdrawParams,
18456
+ new (0, _anchor.BN)(rebalancePosition.lbPair.activeId)
18457
+ );
18188
18458
  return {
18189
18459
  rebalancePosition,
18190
- simulationResult
18460
+ simulationResult,
18461
+ ...binArrayQuoteResult
18462
+ };
18463
+ }
18464
+ async quoteBinArrayAccountsRentalCost(deposits, withdraws, activeId) {
18465
+ const { binArrayBitmap, binArrayIndexes } = getRebalanceBinArrayIndexesAndBitmapCoverage(
18466
+ deposits,
18467
+ withdraws,
18468
+ activeId.toNumber(),
18469
+ this.pubkey,
18470
+ this.program.programId
18471
+ );
18472
+ const binArrayPublicKeys = binArrayIndexes.map((index) => {
18473
+ const [binArrayPubkey] = deriveBinArray(
18474
+ this.pubkey,
18475
+ index,
18476
+ this.program.programId
18477
+ );
18478
+ return binArrayPubkey;
18479
+ });
18480
+ const accountPublicKeys = [...binArrayPublicKeys];
18481
+ if (!binArrayBitmap.equals(_web3js.PublicKey.default)) {
18482
+ accountPublicKeys.push(binArrayBitmap);
18483
+ }
18484
+ const accounts = await chunkedGetMultipleAccountInfos(
18485
+ this.program.provider.connection,
18486
+ binArrayPublicKeys
18487
+ );
18488
+ const binArrayAccounts = accounts.splice(0, binArrayPublicKeys.length);
18489
+ let binArrayCount = 0;
18490
+ let bitmapExtensionCost = 0;
18491
+ const binArraySet = /* @__PURE__ */ new Set();
18492
+ for (let i = 0; i < binArrayAccounts.length; i++) {
18493
+ const binArrayAccount = binArrayAccounts[i];
18494
+ const binArrayPubkey = binArrayPublicKeys[i];
18495
+ if (!binArrayAccount) {
18496
+ binArrayCount++;
18497
+ } else {
18498
+ binArraySet.add(binArrayPubkey.toBase58());
18499
+ }
18500
+ }
18501
+ if (!binArrayBitmap.equals(_web3js.PublicKey.default)) {
18502
+ const bitmapAccount = accounts.pop();
18503
+ if (!bitmapAccount) {
18504
+ bitmapExtensionCost = BIN_ARRAY_BITMAP_FEE;
18505
+ }
18506
+ }
18507
+ return {
18508
+ binArrayCost: binArrayCount * BIN_ARRAY_FEE,
18509
+ binArrayCount,
18510
+ binArrayExistence: binArraySet,
18511
+ bitmapExtensionCost
18191
18512
  };
18192
18513
  }
18193
18514
  /**
@@ -18217,9 +18538,15 @@ var DLMM = class {
18217
18538
  withdraws,
18218
18539
  deposits
18219
18540
  );
18541
+ const binArrayQuoteResult = await this.quoteBinArrayAccountsRentalCost(
18542
+ simulationResult.depositParams,
18543
+ simulationResult.withdrawParams,
18544
+ new (0, _anchor.BN)(rebalancePosition.lbPair.activeId)
18545
+ );
18220
18546
  return {
18221
18547
  rebalancePosition,
18222
- simulationResult
18548
+ simulationResult,
18549
+ ...binArrayQuoteResult
18223
18550
  };
18224
18551
  }
18225
18552
  /**
@@ -18297,12 +18624,10 @@ var DLMM = class {
18297
18624
  chunk,
18298
18625
  activeId
18299
18626
  );
18300
- console.log("\u{1F680} ~ DLMM ~ chunkedDepositParams:", chunkedDepositParams);
18301
18627
  const chunkedWithdrawParams = splitWithdrawParamsForChunk(
18302
18628
  withdrawParams,
18303
18629
  chunk
18304
18630
  );
18305
- console.log("\u{1F680} ~ DLMM ~ chunkedWithdrawParams:", chunkedWithdrawParams);
18306
18631
  if (chunkedDepositParams.length === 0 && chunkedWithdrawParams.length === 0)
18307
18632
  continue;
18308
18633
  const { slices, accounts: transferHookAccounts } = this.getPotentialToken2022IxDataAndAccounts(0 /* Liquidity */);
@@ -19299,5 +19624,15 @@ var src_default = DLMM;
19299
19624
 
19300
19625
 
19301
19626
 
19302
- exports.ADMIN = ADMIN; exports.ActionType = ActionType; exports.ActivationType = ActivationType; exports.BASIS_POINT_MAX = BASIS_POINT_MAX; exports.BIN_ARRAY_BITMAP_FEE = BIN_ARRAY_BITMAP_FEE; exports.BIN_ARRAY_BITMAP_FEE_BN = BIN_ARRAY_BITMAP_FEE_BN; exports.BIN_ARRAY_BITMAP_SIZE = BIN_ARRAY_BITMAP_SIZE; exports.BIN_ARRAY_FEE = BIN_ARRAY_FEE; exports.BIN_ARRAY_FEE_BN = BIN_ARRAY_FEE_BN; exports.BinLiquidity = BinLiquidity; exports.BitmapType = BitmapType; exports.ClockLayout = ClockLayout; exports.DEFAULT_BIN_PER_POSITION = DEFAULT_BIN_PER_POSITION; exports.DLMMError = DLMMError; exports.DlmmSdkError = DlmmSdkError; exports.EXTENSION_BINARRAY_BITMAP_SIZE = EXTENSION_BINARRAY_BITMAP_SIZE; exports.FEE_PRECISION = FEE_PRECISION; exports.IDL = dlmm_default; exports.ILM_BASE = ILM_BASE; exports.LBCLMM_PROGRAM_IDS = LBCLMM_PROGRAM_IDS; exports.MAX_ACTIVE_BIN_SLIPPAGE = MAX_ACTIVE_BIN_SLIPPAGE; exports.MAX_BINS_PER_POSITION = MAX_BINS_PER_POSITION; exports.MAX_BIN_ARRAY_SIZE = MAX_BIN_ARRAY_SIZE; exports.MAX_BIN_LENGTH_ALLOWED_IN_ONE_TX = MAX_BIN_LENGTH_ALLOWED_IN_ONE_TX; exports.MAX_CLAIM_ALL_ALLOWED = MAX_CLAIM_ALL_ALLOWED; exports.MAX_EXTRA_BIN_ARRAYS = MAX_EXTRA_BIN_ARRAYS; exports.MAX_FEE_RATE = MAX_FEE_RATE; exports.MAX_RESIZE_LENGTH = MAX_RESIZE_LENGTH; exports.MEMO_PROGRAM_ID = MEMO_PROGRAM_ID; exports.Network = Network; exports.POOL_FEE = POOL_FEE; exports.POOL_FEE_BN = POOL_FEE_BN; exports.POSITION_BIN_DATA_SIZE = POSITION_BIN_DATA_SIZE; exports.POSITION_FEE = POSITION_FEE; exports.POSITION_FEE_BN = POSITION_FEE_BN; exports.POSITION_MAX_LENGTH = POSITION_MAX_LENGTH; exports.POSITION_MIN_SIZE = POSITION_MIN_SIZE; exports.PRECISION = PRECISION; exports.PairStatus = PairStatus; exports.PairType = PairType; exports.PositionVersion = PositionVersion; exports.ResizeSide = ResizeSide; exports.SCALE = SCALE; exports.SCALE_OFFSET = SCALE_OFFSET; exports.SIMULATION_USER = SIMULATION_USER; exports.Strategy = Strategy; exports.StrategyType = StrategyType; exports.TOKEN_ACCOUNT_FEE = TOKEN_ACCOUNT_FEE; exports.TOKEN_ACCOUNT_FEE_BN = TOKEN_ACCOUNT_FEE_BN; exports.U64_MAX = U64_MAX; exports.autoFillXByStrategy = autoFillXByStrategy; exports.autoFillXByWeight = autoFillXByWeight; exports.autoFillYByStrategy = autoFillYByStrategy; exports.autoFillYByWeight = autoFillYByWeight; exports.binIdToBinArrayIndex = binIdToBinArrayIndex; exports.calculateBidAskDistribution = calculateBidAskDistribution; exports.calculateNormalDistribution = calculateNormalDistribution; exports.calculateSpotDistribution = calculateSpotDistribution; exports.capSlippagePercentage = capSlippagePercentage; exports.chunkedFetchMultipleBinArrayBitmapExtensionAccount = chunkedFetchMultipleBinArrayBitmapExtensionAccount; exports.chunkedFetchMultiplePoolAccount = chunkedFetchMultiplePoolAccount; exports.chunkedGetMultipleAccountInfos = chunkedGetMultipleAccountInfos; exports.chunks = chunks; exports.computeFee = computeFee; exports.computeFeeFromAmount = computeFeeFromAmount; exports.computeProtocolFee = computeProtocolFee; exports.createProgram = createProgram; exports.decodeAccount = decodeAccount; exports.default = src_default; exports.deriveBinArray = deriveBinArray; exports.deriveBinArrayBitmapExtension = deriveBinArrayBitmapExtension; exports.deriveCustomizablePermissionlessLbPair = deriveCustomizablePermissionlessLbPair; exports.deriveEventAuthority = deriveEventAuthority; exports.deriveLbPair = deriveLbPair; exports.deriveLbPair2 = deriveLbPair2; exports.deriveLbPairWithPresetParamWithIndexKey = deriveLbPairWithPresetParamWithIndexKey; exports.deriveOracle = deriveOracle; exports.derivePermissionLbPair = derivePermissionLbPair; exports.derivePosition = derivePosition; exports.derivePresetParameter = derivePresetParameter; exports.derivePresetParameter2 = derivePresetParameter2; exports.derivePresetParameterWithIndex = derivePresetParameterWithIndex; exports.deriveReserve = deriveReserve; exports.deriveRewardVault = deriveRewardVault; exports.deriveTokenBadge = deriveTokenBadge; exports.enumerateBins = enumerateBins; exports.findNextBinArrayIndexWithLiquidity = findNextBinArrayIndexWithLiquidity; exports.findNextBinArrayWithLiquidity = findNextBinArrayWithLiquidity; exports.fromWeightDistributionToAmount = fromWeightDistributionToAmount; exports.fromWeightDistributionToAmountOneSide = fromWeightDistributionToAmountOneSide; exports.getAccountDiscriminator = getAccountDiscriminator; exports.getBaseFee = getBaseFee; exports.getBinArrayLowerUpperBinId = getBinArrayLowerUpperBinId; exports.getBinArraysRequiredByPositionRange = getBinArraysRequiredByPositionRange; exports.getBinFromBinArray = getBinFromBinArray; exports.getBinIdIndexInBinArray = getBinIdIndexInBinArray; exports.getEstimatedComputeUnitIxWithBuffer = getEstimatedComputeUnitIxWithBuffer; exports.getEstimatedComputeUnitUsageWithBuffer = getEstimatedComputeUnitUsageWithBuffer; exports.getOrCreateATAInstruction = getOrCreateATAInstruction; exports.getOutAmount = getOutAmount; exports.getPriceOfBinByBinId = getPriceOfBinByBinId; exports.getTokenBalance = getTokenBalance; exports.getTokenDecimals = getTokenDecimals; exports.getTokenProgramId = getTokenProgramId; exports.getTokensMintFromPoolAddress = getTokensMintFromPoolAddress; exports.getTotalFee = getTotalFee; exports.getVariableFee = getVariableFee; exports.isBinIdWithinBinArray = isBinIdWithinBinArray; exports.isOverflowDefaultBinArrayBitmap = isOverflowDefaultBinArrayBitmap; exports.parseLogs = parseLogs; exports.range = range; exports.swapExactInQuoteAtBin = swapExactInQuoteAtBin; exports.swapExactOutQuoteAtBin = swapExactOutQuoteAtBin; exports.toAmountAskSide = toAmountAskSide; exports.toAmountBidSide = toAmountBidSide; exports.toAmountBothSide = toAmountBothSide; exports.toAmountsBothSideByStrategy = toAmountsBothSideByStrategy; exports.toStrategyParameters = toStrategyParameters; exports.toWeightDistribution = toWeightDistribution; exports.unwrapSOLInstruction = unwrapSOLInstruction; exports.updateBinArray = updateBinArray; exports.wrapSOLInstruction = wrapSOLInstruction;
19627
+
19628
+
19629
+
19630
+
19631
+
19632
+
19633
+
19634
+
19635
+
19636
+
19637
+ exports.ADMIN = ADMIN; exports.ActionType = ActionType; exports.ActivationType = ActivationType; exports.BASIS_POINT_MAX = BASIS_POINT_MAX; exports.BIN_ARRAY_BITMAP_FEE = BIN_ARRAY_BITMAP_FEE; exports.BIN_ARRAY_BITMAP_FEE_BN = BIN_ARRAY_BITMAP_FEE_BN; exports.BIN_ARRAY_BITMAP_SIZE = BIN_ARRAY_BITMAP_SIZE; exports.BIN_ARRAY_FEE = BIN_ARRAY_FEE; exports.BIN_ARRAY_FEE_BN = BIN_ARRAY_FEE_BN; exports.BinLiquidity = BinLiquidity; exports.BitmapType = BitmapType; exports.ClockLayout = ClockLayout; exports.DEFAULT_BIN_PER_POSITION = DEFAULT_BIN_PER_POSITION; exports.DLMMError = DLMMError; exports.DlmmSdkError = DlmmSdkError; exports.EXTENSION_BINARRAY_BITMAP_SIZE = EXTENSION_BINARRAY_BITMAP_SIZE; exports.FEE_PRECISION = FEE_PRECISION; exports.IDL = dlmm_default; exports.ILM_BASE = ILM_BASE; exports.LBCLMM_PROGRAM_IDS = LBCLMM_PROGRAM_IDS; exports.MAX_ACTIVE_BIN_SLIPPAGE = MAX_ACTIVE_BIN_SLIPPAGE; exports.MAX_BINS_PER_POSITION = MAX_BINS_PER_POSITION; exports.MAX_BIN_ARRAY_SIZE = MAX_BIN_ARRAY_SIZE; exports.MAX_BIN_LENGTH_ALLOWED_IN_ONE_TX = MAX_BIN_LENGTH_ALLOWED_IN_ONE_TX; exports.MAX_CLAIM_ALL_ALLOWED = MAX_CLAIM_ALL_ALLOWED; exports.MAX_EXTRA_BIN_ARRAYS = MAX_EXTRA_BIN_ARRAYS; exports.MAX_FEE_RATE = MAX_FEE_RATE; exports.MAX_RESIZE_LENGTH = MAX_RESIZE_LENGTH; exports.MEMO_PROGRAM_ID = MEMO_PROGRAM_ID; exports.Network = Network; exports.POOL_FEE = POOL_FEE; exports.POOL_FEE_BN = POOL_FEE_BN; exports.POSITION_BIN_DATA_SIZE = POSITION_BIN_DATA_SIZE; exports.POSITION_FEE = POSITION_FEE; exports.POSITION_FEE_BN = POSITION_FEE_BN; exports.POSITION_MAX_LENGTH = POSITION_MAX_LENGTH; exports.POSITION_MIN_SIZE = POSITION_MIN_SIZE; exports.PRECISION = PRECISION; exports.PairStatus = PairStatus; exports.PairType = PairType; exports.PositionVersion = PositionVersion; exports.RebalancePosition = RebalancePosition; exports.ResizeSide = ResizeSide; exports.SCALE = SCALE; exports.SCALE_OFFSET = SCALE_OFFSET; exports.SIMULATION_USER = SIMULATION_USER; exports.Strategy = Strategy; exports.StrategyType = StrategyType; exports.TOKEN_ACCOUNT_FEE = TOKEN_ACCOUNT_FEE; exports.TOKEN_ACCOUNT_FEE_BN = TOKEN_ACCOUNT_FEE_BN; exports.U64_MAX = U64_MAX; exports.autoFillXByStrategy = autoFillXByStrategy; exports.autoFillXByWeight = autoFillXByWeight; exports.autoFillYByStrategy = autoFillYByStrategy; exports.autoFillYByWeight = autoFillYByWeight; exports.binIdToBinArrayIndex = binIdToBinArrayIndex; exports.buildLiquidityStrategyParameters = buildLiquidityStrategyParameters; exports.calculateBidAskDistribution = calculateBidAskDistribution; exports.calculateNormalDistribution = calculateNormalDistribution; exports.calculateSpotDistribution = calculateSpotDistribution; exports.capSlippagePercentage = capSlippagePercentage; exports.chunkedFetchMultipleBinArrayBitmapExtensionAccount = chunkedFetchMultipleBinArrayBitmapExtensionAccount; exports.chunkedFetchMultiplePoolAccount = chunkedFetchMultiplePoolAccount; exports.chunkedGetMultipleAccountInfos = chunkedGetMultipleAccountInfos; exports.chunks = chunks; exports.computeFee = computeFee; exports.computeFeeFromAmount = computeFeeFromAmount; exports.computeProtocolFee = computeProtocolFee; exports.createProgram = createProgram; exports.decodeAccount = decodeAccount; exports.default = src_default; exports.deriveBinArray = deriveBinArray; exports.deriveBinArrayBitmapExtension = deriveBinArrayBitmapExtension; exports.deriveCustomizablePermissionlessLbPair = deriveCustomizablePermissionlessLbPair; exports.deriveEventAuthority = deriveEventAuthority; exports.deriveLbPair = deriveLbPair; exports.deriveLbPair2 = deriveLbPair2; exports.deriveLbPairWithPresetParamWithIndexKey = deriveLbPairWithPresetParamWithIndexKey; exports.deriveOracle = deriveOracle; exports.derivePermissionLbPair = derivePermissionLbPair; exports.derivePosition = derivePosition; exports.derivePresetParameter = derivePresetParameter; exports.derivePresetParameter2 = derivePresetParameter2; exports.derivePresetParameterWithIndex = derivePresetParameterWithIndex; exports.deriveReserve = deriveReserve; exports.deriveRewardVault = deriveRewardVault; exports.deriveTokenBadge = deriveTokenBadge; exports.enumerateBins = enumerateBins; exports.findNextBinArrayIndexWithLiquidity = findNextBinArrayIndexWithLiquidity; exports.findNextBinArrayWithLiquidity = findNextBinArrayWithLiquidity; exports.fromWeightDistributionToAmount = fromWeightDistributionToAmount; exports.fromWeightDistributionToAmountOneSide = fromWeightDistributionToAmountOneSide; exports.getAccountDiscriminator = getAccountDiscriminator; exports.getAmountInBinsAskSide = getAmountInBinsAskSide; exports.getAmountInBinsBidSide = getAmountInBinsBidSide; exports.getAutoFillAmountByRebalancedPosition = getAutoFillAmountByRebalancedPosition; exports.getBaseFee = getBaseFee; exports.getBinArrayLowerUpperBinId = getBinArrayLowerUpperBinId; exports.getBinArraysRequiredByPositionRange = getBinArraysRequiredByPositionRange; exports.getBinFromBinArray = getBinFromBinArray; exports.getBinIdIndexInBinArray = getBinIdIndexInBinArray; exports.getEstimatedComputeUnitIxWithBuffer = getEstimatedComputeUnitIxWithBuffer; exports.getEstimatedComputeUnitUsageWithBuffer = getEstimatedComputeUnitUsageWithBuffer; exports.getLiquidityStrategyParameterBuilder = getLiquidityStrategyParameterBuilder; exports.getOrCreateATAInstruction = getOrCreateATAInstruction; exports.getOutAmount = getOutAmount; exports.getPriceOfBinByBinId = getPriceOfBinByBinId; exports.getRebalanceBinArrayIndexesAndBitmapCoverage = getRebalanceBinArrayIndexesAndBitmapCoverage; exports.getTokenBalance = getTokenBalance; exports.getTokenDecimals = getTokenDecimals; exports.getTokenProgramId = getTokenProgramId; exports.getTokensMintFromPoolAddress = getTokensMintFromPoolAddress; exports.getTotalFee = getTotalFee; exports.getVariableFee = getVariableFee; exports.isBinIdWithinBinArray = isBinIdWithinBinArray; exports.isOverflowDefaultBinArrayBitmap = isOverflowDefaultBinArrayBitmap; exports.parseLogs = parseLogs; exports.range = range; exports.suggestBalancedXParametersFromY = suggestBalancedXParametersFromY; exports.suggestBalancedYParametersFromX = suggestBalancedYParametersFromX; exports.swapExactInQuoteAtBin = swapExactInQuoteAtBin; exports.swapExactOutQuoteAtBin = swapExactOutQuoteAtBin; exports.toAmountAskSide = toAmountAskSide; exports.toAmountBidSide = toAmountBidSide; exports.toAmountBothSide = toAmountBothSide; exports.toAmountIntoBins = toAmountIntoBins; exports.toAmountsBothSideByStrategy = toAmountsBothSideByStrategy; exports.toStrategyParameters = toStrategyParameters; exports.toWeightDistribution = toWeightDistribution; exports.unwrapSOLInstruction = unwrapSOLInstruction; exports.updateBinArray = updateBinArray; exports.wrapSOLInstruction = wrapSOLInstruction;
19303
19638
  //# sourceMappingURL=index.js.map