@kamino-finance/scope-sdk
Version:
Scope Oracle SDK
227 lines (209 loc) • 8.36 kB
text/typescript
/* eslint-disable @typescript-eslint/no-unused-vars */
import {
address,
Address,
fetchEncodedAccount,
fetchEncodedAccounts,
GetAccountInfoApi,
GetMultipleAccountsApi,
Rpc,
} from "@solana/kit"
/* eslint-enable @typescript-eslint/no-unused-vars */
import BN from "bn.js" // eslint-disable-line @typescript-eslint/no-unused-vars
import * as borsh from "@coral-xyz/borsh" // eslint-disable-line @typescript-eslint/no-unused-vars
import { borshAddress } from "../utils" // eslint-disable-line @typescript-eslint/no-unused-vars
import * as types from "../types" // eslint-disable-line @typescript-eslint/no-unused-vars
import { PROGRAM_ID } from "../programId"
export interface PersonalPositionStateFields {
/** Bump to identify PDA */
bump: number
/** Mint address of the tokenized position */
nftMint: Address
/** The ID of the pool with which this token is connected */
poolId: Address
/** The lower bound tick of the position */
tickLowerIndex: number
/** The upper bound tick of the position */
tickUpperIndex: number
/** The amount of liquidity owned by this position */
liquidity: BN
/** The token_0 fee growth of the aggregate position as of the last action on the individual position */
feeGrowthInside0LastX64: BN
/** The token_1 fee growth of the aggregate position as of the last action on the individual position */
feeGrowthInside1LastX64: BN
/** The fees owed to the position owner in token_0, as of the last computation */
tokenFeesOwed0: BN
/** The fees owed to the position owner in token_1, as of the last computation */
tokenFeesOwed1: BN
rewardInfos: Array<types.PositionRewardInfoFields>
padding: Array<BN>
}
export interface PersonalPositionStateJSON {
/** Bump to identify PDA */
bump: number
/** Mint address of the tokenized position */
nftMint: string
/** The ID of the pool with which this token is connected */
poolId: string
/** The lower bound tick of the position */
tickLowerIndex: number
/** The upper bound tick of the position */
tickUpperIndex: number
/** The amount of liquidity owned by this position */
liquidity: string
/** The token_0 fee growth of the aggregate position as of the last action on the individual position */
feeGrowthInside0LastX64: string
/** The token_1 fee growth of the aggregate position as of the last action on the individual position */
feeGrowthInside1LastX64: string
/** The fees owed to the position owner in token_0, as of the last computation */
tokenFeesOwed0: string
/** The fees owed to the position owner in token_1, as of the last computation */
tokenFeesOwed1: string
rewardInfos: Array<types.PositionRewardInfoJSON>
padding: Array<string>
}
export class PersonalPositionState {
/** Bump to identify PDA */
readonly bump: number
/** Mint address of the tokenized position */
readonly nftMint: Address
/** The ID of the pool with which this token is connected */
readonly poolId: Address
/** The lower bound tick of the position */
readonly tickLowerIndex: number
/** The upper bound tick of the position */
readonly tickUpperIndex: number
/** The amount of liquidity owned by this position */
readonly liquidity: BN
/** The token_0 fee growth of the aggregate position as of the last action on the individual position */
readonly feeGrowthInside0LastX64: BN
/** The token_1 fee growth of the aggregate position as of the last action on the individual position */
readonly feeGrowthInside1LastX64: BN
/** The fees owed to the position owner in token_0, as of the last computation */
readonly tokenFeesOwed0: BN
/** The fees owed to the position owner in token_1, as of the last computation */
readonly tokenFeesOwed1: BN
readonly rewardInfos: Array<types.PositionRewardInfo>
readonly padding: Array<BN>
static readonly discriminator = Buffer.from([
70, 111, 150, 126, 230, 15, 25, 117,
])
static readonly layout = borsh.struct<PersonalPositionState>([
borsh.u8("bump"),
borshAddress("nftMint"),
borshAddress("poolId"),
borsh.i32("tickLowerIndex"),
borsh.i32("tickUpperIndex"),
borsh.u128("liquidity"),
borsh.u128("feeGrowthInside0LastX64"),
borsh.u128("feeGrowthInside1LastX64"),
borsh.u64("tokenFeesOwed0"),
borsh.u64("tokenFeesOwed1"),
borsh.array(types.PositionRewardInfo.layout(), 3, "rewardInfos"),
borsh.array(borsh.u64(), 8, "padding"),
])
constructor(fields: PersonalPositionStateFields) {
this.bump = fields.bump
this.nftMint = fields.nftMint
this.poolId = fields.poolId
this.tickLowerIndex = fields.tickLowerIndex
this.tickUpperIndex = fields.tickUpperIndex
this.liquidity = fields.liquidity
this.feeGrowthInside0LastX64 = fields.feeGrowthInside0LastX64
this.feeGrowthInside1LastX64 = fields.feeGrowthInside1LastX64
this.tokenFeesOwed0 = fields.tokenFeesOwed0
this.tokenFeesOwed1 = fields.tokenFeesOwed1
this.rewardInfos = fields.rewardInfos.map(
(item) => new types.PositionRewardInfo({ ...item })
)
this.padding = fields.padding
}
static async fetch(
rpc: Rpc<GetAccountInfoApi>,
address: Address,
programId: Address = PROGRAM_ID
): Promise<PersonalPositionState | null> {
const info = await fetchEncodedAccount(rpc, address)
if (!info.exists) {
return null
}
if (info.programAddress !== programId) {
throw new Error("account doesn't belong to this program")
}
return this.decode(Buffer.from(info.data))
}
static async fetchMultiple(
rpc: Rpc<GetMultipleAccountsApi>,
addresses: Address[],
programId: Address = PROGRAM_ID
): Promise<Array<PersonalPositionState | null>> {
const infos = await fetchEncodedAccounts(rpc, addresses)
return infos.map((info) => {
if (!info.exists) {
return null
}
if (info.programAddress !== programId) {
throw new Error("account doesn't belong to this program")
}
return this.decode(Buffer.from(info.data))
})
}
static decode(data: Buffer): PersonalPositionState {
if (!data.slice(0, 8).equals(PersonalPositionState.discriminator)) {
throw new Error("invalid account discriminator")
}
const dec = PersonalPositionState.layout.decode(data.slice(8))
return new PersonalPositionState({
bump: dec.bump,
nftMint: dec.nftMint,
poolId: dec.poolId,
tickLowerIndex: dec.tickLowerIndex,
tickUpperIndex: dec.tickUpperIndex,
liquidity: dec.liquidity,
feeGrowthInside0LastX64: dec.feeGrowthInside0LastX64,
feeGrowthInside1LastX64: dec.feeGrowthInside1LastX64,
tokenFeesOwed0: dec.tokenFeesOwed0,
tokenFeesOwed1: dec.tokenFeesOwed1,
rewardInfos: dec.rewardInfos.map(
(
item: any /* eslint-disable-line @typescript-eslint/no-explicit-any */
) => types.PositionRewardInfo.fromDecoded(item)
),
padding: dec.padding,
})
}
toJSON(): PersonalPositionStateJSON {
return {
bump: this.bump,
nftMint: this.nftMint,
poolId: this.poolId,
tickLowerIndex: this.tickLowerIndex,
tickUpperIndex: this.tickUpperIndex,
liquidity: this.liquidity.toString(),
feeGrowthInside0LastX64: this.feeGrowthInside0LastX64.toString(),
feeGrowthInside1LastX64: this.feeGrowthInside1LastX64.toString(),
tokenFeesOwed0: this.tokenFeesOwed0.toString(),
tokenFeesOwed1: this.tokenFeesOwed1.toString(),
rewardInfos: this.rewardInfos.map((item) => item.toJSON()),
padding: this.padding.map((item) => item.toString()),
}
}
static fromJSON(obj: PersonalPositionStateJSON): PersonalPositionState {
return new PersonalPositionState({
bump: obj.bump,
nftMint: address(obj.nftMint),
poolId: address(obj.poolId),
tickLowerIndex: obj.tickLowerIndex,
tickUpperIndex: obj.tickUpperIndex,
liquidity: new BN(obj.liquidity),
feeGrowthInside0LastX64: new BN(obj.feeGrowthInside0LastX64),
feeGrowthInside1LastX64: new BN(obj.feeGrowthInside1LastX64),
tokenFeesOwed0: new BN(obj.tokenFeesOwed0),
tokenFeesOwed1: new BN(obj.tokenFeesOwed1),
rewardInfos: obj.rewardInfos.map((item) =>
types.PositionRewardInfo.fromJSON(item)
),
padding: obj.padding.map((item) => new BN(item)),
})
}
}