@symmetry-hq/sdk 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (121) hide show
  1. package/dist/src/constants.d.ts +23 -0
  2. package/dist/src/constants.js +38 -0
  3. package/dist/src/index.d.ts +804 -0
  4. package/dist/src/index.js +2097 -0
  5. package/dist/src/instructions/automation/auction.d.ts +6 -0
  6. package/dist/src/instructions/automation/auction.js +40 -0
  7. package/dist/src/instructions/automation/claimBounty.d.ts +12 -0
  8. package/dist/src/instructions/automation/claimBounty.js +44 -0
  9. package/dist/src/instructions/automation/flashSwap.d.ts +21 -0
  10. package/dist/src/instructions/automation/flashSwap.js +74 -0
  11. package/dist/src/instructions/automation/priceUpdate.d.ts +19 -0
  12. package/dist/src/instructions/automation/priceUpdate.js +89 -0
  13. package/dist/src/instructions/automation/rebalanceIntent.d.ts +32 -0
  14. package/dist/src/instructions/automation/rebalanceIntent.js +117 -0
  15. package/dist/src/instructions/automation/rebalanceSwap.d.ts +11 -0
  16. package/dist/src/instructions/automation/rebalanceSwap.js +42 -0
  17. package/dist/src/instructions/management/addBounty.d.ts +7 -0
  18. package/dist/src/instructions/management/addBounty.js +41 -0
  19. package/dist/src/instructions/management/admin.d.ts +9 -0
  20. package/dist/src/instructions/management/admin.js +53 -0
  21. package/dist/src/instructions/management/claimFees.d.ts +15 -0
  22. package/dist/src/instructions/management/claimFees.js +95 -0
  23. package/dist/src/instructions/management/createBasket.d.ts +21 -0
  24. package/dist/src/instructions/management/createBasket.js +98 -0
  25. package/dist/src/instructions/management/edit.d.ts +51 -0
  26. package/dist/src/instructions/management/edit.js +477 -0
  27. package/dist/src/instructions/management/luts.d.ts +30 -0
  28. package/dist/src/instructions/management/luts.js +99 -0
  29. package/dist/src/instructions/pda.d.ts +25 -0
  30. package/dist/src/instructions/pda.js +128 -0
  31. package/dist/src/instructions/user/deposit.d.ts +20 -0
  32. package/dist/src/instructions/user/deposit.js +100 -0
  33. package/dist/src/instructions/user/withdraw.d.ts +8 -0
  34. package/dist/src/instructions/user/withdraw.js +36 -0
  35. package/dist/src/jup.d.ts +49 -0
  36. package/dist/src/jup.js +80 -0
  37. package/dist/src/keeperMonitor.d.ts +52 -0
  38. package/dist/src/keeperMonitor.js +624 -0
  39. package/dist/src/layouts/basket.d.ts +191 -0
  40. package/dist/src/layouts/basket.js +51 -0
  41. package/dist/src/layouts/config.d.ts +281 -0
  42. package/dist/src/layouts/config.js +237 -0
  43. package/dist/src/layouts/fraction.d.ts +20 -0
  44. package/dist/src/layouts/fraction.js +164 -0
  45. package/dist/src/layouts/intents/bounty.d.ts +18 -0
  46. package/dist/src/layouts/intents/bounty.js +19 -0
  47. package/dist/src/layouts/intents/intent.d.ts +209 -0
  48. package/dist/src/layouts/intents/intent.js +97 -0
  49. package/dist/src/layouts/intents/rebalanceIntent.d.ts +212 -0
  50. package/dist/src/layouts/intents/rebalanceIntent.js +94 -0
  51. package/dist/src/layouts/lookupTable.d.ts +7 -0
  52. package/dist/src/layouts/lookupTable.js +10 -0
  53. package/dist/src/layouts/oracle.d.ts +63 -0
  54. package/dist/src/layouts/oracle.js +96 -0
  55. package/dist/src/states/basket.d.ts +14 -0
  56. package/dist/src/states/basket.js +479 -0
  57. package/dist/src/states/config.d.ts +3 -0
  58. package/dist/src/states/config.js +71 -0
  59. package/dist/src/states/intents/intent.d.ts +10 -0
  60. package/dist/src/states/intents/intent.js +316 -0
  61. package/dist/src/states/intents/rebalanceIntent.d.ts +42 -0
  62. package/dist/src/states/intents/rebalanceIntent.js +680 -0
  63. package/dist/src/states/oracles/constants.d.ts +9 -0
  64. package/dist/src/states/oracles/constants.js +15 -0
  65. package/dist/src/states/oracles/oracle.d.ts +24 -0
  66. package/dist/src/states/oracles/oracle.js +168 -0
  67. package/dist/src/states/oracles/pythOracle.d.ts +132 -0
  68. package/dist/src/states/oracles/pythOracle.js +609 -0
  69. package/dist/src/states/oracles/raydiumClmmOracle.d.ts +184 -0
  70. package/dist/src/states/oracles/raydiumClmmOracle.js +843 -0
  71. package/dist/src/states/oracles/raydiumCpmmOracle.d.ts +120 -0
  72. package/dist/src/states/oracles/raydiumCpmmOracle.js +540 -0
  73. package/dist/src/states/oracles/switchboardOracle.d.ts +0 -0
  74. package/dist/src/states/oracles/switchboardOracle.js +1 -0
  75. package/dist/src/states/withdrawBasketFees.d.ts +10 -0
  76. package/dist/src/states/withdrawBasketFees.js +154 -0
  77. package/dist/src/txUtils.d.ts +65 -0
  78. package/dist/src/txUtils.js +306 -0
  79. package/dist/test.d.ts +1 -0
  80. package/dist/test.js +561 -0
  81. package/package.json +31 -0
  82. package/src/constants.ts +40 -0
  83. package/src/index.ts +2431 -0
  84. package/src/instructions/automation/auction.ts +55 -0
  85. package/src/instructions/automation/claimBounty.ts +69 -0
  86. package/src/instructions/automation/flashSwap.ts +104 -0
  87. package/src/instructions/automation/priceUpdate.ts +117 -0
  88. package/src/instructions/automation/rebalanceIntent.ts +181 -0
  89. package/src/instructions/management/addBounty.ts +55 -0
  90. package/src/instructions/management/admin.ts +72 -0
  91. package/src/instructions/management/claimFees.ts +129 -0
  92. package/src/instructions/management/createBasket.ts +138 -0
  93. package/src/instructions/management/edit.ts +602 -0
  94. package/src/instructions/management/luts.ts +157 -0
  95. package/src/instructions/pda.ts +151 -0
  96. package/src/instructions/user/deposit.ts +143 -0
  97. package/src/instructions/user/withdraw.ts +53 -0
  98. package/src/jup.ts +113 -0
  99. package/src/keeperMonitor.ts +585 -0
  100. package/src/layouts/basket.ts +233 -0
  101. package/src/layouts/config.ts +576 -0
  102. package/src/layouts/fraction.ts +164 -0
  103. package/src/layouts/intents/bounty.ts +35 -0
  104. package/src/layouts/intents/intent.ts +324 -0
  105. package/src/layouts/intents/rebalanceIntent.ts +306 -0
  106. package/src/layouts/lookupTable.ts +14 -0
  107. package/src/layouts/oracle.ts +157 -0
  108. package/src/states/basket.ts +527 -0
  109. package/src/states/config.ts +62 -0
  110. package/src/states/intents/intent.ts +311 -0
  111. package/src/states/intents/rebalanceIntent.ts +751 -0
  112. package/src/states/oracles/constants.ts +13 -0
  113. package/src/states/oracles/oracle.ts +212 -0
  114. package/src/states/oracles/pythOracle.ts +874 -0
  115. package/src/states/oracles/raydiumClmmOracle.ts +1193 -0
  116. package/src/states/oracles/raydiumCpmmOracle.ts +784 -0
  117. package/src/states/oracles/switchboardOracle.ts +0 -0
  118. package/src/states/withdrawBasketFees.ts +160 -0
  119. package/src/txUtils.ts +424 -0
  120. package/test.ts +609 -0
  121. package/tsconfig.json +101 -0
@@ -0,0 +1,527 @@
1
+ import Decimal from 'decimal.js';
2
+ import { createHash } from 'crypto';
3
+ import {
4
+ AccountInfo, Connection, GetProgramAccountsFilter, GetProgramAccountsResponse, PublicKey
5
+ } from '@solana/web3.js';
6
+
7
+ import {
8
+ BASKETS_V3_PROGRAM_ID, MAX_MANAGERS_PER_BASKET, PYTHNET_CUSTODY_PRICE_USDC_ACCOUNT,
9
+ PYTHNET_CUSTODY_PRICE_WSOL_ACCOUNT
10
+ } from '../constants';
11
+ import { Basket, BASKET_TYPES_STRINGS, BasketLayout, FormattedAccumulatedFees, FormattedAsset, FormattedLookupTables } from '../layouts/basket';
12
+ import {
13
+ ORACLE_TYPES_STRINGS,
14
+ OracleType, PYTH_USDC_ORACLE_SETTINGS, PYTH_WSOL_ORACLE_SETTINGS,
15
+ } from '../layouts/oracle';
16
+ import { getMultipleAccountsInfoBatched, getMultipleAddressLookupTableAccounts } from '../txUtils';
17
+ import { PriceAggregator } from './oracles/oracle';
18
+ import { PythOracle } from './oracles/pythOracle';
19
+ import {
20
+ getNextTickArrayStartIndex, getPdaTickArrayAddress, getTickArrayStartIndexByTick, PoolState
21
+ } from './oracles/raydiumClmmOracle';
22
+ import { fractionToDecimal } from '../layouts/fraction';
23
+ import {
24
+ FormattedBasket,
25
+ FormattedCreatorSettings,
26
+ FormattedManagersSettings,
27
+ FormattedFeeSettings,
28
+ FormattedScheduleSettings,
29
+ FormattedAutomationSettings,
30
+ FormattedLpSettings,
31
+ FormattedMetadataSettings,
32
+ FormattedDepositsSettings,
33
+ FormattedForceRebalanceSettings,
34
+ FormattedCustomRebalanceSettings,
35
+ FormattedAddTokenSettings,
36
+ FormattedUpdateWeightsSettings,
37
+ FormattedMakeDirectSwapSettings,
38
+ } from '../layouts/basket';
39
+
40
+ // export async function computeTokenMintsHash(mints: string[]): Promise<Buffer> {
41
+ // let hashBuffer = new Uint8Array(32)
42
+ // for (const mint of mints) {
43
+ // const mintBytes = new PublicKey(mint).toBytes()
44
+
45
+ // const combined = new Uint8Array(hashBuffer.length + mintBytes.length)
46
+ // combined.set(hashBuffer, 0)
47
+ // combined.set(mintBytes, hashBuffer.length)
48
+
49
+ // const hashArrayBuffer = await crypto.subtle.digest("SHA-256", combined)
50
+ // hashBuffer = new Uint8Array(hashArrayBuffer)
51
+ // }
52
+
53
+ // return Buffer.from(hashBuffer)
54
+ // }
55
+
56
+ export function computeTokenMintsHash(mints: string[]): number[] {
57
+ let hashBuffer = Buffer.alloc(32);
58
+ for (const mint of mints) {
59
+ hashBuffer = createHash('sha256').update(Buffer.concat([hashBuffer, new PublicKey(mint).toBytes()])).digest();
60
+ }
61
+ return Array.from(hashBuffer);
62
+ }
63
+
64
+ export function addFieldsToBasket(basket: Basket): Basket {
65
+ let metadataSettings = basket.settings.metadata;
66
+ let metadata = {
67
+ symbol: Buffer.from(metadataSettings.symbol.slice(0, metadataSettings.symbolLength)).toString(),
68
+ name: Buffer.from(metadataSettings.name.slice(0, metadataSettings.nameLength)).toString(),
69
+ uri: Buffer.from(metadataSettings.uri.slice(0, metadataSettings.uriLength)).toString()
70
+ }
71
+ basket.composition = basket.composition.slice(0, basket.numTokens);
72
+ for (let i = 0; i < basket.numTokens; i++) {
73
+ let agg = basket.composition[i].oracleAggregator;
74
+ basket.composition[i].oracleAggregator.oracles = agg.oracles.slice(0, agg.numOracles);
75
+ for (let j = 0; j < agg.numOracles; j++) {
76
+ let lutIds = agg.oracles[j].accountsToLoadLutIds;
77
+ let lutIndices = agg.oracles[j].accountsToLoadLutIndices;
78
+ basket.composition[i].oracleAggregator.oracles[j].accountsToLoadLutIds =
79
+ lutIds.slice(0, agg.oracles[j].oracleSettings.numRequiredAccounts);
80
+ basket.composition[i].oracleAggregator.oracles[j].accountsToLoadLutIndices =
81
+ lutIndices.slice(0, agg.oracles[j].oracleSettings.numRequiredAccounts);
82
+ }
83
+ }
84
+ let totalManagers = MAX_MANAGERS_PER_BASKET;
85
+ while (totalManagers > 0 && basket.settings.managers.managers[totalManagers - 1].equals(PublicKey.default)) {
86
+ totalManagers--;
87
+ }
88
+ basket.settings.managers.managers = basket.settings.managers.managers.slice(0, totalManagers);
89
+ basket.settings.managers.managersWeightBps = basket.settings.managers.managersWeightBps.slice(0, totalManagers);
90
+ basket = {...basket, metadata: metadata};
91
+ let creator_settings: FormattedCreatorSettings = {
92
+ creator: basket.settings.creator.toBase58(),
93
+ }
94
+ let manager_settings: FormattedManagersSettings = {
95
+ managers: basket.settings.managers.managers.map((manager, i) => ({
96
+ pubkey: manager.toBase58(),
97
+ fee_split_weight_bps: basket.settings.managers.managersWeightBps[i],
98
+ authorities: {
99
+ managers: (((2 ** i) & basket.settings.managersAuthorityBitmask) != 0),
100
+ fees: (((2 ** i) & basket.settings.feesAuthorityBitmask) != 0),
101
+ schedule: (((2 ** i) & basket.settings.scheduleAuthorityBitmask) != 0),
102
+ automation: (((2 ** i) & basket.settings.automationAuthorityBitmask) != 0),
103
+ lp: (((2 ** i) & basket.settings.lpAuthorityBitmask) != 0),
104
+ metadata: (((2 ** i) & basket.settings.metadataAuthorityBitmask) != 0),
105
+ deposits: (((2 ** i) & basket.settings.depositsAuthorityBitmask) != 0),
106
+ force_rebalance: (((2 ** i) & basket.settings.forceRebalanceAuthorityBitmask) != 0),
107
+ custom_rebalance: (((2 ** i) & basket.settings.customRebalanceAuthorityBitmask) != 0),
108
+ add_token: (((2 ** i) & basket.settings.addTokenIntentAuthorityBitmask) != 0),
109
+ update_weights: (((2 ** i) & basket.settings.updateWeightsIntentAuthorityBitmask) != 0),
110
+ make_direct_swap: (((2 ** i) & basket.settings.makeDirectSwapIntentAuthorityBitmask) != 0),
111
+ },
112
+ })),
113
+ modification_delay: parseInt(basket.settings.managers.modificationDelay.toString()),
114
+ updated_at: parseInt(basket.settings.managersLastUpdateTimestamp.toString()),
115
+ }
116
+ let fee_settings: FormattedFeeSettings = {
117
+ host_deposit_fee_bps: basket.settings.fees.hostDepositFeeBps,
118
+ host_withdraw_fee_bps: basket.settings.fees.hostWithdrawFeeBps,
119
+ host_management_fee_bps: basket.settings.fees.hostManagementFeeBps,
120
+ host_performance_fee_bps: basket.settings.fees.hostPerformanceFeeBps,
121
+ creator_deposit_fee_bps: basket.settings.fees.creatorDepositFeeBps,
122
+ creator_withdraw_fee_bps: basket.settings.fees.creatorWithdrawFeeBps,
123
+ creator_management_fee_bps: basket.settings.fees.creatorManagementFeeBps,
124
+ creator_performance_fee_bps: basket.settings.fees.creatorPerformanceFeeBps,
125
+ managers_deposit_fee_bps: basket.settings.fees.managersDepositFeeBps,
126
+ managers_withdraw_fee_bps: basket.settings.fees.managersWithdrawFeeBps,
127
+ managers_management_fee_bps: basket.settings.fees.managersManagementFeeBps,
128
+ managers_performance_fee_bps: basket.settings.fees.managersPerformanceFeeBps,
129
+ basket_deposit_fee_bps: basket.settings.fees.basketDepositFeeBps,
130
+ basket_withdraw_fee_bps: basket.settings.fees.basketWithdrawFeeBps,
131
+ modification_delay: parseInt(basket.settings.fees.modificationDelay.toString()),
132
+ updated_at: parseInt(basket.settings.feesLastUpdateTimestamp.toString()),
133
+ }
134
+ let schedule_settings: FormattedScheduleSettings = {
135
+ cycle_start_time: parseInt(basket.settings.schedule.cycleStartTime.toString()),
136
+ cycle_duration: parseInt(basket.settings.schedule.cycleDuration.toString()),
137
+ deposits_start: parseInt(basket.settings.schedule.depositsStart.toString()),
138
+ deposits_end: parseInt(basket.settings.schedule.depositsEnd.toString()),
139
+ automation_start: parseInt(basket.settings.schedule.automationStart.toString()),
140
+ automation_end: parseInt(basket.settings.schedule.automationEnd.toString()),
141
+ management_start: parseInt(basket.settings.schedule.managementStart.toString()),
142
+ management_end: parseInt(basket.settings.schedule.managementEnd.toString()),
143
+ modification_delay: parseInt(basket.settings.schedule.modificationDelay.toString()),
144
+ updated_at: parseInt(basket.settings.scheduleLastUpdateTimestamp.toString()),
145
+ }
146
+ let automation_settings: FormattedAutomationSettings = {
147
+ enabled: basket.settings.automation.allowAutomation == 1 ? true : false,
148
+ rebalance_slippage_threshold_bps: basket.settings.automation.rebalanceSlippageThresholdBps,
149
+ per_trade_rebalance_slippage_threshold_bps: basket.settings.automation.perTradeRebalanceSlippageThresholdBps,
150
+ rebalance_activation_threshold_abs_bps: basket.settings.automation.rebalanceActivationThresholdAbsBps,
151
+ rebalance_activation_threshold_rel_bps: basket.settings.automation.rebalanceActivationThresholdRelBps,
152
+ rebalance_activation_cooldown: parseInt(basket.settings.automation.rebalanceActivationCooldown.toString()),
153
+ modification_delay: parseInt(basket.settings.automation.modificationDelay.toString()),
154
+ updated_at: parseInt(basket.settings.automationLastUpdateTimestamp.toString()),
155
+ }
156
+ let lp_settings: FormattedLpSettings = {
157
+ enabled: basket.settings.lp.allowLp == 1 ? true : false,
158
+ lp_threshold_bps: basket.settings.lp.lpThresholdBps,
159
+ modification_delay: parseInt(basket.settings.lp.modificationDelay.toString()),
160
+ updated_at: parseInt(basket.settings.lpLastUpdateTimestamp.toString()),
161
+ }
162
+ let metadata_settings: FormattedMetadataSettings = {
163
+ symbol: metadata.symbol,
164
+ name: metadata.name,
165
+ uri: metadata.uri,
166
+ modification_delay: parseInt(basket.settings.metadata.modificationDelay.toString()),
167
+ updated_at: parseInt(basket.settings.metadataLastUpdateTimestamp.toString()),
168
+ }
169
+ let deposits_settings: FormattedDepositsSettings = {
170
+ enabled: basket.settings.depositsAreAllowed == 1 ? true : false,
171
+ }
172
+ let force_rebalance_settings: FormattedForceRebalanceSettings = {
173
+ enabled: basket.settings.forceRebalanceIsAllowed == 1 ? true : false,
174
+ modification_delay: parseInt(basket.settings.forceRebalanceModificationDelay.toString()),
175
+ updated_at: parseInt(basket.settings.forceRebalanceLastUpdateTimestamp.toString()),
176
+ }
177
+ let custom_rebalance_settings: FormattedCustomRebalanceSettings = {
178
+ enabled: basket.settings.customRebalanceIsAllowed == 1 ? true : false,
179
+ modification_delay: parseInt(basket.settings.customRebalanceModificationDelay.toString()),
180
+ updated_at: parseInt(basket.settings.customRebalanceLastUpdateTimestamp.toString()),
181
+ }
182
+ let add_token_settings: FormattedAddTokenSettings = {
183
+ modification_delay: parseInt(basket.settings.addTokenDelay.toString()),
184
+ updated_at: parseInt(basket.settings.addTokenLastUpdateTimestamp.toString()),
185
+ }
186
+ let update_weights_settings: FormattedUpdateWeightsSettings = {
187
+ modification_delay: parseInt(basket.settings.updateWeightsDelay.toString()),
188
+ updated_at: parseInt(basket.settings.updateWeightsLastUpdateTimestamp.toString()),
189
+ }
190
+ let make_direct_swap_settings: FormattedMakeDirectSwapSettings = {
191
+ modification_delay: parseInt(basket.settings.makeDirectSwapDelay.toString()),
192
+ updated_at: parseInt(basket.settings.makeDirectSwapLastUpdateTimestamp.toString()),
193
+ }
194
+ let accumulated_fees: FormattedAccumulatedFees = {
195
+ symmetry_fees: parseInt(basket.accumulatedFees.symmetryFees.toString()),
196
+ creator_fees: parseInt(basket.accumulatedFees.creatorFees.toString()),
197
+ host_fees: parseInt(basket.accumulatedFees.hostFees.toString()),
198
+ managers_fees: parseInt(basket.accumulatedFees.managersFees.toString()),
199
+ };
200
+ let lookup_tables: FormattedLookupTables = {
201
+ active: [
202
+ {
203
+ pubkey: basket.lookupTables.active[0].toBase58(),
204
+ contents: basket.lutPubkeys?.[0]?.state.addresses.map(address => address.toBase58()) || [],
205
+ },
206
+ {
207
+ pubkey: basket.lookupTables.active[1].toBase58(),
208
+ contents: basket.lutPubkeys?.[1]?.state.addresses.map(address => address.toBase58()) || [],
209
+ },
210
+ ],
211
+ temp: [
212
+ {
213
+ pubkey: basket.lookupTables.temp[0].toBase58(),
214
+ contents: basket.lutPubkeys?.[2]?.state.addresses.map(address => address.toBase58()) || [],
215
+ },
216
+ {
217
+ pubkey: basket.lookupTables.temp[1].toBase58(),
218
+ contents: basket.lutPubkeys?.[3]?.state.addresses.map(address => address.toBase58()) || [],
219
+ },
220
+ ],
221
+ };
222
+ let composition: FormattedAsset[] = basket.composition.map(asset => ({
223
+ mint: asset.mint.toBase58(),
224
+ amount: parseInt(asset.amount.toString()),
225
+ weight: asset.weight,
226
+ active: asset.active == 1 ? true : false,
227
+ oracle_aggregator: {
228
+ num_oracles: asset.oracleAggregator.numOracles,
229
+ min_oracles_thresh: asset.oracleAggregator.minOraclesThresh,
230
+ oracles: asset.oracleAggregator.oracles.map(oracle => ({
231
+ oracle_settings: {
232
+ oracle_type: ORACLE_TYPES_STRINGS.get(oracle.oracleSettings.oracleType) ?? "example",
233
+ account_lut_id: oracle.accountsToLoadLutIds[0],
234
+ account_lut_index: oracle.accountsToLoadLutIndices[0],
235
+ account: lookup_tables.active[oracle.accountsToLoadLutIds[0]].contents[oracle.accountsToLoadLutIndices[0]],
236
+ num_required_accounts: oracle.oracleSettings.numRequiredAccounts,
237
+ weight: oracle.oracleSettings.weight,
238
+ is_required: oracle.oracleSettings.isRequired == 1 ? true : false,
239
+ conf_thresh_bps: oracle.oracleSettings.confThreshBps,
240
+ volatility_thresh_bps: oracle.oracleSettings.volatilityThreshBps,
241
+ max_slippage_bps: oracle.oracleSettings.maxSlippageBps,
242
+ min_liquidity: parseInt(oracle.oracleSettings.minLiquidity.toString()),
243
+ staleness_thresh: parseInt(oracle.oracleSettings.stalenessThresh.toString()),
244
+ staleness_conf_rate_bps: oracle.oracleSettings.stalenessConfRateBps,
245
+ token_decimals: oracle.oracleSettings.tokenDecimals,
246
+ twap_seconds_ago: parseInt(oracle.oracleSettings.twapSecondsAgo.toString()),
247
+ twap_secondary_seconds_ago: parseInt(oracle.oracleSettings.twapSecondarySecondsAgo.toString()),
248
+ quote: oracle.oracleSettings.quote == 0 ? "usdc" : "wsol",
249
+ side: oracle.oracleSettings.side == 0 ? "base" : "quote",
250
+ },
251
+ accounts_to_load_lut_ids: oracle.accountsToLoadLutIds.map(id => id),
252
+ accounts_to_load_lut_indices: oracle.accountsToLoadLutIndices.map(index => index),
253
+ })),
254
+ min_conf_bps: asset.oracleAggregator.minConfBps,
255
+ conf_thresh_bps: asset.oracleAggregator.confThreshBps,
256
+ conf_multiplier: fractionToDecimal(asset.oracleAggregator.confMultiplier).toNumber(),
257
+ },
258
+ }));
259
+ let formatted: FormattedBasket = {
260
+ pubkey: basket.ownAddress.toBase58(),
261
+ name: basket.metadata?.name || "",
262
+ symbol: basket.metadata?.symbol || "",
263
+ uri: basket.metadata?.uri || "",
264
+
265
+ version: basket.version,
266
+ own_address: basket.ownAddress.toBase58(),
267
+ mint: basket.mint.toBase58(),
268
+ supply_outstanding: parseInt(basket.supplyOutstanding.toString()),
269
+
270
+ creator: basket.settings.creator.toBase58(),
271
+ host: basket.settings.host.toBase58(),
272
+ basket_type: BASKET_TYPES_STRINGS.get(basket.settings.basketType) ?? "private",
273
+ bounty_mint: basket.settings.bountyMint.toBase58(),
274
+ bounty_balance: parseInt(basket.settings.bountyBalance.toString()),
275
+ start_price: fractionToDecimal(basket.settings.startPrice).toNumber(),
276
+ high_watermark: fractionToDecimal(basket.settings.highWaterMark).toNumber(),
277
+ active_rebalance: parseInt(basket.settings.activeRebalance.toString()),
278
+ active_withdraws: parseInt(basket.settings.activeWithdraws.toString()),
279
+ active_managements: parseInt(basket.settings.activeManagements.toString()),
280
+ last_automation_execution_timestamp: parseInt(basket.settings.lastAutomationExecutionTimestamp.toString()),
281
+ creator_settings: creator_settings,
282
+ manager_settings: manager_settings,
283
+ fee_settings: fee_settings,
284
+ schedule_settings: schedule_settings,
285
+ automation_settings: automation_settings,
286
+ lp_settings: lp_settings,
287
+ metadata_settings: metadata_settings,
288
+ deposits_settings: deposits_settings,
289
+ force_rebalance_settings: force_rebalance_settings,
290
+ custom_rebalance_settings: custom_rebalance_settings,
291
+ add_token_settings: add_token_settings,
292
+ update_weights_settings: update_weights_settings,
293
+ make_direct_swap_settings: make_direct_swap_settings,
294
+
295
+ accumulated_fees: accumulated_fees,
296
+ lookup_tables: lookup_tables,
297
+ composition: composition,
298
+ }
299
+ basket = {...basket, formatted: formatted};
300
+ return basket;
301
+ }
302
+
303
+ export async function addLutsToBaskets(
304
+ connection: Connection,
305
+ baskets: Basket[],
306
+ ): Promise<Basket[]> {
307
+ let allLutPubkeys: PublicKey[][] = [];
308
+ for (let basket of baskets)
309
+ allLutPubkeys.push([...basket.lookupTables.active, ...basket.lookupTables.temp]);
310
+ let allInfos = (await getMultipleAddressLookupTableAccounts(connection, [allLutPubkeys]))[0];
311
+ for (let i = 0; i < baskets.length; i++) {
312
+ baskets[i].lutPubkeys = allInfos[i];
313
+ }
314
+ return baskets;
315
+ }
316
+
317
+ export async function fetchBasket(
318
+ connection: Connection,
319
+ basketAddress: PublicKey,
320
+ ): Promise<Basket> {
321
+ const basketAi = await connection.getAccountInfo(basketAddress);
322
+ if (!basketAi) throw new Error("Basket not found");
323
+ let basket = BasketLayout.decode(basketAi.data.slice(8));
324
+ basket = (await addLutsToBaskets(connection, [basket]))[0];
325
+ basket = addFieldsToBasket(basket);
326
+ return basket;
327
+ }
328
+
329
+ export async function fetchBasketsMultiple(
330
+ connection: Connection,
331
+ basketAddresses: PublicKey[],
332
+ ): Promise<Map<string, Basket>> {
333
+ let multipleAccountsInfo = await getMultipleAccountsInfoBatched(connection, basketAddresses);
334
+ let baskets: Basket[] = basketAddresses.map(address => {
335
+ let ai = multipleAccountsInfo.get(address.toBase58());
336
+ if (!ai) return null;
337
+ return BasketLayout.decode(ai.data.slice(8))
338
+ }).filter(basket => basket !== null);
339
+ baskets = await addLutsToBaskets(connection, baskets);
340
+ baskets = baskets.map(basket => addFieldsToBasket(basket));
341
+ let basketsMap: Map<string, Basket> = new Map();
342
+ for (let basket of baskets) {
343
+ basketsMap.set(basket.ownAddress.toBase58(), basket);
344
+ }
345
+ return basketsMap;
346
+ }
347
+
348
+ export interface BasketFilter {
349
+ type: "host" | "creator" | "manager";
350
+ pubkey: string;
351
+ }
352
+
353
+ export async function fetchBaskets(
354
+ connection: Connection,
355
+ filter?: BasketFilter,
356
+ ): Promise<Basket[]> {
357
+ // Managers array begins at this offset in Basket account data.
358
+ const MANAGERS_ARRAY_OFFSET = 8 + 1 + 32 + 32 + 8 + 32 + 32 + 1 + 32 + 8 + 16 + 16 + 8 + 8 + 8 + 8;
359
+ const MANAGER_PUBKEY_SIZE = 32;
360
+
361
+ if (filter?.type === "manager") {
362
+ try {
363
+ const managerPubkey = new PublicKey(filter.pubkey);
364
+ const managerBytes = Buffer.from(managerPubkey.toBytes());
365
+ const managerSliceLength = MAX_MANAGERS_PER_BASKET * MANAGER_PUBKEY_SIZE;
366
+
367
+ // Request only manager bytes for each basket (single GPA), then filter locally.
368
+ const managerSlices = await connection.getProgramAccounts(BASKETS_V3_PROGRAM_ID, {
369
+ commitment: "confirmed",
370
+ filters: [{ dataSize: 8 + BasketLayout.getSpan() }],
371
+ dataSlice: {
372
+ offset: MANAGERS_ARRAY_OFFSET,
373
+ length: managerSliceLength,
374
+ },
375
+ encoding: 'base64',
376
+ });
377
+
378
+ const matchedPubkeys: PublicKey[] = [];
379
+ for (const basketAccount of managerSlices) {
380
+ const managersData = basketAccount.account.data;
381
+ for (let i = 0; i < MAX_MANAGERS_PER_BASKET; i++) {
382
+ const start = i * MANAGER_PUBKEY_SIZE;
383
+ const end = start + MANAGER_PUBKEY_SIZE;
384
+ if (managersData.subarray(start, end).equals(managerBytes)) {
385
+ matchedPubkeys.push(basketAccount.pubkey);
386
+ break;
387
+ }
388
+ }
389
+ }
390
+
391
+ if (matchedPubkeys.length === 0) {
392
+ return [];
393
+ }
394
+
395
+ const matchedAccountsInfo = await getMultipleAccountsInfoBatched(connection, matchedPubkeys);
396
+ const accounts: AccountInfo<Buffer>[] = [];
397
+ for (const pubkey of matchedPubkeys) {
398
+ const accountInfo = matchedAccountsInfo.get(pubkey.toBase58());
399
+ if (accountInfo) {
400
+ accounts.push(accountInfo);
401
+ }
402
+ }
403
+
404
+ let baskets: Basket[] = accounts.map(account => BasketLayout.decode(account.data.slice(8)));
405
+ baskets = await addLutsToBaskets(connection, baskets);
406
+ baskets = baskets.map(basket => addFieldsToBasket(basket));
407
+ return baskets;
408
+ } catch (error) {
409
+ // Fallback for RPC providers that might not support/serve this path reliably.
410
+ }
411
+ }
412
+
413
+ let filters: GetProgramAccountsFilter[] = [{ dataSize: 8 + BasketLayout.getSpan() }];
414
+ if (filter?.type === "creator" || filter?.type === "host") {
415
+ filters.push({
416
+ memcmp: {
417
+ offset: filter?.type === "creator" ? 8 + 1 + 32 + 32 + 8 : 8 + 1 + 32 + 32 + 8 + 32,
418
+ bytes: filter.pubkey
419
+ }
420
+ });
421
+ }
422
+ let accounts = await connection.getProgramAccounts(BASKETS_V3_PROGRAM_ID, {
423
+ commitment: "confirmed",
424
+ filters: filters,
425
+ encoding: 'base64'
426
+ });
427
+ let baskets: Basket[] = accounts.map(account => BasketLayout.decode(account.account.data.slice(8)));
428
+ baskets = await addLutsToBaskets(connection, baskets);
429
+ baskets = baskets.map(basket => addFieldsToBasket(basket));
430
+ return baskets;
431
+ }
432
+
433
+ export async function fetchBasketOracleAccountInfos(
434
+ connection: Connection,
435
+ basket: Basket,
436
+ ): Promise<Map<string, AccountInfo<Buffer> | null>> {
437
+ let allKeys: PublicKey[] = [];
438
+ for (let i = 0; i < basket.numTokens; i++) {
439
+ let agg = basket.composition[i].oracleAggregator;
440
+ for (let j = 0; j < agg.numOracles; j++) {
441
+ const oracleData = agg.oracles[j];
442
+ for (let k = 0; k < oracleData.oracleSettings.numRequiredAccounts; k++) {
443
+ const lutId = oracleData.accountsToLoadLutIds[k];
444
+ const lutIdx = oracleData.accountsToLoadLutIndices[k];
445
+ if (lutId === 0 && lutIdx === 0) continue;
446
+ if (!basket.lutPubkeys) continue;
447
+ allKeys.push(basket.lutPubkeys[lutId].state.addresses[lutIdx]);
448
+ }
449
+ }
450
+ }
451
+ allKeys.push(PYTHNET_CUSTODY_PRICE_WSOL_ACCOUNT);
452
+ allKeys.push(PYTHNET_CUSTODY_PRICE_USDC_ACCOUNT);
453
+
454
+ // load known accounts
455
+ const baseInfos = await getMultipleAccountsInfoBatched(connection, allKeys);
456
+
457
+ // derive and bind tick array PDAs for Raydium CLMM, then batch load them
458
+ const derivedTickArrayKeys: PublicKey[] = [];
459
+ for (let i = 0; i < basket.numTokens; i++) {
460
+ const agg = basket.composition[i].oracleAggregator;
461
+ for (let j = 0; j < agg.numOracles; j++) {
462
+ const oracleData = agg.oracles[j];
463
+ if (oracleData.oracleSettings.oracleType !== OracleType.RaydiumClmm) continue;
464
+
465
+ const lutId = oracleData.accountsToLoadLutIds[0];
466
+ const lutIdx = oracleData.accountsToLoadLutIndices[0];
467
+ if (!basket.lutPubkeys) continue;
468
+ const poolPk = basket.lutPubkeys[lutId].state.addresses[lutIdx];
469
+ const poolAi = baseInfos.get(poolPk.toBase58());
470
+ if (!poolAi) continue;
471
+
472
+ const [poolState] = PoolState.decode(poolAi.data, 8);
473
+ const currStart = getTickArrayStartIndexByTick(poolState.tickCurrent, poolState.tickSpacing);
474
+ const prevStart = getNextTickArrayStartIndex(currStart, poolState.tickSpacing, true);
475
+ const nextStart = getNextTickArrayStartIndex(currStart, poolState.tickSpacing, false);
476
+ [prevStart, currStart, nextStart].forEach(start =>
477
+ derivedTickArrayKeys.push(getPdaTickArrayAddress(poolPk, start))
478
+ );
479
+ }
480
+ }
481
+
482
+ let derivedInfos = new Map<string, AccountInfo<Buffer> | null>();
483
+ derivedInfos = await getMultipleAccountsInfoBatched(connection, derivedTickArrayKeys);
484
+
485
+ derivedInfos.forEach((v, k) => baseInfos.set(k, v));
486
+ return baseInfos;
487
+ }
488
+
489
+ export async function loadBasketPrice(
490
+ basket: Basket,
491
+ connection: Connection,
492
+ ): Promise<Basket> {
493
+ let oracleAccountInfos = await fetchBasketOracleAccountInfos(connection, basket);
494
+
495
+ let solPrice = PythOracle.fetch(
496
+ PYTH_WSOL_ORACLE_SETTINGS, //@ts-ignore
497
+ [oracleAccountInfos.get(PYTHNET_CUSTODY_PRICE_WSOL_ACCOUNT.toBase58())]
498
+ );
499
+ let usdcPrice = PythOracle.fetch(
500
+ PYTH_USDC_ORACLE_SETTINGS, //@ts-ignore
501
+ [oracleAccountInfos.get(PYTHNET_CUSTODY_PRICE_USDC_ACCOUNT.toBase58())]
502
+ );
503
+
504
+ let totalValue = new Decimal(0);
505
+
506
+ for (let i = 0; i < basket.numTokens; i++) {
507
+ const oraclePrice = PriceAggregator.fetch(
508
+ basket.composition[i].oracleAggregator, //@ts-ignore
509
+ [basket.lutPubkeys[0], basket.lutPubkeys[1]],
510
+ oracleAccountInfos,
511
+ solPrice,
512
+ usdcPrice
513
+ );
514
+ let amount = new Decimal(basket.composition[i].amount.toString());
515
+ const value = oraclePrice.price.mul(amount);
516
+ basket.composition[i].price = oraclePrice;
517
+ basket.composition[i].value = value;
518
+ totalValue = totalValue.add(value);
519
+ }
520
+
521
+ basket.tvl = totalValue;
522
+ let supplyOutstanding = new Decimal(basket.supplyOutstanding.toString());
523
+ if (supplyOutstanding.gt(new Decimal(0)))
524
+ basket.price = totalValue.div(supplyOutstanding);
525
+
526
+ return basket;
527
+ }
@@ -0,0 +1,62 @@
1
+ import { Connection } from "@solana/web3.js";
2
+ import { FormattedGlobalConfig, GlobalConfig, GlobalConfigLayout } from "../layouts/config";
3
+ import { getGlobalConfigPda } from "../instructions/pda";
4
+ import { fractionToDecimal } from "../layouts/fraction";
5
+
6
+
7
+ export async function fetchGlobalConfig(connection: Connection): Promise<GlobalConfig> {
8
+ let globalConfig = getGlobalConfigPda();
9
+ let globalConfigData = await connection.getAccountInfo(globalConfig);
10
+ let globalConfigState: GlobalConfig = GlobalConfigLayout.decode(globalConfigData?.data.slice(8) ?? Buffer.alloc(0));
11
+ let formatted: FormattedGlobalConfig = {
12
+ pubkey: globalConfig.toBase58(),
13
+ admin: globalConfigState.admin.toBase58(),
14
+ owners: globalConfigState.owners.map(owner => owner.toBase58()),
15
+ total_number_of_baskets: parseInt(globalConfigState.totalNumberOfBaskets.toString()),
16
+ allow_interactions: globalConfigState.allowInteractions === 1,
17
+ allow_creation: globalConfigState.allowCreation === 1,
18
+ allow_management: globalConfigState.allowManagement === 1,
19
+ allow_automation: globalConfigState.allowAutomation === 1,
20
+ allow_deposits: globalConfigState.allowDeposits === 1,
21
+ allow_withdraws: globalConfigState.allowWithdraws === 1,
22
+ max_deposit_fee_bps: globalConfigState.maxDepositFeeBps,
23
+ max_withdraw_fee_bps: globalConfigState.maxWithdrawFeeBps,
24
+ max_management_fee_bps: globalConfigState.maxManagementFeeBps,
25
+ max_performance_fee_bps: globalConfigState.maxPerformanceFeeBps,
26
+ symmetry_fee_collector: globalConfigState.symmetryFeeCollector.toBase58(),
27
+ symmetry_deposit_fee_bps: globalConfigState.symmetryDepositFeeBps,
28
+ symmetry_deposit_fee_share_bps: globalConfigState.symmetryDepositFeeShareBps,
29
+ symmetry_withdraw_fee_bps: globalConfigState.symmetryWithdrawFeeBps,
30
+ symmetry_withdraw_fee_share_bps: globalConfigState.symmetryWithdrawFeeShareBps,
31
+ symmetry_management_fee_bps: globalConfigState.symmetryManagementFeeBps,
32
+ symmetry_management_fee_share_bps: globalConfigState.symmetryManagementFeeShareBps,
33
+ symmetry_performance_fee_bps: globalConfigState.symmetryPerformanceFeeBps,
34
+ symmetry_performance_fee_share_bps: globalConfigState.symmetryPerformanceFeeShareBps,
35
+ symmetry_trade_fee_bps: globalConfigState.symmetryTradeFeeBps,
36
+ symmetry_limit_order_fee_bps: globalConfigState.symmetryLimitOrderFeeBps,
37
+ bounty_mint: globalConfigState.bountyMint.toBase58(),
38
+ min_bounty_for_basket_automation: parseInt(globalConfigState.minBountyForBasketAutomation.toString()),
39
+ bounty_bond_amount: parseInt(globalConfigState.bountyBondAmount.toString()),
40
+ bounty_per_task: {
41
+ min_bounty: parseInt(globalConfigState.bountyPerTask.minBounty.toString()),
42
+ max_bounty: parseInt(globalConfigState.bountyPerTask.maxBounty.toString()),
43
+ min_bounty_until: parseInt(globalConfigState.bountyPerTask.minBountyUntil.toString()),
44
+ max_bounty_after: parseInt(globalConfigState.bountyPerTask.maxBountyAfter.toString()),
45
+ },
46
+ bounty_per_price_update_task_divisor: parseInt(globalConfigState.bountyPerPriceUpdateTaskDivisor.toString()),
47
+ min_remaining_value: fractionToDecimal(globalConfigState.minRemainingValue).toNumber(),
48
+ min_mint_ratio: fractionToDecimal(globalConfigState.minMintRatio).toNumber(),
49
+ apr_bps_per_year: globalConfigState.aprBpsPerYear,
50
+ rebalance_intent_lifetime: parseInt(globalConfigState.rebalanceIntentLifetime.toString()),
51
+ rebalance_auction_1_timeframe: parseInt(globalConfigState.rebalanceAuction1Timeframe.toString()),
52
+ rebalance_auction_2_timeframe: parseInt(globalConfigState.rebalanceAuction2Timeframe.toString()),
53
+ rebalance_auction_3_timeframe: parseInt(globalConfigState.rebalanceAuction3Timeframe.toString()),
54
+ price_update_delay_after_creation: parseInt(globalConfigState.priceUpdateDelayAfterCreation.toString()),
55
+ price_update_lifetime: parseInt(globalConfigState.priceUpdateLifetime.toString()),
56
+ price_update_reexecution_delay: parseInt(globalConfigState.priceUpdateReexecutionDelay.toString()),
57
+ };
58
+ return {
59
+ ...globalConfigState,
60
+ formatted: formatted,
61
+ };
62
+ }