@gala-chain/launchpad-sdk 4.0.1-beta.9 → 4.0.4-beta.0

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 (72) hide show
  1. package/dist/index.cjs.js +1 -1
  2. package/dist/index.d.ts +5 -6
  3. package/dist/index.d.ts.map +1 -1
  4. package/dist/index.esm.js +1 -1
  5. package/dist/index.js +1 -1
  6. package/dist/src/LaunchpadSDK.d.ts +103 -89
  7. package/dist/src/LaunchpadSDK.d.ts.map +1 -1
  8. package/dist/src/api/LaunchpadAPI.d.ts +20 -4
  9. package/dist/src/api/LaunchpadAPI.d.ts.map +1 -1
  10. package/dist/src/constants/counts.d.ts +4 -7
  11. package/dist/src/constants/counts.d.ts.map +1 -1
  12. package/dist/src/constants/endpoints.d.ts +0 -3
  13. package/dist/src/constants/endpoints.d.ts.map +1 -1
  14. package/dist/src/constants/version.generated.d.ts +1 -1
  15. package/dist/src/index.d.ts +5 -6
  16. package/dist/src/index.d.ts.map +1 -1
  17. package/dist/src/schemas/launchpad.d.ts +24 -6
  18. package/dist/src/schemas/launchpad.d.ts.map +1 -1
  19. package/dist/src/schemas/pagination.d.ts +0 -16
  20. package/dist/src/schemas/pagination.d.ts.map +1 -1
  21. package/dist/src/schemas/primitives.d.ts +0 -9
  22. package/dist/src/schemas/primitives.d.ts.map +1 -1
  23. package/dist/src/schemas/user.d.ts +48 -16
  24. package/dist/src/schemas/user.d.ts.map +1 -1
  25. package/dist/src/schemas/validators.d.ts +1 -13
  26. package/dist/src/schemas/validators.d.ts.map +1 -1
  27. package/dist/src/services/GSwapService.d.ts +11 -1
  28. package/dist/src/services/GSwapService.d.ts.map +1 -1
  29. package/dist/src/services/GalaChainGatewayClient.d.ts.map +1 -1
  30. package/dist/src/services/GalaChainService.d.ts +9 -1
  31. package/dist/src/services/GalaChainService.d.ts.map +1 -1
  32. package/dist/src/services/ImageService.d.ts.map +1 -1
  33. package/dist/src/services/LaunchpadService.d.ts +2 -34
  34. package/dist/src/services/LaunchpadService.d.ts.map +1 -1
  35. package/dist/src/services/PoolService.d.ts.map +1 -1
  36. package/dist/src/services/UserService.d.ts.map +1 -1
  37. package/dist/src/types/dto.d.ts +6 -0
  38. package/dist/src/types/dto.d.ts.map +1 -1
  39. package/dist/src/types/galachain-api.types.d.ts +39 -16
  40. package/dist/src/types/galachain-api.types.d.ts.map +1 -1
  41. package/dist/src/types/gswap-responses.types.d.ts +8 -5
  42. package/dist/src/types/gswap-responses.types.d.ts.map +1 -1
  43. package/dist/src/types/gswap.dto.d.ts +13 -5
  44. package/dist/src/types/gswap.dto.d.ts.map +1 -1
  45. package/dist/src/types/launchpad.dto.d.ts +38 -3
  46. package/dist/src/types/launchpad.dto.d.ts.map +1 -1
  47. package/dist/src/types/options.dto.d.ts +0 -30
  48. package/dist/src/types/options.dto.d.ts.map +1 -1
  49. package/dist/src/types/result.types.d.ts +1 -18
  50. package/dist/src/types/result.types.d.ts.map +1 -1
  51. package/dist/src/types/user.dto.d.ts +65 -35
  52. package/dist/src/types/user.dto.d.ts.map +1 -1
  53. package/dist/src/types/websocket-data.types.d.ts +0 -8
  54. package/dist/src/types/websocket-data.types.d.ts.map +1 -1
  55. package/dist/src/utils/auto-pagination.d.ts +2 -5
  56. package/dist/src/utils/auto-pagination.d.ts.map +1 -1
  57. package/dist/src/utils/error-factories.d.ts +2 -2
  58. package/dist/src/utils/query-params.d.ts +0 -1
  59. package/dist/src/utils/query-params.d.ts.map +1 -1
  60. package/dist/src/utils/response-handlers.d.ts +10 -10
  61. package/dist/src/utils/validation-helpers.d.ts +3 -12
  62. package/dist/src/utils/validation-helpers.d.ts.map +1 -1
  63. package/dist/src/utils/validation.d.ts +2 -2
  64. package/dist/src/utils/websocket-validators.d.ts +1 -5
  65. package/dist/src/utils/websocket-validators.d.ts.map +1 -1
  66. package/package.json +1 -1
  67. package/dist/src/services/CommentService.d.ts +0 -63
  68. package/dist/src/services/CommentService.d.ts.map +0 -1
  69. package/dist/src/services/FaucetService.d.ts +0 -55
  70. package/dist/src/services/FaucetService.d.ts.map +0 -1
  71. package/dist/src/types/comment.dto.d.ts +0 -169
  72. package/dist/src/types/comment.dto.d.ts.map +0 -1
package/dist/index.esm.js CHANGED
@@ -1 +1 @@
1
- import e from"axios";import{ethers as t,Wallet as n,Mnemonic as o,HDNodeWallet as i}from"ethers";import{SigningType as r,SigningClient as a,ChainCallDTO as s,calculatePersonalSignPrefix as c}from"@gala-chain/connect";import{z as l}from"zod";import u from"bignumber.js";import{tickToSqrtPrice as d,getLiquidityForAmounts as h,getAmountsForLiquidity as g,liquidity0 as p,liquidity1 as m,getAmount0Delta as f,getAmount1Delta as y,Pool as w,TickData as k,CompositePoolDto as v,GetCompositePoolDto as b,QuoteExactAmountDto as T,quoteExactAmount as S,sqrtPriceToTick as A,computeSwapStep as E}from"@gala-chain/dex";import{TokenClassKey as I,TokenBalance as P}from"@gala-chain/api";import{v4 as N}from"uuid";import{io as D}from"socket.io-client";import*as F from"path";import*as C from"fs";import*as x from"dotenv";import $ from"crypto";if("undefined"==typeof File){const{File:e}=require("web-file-polyfill");global.File=e}var L,O;!function(e){e.WALLET_NOT_CONNECTED="WALLET_NOT_CONNECTED",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.INVALID_ADDRESS="INVALID_ADDRESS",e.MESSAGE_GENERATION_FAILED="MESSAGE_GENERATION_FAILED"}(L||(L={}));class _ extends Error{constructor(e,t,n){super(t),this.type=e,this.originalError=n,this.name="AuthError"}}class B{constructor(e){if(this.wallet=e.wallet,this.messagePrefix=e.messagePrefix||"Create a GalaChain Wallet",""===e.messagePrefix)throw new _(L.SIGNATURE_FAILED,"Message prefix cannot be empty")}hasWallet(){return void 0!==this.wallet}setWallet(e){if(void 0!==e){if("object"!=typeof e||!("address"in e))throw new _(L.WALLET_NOT_CONNECTED,"Invalid wallet: must be an ethers Wallet instance or undefined");if(!e.address||"string"!=typeof e.address)throw new _(L.INVALID_ADDRESS,"Wallet address is not available")}this.wallet=e}async generateSignature(){this.validateWallet();try{const e=Date.now(),t=`${this.messagePrefix} ${e}`,n=await this.wallet.signMessage(t);return{message:t,signature:n,address:this.formatAddress(this.wallet.address),timestamp:e}}catch(e){if(e instanceof _)throw e;throw new _(L.SIGNATURE_FAILED,"Failed to generate signature for authentication",e instanceof Error?e:new Error(String(e)))}}getAddress(){return this.validateWallet(),this.formatAddress(this.wallet.address)}getEthereumAddress(){return this.validateWallet(),this.wallet.address}getPrivateKey(){if(this.validateWallet(),!this.wallet.privateKey)throw new _(L.WALLET_NOT_CONNECTED,"Wallet private key not available for @gala-chain signing");return this.wallet.privateKey}formatAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new _(L.INVALID_ADDRESS,`Invalid Ethereum address format: ${e}`);return`eth|${t}`}async signMessage(e){this.validateWallet();try{return{message:e,signature:await this.wallet.signMessage(e),address:this.wallet.address,timestamp:Date.now()}}catch(e){if(e instanceof _)throw e;const t=e instanceof Error?e.message:String(e);throw new _(L.SIGNATURE_FAILED,t,e instanceof Error?e:new Error(String(e)))}}async generateAuthHeaders(e,t){this.validateWallet();try{const n=Date.now(),o=`${this.messagePrefix} ${t.toUpperCase()} ${e} ${n}`,i=await this.wallet.signMessage(o);return{"x-signature":i,"x-address":this.formatAddress(this.wallet.address),"x-message":o,"x-timestamp":n.toString()}}catch(e){if(e instanceof _)throw e;throw new _(L.SIGNATURE_FAILED,"Failed to generate authentication headers",e instanceof Error?e:new Error(String(e)))}}async signTypedData(e,t,n){this.validateWallet();try{return await this.wallet.signTypedData(e,t,n)}catch(e){if(e instanceof _)throw e;throw new _(L.SIGNATURE_FAILED,"Failed to sign typed data",e instanceof Error?e:new Error(String(e)))}}async generateCustomSignature(e){if(!e||"string"!=typeof e||0===e.trim().length)throw new _(L.SIGNATURE_FAILED,"Custom message must be a non-empty string");this.validateWallet();try{const t=await this.wallet.signMessage(e);return{message:e,signature:t,address:this.formatAddress(this.wallet.address),timestamp:Date.now()}}catch(e){if(e instanceof _)throw e;throw new _(L.SIGNATURE_FAILED,"Failed to generate custom message signature",e instanceof Error?e:new Error(String(e)))}}validateWallet(){if(!this.wallet)throw new _(L.WALLET_NOT_CONNECTED,"Wallet is required for authentication");if(!this.wallet.address)throw new _(L.WALLET_NOT_CONNECTED,"Wallet address is not available");if(!this.wallet.privateKey&&!this.wallet.signMessage)throw new _(L.WALLET_NOT_CONNECTED,"Wallet must have a private key for signing messages")}}!function(e){e.DEBUG="DEBUG",e.INFO="INFO",e.WARN="WARN",e.ERROR="ERROR"}(O||(O={}));class U{constructor(e){this.levelPriority={[O.DEBUG]:0,[O.INFO]:1,[O.WARN]:2,[O.ERROR]:3},this.debugEnabled=e.debug,this.context=e.context||"SDK",this.minLevel=e.minLevel||(e.debug?O.DEBUG:O.INFO)}debug(e,t){this.log(O.DEBUG,e,t)}info(e,t){this.log(O.INFO,e,t)}warn(e,t){this.log(O.WARN,e,t)}error(e,t){this.log(O.ERROR,e,t)}log(e,t,n){if(this.levelPriority[e]<this.levelPriority[this.minLevel])return;if(e===O.DEBUG&&!this.debugEnabled)return;const o=`[${(new Date).toISOString()}] [${this.context}] [${e}]`,i=this.getConsoleMethod(e);void 0!==n?n instanceof Error?i(`${o} ${t}`,n.message,n.stack):i(`${o} ${t}`,n):i(`${o} ${t}`)}getConsoleMethod(e){switch(e){case O.DEBUG:return console.debug;case O.INFO:return console.info;case O.WARN:return console.warn;case O.ERROR:return console.error;default:return console.log}}child(e){return new U({debug:this.debugEnabled,context:`${this.context}:${e}`,minLevel:this.minLevel})}isDebugEnabled(){return this.debugEnabled&&this.levelPriority[O.DEBUG]>=this.levelPriority[this.minLevel]}}function M(e){if(!e||"object"!=typeof e)return{};const t={};for(const[n,o]of Object.entries(e))null!=o&&("string"==typeof o?t[n]=o:"number"==typeof o||"boolean"==typeof o?t[n]=o.toString():Array.isArray(o)?t[n]=o.join(","):t[n]="object"==typeof o?JSON.stringify(o):String(o));return t}class R{constructor(t,n={}){this.auth=t,this.debug=n.debug??!1,this.logger=new U({debug:this.debug,context:"HttpClient"}),this.axios=e.create({baseURL:n.baseUrl||"https://lpad-backend-dev1.defi.gala.com",timeout:n.timeout||3e4,headers:{Accept:"application/json",...n.headers}}),this.setupInterceptors()}async request(e){try{const t={method:e.method,url:e.url,data:e.data,...e.params&&{params:M(e.params)},...e.headers&&{headers:e.headers},...e.timeout&&{timeout:e.timeout}};e.headers&&this.logger.debug("Custom headers provided:",e.headers),e.data instanceof FormData&&(t.headers&&t.headers["Content-Type"]&&delete t.headers["Content-Type"],this.logger.debug("FormData detected - removing Content-Type header for multipart upload"));const n=e.data instanceof FormData?"[FormData object - multipart/form-data]":e.data;this.logger.debug("Request:",{method:e.method,url:e.url,fullUrl:`${this.axios.defaults.baseURL}${e.url}`,baseURL:this.axios.defaults.baseURL,params:t.params,data:n,isFormData:e.data instanceof FormData,contentType:t.headers?.["Content-Type"]||"not set"});const o=await this.axios.request(t);return this.logger.debug("Response:",{status:o.status,data:o.data}),o.data}catch(e){throw this.logger.error("Error:",e),e}}async get(e,t,n){return this.request({method:"GET",url:e,...t&&{params:t},...n&&{headers:n}})}async post(e,t,n){return this.request({method:"POST",url:e,data:t,...n&&{headers:n}})}async put(e,t,n){return this.request({method:"PUT",url:e,data:t,...n&&{headers:n}})}async delete(e,t,n){return this.request({method:"DELETE",url:e,...t&&{params:t},...n&&{headers:n}})}async patch(e,t,n){return this.request({method:"PATCH",url:e,data:t,...n&&{headers:n}})}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.auth.getEthereumAddress()}async signMessage(e){return(await this.auth.signMessage(e)).signature}async signTypedData(e,t,n){return await this.auth.signTypedData(e,t,n)}async signCustomMessage(e){try{const t=await this.auth.generateCustomSignature(e);return this.logger.debug("Generated custom signature:",{message:e,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}),{signature:t.signature,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}}catch(e){throw this.logger.error("Custom signature generation failed:",e),new Error(`Failed to generate custom signature for message: ${e instanceof Error?e.message:"Unknown error"}`)}}async signWithGalaChain(e,t,n=r.SIGN_TYPED_DATA){const o=this.auth.getPrivateKey(),i=new a(o);return await i.sign(e,t,n)}setupInterceptors(){this.requestInterceptorId=this.axios.interceptors.request.use(async e=>{try{if(e.headers||(e.headers={}),this.auth.hasWallet()){const t=await this.auth.generateSignature();e.headers.Sign=t.signature,this.logger.debug("Added signature header:",{address:t.address,message:t.message,timestamp:t.timestamp})}else this.logger.debug("No wallet configured - skipping signature header");return e.data instanceof FormData||(e.headers["Content-Type"]="application/json"),this.logger.debug("Final request headers being sent:",e.headers),e}catch(e){throw this.logger.error("Failed to add signature:",e),e}},e=>Promise.reject(e)),this.responseInterceptorId=this.axios.interceptors.response.use(e=>e,e=>{if(e.response){const t={message:e.response.data?.message||e.message,error:e.response.data?.error,statusCode:e.response.status,details:e.response.data?.details,timestamp:e.response.data?.timestamp,path:e.response.data?.path};e.launchpadError=t,this.logger.error("Backend error:",t)}else e.request?this.logger.error("Network error:",e.message):this.logger.error("Request setup error:",e.message);return Promise.reject(e)})}cleanup(){void 0!==this.requestInterceptorId&&(this.axios.interceptors.request.eject(this.requestInterceptorId),this.requestInterceptorId=void 0),void 0!==this.responseInterceptorId&&(this.axios.interceptors.response.eject(this.responseInterceptorId),this.responseInterceptorId=void 0),this.logger.debug("Interceptors cleaned up")}}const q="Token name is required and must be a string",K=e=>`Could not find vault address for token: ${e}`;class G extends Error{constructor(e,t,n){super(e),this.field=t,this.code=n,this.name="ValidationError"}}class W extends Error{constructor(e,t,n){super(e),this.statusCode=t,this.originalError=n,this.name="NetworkError"}}class j extends Error{constructor(e,t){super(e),this.field=t,this.name="ConfigurationError"}}class V extends Error{constructor(e,t,n){super(e),this.transactionId=t,this.code=n,this.name="TransactionError"}}class z extends Error{constructor(e,t,n){super(e),this.originalError=t,this.code=n,this.name="GSwapQuoteError"}}class H extends Error{constructor(e,t,n,o){super(e),this.originalError=t,this.transactionHash=n,this.code=o,this.name="GSwapSwapError"}}class Q extends Error{constructor(e,t,n,o,i){super(e),this.originalError=t,this.tokenA=n,this.tokenB=o,this.code=i,this.name="GSwapPoolError"}}class X extends Error{constructor(e,t,n,o){super(e),this.originalError=t,this.walletAddress=n,this.code=o,this.name="GSwapAssetError"}}class Z extends Error{constructor(e,t,n){super(e),this.originalError=t,this.code=n,this.name="GSwapPositionError"}}class Y extends G{constructor(e,t){super(e,"dexQuote","DEX_QUOTE_ERROR"),this.context=t,this.name="DexQuoteError"}}class J extends G{constructor(e){super(e,"dexPool","DEX_POOL_NOT_FOUND"),this.name="DexPoolNotFoundError"}}function ee(e,t){return new G(`Token "${e}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND")}function te(e,t){const n=t||e.charAt(0).toUpperCase()+e.slice(1);return new G(`${n} is required`,e,"REQUIRED_FIELD")}function ne(e,t,n){const o=n||e.charAt(0).toUpperCase()+e.slice(1);return new G(`${o} must be ${t}`,e,"INVALID_FORMAT")}function oe(e,t,n){return new W(e,t,n)}function ie(e,t){return new j(e,t)}function re(e,t,n){return new V(e,t,n)}function ae(e,t,n){const{MIN_PAGE:o,MAX_PAGE:i,MIN_LIMIT:r,MAX_LIMIT:a}=n.PAGINATION;if("number"!=typeof e||e<o||e>i)throw new G(`Page must be a number between ${o} and ${i}`,"page","INVALID_PAGE");if("number"!=typeof t||t<r||t>a)throw new G(`Limit must be a number between ${r} and ${a}`,"limit","INVALID_LIMIT")}const se={ETH_ADDRESS:/^0x[0-9a-fA-F]{40}$/,BACKEND_ADDRESS:/^eth\|[0-9a-fA-F]{40}$/};function ce(e){return"string"==typeof e&&e.trim().length>0}function le(e){return!(!e||"string"!=typeof e)&&(se.ETH_ADDRESS.test(e)||se.BACKEND_ADDRESS.test(e))}function ue(e){return e.startsWith("0x")?`eth|${e.slice(2)}`:e}const de=l.string().min(3,"Token name must be at least 3 characters").max(20,"Token name must be at most 20 characters").regex(/^[a-zA-Z0-9]{3,20}$/,"Token name can only contain letters and numbers"),he=l.string().min(1,"Token symbol must be at least 1 character").max(8,"Token symbol must be at most 8 characters").regex(/^[A-Z]{1,8}$/,"Token symbol must be uppercase letters only"),ge=l.string().min(1,"Token description is required").max(500,"Token description must be at most 500 characters"),pe=l.string().min(1,"Token name must be at least 1 character").max(50,"Token name must be at most 50 characters"),me=l.string().min(1,"Search query must be at least 1 character").max(100,"Search query must be at most 100 characters"),fe=l.string().min(1,"Full name is required").max(100,"Full name must be at most 100 characters").regex(/^[a-zA-Z\s]+$/,"Full name can only contain letters and spaces"),ye=l.string().regex(se.BACKEND_ADDRESS,"Address must be in format: eth|[40-hex-chars]"),we=l.string().regex(se.ETH_ADDRESS,"Invalid Ethereum address format"),ke=l.string().refine(e=>se.BACKEND_ADDRESS.test(e)||se.ETH_ADDRESS.test(e),"Address must be either eth|[40-hex-chars] or 0x[40-hex-chars] format").transform(e=>e.startsWith("0x")?`eth|${e.slice(2)}`:e),ve=l.string().refine(e=>se.BACKEND_ADDRESS.test(e)||/^service\|Token\$Unit\$[A-Z0-9]+\$eth:[0-9a-fA-F]{40}\$launchpad$/.test(e),"Invalid vault address format"),be=l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>0,"Amount must be greater than zero"),Te=l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>=0,"Amount must be zero or greater"),Se=l.string().regex(/^(?!0+(\.0+)?$)\d+(\.\d+)?$/,"Amount must be a positive, non-zero number"),Ae=l.string().url("Must be a valid URL").regex(/^https?:\/\//,"URL must start with http:// or https://"),Ee=l.string().optional().refine(e=>!e||/^https?:\/\/.+\..+/.test(e),"Must be a valid URL if provided"),Ie=l.number().int("Page must be an integer").min(1,"Page must be at least 1").max(1e3,"Page must be at most 1000").default(1);function Pe(e=100){return l.number().int("Limit must be an integer").min(1,"Limit must be at least 1").max(e,`Limit must be at most ${e}`).default(10)}const Ne=Pe(100),De=Pe(20),Fe=Pe(20),Ce=l.number().int("File size must be an integer").min(1,"File must be at least 1 byte").max(10485760,"File must be at most 10MB"),xe=l.string().max(255,"Filename must be at most 255 characters"),$e=l.enum(["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"]),Le=l.string().min(1,"Comment message is required").max(500,"Comment must be at most 500 characters"),Oe=l.string().datetime("Must be a valid ISO 8601 date string"),_e=l.number().int("Timestamp must be an integer").min(0,"Timestamp must be non-negative"),Be=l.string().regex(/^0x[a-fA-F0-9]{64}$/,"Private key must be format: 0x + 64 hex characters"),Ue=l.string().regex(/^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$/,"Transaction ID must be in UUID format"),Me=l.string().regex(/^galaconnect-operation-[a-z0-9-]+$/,"Unique key must be format: galaconnect-operation-{unique-id}"),Re=l.object({websiteUrl:Ee,telegramUrl:Ee,twitterUrl:Ee}).refine(e=>e.websiteUrl||e.telegramUrl||e.twitterUrl,"At least one social URL (website, telegram, or twitter) is required"),qe=l.string().min(1,"Token category must not be empty").default("Unit"),Ke=l.string().min(1,"Token collection must not be empty").default("Token"),Ge=l.object({minFeePortion:be,maxFeePortion:be}),We=l.object({tokenName:de,tokenSymbol:he,tokenDescription:ge,tokenImage:l.union([l.instanceof(File),l.instanceof(Buffer),l.string().url("Token image must be a valid URL")]).optional(),preBuyQuantity:Te.default("0"),websiteUrl:Ee,telegramUrl:Ee,twitterUrl:Ee,tokenCategory:qe,tokenCollection:Ke,reverseBondingCurveConfiguration:Ge.optional(),privateKey:Be.optional()}),je=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),tokenName:de}),Ve=l.enum(["recent","popular"]),ze=l.object({tokenName:de.optional(),symbol:he.optional()}).refine(e=>e.tokenName||e.symbol,"At least one of tokenName or symbol is required"),He=l.enum(["NATIVE","MEME"]),Qe=l.enum(["IN","OUT"]),Xe=l.object({from:l.number().int("From timestamp must be an integer").min(173e6,"From timestamp must be at least 173000000"),to:l.number().int("To timestamp must be an integer").min(173e6,"To timestamp must be at least 173000000"),resolution:l.number().int("Resolution must be an integer").min(1,"Resolution must be at least 1"),tokenName:de}),Ze=l.object({tokenName:de,slippageToleranceFactor:l.number().min(0).max(1).optional(),maxAcceptableReverseBondingCurveFeeSlippageFactor:l.number().min(0).max(1).optional(),privateKey:Be.optional()}),Ye=[".png",".jpg",".jpeg",".gif",".webp",".svg"],Je=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),name:xe,size:Ce,type:$e}),et=l.instanceof(File).refine(e=>e.size>=1&&e.size<=10485760,"File size must be between 1 byte and 10MB").refine(e=>["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"].includes(e.type),"File must be a valid image type (PNG, JPG, JPEG, GIF, WebP, or SVG)").refine(e=>e.name.length<=255,"Filename must be at most 255 characters"),tt=l.instanceof(Buffer).refine(e=>e.length>=1&&e.length<=10485760,"Buffer size must be between 1 byte and 10MB"),nt=l.union([et,tt]),ot=l.enum([".png",".jpg",".jpeg",".gif",".webp",".svg"]),it=xe.refine(e=>{const t=e.slice(e.lastIndexOf(".")).toLowerCase();return Ye.includes(t)},`Filename must end with one of: ${Ye.join(", ")}`),rt=l.object({page:Ie,limit:Ne}),at=l.object({page:Ie,limit:De}),st=l.object({page:Ie,limit:Fe}),ct=l.object({page:Ie,limit:Pe(50)}),lt=rt.extend({type:l.enum(["recent","popular"]).optional(),tokenName:l.string().min(1).max(50).optional(),search:l.string().min(1).max(100).optional()}),ut=at.extend({tokenName:l.string().min(1).max(50).optional(),search:l.string().min(1).max(100).optional()}),dt=st.extend({tradeType:l.enum(["BUY","SELL"]).optional(),tokenName:l.string().min(1).max(50).optional(),userAddress:l.string().regex(/^(0x[a-fA-F0-9]{40}|eth\|[a-fA-F0-9]{40})$/).optional(),startDate:l.string().datetime().optional(),endDate:l.string().datetime().optional(),sortOrder:l.enum(["ASC","DESC"]).default("DESC")}),ht=l.object({page:l.number().int().min(1),limit:l.number().int().min(1),total:l.number().int().min(0),totalPages:l.number().int().min(0),hasNext:l.boolean(),hasPrevious:l.boolean()});function gt(e){return l.object({data:l.array(e),page:l.number().int().min(1),limit:l.number().int().min(1),total:l.number().int().min(0),totalPages:l.number().int().min(0),hasNext:l.boolean(),hasPrevious:l.boolean()})}const pt=l.enum(["all","DEFI","ASSET"]),mt=at.extend({type:pt.optional(),address:ke.optional(),search:me.optional(),tokenName:pe.optional()}),ft=l.object({walletAddress:ke,amount:Se}),yt=l.object({address:ke.optional(),refresh:l.boolean().optional()}),wt=l.object({profileImage:l.string(),fullName:fe,address:ke,privateKey:Be.optional()}),kt=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),address:ke.optional(),privateKey:Be.optional()}),vt=l.object({address:ke,tokenId:l.union([l.string(),l.object({collection:l.string(),category:l.string(),type:l.string(),additionalKey:l.string()}),l.object({collection:l.string(),category:l.string(),type:l.string(),additionalKey:l.string(),instance:l.string()})]).optional(),tokenName:pe.optional()}).refine(e=>void 0!==e.tokenId||void 0!==e.tokenName,"At least one token identifier (tokenId or tokenName) is required"),bt=l.enum(["buy","sell"]),Tt=l.enum(["BUY","SELL"]),St=l.object({tradeType:bt,tokenAmount:be,vaultAddress:ve,userAddress:ke,slippageTolerance:be.optional(),deadline:l.number().int().positive().optional()}),At=l.object({tokenSymbol:he,nativeTokenQuantity:be,expectedToken:be,maxAcceptableReverseBondingCurveFee:Te.default("0").optional()}),Et=l.object({tokenSymbol:he,tokenQuantity:be,expectedNativeToken:be,maxAcceptableReverseBondingCurveFee:Te.default("0").optional()}),It=st.extend({tokenName:pe.optional()}),Pt=l.object({page:l.number().int().min(1).max(1e3).default(1).optional(),limit:l.number().int().min(1).max(20).default(10).optional()}),Nt=l.enum(["NATIVE","MEME"]),Dt=l.enum(["IN","OUT"]),Ft=l.object({type:Nt,method:Dt,vaultAddress:ve,amount:be}),Ct=l.object({nativeTokenQuantity:be}),xt=l.object({vaultAddress:ve}),$t=l.object({minFeePortion:be,maxFeePortion:be});function Lt(e){return t=>{const n=e.safeParse(t);return{success:n.success,data:n.success?n.data:void 0,errors:n.success?void 0:n.error.errors.map(e=>e.message)}}}const Ot=Lt(de),_t=Lt(he),Bt=Lt(ge),Ut=Lt(ke),Mt=Lt(ve),Rt=Lt(be),qt=Lt(Se),Kt=Lt(fe),Gt=Lt(me),Wt=Lt(pe),jt=Lt(We),Vt=Lt(Re),zt=Lt(je),Ht=Lt(ze),Qt=Lt(mt),Xt=Lt(ft),Zt=Lt(yt),Yt=Lt(wt),Jt=Lt(kt),en=Lt(vt),tn=Lt(St),nn=Lt(At),on=Lt(Et),rn=Lt(It),an=Lt(Pt),sn=Lt(Ft),cn=Lt(Ct),ln=Lt(xt);function un(e,t){throw new G(e.join("; "),t,"VALIDATION_ERROR")}function dn(e){const t=Ot(e);!t.success&&t.errors&&un(t.errors,"tokenName")}function hn(e){const t=lt.safeParse(e);t.success||un(t.error.errors.map(e=>e.message),"pagination")}function gn(e){const t=Ht(e);!t.success&&t.errors&&un(t.errors,"options")}function pn(e){const t=sn(e);!t.success&&t.errors&&un(t.errors,"options")}function mn(e){const t=Xe.safeParse(e);t.success||un(t.error.errors.map(e=>e.message),"options")}function fn(e){const t=ke.safeParse(e);if(!t.success)throw new G("Ethereum address must be 40 hex characters (with or without 0x prefix)","ethereumAddress","INVALID_FORMAT");return t.data}function yn(e,t,n=!0){if(!e||""===e.trim())throw new G(`${t} cannot be empty or whitespace-only. Provide a valid numeric string or omit the parameter to auto-fetch.`,t,"INVALID_NUMERIC_STRING");if(/[eE]/.test(e))throw new G(`${t} cannot use scientific notation. Use standard decimal format (e.g., "1000" instead of "1e3").`,t,"INVALID_NUMERIC_STRING");const o=parseFloat(e);if(isNaN(o))throw new G(`${t} must be a valid numeric string. Received: "${e}"`,t,"INVALID_NUMERIC_STRING");if(!isFinite(o))throw new G(`${t} must be a finite number. Cannot be Infinity or -Infinity.`,t,"INVALID_NUMERIC_STRING");if(o<0)throw new G(`${t} must be non-negative. Received: "${e}"`,t,"INVALID_NUMERIC_STRING");if(!n&&0===o)throw new G(`${t} must be greater than zero. Received: "${e}"`,t,"INVALID_NUMERIC_STRING")}var wn=Object.freeze({__proto__:null,normalizeAddressInput:function(e){if(!e)return;const t=ke.safeParse(e);if(!t.success)throw new G(`Invalid address format: ${e}. Must be either "0x..." (Ethereum) or "eth|..." (GalaChain) format`,"address","INVALID_FORMAT");return t.data},toBackendAddressFormat:fn,validateCheckPoolOptions:gn,validateGetAmountOptions:pn,validateGetGraphOptions:mn,validateNumericString:yn,validatePagination:hn,validateTokenName:dn});function kn(e,t){const n=e,o=Number(n.page)||t.page,i=Number(n.limit)||t.limit,r=n.data,a=Number(n.total)||Number(r?.count)||0;return{page:o,limit:i,total:a,totalPages:Math.ceil(a/i)}}function vn(e,t){return{hasNext:e<t,hasPrevious:e>1}}function bn(e,t,n=!1){const o=!0===e.error||200!==e.status,i=n&&!e.data;if(o||i)throw new Error(e.message||t)}const Tn="/launchpad/upload-image",Sn="/launchpad/fetch-pool",An="/launchpad/check-pool",En="/launchpad/get-graph-data",In="/holders",Pn="/launchpad/get-badge/",Nn="/trade/",Dn="/token/commment",Fn="/token/commment",Cn="/user/profile",xn="/user/profile",$n="/user/token-list",Ln="/user/token-hold",On="/user/transfer-faucets",_n=5,Bn={DEFAULT_PAGE:1,DEFAULT_LIMIT:10,BACKEND_MAX_PAGE_SIZE:20,SAFETY_MAX_PAGES:100},Un={MAX_CONCURRENT_FETCHES:5,BACKEND_PAGE_SIZE:20};class Mn{constructor(e,t=!1){this.http=e,this.logger=new U({debug:t,context:this.constructor.name})}}class Rn{constructor(e=!1){this.logger=new U({debug:e,context:this.constructor.name})}}function qn(e,t){return"string"==typeof e[t]}function Kn(e,t){return void 0===e[t]||"string"==typeof e[t]}function Gn(e,t){return void 0===e[t]||"number"==typeof e[t]}function Wn(e){return void 0===e.calculateAmountMode||"local"===e.calculateAmountMode||"external"===e.calculateAmountMode}function jn(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&Gn(t,"from")&&Gn(t,"to")&&Gn(t,"resolution")}class Vn extends Mn{constructor(e,t=!1){super(e,t)}async fetchSinglePage(e){const t={page:e.page.toString(),limit:e.limit.toString()};void 0!==e.type&&(t.type=e.type),void 0!==e.tokenName&&(t.tokenName=e.tokenName),void 0!==e.search&&(t.search=e.search);const n=M(t),o=await this.http.get(Sn,n);if(!o)throw oe("No response from pool service",500);bn(o,"Failed to fetch pools",!0);const i=function(e){if(!e)return[];let t=[];if(e.tokens)if(Array.isArray(e.tokens))t=e.tokens.map(e=>({...e,createdAt:e.created_at||e.createdAt||""}));else{const n=e.tokens;t=[{...n,createdAt:n.created_at||n.createdAt||""}]}else e.pools&&Array.isArray(e.pools)&&(t=e.pools.map(e=>({...e,createdAt:e.created_at||e.createdAt||""})));return t}(o.data),r=o.data.count??o.data.total??0;return{pools:i,total:r,totalPages:e.limit>0?Math.ceil(r/e.limit):1}}async fetchMultiplePages(e){const{startPage:t,totalPages:n,pageSize:o,...i}=e,r=[];if(n&&n>=t){const e=[];for(let o=t;o<=n;o++)e.push(o);for(let t=0;t<e.length;t+=5){const n=e.slice(t,t+5).map(e=>this.fetchSinglePage({...i,page:e,limit:o}).catch(e=>{if(400===e?.launchpadError?.statusCode)return{pools:[],total:0,totalPages:0};throw e}));(await Promise.all(n)).forEach(e=>{r.push(...e.pools)})}return r}let a=t,s=!0;for(;s;){const e=[];for(let t=0;t<5&&s;t++)e.push(a+t);const t=e.map(e=>this.fetchSinglePage({...i,page:e,limit:o}).catch(e=>{if(400===e?.launchpadError?.statusCode)return{pools:[],total:0,totalPages:0};throw e})),n=await Promise.all(t);for(const e of n){if(0===e.pools.length){s=!1;break}r.push(...e.pools)}a+=e.length,a>Bn.SAFETY_MAX_PAGES&&(s=!1)}return r}async fetchPools(e={}){const t=e.page||Bn.DEFAULT_PAGE,n=e.limit??Bn.DEFAULT_LIMIT;let o;if(0!==n&&hn({page:t,limit:n}),e.tokenName&&dn(e.tokenName),"recent"===e.type?o="RECENT":"popular"===e.type&&(o="POPULAR"),n>0&&n<=20){const i=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:t,limit:n});return{pools:i.pools,page:t,limit:n,total:i.total,totalPages:i.totalPages,hasNext:t<i.totalPages,hasPrevious:t>1}}if(0===n){const t=Bn.BACKEND_MAX_PAGE_SIZE,n=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:1,limit:t}),i=[...n.pools];if(n.pools.length>0){const r=await this.fetchMultiplePages({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},startPage:2,totalPages:n.totalPages>1?n.totalPages:null,pageSize:t});i.push(...r)}return{pools:i,page:1,limit:i.length,total:n.total||i.length,totalPages:1,hasNext:!1,hasPrevious:!1}}const i=Bn.BACKEND_MAX_PAGE_SIZE,r=Math.ceil(n/i),a=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:t,limit:i}),s=[...a.pools],c=Math.min(r,a.totalPages-t+1);if(c>1){const n=t+c-1,r=await this.fetchMultiplePages({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},startPage:t+1,totalPages:n,pageSize:i});s.push(...r)}const l=s.slice(0,n);return{pools:l,page:t,limit:n,total:a.total,totalPages:a.totalPages,hasNext:t<a.totalPages&&l.length<a.total,hasPrevious:t>1}}async fetchAllPools(e){return this.fetchPools({...e,limit:0})}async checkPool(e){gn(e),e.tokenName&&dn(e.tokenName);const t=M(e),n=await this.http.get(An,t);if(!n)throw oe("No response from pool service",500);bn(n,"Failed to check pool");const o=n.data;return e.symbol?o?.isSymbolExist??!1:e.tokenName?o?.isNameExist??!1:o?.exists??!1}async isTokenNameAvailable(e){try{return!await this.checkPool({tokenName:e})}catch{return!1}}async isTokenSymbolAvailable(e){try{return!await this.checkPool({symbol:e})}catch{return!1}}async fetchVolumeData(e){if(!jn(e))throw new G("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:n,to:o,resolution:i}=e;if(dn(t),!n||!o||!i)throw new G("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const r={tokenName:t,from:n,to:o,resolution:i};mn(r);const a=M(r),s=await this.http.get(En,a);if(!s)throw oe("No response from pool service",500);return bn(s,"Failed to fetch graph data",!0),{dataPoints:s.data}}async fetchTokenDistribution(e){if(!e)throw te("tokenName","Token name");dn(e);const t=await this.resolveTokenNameToVault(e);if(!t)throw ee(e);const n=encodeURIComponent(t);let o;try{o=await this.http.get(`${In}/${n}`)}catch(t){if(t&&"object"==typeof t&&"response"in t){const n=t;if(500===n.response?.status)throw oe(`Token distribution data temporarily unavailable for ${e}. This is a backend issue - please try again later.`,500)}throw t}if(!o)throw oe("No response from pool service",500);bn(o,"Failed to fetch token distribution",!0);const i=o.data;if(!Array.isArray(i))throw oe("Invalid API response: expected array of holders",o.status);for(const e of i){if(!e.owner||"string"!=typeof e.owner)throw oe("Invalid holder data: missing or invalid owner field",o.status);if(!e.quantity||"string"!=typeof e.quantity)throw oe("Invalid holder data: missing or invalid quantity field",o.status);const t=parseFloat(e.quantity);if(isNaN(t)||!isFinite(t))throw oe(`Invalid holder quantity: "${e.quantity}"`,o.status)}const r=i.reduce((e,t)=>e.plus(t.quantity),new u(0));return{holders:i.map(e=>{const t=new u(e.quantity),n=r.isZero()?0:t.dividedBy(r).multipliedBy(100).toNumber();return{address:e.owner,balance:e.quantity,percentage:n}}),totalSupply:r.toFixed(),totalHolders:i.length,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw te("tokenName","Token name");dn(e);const t=await this.http.get(Pn,{tokenName:e});if(!t)throw oe("No response from pool service",500);return bn(t,"Failed to fetch token badges",!0),{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadge(e){const{tokenName:t,badgeType:n,badgeName:o}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const i=("volume"===n?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===o);return i?.isActive||!1}catch{return!1}}async resolveTokenNameToVault(e){try{const t=await this.fetchPools({tokenName:e});if(t.pools&&Array.isArray(t.pools)&&t.pools.length>0)return t.pools[0].vaultAddress||null;if(t.pools&&"object"==typeof t.pools){const e=t.pools.tokens;return e?.vaultAddress||null}return null}catch{return null}}}function zn(e,t={}){const{stringifyFields:n=[],optionalFields:o=[],fieldMappings:i={}}=t,r={};for(const[t,a]of Object.entries(e)){const e=t;if(o.includes(e)&&void 0===a)continue;if(o.includes(e)&&"string"==typeof a&&0===a.trim().length)continue;const s=i[e],c=s?String(s):t;n.includes(e)?r[c]=String(a):r[c]=a}return M(r)}const Hn={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20}};class Qn extends Mn{constructor(e,t=!1){super(e,t)}async fetchTrades(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&(void 0===t.tradeType||"buy"===t.tradeType||"sell"===t.tradeType)&&Kn(t,"userAddress")&&Gn(t,"page")&&Gn(t,"limit")}(e))throw new G("Invalid options provided. Expected { tokenName: string, tradeType?: string, userAddress?: string, page?: number, limit?: number, startDate?: Date, endDate?: Date, sortOrder?: string }","options","INVALID_OPTIONS");const{tokenName:t,tradeType:n,userAddress:o,page:i=Bn.DEFAULT_PAGE,limit:r=Bn.DEFAULT_LIMIT,startDate:a,endDate:s,sortOrder:c}=e;if(!ce(t))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");ae(i,r,Hn);const l=function(e,t,n){return zn({tokenName:e,page:t,limit:n},{stringifyFields:["page","limit"]})}(t,i,r),u=await this.http.get(Nn,l);if(!u)throw new G("No response from trade service","response","NO_RESPONSE");const d=(h=u.data)?Array.isArray(h)?h:h.trades:[];var h;const g=kn(u,{page:i,limit:r}),p=vn(g.page,g.totalPages);return{trades:d,...g,...p}}}const Xn=new U({debug:!1,context:"DateUtils"});function Zn(e,t){if(!e)return t||new Date;if(e instanceof Date)return isNaN(e.getTime())?t||new Date:e;try{const n=new Date(e);return isNaN(n.getTime())?(Xn.warn(`Invalid date string received: "${e}". Using fallback.`),t||new Date):n}catch(n){return Xn.warn(`Date parsing error for "${e}":`,n),t||new Date}}const Yn={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:50},CONTENT:{MIN_LENGTH:1,MAX_LENGTH:500}};class Jn extends Mn{constructor(e,t,n=!1){super(e,n),this.poolService=t}async fetchComments(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&Gn(t,"page")&&Gn(t,"limit")}(e))throw new G("Invalid options provided. Expected { tokenName: string, page?: number, limit?: number }","options","INVALID_OPTIONS");const{tokenName:t,page:n=Bn.DEFAULT_PAGE,limit:o=Bn.DEFAULT_LIMIT}=e;if(!ce(t))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");ae(n,o,Yn);const i=await this.poolService.resolveTokenNameToVault(t);if(!i)throw ee(t);const r=zn({vaultAddress:i,page:n,limit:o},{stringifyFields:["page","limit"]});let a;try{a=await this.http.get(Dn,r)}catch(e){if(e&&"object"==typeof e&&"response"in e){const t=e;if(404===t.response?.status)throw new G("Comment feature is not available in this environment. Comments may only be supported in production.","endpoint","FEATURE_UNAVAILABLE")}throw e}if(!a)throw new G("No response from comment service","response","NO_RESPONSE");if(bn(a,"Failed to fetch comments"),!a.data)throw new G("Invalid response structure from comment service: no data field","response.data","INVALID_RESPONSE");if(0===Object.keys(a.data).length)return{comments:[],total:0};if(!Array.isArray(a.data.comments))throw new G("Invalid response structure from comment service: comments must be an array, got "+typeof a.data.comments,"response.data.comments","INVALID_RESPONSE");return{comments:a.data.comments.map(e=>({...e,createdAt:Zn(e.createdAt)})),total:a.data.count||0}}async postComment(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&qn(t,"content")}(e))throw new G("Invalid options provided. Expected { tokenName: string, content: string }","options","INVALID_OPTIONS");const{tokenName:t,content:n}=e;if(!ce(t))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");if(!function(e){if(!e||"string"!=typeof e)return!1;const t=e.trim();return t.length>=Yn.CONTENT.MIN_LENGTH&&t.length<=Yn.CONTENT.MAX_LENGTH}(n))throw new G(`Comment content must be between ${Yn.CONTENT.MIN_LENGTH} and ${Yn.CONTENT.MAX_LENGTH} characters`,"content","INVALID_CONTENT");const o=await this.poolService.resolveTokenNameToVault(t);if(!o)throw ee(t);const i=this.http.getAddress();if(!i)throw new G("User address not available - wallet not configured","userAddress","NO_WALLET");const r={userAddress:i,vaultAddress:o,content:n};let a;try{a=await this.http.post(Fn,r)}catch(e){if(e&&"object"==typeof e&&"response"in e){const t=e;if(404===t.response?.status)throw new G("Comment feature is not available in this environment. Comments may only be supported in production.","endpoint","FEATURE_UNAVAILABLE")}throw e}if(!a)throw new G("No response from comment service","response","NO_RESPONSE");bn(a,"Failed to create comment")}}function eo(e){return e instanceof Error}function to(e){return eo(e)||function(e){return"object"==typeof e&&null!==e&&"message"in e&&"string"==typeof e.message}(e)?e.message:"string"==typeof e?e:String(e)}function no(e){return"object"==typeof e&&null!==e&&"message"in e&&("response"in e||"request"in e||"config"in e)}function oo(e,t="image",n){const o=new FormData;if("undefined"!=typeof File&&e instanceof File)o.append(t,e);else{if(!Buffer.isBuffer(e))throw ne("file","a File object (browser) or Buffer (Node.js)");{const i=new Blob([e],{type:"image/png"});o.append(t,i,n)}}return o}const io={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20},USER_ADDRESS:{PATTERN:/^eth\|[0-9a-fA-F]{40}$/},TOKEN_NAME:{MIN_LENGTH:1,MAX_LENGTH:50},SEARCH:{MIN_LENGTH:1,MAX_LENGTH:100},FAUCET_AMOUNT:{POSITIVE_NON_ZERO_DECIMAL:/^(?!0+(\.0+)?$)\d+(\.\d+)?$/},PROFILE:{FULL_NAME:{MIN_LENGTH:1,MAX_LENGTH:100,ALPHABETS_ONLY_PATTERN:/^[a-zA-Z]+(?:\s[a-zA-Z]+)?$/}}};function ro(e){return!(!e||"string"!=typeof e)&&io.USER_ADDRESS.PATTERN.test(e)}const ao="Update profile";class so extends Mn{constructor(e,t=!1){super(e,t)}async fetchProfile(e){const t=e??this.http.getAddress();if(!t||!ro(t))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");const n={userAddress:t},o=await this.http.get(Cn,n);if(!o)throw new G("No response from user service","response","NO_RESPONSE");return o}async updateProfile(e){this.validateUpdateProfileData(e);let t=e.profileImage;if(!t||""===t.trim())try{const n=await this.fetchProfile(e.address);t=n.data?.profileImage||""}catch{t=""}const n={profileImage:t,fullName:e.fullName,userAddress:e.address},o=await this.http.signCustomMessage(ao);if(!o)throw new G("Failed to generate signature - wallet not configured","signature","NO_SIGNATURE");const i={address:o.address,message:ao,publickey:o.ethereumAddress,sign:o.signature},r=await this.http.put(xn,n,i);if(!r)throw new G("No response from user service","response","NO_RESPONSE");bn(r,"Profile update failed")}async uploadProfileImage(e){this.validateUploadProfileImageOptions(e);const t=e.address??this.http.getAddress();if(!t)throw new G("Wallet address not available - wallet not configured","address","NO_WALLET");try{const n=`profile-image-${t}.png`,o=oo(e.file,"image",n),i=await this.http.request({method:"POST",url:`${Tn}?tokenName=${encodeURIComponent(t)}`,data:o,headers:{}});if(!i)throw new G("No response from user service","response","NO_RESPONSE");return bn(i,"Image upload failed"),"string"==typeof i.data?i.data:""}catch(e){if(e instanceof G)throw e;throw new G(`Profile image upload failed: ${to(e)}`,"file","UPLOAD_FAILED")}}async fetchTokenList(e){return this.buildFetchRequest($n,e,{includeType:!0,errorMessage:"Failed to fetch token list"})}async fetchTokensHeld(e){return this.buildFetchRequest(Ln,e,{includeType:!1,errorMessage:"Failed to fetch tokens held"})}async fetchTokensCreated(e={}){const{page:t=Bn.DEFAULT_PAGE,limit:n=Bn.DEFAULT_LIMIT,search:o,tokenName:i}=e,r=this.http.getAddress();if(!r)throw new G("Wallet address not available - wallet not configured","address","NO_WALLET");const a={type:"DEFI",address:r,page:t,limit:n};return void 0!==o&&(a.search=o),void 0!==i&&(a.tokenName=i),this.fetchTokenList(a)}async buildFetchRequest(e,t,n){this.validateGetTokenListOptions(t);const o={page:t.page,limit:t.limit,address:t.address,search:t.search,tokenName:t.tokenName},i=zn(n.includeType?{...o,type:"all"!==t.type&&t.type?t.type:"DEFI"}:o,{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),r=await this.http.get(e,i);if(!r)throw new G("No response from user service","response","NO_RESPONSE");bn(r,n.errorMessage,!0);const a=(s=r.data)?Array.isArray(s)?s:s.token:[];var s;const c=kn(r,{page:t.page||Bn.DEFAULT_PAGE,limit:t.limit||Bn.DEFAULT_LIMIT}),l=vn(c.page,c.totalPages);return{tokens:a,...c,...l}}validateGetTokenListOptions(e){if(ae(e.page,e.limit,io),void 0!==e.address&&!ro(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(void 0!==e.search&&e.search.trim().length>0&&!((t=e.search)&&"string"==typeof t&&t.length>=io.SEARCH.MIN_LENGTH&&t.length<=io.SEARCH.MAX_LENGTH))throw new G(`Search query must be between ${io.SEARCH.MIN_LENGTH} and ${io.SEARCH.MAX_LENGTH} characters`,"search","INVALID_SEARCH");var t,n;if(void 0!==e.tokenName&&e.tokenName.trim().length>0&&!((n=e.tokenName)&&"string"==typeof n&&n.length>=io.TOKEN_NAME.MIN_LENGTH&&n.length<=io.TOKEN_NAME.MAX_LENGTH))throw new G(`Token name must be between ${io.TOKEN_NAME.MIN_LENGTH} and ${io.TOKEN_NAME.MAX_LENGTH} characters`,"tokenName","INVALID_TOKEN_NAME")}validateUpdateProfileData(e){if(!ro(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(!((t=e.fullName)&&"string"==typeof t&&t.length>=io.PROFILE.FULL_NAME.MIN_LENGTH&&t.length<=io.PROFILE.FULL_NAME.MAX_LENGTH&&io.PROFILE.FULL_NAME.ALPHABETS_ONLY_PATTERN.test(t)))throw new G(`Full name must be between ${io.PROFILE.FULL_NAME.MIN_LENGTH} and ${io.PROFILE.FULL_NAME.MAX_LENGTH} characters`,"fullName","INVALID_FULL_NAME");var t}validateUploadProfileImageOptions(e){if(e.address&&!ro(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS")}}class co extends Error{constructor(e,t,n){super(e),this.filename=t,this.mimeType=n,this.name="FileValidationError"}}function lo(e,t,n){if(!e)throw new co("File is required",t,n);if("undefined"!=typeof File&&e instanceof File){const t=et.safeParse(e);if(!t.success){const n=t.error.errors.map(e=>e.message).join("; ");throw new co(n,e.name,e.type)}return}if(Buffer.isBuffer(e)){if(!t)throw new co("Filename is required when uploading Buffer objects",t,n);const o=tt.safeParse(e);if(!o.success){const e=o.error.errors.map(e=>e.message).join("; ");throw new co(e,t,n)}if(t.length>255)throw new co(`Filename length ${t.length} exceeds maximum allowed length of 255 characters`,t,n);const i=["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"];if(!i.includes(n))throw new co(`Invalid file type "${n}" is not allowed. Allowed types: ${i.join(", ")}`,t,n);const r=function(e){if(!e)return"";const t=e.lastIndexOf(".");if(-1===t)return"";return e.substring(t).toLowerCase()}(t),a=[".png",".jpg",".jpeg",".gif",".webp",".svg"];if(!a.includes(r))throw new co(`File extension "${r}" is not allowed. Allowed extensions: ${a.join(", ")}`,t,n);const s=function(e){switch(e.toLowerCase()){case".png":return"image/png";case".jpg":case".jpeg":return"image/jpeg";case".gif":return"image/gif";case".webp":return"image/webp";case".svg":return"image/svg+xml";default:return"application/octet-stream"}}(r);if(s!==n&&"application/octet-stream"!==s)throw new co(`File extension "${r}" does not match MIME type "${n}"`,t,n);return}throw new co("File must be a File object (browser) or Buffer (Node.js)",t,n)}class uo extends Mn{constructor(e,t=!1){super(e,t)}async uploadImageByTokenName(e){const{tokenName:t,options:n}=e;dn(t);const o=`${t}.png`;lo(n.file,o,"image/png");try{const e=`${n.tokenName??t}.png`,o=oo(n.file,"image",e),i=await this.http.request({method:"POST",url:`${Tn}?tokenName=${encodeURIComponent(n.tokenName??t)}`,data:o,headers:{}});if(!i)throw ie("No response from image upload service","response");return bn(i,"Image upload failed"),"string"==typeof i.data?i.data:""}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw ie("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.","FormData");throw e}}}class ho extends Mn{constructor(e,t=!1){super(e,t)}async transferFaucets(e){this.validateTransferFaucetsData(e);const t={userAddress:e.walletAddress,amount:e.amount},n=await this.http.post(On,t);if(!n)throw new G("No response from faucet service","response","NO_RESPONSE");bn(n,"Faucet transfer failed")}validateTransferFaucetsData(e){if(!ro(e.walletAddress))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(!(t=e.amount)||"string"!=typeof t||!io.FAUCET_AMOUNT.POSITIVE_NON_ZERO_DECIMAL.test(t))throw new G("Amount must be a positive decimal string greater than zero","amount","INVALID_AMOUNT");var t}}class go{constructor(e,t=!1){this.http=e,this.poolService=new Vn(e,t),this.tradeService=new Qn(e,t),this.commentService=new Jn(e,this.poolService,t),this.userService=new so(e,t),this.imageService=new uo(e,t),this.faucetService=new ho(e,t)}async uploadImageByTokenName(e){return this.imageService.uploadImageByTokenName(e)}async fetchPools(e={}){return this.poolService.fetchPools(e)}async fetchAllPools(e){return this.poolService.fetchAllPools(e)}async checkPool(e){return this.poolService.checkPool(e)}async isTokenNameAvailable(e){return this.poolService.isTokenNameAvailable(e)}async isTokenSymbolAvailable(e){return this.poolService.isTokenSymbolAvailable(e)}async fetchVolumeData(e){return this.poolService.fetchVolumeData(e)}async fetchTokenDistribution(e){return this.poolService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.poolService.fetchTokenBadges(e)}async hasTokenBadge(e){return this.poolService.hasTokenBadge(e)}async fetchTrades(e){return this.tradeService.fetchTrades(e)}async fetchComments(e){return this.commentService.fetchComments(e)}async postComment(e){return this.commentService.postComment(e)}async fetchProfile(e){return this.userService.fetchProfile(e)}async updateProfile(e){return this.userService.updateProfile(e)}async uploadProfileImage(e){return this.userService.uploadProfileImage(e)}async fetchTokenList(e){return this.userService.fetchTokenList(e)}async fetchTokensHeld(e){return this.userService.fetchTokensHeld(e)}async fetchTokensCreated(e={}){return this.userService.fetchTokensCreated(e)}async transferFaucets(e){return this.faucetService.transferFaucets(e)}getAddress(){return this.http.getAddress()}validateTokenName(e){return dn(e)}}const po={MAX_UNIQUE_KEY_LENGTH:64,UNIQUE_KEY_PATTERN:/^(galaswap-operation-|galaconnect-operation-)/,TOKEN_NAME_PATTERN:/^[a-zA-Z0-9]+$/};var mo;!function(e){e.INVALID_RECIPIENT="INVALID_RECIPIENT",e.INVALID_AMOUNT="INVALID_AMOUNT",e.INSUFFICIENT_BALANCE="INSUFFICIENT_BALANCE",e.TOKEN_NOT_FOUND="TOKEN_NOT_FOUND",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.NETWORK_ERROR="NETWORK_ERROR",e.DUPLICATE_TRANSFER="DUPLICATE_TRANSFER",e.TRANSFER_LIMIT_EXCEEDED="TRANSFER_LIMIT_EXCEEDED",e.WALLET_REQUIRED="WALLET_REQUIRED"}(mo||(mo={}));class fo extends Error{constructor(e,t,n){super(e),this.type=t,this.details=n,this.name="TransferError"}}function yo(e,t="0"){return new u(null==e||""===e?t:e)}function wo(e){return d(e)}function ko(e){const t=yo(e),n=Math.log(1.0001),o=t.toNumber();return Math.log(o)/n}function vo(e,t=!1){const n=yo(e),o=new u(1).dividedBy(n);return t?o.toFixed():o}function bo(...e){e.forEach((e,t)=>{if(e.isNaN())throw new Error(`Value at index ${t} must be a valid number, got: NaN`);if(!e.isFinite())throw new Error(`Value at index ${t} must be finite, got: ${e.toString()}`);if(e.isLessThanOrEqualTo(0))throw new Error(`Value at index ${t} must be positive, got: ${e.toString()}`)})}function To(...e){e.forEach((e,t)=>{if(e.isNaN())throw new Error(`Value at index ${t} must be a valid number, got: NaN`);if(!e.isFinite())throw new Error(`Value at index ${t} must be finite, got: ${e.toString()}`);if(e.isLessThan(0))throw new Error(`Value at index ${t} must be non-negative, got: ${e.toString()}`)})}function So(e,t,n){if(e.isNaN())throw new Error(`${t} must be a valid number, got: NaN`);if(!e.isFinite()){const o=n?`${t} must be finite ${n}, got: ${e.toString()}`:`${t} must be finite, got: ${e.toString()}`;throw new Error(o)}if(e.isLessThanOrEqualTo(0)){const o=n?`${t} must be positive ${n}, got: ${e.toString()}`:`${t} must be positive, got: ${e.toString()}`;throw new Error(o)}}class Ao{static validateAmount(e){const t=new u(e);try{So(t,"amount","for transfer")}catch(t){throw new fo(t.message,mo.INVALID_AMOUNT,{amount:e})}}static validateUniqueKey(e){if(e){if(e.length>po.MAX_UNIQUE_KEY_LENGTH)throw new G(`Unique key too long. Maximum length: ${po.MAX_UNIQUE_KEY_LENGTH}`);if(!po.UNIQUE_KEY_PATTERN.test(e))throw new fo('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',mo.INVALID_AMOUNT,{uniqueKey:e})}}}class Eo{toLaunchpadFormat(e){if(!e)return"unknown|Unit|none|none";if("string"==typeof e)return e.includes("|")?e:`${e}|Unit|none|none`;return`${e.collection||e.symbol||"unknown"}|${e.category||"Unit"}|${e.type||"none"}|${e.additionalKey||"none"}`}toTokenClass(e){if("object"==typeof e&&null!==e)return{collection:e.collection||"Token",category:e.category||"Unit",type:e.type||e.symbol||"unknown",additionalKey:e.additionalKey||"none"};if("string"!=typeof e)throw new Error("Invalid token format: expected string or object, got "+typeof e);const t=e.split("|");if(t.length<3)throw new Error(`Invalid token format: ${e}`);return{collection:t[0]||"Token",category:t[1]||"Unit",type:t[2]||"none",additionalKey:t[3]||"none"}}isGSwapFormat(e){return"string"==typeof e&&e.includes("|")}normalize(e){return"string"==typeof e&&this.isGSwapFormat(e)?e:this.toLaunchpadFormat(e)}}function Io(e){if(!e||"string"!=typeof e)throw new Error("Invalid token format: token must be a non-empty string");return e.replace(/\|/g,"$")}function Po(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("$");if(t.length<4)throw new Error(`Invalid dollar-delimited token format. Expected at least 4 parts separated by $, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("$");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){throw new G(`Invalid dollar-delimited token: "${e}". Expected format: "Token$Unit$SYMBOL$additionalKey". Error: ${t instanceof Error?t.message:String(t)}`,"dollarToken","INVALID_DOLLAR_DELIMITED_TOKEN_FORMAT")}}class No extends s{constructor(e){super(),this.from=e.from,this.to=e.to,this.quantity=e.quantity,this.tokenInstance=e.tokenInstance,this.uniqueKey=e.uniqueKey,e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o,i){let r;if("string"==typeof o){r={...Po(o),instance:"0"}}else r={collection:o.collection,category:o.category,type:o.type,additionalKey:o.additionalKey,instance:"0"};return new No({from:e,to:t,quantity:n,tokenInstance:r,uniqueKey:i||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n,o){return new No({from:e,to:t,quantity:n,tokenInstance:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"},uniqueKey:o||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){return`${this.tokenInstance.collection}$${this.tokenInstance.category}$${this.tokenInstance.type}$${this.tokenInstance.additionalKey}`}toSigningPayload(){return{from:this.from,to:this.to,quantity:this.quantity,tokenInstance:this.tokenInstance,uniqueKey:this.uniqueKey}}}class Do extends s{constructor(e){super(),this.lockAuthority=e.lockAuthority,this.tokenInstances=e.tokenInstances,this.uniqueKey=e.uniqueKey,void 0!==e.expires&&(this.expires=e.expires),void 0!==e.name&&(this.name=e.name),e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o,i){let r;if("string"==typeof o){r={...Po(o),instance:"0"}}else r={collection:o.collection,category:o.category,type:o.type,additionalKey:o.additionalKey,instance:"0"};return new Do({lockAuthority:t||e,tokenInstances:[{owner:e,quantity:n,tokenInstanceKey:r}],...void 0!==i?.expires&&{expires:i.expires},...void 0!==i?.name&&{name:i.name},uniqueKey:i?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n,o){return new Do({lockAuthority:t||e,tokenInstances:[{owner:e,quantity:n,tokenInstanceKey:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}],...void 0!==o?.expires&&{expires:o.expires},...void 0!==o?.name&&{name:o.name},uniqueKey:o?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){if(0===this.tokenInstances.length)return;const e=this.tokenInstances[0].tokenInstanceKey;return`${e.collection}$${e.category}$${e.type}$${e.additionalKey}`}toSigningPayload(){return{lockAuthority:this.lockAuthority,tokenInstances:this.tokenInstances,...void 0!==this.expires&&{expires:this.expires},...void 0!==this.name&&{name:this.name},uniqueKey:this.uniqueKey}}}class Fo extends s{constructor(e){super(),this.tokenInstances=e.tokenInstances,this.uniqueKey=e.uniqueKey,void 0!==e.name&&(this.name=e.name),e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o){let i;if("string"==typeof n){i={...Po(n),instance:"0"}}else i={collection:n.collection,category:n.category,type:n.type,additionalKey:n.additionalKey,instance:"0"};return new Fo({tokenInstances:[{owner:e,quantity:t,tokenInstanceKey:i}],...void 0!==o?.name&&{name:o.name},uniqueKey:o?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n){return new Fo({tokenInstances:[{owner:e,quantity:t,tokenInstanceKey:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}],...void 0!==n?.name&&{name:n.name},uniqueKey:n?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){if(0===this.tokenInstances.length)return;const e=this.tokenInstances[0].tokenInstanceKey;return`${e.collection}$${e.category}$${e.type}$${e.additionalKey}`}toSigningPayload(){return{tokenInstances:this.tokenInstances,...void 0!==this.name&&{name:this.name},uniqueKey:this.uniqueKey}}}class Co{constructor(e){this.wallet=e}static generateUniqueKey(){return`${Date.now()}_${Math.random().toString(36).substring(2,8)}`}async signTransferToken(e){const t={name:"GalaChain",chainId:1},n={TransferToken:[{name:"from",type:"string"},{name:"to",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstance",type:"TokenInstance"},{name:"uniqueKey",type:"string"}],TokenInstance:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]};return{signature:await this.wallet.signTypedData(t,n,e),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}async signLockToken(e){const t={name:"GalaChain",chainId:1},n={LockToken:[{name:"lockAuthority",type:"string"},{name:"tokenInstances",type:"TokenInstanceQuantity[]"},{name:"uniqueKey",type:"string"},...void 0!==e.expires?[{name:"expires",type:"uint256"}]:[],...void 0!==e.name?[{name:"name",type:"string"}]:[]],TokenInstanceQuantity:[{name:"owner",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstanceKey",type:"TokenInstanceKey"}],TokenInstanceKey:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]},o={lockAuthority:e.lockAuthority,tokenInstances:e.tokenInstances,uniqueKey:e.uniqueKey};void 0!==e.expires&&(o.expires=e.expires),void 0!==e.name&&(o.name=e.name);return{signature:await this.wallet.signTypedData(t,n,o),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}async signUnlockToken(e){const t={name:"GalaChain",chainId:1},n={UnlockToken:[{name:"tokenInstances",type:"TokenInstanceQuantity[]"},{name:"uniqueKey",type:"string"},...void 0!==e.name?[{name:"name",type:"string"}]:[]],TokenInstanceQuantity:[{name:"owner",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstanceKey",type:"TokenInstanceKey"}],TokenInstanceKey:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]},o={tokenInstances:e.tokenInstances,uniqueKey:e.uniqueKey};void 0!==e.name&&(o.name=e.name);return{signature:await this.wallet.signTypedData(t,n,o),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}static toGalaChainAddress(e){const n=e.replace("0x","");return`eth|${t.getAddress(`0x${n}`).replace("0x","")}`}static fromGalaChainAddress(e){return e.startsWith("eth|")?e.substring(4):e}static createGALATokenInstance(){return{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}static createTokenInstanceFromClassKey(e){return{...Po(e),instance:"0"}}}function xo(e){if("string"==typeof e){const t=e.split("|");if(t.length<4)throw new G(`Invalid tokenId string format: "${e}". Expected format: "collection|category|type|additionalKey" or "collection|category|type|additionalKey|instance"`,"tokenId","INVALID_TOKEN_ID_FORMAT");if(!(t[0]&&t[1]&&t[2]&&t[3]))throw new G(`Invalid tokenId string format: "${e}". All components (collection, category, type, additionalKey) must be non-empty`,"tokenId","INVALID_TOKEN_ID_FORMAT");return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3],instance:t[4]||"0"}}if("object"==typeof e&&null!==e){if(!(e.collection&&e.category&&e.type&&e.additionalKey))throw new G("Invalid tokenId object format. All fields (collection, category, type, additionalKey) are required","tokenId","INVALID_TOKEN_ID_FORMAT");return"instance"in e&&void 0!==e.instance?e:{...e,instance:"0"}}throw new G(`Invalid tokenId type: ${typeof e}. Expected string, TokenClassKey, or TokenInstanceKey`,"tokenId","INVALID_TOKEN_ID_TYPE")}function $o(e){try{const[t,n]=e.split("|");if(!n)throw new Error("Missing token part after service");const o=n.split("$");if(o.length<4)throw new Error(`Invalid vault address format. Expected at least 4 parts separated by $, got ${o.length}`);if(!(o[0]&&o[1]&&o[2]&&o[3]))throw new Error("All vault address components (collection, category, type, additionalKey) must be non-empty");const i=o.slice(3,-1),r=i.length>0?i.join("$"):o[3];return{collection:o[0],category:o[1],type:o[2],additionalKey:r}}catch(t){throw new G(`Invalid vault address: "${e}". Expected format: "service|Token$Unit$SYMBOL$additionalKey$launchpad". Error: ${t instanceof Error?t.message:String(t)}`,"vaultAddress","INVALID_VAULT_ADDRESS_FORMAT")}}function Lo(e){return{...$o(e),instance:"0"}}function Oo(e){return $o(e).type}var _o,Bo=Object.freeze({__proto__:null,extractTokenSymbolFromVault:Oo,normalizeToTokenInstanceKey:xo,parseVaultAddressToTokenClassKey:$o,parseVaultAddressToTokenInstance:Lo});function Uo(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(!("string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0))||void 0!==t.lockAuthority&&"string"!=typeof t.lockAuthority||void 0!==t.expires&&"number"!=typeof t.expires||void 0!==t.name&&"string"!=typeof t.name||void 0!==t.uniqueKey&&"string"!=typeof t.uniqueKey)}function Mo(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0)&&(void 0===t.name||"string"==typeof t.name)&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}!function(e){e.TOKEN_NOT_FOUND="TOKEN_NOT_FOUND",e.INVALID_AMOUNT="INVALID_AMOUNT",e.INSUFFICIENT_BALANCE="INSUFFICIENT_BALANCE",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.NETWORK_ERROR="NETWORK_ERROR",e.WALLET_REQUIRED="WALLET_REQUIRED",e.LOCK_NOT_FOUND="LOCK_NOT_FOUND",e.LOCK_EXPIRED="LOCK_EXPIRED",e.INSUFFICIENT_LOCKED_BALANCE="INSUFFICIENT_LOCKED_BALANCE",e.NOT_LOCK_AUTHORITY="NOT_LOCK_AUTHORITY",e.LOCK_NAME_MISMATCH="LOCK_NAME_MISMATCH"}(_o||(_o={}));class Ro extends Error{constructor(e,t,n){super(e),this.type=t,this.details=n,this.name="LockError"}}const qo="gala-transfer-successful",Ko="token-transfer-successful",Go="transfer-successful-no-id";class Wo extends Mn{constructor(e,t,n,o=!1,i){super(e,o),this.wallet=t,this.tokenResolver=n,this.publicAxios=i,this.signatureHelper=t?new Co(t):void 0}async fetchPoolDetails(e){this.validateFetchPoolDetailsData(e);const t=await this.http.post("/api/asset/launchpad-contract/FetchSaleDetails",e);if(!t)throw oe("No response from GalaChain service",500);if(1!==t.Status)throw oe(`Failed to fetch pool details: Status ${t.Status}`,t.Status);const n=t.Data.reverseBondingCurveConfiguration,o=n?.minFeePortion??"0",i=n?.maxFeePortion??"0",r=(await import("bignumber.js")).default,a=!new r(o).isZero()||!new r(i).isZero(),s=t.Data;return s.reverseBondingCurveMinFeePortion=o,s.reverseBondingCurveMaxFeePortion=i,s.hasReverseBondingCurveFee=a,s.isGraduated="Finished"===t.Data.saleStatus,delete s.reverseBondingCurveConfiguration,t}async fetchLaunchTokenFee(){const e=await this.http.post("/api/asset/launchpad-contract/FetchLaunchpadFeeAmount",{});if(!e)throw oe("No response from GalaChain service",500);if(1!==e.Status)throw oe(`Failed to fetch launch token fee: Status ${e.Status}`,e.Status);return e.Data.feeAmount}validateFetchPoolDetailsData(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.vaultAddress)throw new G("Invalid fetch pool details data: missing required fields","data","INVALID_TYPE");var t;if(!e.vaultAddress||"string"!=typeof e.vaultAddress)throw new G("Vault address is required and must be a string","vaultAddress","INVALID_VAULT_ADDRESS");if(!e.vaultAddress.startsWith("service|Token$Unit$"))throw new G("Vault address must be in service format: service|Token$Unit$...","vaultAddress","INVALID_VAULT_ADDRESS")}async fetchGalaBalance(e){return this.fetchTokenBalance(e)}async fetchTokenBalance(e){try{const t=await this.http.post("/api/asset/token-contract/FetchBalances",e);if(!t)return null;if(1!==t.Status||!t.Data||0===t.Data.length)return null;const n=t.Data.find(t=>t.collection===e.collection&&t.category===e.category&&t.additionalKey===e.additionalKey&&t.type===e.type);if(!n||"0"===n.quantity)return null;const o=`${n.collection}|${n.category}|${n.additionalKey}|${n.type}`;return{quantity:n.quantity,collection:n.collection,category:n.category,tokenId:o}}catch(e){throw oe(`Failed to fetch token balance from GalaChain: ${to(e)}`,void 0,eo(e)?e:void 0)}}async fetchTokenClassFromChain(e){try{const t="string"==typeof e?xo(e):e,n={tokenClasses:[{collection:t.collection,category:t.category,type:t.type,additionalKey:t.additionalKey}]},o=(await this.publicAxios.post("/api/asset/token-contract/FetchTokenClasses",n)).data;if(1!==o.Status)throw oe(`Failed to fetch token class from GalaChain: Status ${o.Status}${o.Message?` - ${o.Message}`:""}`,o.Status);if(!o.Data||0===o.Data.length)throw oe(`Token not found on GalaChain: ${t.collection}|${t.category}|${t.type}|${t.additionalKey}`,404);return o.Data[0]}catch(e){if(no(e)&&404===e.response?.status)throw oe("Token not found on GalaChain",404,eo(e)?e:void 0);const t=to(e);if(t.includes("Token not found")||t.includes("Status"))throw e;throw oe(`Failed to fetch token class from GalaChain: ${t}`,void 0,eo(e)?e:void 0)}}async fetchTokenClassesWithSupply(e){try{if(!e||0===e.length)throw new G("tokenClasses array must not be empty");const t={tokenClasses:e},n=(await this.publicAxios.post("/api/asset/token-contract/FetchTokenClassesWithSupply",t)).data;if(1!==n.Status)throw oe(`Failed to fetch token classes with supply from GalaChain: Status ${n.Status}${n.Message?` - ${n.Message}`:""}`,n.Status);if(!n.Data||0===n.Data.length)throw oe("No token supply data found for requested token classes",404);return n.Data}catch(e){if(no(e)&&404===e.response?.status)throw oe("Token supply data not found on GalaChain",404,eo(e)?e:void 0);const t=to(e);if(t.includes("Token")||t.includes("Status")||t.includes("tokenClasses"))throw e;throw oe(`Failed to fetch token classes with supply from GalaChain: ${t}`,void 0,eo(e)?e:void 0)}}async transferGala(e){if(this.validateTransferGalaData(e),!this.wallet||!this.signatureHelper)throw new fo("Wallet required for GALA transfer operations",mo.WALLET_REQUIRED);try{const t=ue(e.recipientAddress),n=ue(this.wallet.address),o=No.forGALA(n,t,e.amount,e.uniqueKey),i=await this.signatureHelper.signTransferToken(o.toSigningPayload()),r=new No({...o.toSigningPayload(),signedPayload:i});this.logger.debug("[DEBUG] Full GALA Transfer Request Payload:",JSON.stringify(r,null,2));const a=await this.http.post("/api/asset/token-contract/TransferToken",r);if(!a)throw new fo("No response from GalaChain transfer service",mo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Transfer response:",JSON.stringify(a,null,2)),this.extractTransactionIdFromResponse(a,"gala")}catch(t){throw this.handleTransferError(t,"GALA transfer failed",e)}}async transferToken(e){if(this.validateTransferTokenData(e),!this.wallet||!this.signatureHelper)throw new fo("Wallet required for token transfer operations",mo.WALLET_REQUIRED);try{const t=ue(e.to),n=ue(this.wallet.address);let o;if(e.tokenId)o=xo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId),this.logger.debug("[DEBUG] Normalized Token Instance:",JSON.stringify(o,null,2));else{if(!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);o=await this.resolveTokenInstance(e.tokenName)}const i=new No({from:n,to:t,quantity:e.amount,tokenInstance:o,uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),r=await this.signatureHelper.signTransferToken(i.toSigningPayload()),a=new No({...i.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full Transfer Request Payload:",JSON.stringify(a,null,2));const s=await this.http.post("/api/asset/token-contract/TransferToken",a);if(!s)throw new fo("No response from GalaChain transfer service",mo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token transfer response:",JSON.stringify(s,null,2)),this.extractTransactionIdFromResponse(s,"token")}catch(t){throw this.handleTransferError(t,"Token transfer failed",e)}}async resolveTokenClassKey(e){try{const t=await this.tokenResolver.resolveTokenClassKey(e);return this.logger.debug(`[DEBUG] Token class key resolution for '${e}':`,JSON.stringify(t,null,2)),t}catch(t){if(t instanceof fo)throw t;throw new fo(`Failed to resolve token class key for '${e}': ${to(t)}`,mo.TOKEN_NOT_FOUND,{tokenName:e})}}async lockToken(e){if(this.validateLockTokenData(e),!this.wallet||!this.signatureHelper)throw new fo("Wallet required for token lock operations",mo.WALLET_REQUIRED);try{const t=ue(this.wallet.address),n=e.lockAuthority?ue(e.lockAuthority):t;let o;if(e.tokenId)o=xo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId);else{if(!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);o=await this.resolveTokenInstance(e.tokenName)}const i=new Do({lockAuthority:n,tokenInstances:[{owner:t,quantity:e.amount,tokenInstanceKey:o}],...void 0!==e.expires&&{expires:e.expires},...void 0!==e.name&&{name:e.name},uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),r=await this.signatureHelper.signLockToken(i.toSigningPayload()),a=new Do({...i.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full Lock Request Payload:",JSON.stringify(a,null,2));const s=await this.http.post("/api/asset/token-contract/LockToken",a);if(!s)throw new fo("No response from GalaChain lock service",mo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token lock response:",JSON.stringify(s,null,2)),this.extractTransactionIdFromResponse(s,"token")}catch(t){throw this.handleTransferError(t,"Token lock failed",e)}}async unlockToken(e){if(this.validateUnlockTokenData(e),!this.wallet||!this.signatureHelper)throw new fo("Wallet required for token unlock operations",mo.WALLET_REQUIRED);try{const t=ue(this.wallet.address);let n;if(e.tokenId)n=xo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId);else{if(!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);n=await this.resolveTokenInstance(e.tokenName)}const o=new Fo({tokenInstances:[{owner:t,quantity:e.amount,tokenInstanceKey:n}],...void 0!==e.name&&{name:e.name},uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),i=await this.signatureHelper.signUnlockToken(o.toSigningPayload()),r=new Fo({...o.toSigningPayload(),signedPayload:i});this.logger.debug("[DEBUG] Full Unlock Request Payload:",JSON.stringify(r,null,2));const a=await this.http.post("/api/asset/token-contract/UnlockToken",r);if(!a)throw new fo("No response from GalaChain unlock service",mo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token unlock response:",JSON.stringify(a,null,2)),this.extractTransactionIdFromResponse(a,"token")}catch(t){throw this.handleTransferError(t,"Token unlock failed",e)}}validateTransferGalaData(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.recipientAddress&&t.recipientAddress.trim().length>0&&"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}(e))throw new G("Invalid GALA transfer data: missing required fields");if(!le(e.recipientAddress))throw new fo("Invalid recipient address format",mo.INVALID_RECIPIENT,{recipientAddress:e.recipientAddress});Ao.validateAmount(e.amount),Ao.validateUniqueKey(e.uniqueKey)}validateTransferTokenData(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.to&&t.to.trim().length>0&&"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0)&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}(e))throw new G("Invalid token transfer data: missing required fields");if(!le(e.to))throw new fo("Invalid recipient address format",mo.INVALID_RECIPIENT,{recipientAddress:e.to});if(!e.tokenId&&!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);if(e.tokenName&&!po.TOKEN_NAME_PATTERN.test(e.tokenName))throw new fo("Invalid token name format",mo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});Ao.validateAmount(e.amount),Ao.validateUniqueKey(e.uniqueKey)}validateLockTokenData(e){if(!Uo(e))throw new G("Invalid lock token data: missing required fields");if(e.lockAuthority&&!le(e.lockAuthority))throw new fo("Invalid lock authority address format",mo.INVALID_RECIPIENT,{recipientAddress:e.lockAuthority});if(!e.tokenId&&!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);if(e.tokenName&&!po.TOKEN_NAME_PATTERN.test(e.tokenName))throw new fo("Invalid token name format",mo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});if(void 0!==e.expires&&(e.expires<=0||!Number.isInteger(e.expires)))throw new G("Expires must be a positive integer (epoch milliseconds)");Ao.validateAmount(e.amount),Ao.validateUniqueKey(e.uniqueKey)}validateUnlockTokenData(e){if(!Mo(e))throw new G("Invalid unlock token data: missing required fields");if(!e.tokenId&&!e.tokenName)throw new fo("Must provide either tokenId or tokenName for token identification",mo.TOKEN_NOT_FOUND);if(e.tokenName&&!po.TOKEN_NAME_PATTERN.test(e.tokenName))throw new fo("Invalid token name format",mo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});Ao.validateAmount(e.amount),Ao.validateUniqueKey(e.uniqueKey)}async resolveTokenInstance(e){try{const t=await this.tokenResolver.resolveTokenToVault(e);if(t){const n=this.resolveTokenInstanceFromVaultAddress(t);return this.logger.debug(`[DEBUG] Token resolution for '${e}' (launchpad):\n Vault Address: ${t}\n Token Instance: ${JSON.stringify(n,null,2)}`),n}const n={collection:e.trim().toUpperCase(),category:"Unit",type:"none",additionalKey:"none",instance:"0"};return this.logger.debug(`[DEBUG] Token resolution for '${e}' (standard format):\n Token Instance: ${JSON.stringify(n,null,2)}`),n}catch(t){if(t instanceof fo)throw t;throw new fo(`Failed to resolve token '${e}': ${to(t)}`,mo.TOKEN_NOT_FOUND,{tokenName:e})}}resolveTokenInstanceFromVaultAddress(e){try{return Lo(e)}catch(e){if(e instanceof G)throw new fo(`Invalid vault address format: ${e.message}`,mo.TOKEN_NOT_FOUND);throw new fo(`Failed to parse vault address: ${e instanceof Error?e.message:String(e)}`,mo.TOKEN_NOT_FOUND)}}extractTransactionIdFromResponse(e,t){if(e&&"object"==typeof e){if("Status"in e&&1===e.Status&&"Data"in e){const n=e;return Array.isArray(n.Data)&&n.Data.length>0?"gala"===t?qo:Ko:Go}if("transactionId"in e&&"string"==typeof e.transactionId&&e.transactionId)return e.transactionId}throw new fo("Transfer succeeded but transaction ID could not be extracted",mo.NETWORK_ERROR)}handleTransferError(e,t,n){if(e instanceof fo)return e;if(e instanceof G)return new fo(e.message,mo.INVALID_AMOUNT);if(no(e)&&e.response){const t=e.response.status,o=e.response.data;if(400===t)return new fo(("string"==typeof o?.message?o.message:void 0)||"Invalid transfer request",mo.INVALID_AMOUNT);if(403===t)return new fo("Insufficient balance for transfer",mo.INSUFFICIENT_BALANCE);if(404===t){const e={};return"tokenName"in n&&(e.tokenName=n.tokenName),new fo("Token not found",mo.TOKEN_NOT_FOUND,e)}}if("object"==typeof e&&null!==e&&"code"in e&&("ECONNABORTED"===e.code||"ETIMEDOUT"===e.code))return new fo("Transfer request timed out",mo.NETWORK_ERROR);const o=to(e);return new fo(o||t,mo.NETWORK_ERROR)}}class jo{constructor(e,t,n,o=!1){this.dexBackendHttp=e,this.cache=t,this.galaChainService=n,this.logger=new U({debug:o,context:"DexService"})}async fetchTokenPrice(e){const{tokenName:t,tokenId:n}=e;if(!t&&!n)throw te("tokenName or tokenId","Either tokenName (for launchpad tokens) or tokenId (for DEX tokens) is required");if(t&&n)throw new G("tokenName and tokenId are mutually exclusive - provide only one","params","INVALID_PARAMS");if(n)return this.logger.debug(`Fetching spot price by tokenId: ${n}`),this._fetchDexTokenSpotPrice(n);throw new G("tokenName parameter requires LaunchpadSDK routing - call LaunchpadSDK.fetchTokenPrice({tokenName}) instead","tokenName","INVALID_PARAMS")}async _fetchDexTokenSpotPrice(e){if(!e)throw te("tokenId","Token ID");try{const t=xo(e),n=Io(`${t.collection}|${t.category}|${t.type}|${t.additionalKey}`);if(this.logger.debug(`Fetching DEX spot price for token: ${n}`),!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const o=await this.dexBackendHttp.request({method:"GET",url:"/v1/trade/price",params:{token:n}});if(!o.data||"string"!=typeof o.data)throw new G("Invalid price response: data must be a string, got "+typeof o.data,"data","INVALID_RESPONSE");const i=parseFloat(o.data);if(isNaN(i))throw new G(`Invalid price value: could not parse "${o.data}" as number`,"data","INVALID_CALCULATION");const r=`${t.collection}|${t.category}|${t.type}|${t.additionalKey}`;let a;try{if(this.cache){const e=this.cache.getByTokenId(r);if(e?.symbol)return a=e.symbol,this.logger.debug(`DEX spot price for ${a} (cached): $${i}`),{symbol:a,price:i}}this.logger.debug(`Symbol cache miss for ${r}, fetching from API`);a=(await this.fetchTokenDetails(e)).symbol,this.cache&&(this.cache.setByTokenId(r,{symbol:a}),this.logger.debug(`Cached symbol for ${r}: ${a}`)),this.logger.debug(`DEX spot price for ${a}: $${i}`)}catch(e){this.logger.debug(`Could not fetch token details for symbol, falling back to token format parsing: ${e instanceof Error?e.message:String(e)}`),a=("Token"===t.collection?t.type:t.collection).toUpperCase(),this.logger.debug(`DEX spot price for ${a} (fallback): $${i}`)}return{symbol:a,price:i}}catch(e){if(e instanceof G)throw e;throw oe(`Failed to fetch DEX spot price: ${to(e)}`)}}async fetchLaunchpadTokenSpotPrice(e,t,n){if(!e||"string"!=typeof e)throw new Error(q);try{if(n)try{this.logger.debug(`Checking graduation status for token: ${e}`);const t=await n(e);if(t&&t.isGraduated){this.logger.debug(`Token ${e} is graduated, using DEX spot price`);const n=`${t.sellingToken.collection}|${t.sellingToken.category}|${t.sellingToken.type}|${t.sellingToken.additionalKey}`;return this._fetchDexTokenSpotPrice(n)}}catch(t){this.logger.debug(`Could not determine graduation status for ${e}, falling back to bonding curve: ${to(t)}`)}this.logger.debug(`Using bonding curve calculation for token: ${e}`);const o=await t({tokenName:e,amount:"1",type:"native"}),i=await this._fetchDexTokenSpotPrice({collection:"GALA",category:"Unit",type:"none",additionalKey:"none"});if(!i)throw oe("GALA price not available");const r=Number(o.amount)/1e18;if(r<=0)throw new G(`Invalid token amount calculation: ${r}`,"amount","INVALID_CALCULATION");const a=i.price/r;return{symbol:e.toUpperCase(),price:a}}catch(t){if(t instanceof Error)throw oe(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw oe(`Failed to calculate launchpad token spot price for ${e}: ${to(t)}`)}}async fetchTokenDetails(e){this.logger.debug("Fetching token details from GalaChain for tokenId:",e);try{if(!this.galaChainService)throw oe("GalaChainService not available for token metadata fetch",500);const t=await this.galaChainService.fetchTokenClassFromChain(e),n={collection:t.collection,category:t.category,type:t.type,additionalKey:t.additionalKey,symbol:t.symbol,decimals:t.decimals,name:t.name,image:t.image,description:t.description,network:t.network,...void 0!==t.contractAddress&&{contractAddress:t.contractAddress}};return this.logger.debug(`Fetched token details for ${t.symbol} from GalaChain`),n}catch(t){if((t instanceof G||t instanceof Error)&&("NetworkError"===t.name||t.message.includes("Token not found")))throw t;throw oe(`Failed to fetch token details from GalaChain for ${e}: ${to(t)}`,500)}}async fetchAllDexSeasons(){try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const e=await this.dexBackendHttp.request({method:"GET",url:"/leaderboard/seasons"});let t;if(Array.isArray(e))t=e;else{if(!e||"object"!=typeof e)return this.logger.warn("Seasons endpoint returned invalid data:",e),[];if(Array.isArray(e.data))t=e.data;else if(e.data&&Array.isArray(e.data.seasons))t=e.data.seasons;else{if(!Array.isArray(e.seasons))return this.logger.warn("Seasons endpoint returned unexpected structure:",e),[];t=e.seasons}}const n=t.map(e=>({id:e?.id??0,name:e?.name??"",start:e?.start?new Date(e.start):new Date,end:e?.end?new Date(e.end):new Date,rulesId:e?.rules_id??0}));return this.logger.debug(`Fetched ${n.length} DEX seasons`),n}catch(e){if(e instanceof Error&&e.message.includes("not configured"))throw e;if(e&&"object"==typeof e&&"response"in e){const t=e;if(404===t.response?.status)return this.logger.warn("Seasons endpoint not available"),[]}throw oe(`Failed to fetch DEX seasons: ${to(e)}`)}}async fetchCurrentDexSeason(){const e=await this.fetchAllDexSeasons(),t=new Date,n=e.find(e=>t>=e.start&&t<=e.end);return n?this.logger.debug(`Current DEX season: ${n.name} (ID: ${n.id})`):this.logger.debug("No active DEX season found"),n||null}async fetchDexLeaderboardBySeasonId(e){if(!e||"number"!=typeof e||e<1)throw te("seasonId","Season ID must be a positive number");try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const t=await this.dexBackendHttp.request({method:"GET",url:"/leaderboard",params:{seasonId:e.toString()}});let n;if(Array.isArray(t))n=t;else{if(!t||"object"!=typeof t)return this.logger.warn("Leaderboard endpoint returned invalid data:",t),{entries:[],seasonId:e,totalEntries:0};if(t.data&&Array.isArray(t.data.leaderboard))n=t.data.leaderboard;else if(Array.isArray(t.leaderboard))n=t.leaderboard;else{if(!t.data||!Array.isArray(t.data))return this.logger.warn("Leaderboard endpoint returned unexpected structure:",t),{entries:[],seasonId:e,totalEntries:0};n=t.data}}const o=n.map(e=>({wallet:e?.wallet??"",rank:e?.rank??0,totalXp:e?.total_xp??0,distributionPercent:e?.distribution_percent??0,liquidityXp:e?.liquidity_xp??0,tradingXp:e?.trading_xp??0,masteryTitles:(e?.mastery_titles??[]).map(e=>({name:e?.name??"",type:e?.type??"trade",order:e?.order??0}))}));return this.logger.debug(`Fetched leaderboard for season ${e} with ${o.length} entries`),{entries:o,seasonId:e,totalEntries:o.length}}catch(t){if(t instanceof Error&&t.message.includes("must be a positive number"))throw t;throw oe(`Failed to fetch DEX leaderboard for season ${e}: ${to(t)}`)}}async fetchCurrentDexLeaderboard(){const e=await this.fetchCurrentDexSeason();return e?this.fetchDexLeaderboardBySeasonId(e.id):(this.logger.debug("Cannot fetch current leaderboard - no active season"),null)}async fetchDexAggregatedVolumeSummary(){try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const e=await this.dexBackendHttp.request({method:"GET",url:"/explore/volume"}),t={volume1d:e.data.volume1d,volume1dDelta:e.data.volume1dDelta,volume7d:e.data.volume7d,volume7dDelta:e.data.volume7dDelta,volume30d:e.data.volume30d,volume30dDelta:e.data.volume30dDelta};return this.logger.debug(`Fetched DEX volume summary: $${t.volume1d.toFixed(2)} (1d)`),t}catch(e){throw oe(`Failed to fetch DEX volume summary: ${to(e)}`)}}}const Vo=8,zo=18,Ho={collection:"GALA",category:"Unit",type:"none",additionalKey:"none"};function Qo(e,t=18){const n=parseFloat(e);if(0===n)return"0";return n.toFixed(t).replace(/\.?0+$/,"")}function Xo(e){return Qo(e,8)}function Zo(e){return Qo(e,18)}function Yo(e){return{maxAcceptableReverseBondingCurveFee:Xo(e.maxAcceptableReverseBondingCurveFee)}}new U({debug:!1,context:"NumberUtils"});const Jo={BuyNativeDto:class extends s{constructor(e,t,n="0",o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=Xo(t),this.expectedToken=Zo(n),this.extraFees=Yo(o)}},BuyExactDto:class extends s{constructor(e,t,n,o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Zo(t),this.expectedNativeToken=Xo(n),this.extraFees=Yo(o)}},SellExactDto:class extends s{constructor(e,t,n="0",o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Zo(t),this.expectedNativeToken=Xo(n),this.extraFees=Yo(o)}},SellNativeDto:class extends s{constructor(e,t,n,o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=Xo(t),this.expectedToken=Zo(n),this.extraFees=Yo(o)}}};class ei extends Rn{constructor(e,t=!1){super(t),this.walletProvider=e}async signDTO(e,t,n){try{this.logger.debug("🔐 Signing DTO:",{methodName:t,dtoKeys:Object.keys(e)});const n=this.generateEIP712Types(t,e),o=c(e),i={...e,prefix:o},{signature:r,domain:a}=await this.signWithEthersWallet(n,i),s={...e,signature:r,types:n,domain:a};return this.logger.debug("✅ DTO signed successfully:",{payloadKeys:Object.keys(s),signatureLength:r.length}),s}catch(e){this.logger.error("❌ Signature generation failed:",e);throw re(`Failed to sign DTO: ${to(e)}`)}}async signWithEthersWallet(e,t){try{let n,o;if(this.walletProvider.signTypedData&&!this.walletProvider.getNetwork)n={name:"ethereum",chainId:1},o=await this.walletProvider.signTypedData(n,e,t);else{if(!this.walletProvider.getNetwork||!this.walletProvider.signTypedData)throw ie("Wallet provider does not support typed data signing","walletProvider");{const i=await this.walletProvider.getNetwork();n={name:i.name,chainId:Number(i.chainId)},o=await this.walletProvider.signTypedData(n,e,t)}}return{signature:o,domain:n}}catch(e){throw re(`Ethers.js signing failed: ${to(e)}`)}}generateEIP712Types(e,t){const n={};n[e]=[];const o=Object.fromEntries(Object.entries(t).filter(([e,t])=>void 0!==t)),i=(e,t,o,r=!1)=>{if(void 0!==t){if(Array.isArray(t)){if(0===t.length)return;const a=i(e,t[0],o,!0);return r||n[o].push({name:e,type:(a??e)+"[]"}),a?a+"[]":void 0}if("object"==typeof t&&null!==t){if(n[e])throw new G(`Type name collision not supported: ${e}`,"fieldValue","TYPE_COLLISION");return n[e]=[],Object.entries(t).forEach(([t,n])=>{i(t,n,e)}),r||n[o].push({name:e,type:e}),e}{let i;switch(typeof t){case"string":i="string";break;case"number":i="uint256";break;case"boolean":i="bool";break;default:throw new G(`Unsupported type for field "${e}": ${typeof t} (value: ${JSON.stringify(t)})`,"fieldValue","UNSUPPORTED_TYPE")}return r||n[o].push({name:e,type:i}),i}}};return Object.entries(o).forEach(([t,n])=>{i(t,n,e)}),this.logger.debug("📝 Generated EIP-712 types:",n),n}}class ti extends Rn{constructor(e=!1){super(e)}generateStringsInstructions(e){try{this.logger.debug("🔧 Generating stringsInstructions for:",e);const t=this.extractTokenSymbolFromVault(e),n=this.createTokenInstance(t),o=this.createGalaInstance(),i=`$service$${n.toStringKey()}$launchpad`,r=`$tokenBalance$${n.toStringKey()}$${e}`,a=`$tokenBalance$${n.toStringKey()}$${e}`,s=`$tokenBalance$${o.toStringKey()}$${e}`,c=[i,r,a,s,`$tokenBalance$${o.toStringKey()}$${e}`];return this.logger.debug("✅ Generated stringsInstructions:",c),c}catch(e){this.logger.error("❌ Failed to generate stringsInstructions:",e);const t=to(e);throw new G(`Failed to generate stringsInstructions: ${t}`,"vaultAddress","INVALID_VAULT_ADDRESS")}}createTokenInstance(e){const t=new I;return t.collection=e.toLowerCase(),t.category="Unit",t.type="none",t.additionalKey="none",this.logger.debug("🪙 Created token instance:",{symbol:e,lowercaseCollection:e.toLowerCase(),stringKey:t.toStringKey()}),t}createGalaInstance(){const e=new I;return e.collection="GALA",e.category="Unit",e.type="none",e.additionalKey="none",this.logger.debug("🟡 Created GALA instance:",{stringKey:e.toStringKey()}),e}extractTokenSymbolFromVault(e){if(!e||"string"!=typeof e)throw te("vaultAddress","Vault address");try{const t=Oo(e);return this.logger.debug("🔍 Extracted token symbol:",{vaultAddress:e,tokenSymbol:t}),t}catch(e){if(e instanceof G)throw ne("vaultAddress","format: service|Token$Unit$SYMBOL$eth:address$launchpad");throw e}}validateVaultAddress(e){if(!e||"string"!=typeof e)throw te("vaultAddress","Vault address");if(!e.startsWith("service|Token$Unit$"))throw ne("vaultAddress",'starting with "service|Token$Unit$"');if(!e.endsWith("$launchpad"))throw ne("vaultAddress",'ending with "$launchpad"');const t=e.split("$");if(t.length<5)throw ne("vaultAddress",'having at least 5 parts separated by "$"');const n=t[2];if(!n||!/^[A-Za-z]{1,10}$/.test(n))throw ne("vaultAddress","containing a 1-10 letter token symbol (case insensitive)");return this.logger.debug("✅ Vault address validation passed:",e),!0}generateTokenClassKeyString(e,t,n,o){return`${e}$${t}$${n}$${o}`}parseTokenClassKeyString(e){try{return Po(e)}catch(e){if(e instanceof G)throw ne("stringKey","format: collection$category$type$additionalKey (4 parts)");throw e}}}function ni(e,t,n){if(t<0||t>1)throw new Error(`Invalid slippage tolerance factor: ${t}. Must be between 0 and 1 (e.g., 0.05 for 5%)`);const o=new u(e);if(o.isNaN())throw new Error(`Invalid expected amount: ${e}. Must be a valid number`);if(0===t)return e;const i=o.multipliedBy(t);let r;switch(n){case"buy-native":case"sell-exact":r=o.minus(i);break;case"buy-exact":case"sell-native":r=o.plus(i);break;default:throw new Error(`Unknown operation type: ${n}`)}return r.isLessThan(0)&&(r=new u(0)),r.toFixed()}class oi extends Mn{constructor(e,t,n=!1,o,i,r=.05,a=.01){super(e,n),this.tokenResolver=t,this.walletProvider=o,this.userAddress=i,this.defaultSlippageToleranceFactor=r,this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor=a,this.bundleEndpoint="/bundle",o&&i&&(this.signatureService=new ei(o,n),this.tokenKeyService=new ti(n))}async submitTransaction(e){try{this.logger.debug("📦 Submitting bundle transaction:",{method:e.method,stringsInstructionsCount:e.stringsInstructions.length,signedDtoKeys:Object.keys(e.signedDto)}),this.validateBundleData(e);const t=this.formatBundleRequest(e);this.logger.debug("🚀 Bundle request payload:",{...t,signedDto:"[REDACTED - Contains signature]"});const n=await this.http.post(this.bundleEndpoint,t);return n?(this.logger.debug("📥 Bundle API response:",{success:n.success,hasData:!!n.data,error:n.error}),this.handleBundleResponse(n)):{success:!1,error:"No response from bundle API"}}catch(e){return this.logger.error("❌ Bundle transaction submission failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}validateBundleData(e){if(!e)throw te("bundleData","Bundle data");if(!e.signedDto)throw te("signedDto","Signed DTO");if(!e.method||"string"!=typeof e.method)throw te("method","Method name");if(!Array.isArray(e.stringsInstructions))throw ne("stringsInstructions","an array of resource tracking strings");if(0===e.stringsInstructions.length)throw new G("stringsInstructions cannot be empty","stringsInstructions","EMPTY_ARRAY");const t=["BuyWithNative","BuyExactToken","SellExactToken","SellWithNative"];if(!t.includes(e.method))throw ne("method",`one of: ${t.join(", ")}`);e.stringsInstructions.forEach((e,t)=>{if("string"!=typeof e||0===e.length)throw new G(`stringsInstructions[${t}] must be a non-empty string`,`stringsInstructions[${t}]`,"INVALID_INSTRUCTION");if(!e.startsWith("$"))throw new G(`stringsInstructions[${t}] must start with '$': ${e}`,`stringsInstructions[${t}]`,"INVALID_INSTRUCTION_FORMAT")}),this.logger.debug("✅ Bundle data validation passed")}formatBundleRequest(e){return{signedDto:e.signedDto,stringsInstructions:e.stringsInstructions,method:e.method}}handleBundleResponse(e){if(e.data&&!1===e.error)return this.logger.debug("✅ Bundle transaction successful:",e.data),{success:!0,data:e.data};const t=e.error||e.message||"Bundle transaction failed";return this.logger.debug("❌ Bundle transaction failed:",t),{success:!1,error:t}}formatErrorMessage(e){if("string"==typeof e)return e;if(no(e)&&e.response){const t=e.response.data;if(t&&"object"==typeof t){const e=t;if(e.error)return String(e.error);if(e.message)return String(e.message)}}return to(e)||"Unknown bundle transaction error"}async getBundlerTransactionResult(e){try{if(!e||"string"!=typeof e)throw te("transactionId","Transaction ID");this.logger.debug("🔍 Checking bundler transaction result:",e);const t=await this.http.get(`${this.bundleEndpoint}?id=${e}`);return t?(this.logger.debug("📊 Bundler transaction result:",t),{success:!0,data:t}):{success:!1,error:"No response from bundler transaction query"}}catch(e){return this.logger.error("❌ Failed to get bundler transaction result:",e),{success:!1,error:this.formatErrorMessage(e)}}}async cancelTransaction(e){try{if(!e||"string"!=typeof e)throw te("transactionId","Transaction ID");this.logger.debug("🚫 Cancelling transaction:",e);const t=await this.http.delete(`${this.bundleEndpoint}/${e}`);return t?(this.logger.debug("🗑️ Transaction cancellation response:",t),{success:!0,data:t}):{success:!1,error:"No response from transaction cancellation"}}catch(e){return this.logger.error("❌ Failed to cancel transaction:",e),{success:!1,error:this.formatErrorMessage(e)}}}async getHealthStatus(){try{this.logger.debug("🏥 Checking bundle service health");const e=await this.http.get(`${this.bundleEndpoint}/health`);return e?(this.logger.debug("💚 Bundle service health:",e),{success:!0,data:e}):{success:!1,error:"No response from bundle service health check"}}catch(e){return this.logger.error("❌ Bundle service health check failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}async buyToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:n,type:o,expectedAmount:i}=e,{effectiveSlippageFactor:r,effectiveMaxFee:a,vaultAddress:s}=await this.prepareTradingOperation(t,e.maxAcceptableReverseBondingCurveFee,e.maxAcceptableReverseBondingCurveFeeSlippageFactor,e.slippageToleranceFactor);if("native"===o){if(!i)throw new G("expectedAmount is required for native buy operations. Use getBuyTokenAmount() first to calculate expected tokens.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=ni(i,r,"buy-native");this.logger.debug("BuyNative slippage applied:",{originalExpectedTokens:i,slippageFactor:r,adjustedMinTokens:e});const t=new Jo.BuyNativeDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"BuyWithNative",s)}{if(!i)throw new G("expectedAmount is required for exact buy operations. Use getBuyTokenAmount() first to calculate expected GALA cost.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=ni(i,r,"buy-exact");this.logger.debug("BuyExact slippage applied:",{originalExpectedGalaCost:i,slippageFactor:r,adjustedMaxGalaCost:e});const t=new Jo.BuyExactDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"BuyExactToken",s)}}async sellToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:n,type:o,expectedAmount:i}=e,{effectiveSlippageFactor:r,effectiveMaxFee:a,vaultAddress:s}=await this.prepareTradingOperation(t,e.maxAcceptableReverseBondingCurveFee,e.maxAcceptableReverseBondingCurveFeeSlippageFactor,e.slippageToleranceFactor);if("exact"===o){if(!i)throw new G("expectedAmount is required for exact sell operations. Use getSellTokenAmount() first to calculate expected GALA.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=ni(i,r,"sell-exact");this.logger.debug("SellExact slippage applied:",{originalExpectedGala:i,slippageFactor:r,adjustedMinGala:e});const t=new Jo.SellExactDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"SellExactToken",s)}{if(!i)throw new G("expectedAmount is required for native sell operations. Use getSellTokenAmount() first to calculate tokens to sell.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=ni(i,r,"sell-native");this.logger.debug("SellNative slippage applied:",{originalExpectedTokensToSell:i,slippageFactor:r,adjustedMaxTokensToSell:e});const t=new Jo.SellNativeDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"SellWithNative",s)}}async prepareTradingOperation(e,t,n,o){const{effectiveSlippageFactor:i,effectiveMaxFee:r}=this.calculateEffectiveSlippage(t,n,o),a=await this.resolveTokenNameToVault(e);if(!a)throw ee(e);return{effectiveSlippageFactor:i,effectiveMaxFee:r,vaultAddress:a}}calculateEffectiveSlippage(e,t,n){const o=n??this.defaultSlippageToleranceFactor,i=t??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let r=e||"0";return e&&(r=ni(e,i,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:e,slippageFactor:i,adjustedMaxFee:r})),{effectiveSlippageFactor:o,effectiveFeeSlippageFactor:i,effectiveMaxFee:r}}ensureTradingServicesAvailable(){if(!this.signatureService||!this.tokenKeyService)throw ie("Trading services not available. BundleService requires walletProvider and userAddress for trading operations.","walletProvider");if(!this.userAddress)throw te("userAddress","User address")}async executeBundleTransaction(e,t,n){this.ensureTradingServicesAvailable();try{e.uniqueKey=`galaswap - operation - ${N()}-${Date.now()}-${this.userAddress}`;const o=await this.signatureService.signDTO(e,t,this.userAddress),i=this.tokenKeyService.generateStringsInstructions(n),r={stringsInstructions:i,method:t,signedDto:o};this.logger.debug("📦 Bundle transaction data:",{method:t,stringsInstructions:i,dtoKeys:Object.keys(o)});const a=await this.submitTransaction(r);if(a.success&&a.data)return this.logger.debug("✅ Bundle transaction submitted:",a.data),{success:!0,data:{transactionId:a.data,message:"Transaction submitted successfully. Monitor WebSocket for completion."}};throw new Error(String(a.error||"Bundle transaction failed"))}catch(e){throw this.logger.error("❌ Bundle transaction error:",e),e}}async resolveTokenNameToVault(e){return await this.tokenResolver.resolveTokenToVault(e)}}var ii,ri;!function(e){e.PROCESSED="PROCESSED",e.COMPLETED="COMPLETED",e.SUCCESS="SUCCESS",e.FAILED="FAILED",e.ERROR="ERROR",e.PROCESSING="PROCESSING",e.PENDING="PENDING"}(ii||(ii={})),function(e){e.PENDING="pending",e.PROCESSING="processing",e.COMPLETED="completed",e.FAILED="failed",e.TIMEOUT="timeout"}(ri||(ri={}));const ai={[ii.PROCESSED]:ri.COMPLETED,[ii.COMPLETED]:ri.COMPLETED,[ii.SUCCESS]:ri.COMPLETED,[ii.FAILED]:ri.FAILED,[ii.ERROR]:ri.FAILED,[ii.PROCESSING]:ri.PROCESSING,[ii.PENDING]:ri.PENDING};class si extends Rn{constructor(e,t=!1){super(t),this.socket=null,this.listeners=new Map,this.timeouts=new Map,this.reconnectCount=0,this.hasOnAnyListener=!1,this.eventBuffer=new Map,this.eventBufferTimeouts=new Map,this.MAX_BUFFER_SIZE=1e3,this.config={reconnectAttempts:5,reconnectDelay:2e3,timeout:3e5,...e},this.debug=t,this.isSocketIOAvailable=this.checkSocketIOAvailability()}checkSocketIOAvailability(){try{return"function"==typeof D||(this.logger.warn('⚠️ Socket.IO client not available. Install "socket.io-client" package.'),!1)}catch(e){return this.logger.warn("⚠️ Socket.IO availability check failed:",e),!1}}async connect(){return new Promise((e,t)=>{try{if(!this.isSocketIOAvailable){const e=new Error('Socket.IO not available in current environment. Install "socket.io-client" package.');return this.logger.error("❌ Socket.IO connection failed:",e.message),void t(e)}this.logger.debug("🔌 Connecting to Socket.IO server:",this.config.url),this.socket=D(this.config.url,{transports:["websocket"],reconnection:!0,reconnectionAttempts:this.config.reconnectAttempts||5,reconnectionDelay:this.config.reconnectDelay||2e3}),this.socket.on("connect",()=>{this.logger.debug("✅ Socket.IO connected successfully:",this.socket?.id),this.logger.debug("📡 Connected to bundle backend WebSocket:",this.config.url),this.logger.debug("🔗 Ready to monitor transaction updates"),this.reconnectCount=0,e()}),this.socket.on("connect_error",e=>{this.logger.error("❌ Socket.IO connection error:",e),t(e)}),this.socket.on("disconnect",e=>{this.logger.debug(`🔌 Socket.IO disconnected: ${e}`),this.handleReconnect()}),this.socket.on("error",e=>{this.logger.error("❌ Socket.IO error:",e)}),this.socket.onAny((e,...t)=>{if(e&&t.length>0&&"object"==typeof t[0]&&null!==t[0]){const n=t[0],o=n.status||n.Status;if(o&&"string"==typeof o){if(this.logger.debug(`📡 [Event Buffer] Buffering event for ${e}: ${o}`),this.eventBuffer.size>=this.MAX_BUFFER_SIZE){const e=this.eventBuffer.keys().next().value;if(e){const t=this.eventBufferTimeouts.get(e);t&&(clearTimeout(t),this.eventBufferTimeouts.delete(e)),this.eventBuffer.delete(e),this.logger.warn(`📡 [Event Buffer] Buffer full (${this.MAX_BUFFER_SIZE}), dropped oldest: ${e}`)}}this.eventBuffer.set(e,n);const t=this.eventBufferTimeouts.get(e);t&&clearTimeout(t);const i=setTimeout(()=>{this.eventBuffer.has(e)&&(this.logger.debug(`📡 [Event Buffer] Cleaning up orphaned event for ${e}`),this.eventBuffer.delete(e),this.eventBufferTimeouts.delete(e))},3e4);this.eventBufferTimeouts.set(e,i)}}this.debug&&this.logger.debug(`📡 [WebSocket Event] "${e}":`,JSON.stringify(t,null,2))}),this.hasOnAnyListener=!0}catch(e){t(e)}})}async monitorTransaction(e,t){this.listeners.set(e,t),this.logger.debug(`📡 Starting to monitor transaction: ${e}`),this.logger.debug(`📡 WebSocket connected: ${!!this.socket&&this.socket.connected}`);const n=this.eventBuffer.get(e);if(n){this.logger.debug(`📡 [Event Buffer] Found buffered event for ${e}, delivering immediately`),setImmediate(()=>{this.processTransactionEvent(e,n,t)});const o=this.eventBufferTimeouts.get(e);o&&(clearTimeout(o),this.eventBufferTimeouts.delete(e)),this.eventBuffer.delete(e)}const o=this.config.timeout||3e5,i=setTimeout(()=>{if(this.listeners.has(e)){const n=Math.round(o/1e3),i={transactionId:e,status:ri.TIMEOUT,message:`Transaction monitoring timeout - no response after ${n} seconds`,timestamp:Date.now()};this.logger.debug(`📡 Transaction timeout for ${e} (${n}s)`),t(i),this.listeners.delete(e),this.timeouts.delete(e),this.socket?.off(e)}},o);if(this.timeouts.set(e,i),this.socket&&this.socket.connected)this.socket.off(e),this.logger.debug(`📡 Listening for transaction updates: ${e}`),this.logger.debug(`📡 WebSocket connection ID: ${this.socket.id}`),this.logger.debug(`📡 WebSocket URL: ${this.config.url}`),this.socket.on(e,n=>{this.processTransactionEvent(e,n,t)});else{const n={transactionId:e,status:ri.FAILED,message:"WebSocket not connected - cannot monitor transaction",timestamp:Date.now()};t(n),this.listeners.delete(e),this.timeouts.delete(e)}}processTransactionEvent(e,t,n){this.logger.debug(`📡 Socket.IO transaction update for ${e}:`,JSON.stringify(t,null,2));const o=t,i=o?.data,r=o?.status||o?.Status||i?.status||i?.Status;let a=o?.message||o?.Message||i?.message||i?.Message||o?.error||i?.error;a&&"string"==typeof a||(a=r===ii.FAILED||r===ii.ERROR?"Transaction failed - check transaction details":r===ii.COMPLETED||r===ii.PROCESSED||r===ii.SUCCESS?"Transaction completed successfully":r?`Transaction status: ${r}`:"Unknown transaction status");const s=o?.blockHash||i?.blockHash,c=o?.gasUsed||i?.gasUsed,l=o?.Data||i?.Data,u={transactionId:e,status:this.mapSocketStatus(r),message:"string"==typeof a?a:"Transaction update received",timestamp:Date.now(),...s&&{blockHash:s},...c&&{gasUsed:c},...l&&{data:l}};if(this.logger.debug(`📡 Mapped status for ${e}: ${r} -> ${u.status}`),this.logger.debug(`📡 Final message: "${a}"`),n(u),u.status===ri.COMPLETED||u.status===ri.FAILED){this.listeners.delete(e);const t=this.timeouts.get(e);t&&(clearTimeout(t),this.timeouts.delete(e)),this.socket?.off(e),this.logger.debug(`📡 Cleaned up listener for ${e} (${u.status})`)}}async waitForTransaction(e){return new Promise((t,n)=>{this.monitorTransaction(e,e=>{e.status===ri.COMPLETED?t(e):e.status!==ri.FAILED&&e.status!==ri.TIMEOUT||n(new Error(`Transaction ${e.status}: ${e.message}`))})})}mapSocketStatus(e){const t=e?.toUpperCase();return ai[t]||ri.PENDING}async handleReconnect(){this.reconnectCount<this.config.reconnectAttempts?(this.reconnectCount++,this.logger.debug(`🔄 Attempting Socket.IO reconnect ${this.reconnectCount}/${this.config.reconnectAttempts}`),setTimeout(()=>{this.socket&&!this.socket.connected&&this.socket.connect()},this.config.reconnectDelay)):this.logger.error("❌ Socket.IO max reconnection attempts reached")}disconnect(){this.socket&&(this.listeners.forEach((e,t)=>{this.socket?.off(t)}),this.listeners.clear(),this.timeouts.forEach(e=>{clearTimeout(e)}),this.timeouts.clear(),this.eventBuffer.clear(),this.eventBufferTimeouts.forEach(e=>{clearTimeout(e)}),this.eventBufferTimeouts.clear(),this.logger.debug("🧹 Cleared event buffer and timeouts"),this.hasOnAnyListener&&(this.socket.offAny(),this.hasOnAnyListener=!1,this.logger.debug("🧹 Removed onAny debug listener")),this.socket.disconnect(),this.socket=null,this.logger.debug("🔌 Socket.IO disconnected"))}isConnected(){return this.socket?.connected||!1}getSocket(){return this.socket}}class ci extends Rn{constructor(e,t=!1){super(t),this.poolService=e,this.cache=new Map}async resolveTokenToVault(e){if(!ce(e))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");const t=e.trim().toLowerCase(),n=this.get(t);if(n)return n;try{const n=await this.poolService.resolveTokenNameToVault(e);return n&&this.set(t,n),n}catch{return null}}async resolveTokenClassKey(e){const t=await this.resolveTokenToVault(e);if(!t)throw ee(e);return this.parseVaultAddressToTokenClassKey(t)}get(e){return this.cache.get(e.toLowerCase())||null}set(e,t){this.cache.set(e.toLowerCase(),t)}clear(){this.cache.clear()}getStats(){return{size:this.cache.size,keys:Array.from(this.cache.keys())}}preWarm(e){for(const{tokenName:t,vaultAddress:n}of e)this.set(t,n)}parseVaultAddressToTokenClassKey(e){try{return $o(e)}catch(e){if(e instanceof G)throw ne("vaultAddress","format: service|Token$Unit$SYMBOL$eth:address$launchpad","Vault address");throw e}}}async function li(e,t={}){const{maxPages:n=1e4,logger:o,pageSize:i=20}=t,r=[];let a=1,s=!0,c=0;for(;s&&a<=n;){o&&o.debug(`Auto-pagination: fetching page ${a} with limit ${i}`);const t=await e(a,i);if(!t||!Array.isArray(t.items)){o&&o.warn("Auto-pagination: received invalid result structure, stopping");break}if(r.push(...t.items),c=t.total,o&&o.debug(`Auto-pagination: page ${a} returned ${t.items.length} items, hasNext: ${t.hasNext}`),0===t.items.length){o&&o.debug(`Auto-pagination: no items returned on page ${a}, exiting loop`);break}s=t.hasNext,a++}return a>n&&o&&o.warn(`Auto-pagination: exceeded maxPages limit of ${n}, stopping iteration`),o&&o.debug(`Auto-pagination: completed with total items: ${r.length}, total count: ${c}`),{items:r,total:c}}function ui(e,t=e.length,n="items"){return{...{page:1,limit:e.length||0,total:t,totalPages:Math.ceil(t/(e.length||1))||1,hasNext:!1,hasPrevious:!1},[n]:e}}class di extends Mn{constructor(e,t=!1,n){super(e,t),this.tokenResolverService=n}async fetchTokenClassKeyByTokenName(e){if(!this.tokenResolverService)throw ie("TokenResolverService is required for token name resolution. Ensure it is passed to PriceHistoryService constructor.","tokenResolverService");if(!e||"string"!=typeof e||e.length<3||e.length>20)throw ie("Token name must be a string between 3 and 20 characters","tokenName");this.logger.debug(`Resolving token name '${e}' to token class key`);try{const t=await this.tokenResolverService.resolveTokenToVault(e);if(!t)throw ie(`Token '${e}' not found or could not be resolved to vault address`,"tokenName");this.logger.debug(`Resolved '${e}' to vault address: ${t}`);const n=$o(t),o=`${n.collection}|${n.category}|${n.type}|${n.additionalKey}`;return this.logger.debug(`Extracted token class key: ${o}`),o}catch(t){if(t instanceof Error&&t.message.includes("ConfigurationError"))throw t;throw oe(`Failed to resolve token name '${e}': ${to(t)}`,500)}}async fetchPriceHistory(e){this.logger.debug("Fetching price history from DEX Backend API with options:",e),this.validateOptions(e);try{let t=e.tokenId;if(e.tokenName){this.logger.debug(`Resolving token name '${e.tokenName}' to token ID`);const n=await this.fetchTokenClassKeyByTokenName(e.tokenName);t=n,this.logger.debug(`Resolved to token ID: ${n}`)}if(!t)throw ie("Token ID is required but was not provided or resolved","tokenId");const{normalizeToTokenInstanceKey:n}=await Promise.resolve().then(function(){return Bo}),o=n(t),i=Io(`${o.collection}|${o.category}|${o.type}|${o.additionalKey}`),{from:r,to:a,sortOrder:s="DESC",page:c=1,limit:l=10}=e,u={token:i,page:String(c),limit:String(l)};r&&(u.from=r.toISOString()),a&&(u.to=a.toISOString());const d=function(e){if(e)return e.toLowerCase()}(s);d&&(u.order=d),this.logger.debug(`Querying price snapshots for token ${i}, page ${c}, limit ${l}`);const h=await this.http.get("/price-oracle/fetch-price",u);if(!h)throw oe("No response from price history service",500);const g=this.transformApiResponseToPriceHistory(h);return this.logger.debug(`Found ${g.snapshots.length} price snapshots, total ${g.total}`),g}catch(e){if(e instanceof Error&&(e.message.includes("ConfigurationError")||e.message.includes("NetworkError")))throw e;throw oe(`Failed to fetch price history: ${to(e)}`,500)}}transformApiResponseToPriceHistory(e){if(!e?.data?.data||!Array.isArray(e.data.data))throw oe("Invalid API response: missing or invalid data.data array",500);if(!e?.data?.meta)throw oe("Invalid API response: missing data.meta pagination info",500);const t=e.data.data.map(e=>({price:e.price,timestamp:new Date(e.createdAt),tokenId:`${e.collection}|${e.category}|${e.type}|${e.additionalKey}`})),n=e.data.meta,o=n.currentPage??1,i=n.totalPages??1;return{snapshots:t,page:o,limit:n.pageSize??50,total:n.totalItems??0,totalPages:i,hasNext:o<i,hasPrevious:o>1}}async fetchAllPriceHistory(e){this.logger.debug("Fetching all price history with options:",e);const t=await li((t,n)=>this.fetchPriceHistory({...e,page:t,limit:n}).then(e=>({items:e.snapshots,page:e.page,limit:e.limit,total:e.total,totalPages:e.totalPages,hasNext:e.hasNext,hasPrevious:e.hasPrevious})),{maxPages:1e4,logger:this.logger,pageSize:50});return ui(t.items,t.total,"snapshots")}validateOptions(e){const{from:t,to:n,sortOrder:o,page:i=1,limit:r=10}=e;if(function(e,t,n,o={}){const{description:i="parameter",treatEmptyAsNull:r=!0}=o,a=e[t],s=e[n],c=null!=a&&(!r||""!==a),l=null!=s&&(!r||""!==s);if(!c&&!l)throw ie(`Either ${t} or ${n} must be provided (${i})`,n);if(c&&l)throw ie(`Cannot provide both ${t} and ${n}. Provide exactly one (${i}).`,n)}(e,"tokenName","tokenId",{description:"token identifier"}),t&&!(t instanceof Date)&&isNaN(new Date(t).getTime()))throw ie("from must be a valid Date","from");if(n&&!(n instanceof Date)&&isNaN(new Date(n).getTime()))throw ie("to must be a valid Date","to");if(o&&"ASC"!==o&&"DESC"!==o)throw ie('sortOrder must be either "ASC" or "DESC"',"sortOrder");if(!Number.isInteger(i)||i<1)throw ie("page must be a positive integer","page");if(!Number.isInteger(r)||r<1||r>50)throw ie("limit must be between 1 and 50","limit")}}function hi(e){if("object"==typeof e&&null!==e)return function(e){if(!e||"object"!=typeof e)throw new G("Token object must be a non-null object, got "+typeof e,"token","INVALID_TOKEN_OBJECT");const{collection:t,category:n,type:o,additionalKey:i}=e;if(!t||"string"!=typeof t)throw new G("Token.collection must be a non-empty string, got "+typeof t,"token.collection","MISSING_OR_INVALID_COLLECTION");if(!n||"string"!=typeof n)throw new G("Token.category must be a non-empty string, got "+typeof n,"token.category","MISSING_OR_INVALID_CATEGORY");if(!o||"string"!=typeof o)throw new G("Token.type must be a non-empty string, got "+typeof o,"token.type","MISSING_OR_INVALID_TYPE");if(!i||"string"!=typeof i)throw new G("Token.additionalKey must be a non-empty string, got "+typeof i,"token.additionalKey","MISSING_OR_INVALID_ADDITIONAL_KEY");return{collection:t,category:n,type:o,additionalKey:i}}(e);if(!e)throw new G(`Token cannot be null, undefined, or empty. Received: ${JSON.stringify(e)}`,"token","EMPTY_TOKEN");if("string"!=typeof e)throw new G("Token must be a string or TokenClassKey object, got "+typeof e,"token","INVALID_TOKEN_TYPE");if(e.includes("|"))return function(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("|");if(t.length<4)throw new Error(`Invalid pipe-delimited token format. Expected 4+ parts separated by |, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("|");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){const n=e.split("|");throw new G(`Invalid pipe-delimited token: "${e}" (${n.length} parts). Expected format: "collection|category|type|additionalKey" (4 parts minimum). Received: [${n.map(e=>`"${e}"`).join(", ")}]. Error: ${t instanceof Error?t.message:String(t)}`,"token","INVALID_PIPE_DELIMITED_TOKEN")}}(e);if(e.includes("$"))return function(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("$");if(t.length<4)throw new Error(`Invalid dollar-delimited token format. Expected 4+ parts separated by $, got ${t.length}`);const n=t[t.length-1],o=t[t.length-2],i=t[t.length-3],r=t.slice(0,t.length-3).join("$");if(!(r&&i&&o&&n))throw new Error("All components (collection, category, type, additionalKey) must be non-empty");return{collection:r,category:i,type:o,additionalKey:n}}catch(t){const n=e.split("$");throw new G(`Invalid dollar-delimited token: "${e}" (${n.length} parts). Expected format: "collection$category$type$additionalKey" (4 parts minimum). Received: [${n.map(e=>`"${e}"`).join(", ")}]. Error: ${t instanceof Error?t.message:String(t)}`,"token","INVALID_DOLLAR_DELIMITED_TOKEN")}}(e);throw new G(`Plain token string "${e}" (length: ${e.length}) is not allowed - tokens must be delimited with | or $. Expected format: "GALA|Unit|none|none" or "GALA$Unit$none$none". Input: "${e}"`,"token","PLAIN_STRING_NOT_ALLOWED")}class gi extends Error{constructor(e,t,n){super(`API Error [${e}]: ${t}`),this.status=e,this.message=t,this.details=n,this.name="ApiError"}}function pi(e){return"object"==typeof e&&null!==e&&"collection"in e&&"category"in e&&"type"in e&&"additionalKey"in e&&"string"==typeof e.collection&&"string"==typeof e.category&&"string"==typeof e.type&&"string"==typeof e.additionalKey}function mi(e){return"object"==typeof e&&null!==e&&"positionId"in e&&"owner"in e&&"token0"in e&&"token1"in e&&"fee"in e&&"liquidity"in e&&"string"==typeof e.positionId&&"string"==typeof e.owner&&"number"==typeof e.fee}class fi{constructor(t){this.client=e.create({baseURL:t.baseUrl,timeout:t.timeout??3e4})}async getPoolData(e){try{if("string"==typeof e.token0||"string"==typeof e.token1)throw new Error(`GalaChain API getPoolData requires TokenClassKey objects, not strings. Received: token0="${"string"==typeof e.token0?e.token0:"[object]"}", token1="${"string"==typeof e.token1?e.token1:"[object]"}". Convert pipe-delimited tokens using parseToken() before calling getPoolData(). Example: parseToken("GALA|Unit|none|none") → { collection: "GALA", category: "Unit", type: "none", additionalKey: "none" }`);const t=await this.client.post("/api/asset/dexv3-contract/GetPoolData",e);this.validateResponse(t.data);const n=t.data.Data;if(!function(e){if("object"!=typeof e||null===e)return!1;const t=e;return"string"==typeof t.token0&&"string"==typeof t.token1&&pi(t.token0ClassKey)&&pi(t.token1ClassKey)&&"number"==typeof t.fee&&"number"==typeof t.tickSpacing&&"string"==typeof t.liquidity&&"string"==typeof t.sqrtPrice&&"number"==typeof t.tick&&"string"==typeof t.feeGrowthGlobal0&&"string"==typeof t.feeGrowthGlobal1}(n))throw new gi(t.status,"Invalid pool data response format",n);return n}catch(e){throw this.handleError(e,"getPoolData")}}async getPositions(e){try{const t=await this.client.post("/api/asset/dexv3-contract/GetPositions",e);this.validateResponse(t.data);const n=t.data.Data;if(!Array.isArray(n.positions))throw new gi(t.status,"Invalid positions response format: positions must be an array",n);for(const e of n.positions)if(!mi(e))throw new gi(t.status,"Invalid position in response",e);return n}catch(e){throw this.handleError(e,"getPositions")}}async getPositionById(e,t,n,o,i,r,a){try{let s,c;if(void 0!==t&&void 0!==n&&void 0!==o&&void 0!==i&&void 0!==r){s={owner:e,token0:"string"==typeof t?{collection:t,category:"Unit",type:"none",additionalKey:"none"}:t,token1:"string"==typeof n?{collection:n,category:"Unit",type:"none",additionalKey:"none"}:n,fee:o,tickLower:i,tickUpper:r},a&&(s.positionId=a),c=`${e}/${t}/${n}/${o}`}else s={positionId:e},c=e;const l=await this.client.post("/api/asset/dexv3-contract/GetPositions",s);this.validateResponse(l.data);const u=l.data.Data;let d;if(u&&"object"==typeof u&&"positionId"in u&&!("positions"in u))d=u;else{if(!(u&&Array.isArray(u.positions)&&u.positions.length>0))throw new gi(404,`Position not found: ${c}`);d=u.positions[0]}const h={Data:d,Status:l.status};return void 0!==l.data.Message&&(h.Message=l.data.Message),h}catch(t){throw this.handleError(t,`getPositionById(${e})`)}}async getRemoveLiquidityEstimation(e){try{const t=await this.client.post("/api/asset/dexv3-contract/GetRemoveLiquidityEstimation",e);this.validateResponse(t.data);const n=t.data.Data;if("string"!=typeof n.amount0||"string"!=typeof n.amount1||"string"!=typeof n.fee0||"string"!=typeof n.fee1)throw new gi(t.status,"Invalid removal estimation response format",n);return n}catch(e){throw this.handleError(e,"getRemoveLiquidityEstimation")}}validateResponse(e){if(!e||"object"!=typeof e)throw new gi(500,"Invalid response format: not an object");if(!("Data"in e)||!("Status"in e))throw new gi(500,"Invalid response format: missing Data or Status field");if(e.Status>=400)throw new gi(e.Status,e.Message??"Gateway error",e.Data)}handleError(t,n){if(t instanceof gi)return t;if(e.isAxiosError(t)){const e=t.response?.status??500,o=t.response?.data?.Message??t.message,i=t.response?.data?.Data??void 0;return new gi(e,`${n}: ${o}`,i)}return new gi(500,`${n}: ${t instanceof Error?t.message:String(t)}`)}}class yi{constructor(e){this.http=e}async getUserAssets(e,t=20,n=0){try{if(!e||"string"!=typeof e)throw new gi(400,"Invalid wallet address");const o=Math.max(1,Math.floor(n/t)+1),i={};i.address=e,i.page=o,i.limit=t;const r=await this.http.get("/user/assets",i);if(!r||"object"!=typeof r)throw new gi(500,"Invalid response format: not an object");const a=r.data;if(!a||"object"!=typeof a)throw new gi(500,"Invalid response format: missing data wrapper");const s=a.token;if(!Array.isArray(s))throw new gi(500,"Invalid response format: token array must be an array");const c=[];for(const e of s){if("object"!=typeof e||null===e)throw new gi(500,"Invalid asset in response: asset must be an object");const t=e;if("string"!=typeof t.symbol||"string"!=typeof t.name)throw new gi(500,"Invalid asset in response: missing symbol or name",t);const n="number"==typeof t.decimals?t.decimals:"string"==typeof t.decimals?parseInt(t.decimals,10):void 0;if("number"!=typeof n||isNaN(n))throw new gi(500,"Invalid asset in response: decimals must be a number",t);const o={tokenId:t.compositeKey||`${t.symbol}$Unit$none$none`,symbol:t.symbol,name:t.name,decimals:n,balance:t.quantity||"0"};t.image&&(o.imageUrl=t.image),t.verify&&(o.verified=t.verify),c.push(o)}const l={tokens:c,count:a.count??c.length};return void 0!==a.totalValue&&(l.totalValue=a.totalValue),l}catch(t){throw this.handleError(t,`getUserAssets(${e})`)}}handleError(e,t){if(e instanceof gi)return e;if(e instanceof Error&&e instanceof Error&&"response"in e){const n=e,o=n.response?.status??500,i=n.response?.data?.message??n.response?.data?.Message??n.message,r=n.response?.data?.Data??n.response?.data?.data??void 0;return n.response,new gi(o,`${t}: ${i}`,r)}return new gi(500,`${t}: ${e instanceof Error?e.message:String(e)}`)}}function wi(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("|");if(t.length<4)throw new Error(`Invalid pipe-delimited token format. Expected at least 4 parts separated by |, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("|");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){throw new G(`Invalid pipe-delimited token: "${e}". Expected format: "collection|category|type|additionalKey". Error: ${t instanceof Error?t.message:String(t)}`,"pipeDelimitedToken","INVALID_PIPE_DELIMITED_TOKEN_FORMAT")}}const ki=10;class vi extends Rn{constructor(e,t,n){if(super(!1),this.pricingConcurrency=5,this.tokenConverter=new Eo,this.webSocketService=t,this.dexQuoteService=n,this.getWalletAddress=e.getWalletAddress,this.galaChainBaseUrl=e.galaChainBaseUrl,this.bundlerBaseUrl=e.bundlerBaseUrl,this.gatewayBaseUrl=e.gatewayBaseUrl,this.privateKey=e.privateKey,!(e.gatewayBaseUrl&&e.bundlerBaseUrl&&e.dexBackendBaseUrl&&e.dexBackendHttp))throw new Error("GSwapService requires explicit gatewayBaseUrl, bundlerBaseUrl, dexBackendBaseUrl, and dexBackendHttp configuration. These must be provided by LaunchpadSDK to ensure environment alignment.");try{this.gatewayClient=new fi({baseUrl:e.gatewayBaseUrl,timeout:3e4}),this.dexBackendClient=new yi(e.dexBackendHttp),this.logger.debug("HTTP clients initialized successfully",{gatewayUrl:e.gatewayBaseUrl,dexBackendUrl:e.dexBackendBaseUrl})}catch(e){throw this.logger.error("Failed to initialize HTTP clients",e),new Error("Failed to initialize GSwapService HTTP clients")}}setPricingConcurrency(e){if(e<1)throw new Error("Pricing concurrency must be at least 1");e>100&&this.logger.warn("Pricing concurrency > 100 may cause performance issues",{concurrency:e}),this.pricingConcurrency=e,this.logger.debug("Updated pricing concurrency",{concurrency:this.pricingConcurrency})}async getSwapQuoteExactInput(e){try{if(new u(e.amount).isLessThanOrEqualTo(0))throw new z("Amount must be greater than zero",{amount:e.amount,fromToken:e.fromToken,toToken:e.toToken});if(!this.dexQuoteService)throw new z("DexQuoteService not configured - cannot provide quotes",{fromToken:e.fromToken,toToken:e.toToken});this.logger.debug("Getting swap quote for exact input",{fromToken:e.fromToken,toToken:e.toToken,amount:e.amount});const t=this.tokenConverter.toLaunchpadFormat(e.fromToken),n=this.tokenConverter.toLaunchpadFormat(e.toToken),[o,i]=t<n?[t,n]:[n,t],r=[3e3,500,1e4];let a;for(const s of r)try{const r=await this.dexQuoteService.fetchCompositePoolData({token0:o,token1:i,fee:s,gatewayBaseUrl:this.gatewayBaseUrl}),a=await this.dexQuoteService.calculateDexPoolQuoteExactAmount({compositePoolData:r,fromToken:t,toToken:n,amount:e.amount}),c=new u(a.currentSqrtPrice),l=new u(a.newSqrtPrice),d=c.gt(l)?c.minus(l).dividedBy(c):new u(0),h=new u(a.amount0),g=new u(a.amount1),p=h.isNegative(),m=g.isNegative();this.logger.debug("=== AMOUNT SELECTION RAW DATA ===",{"quoteResult.amount0":a.amount0,"quoteResult.amount1":a.amount1,"amount0BN.isNegative()":p,"amount1BN.isNegative()":m});const f=p?h:g;this.logger.debug("=== AMOUNT SELECTION RESULT ===",{selectedFromAmount0:p,selectedAmount:f.toFixed(),selectedAmountAbs:f.absoluteValue().toFixed()});const y=f.absoluteValue().toFixed();return{fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.amount,estimatedOutput:y,feeTier:s,priceImpact:d.toFixed(),executionPrice:this.calculateExecutionPrice(e.amount,y),currentSqrtPrice:a.currentSqrtPrice,newSqrtPrice:a.newSqrtPrice}}catch(e){a=e,this.logger.debug("DexQuoteService failed for fee tier, trying next",{feeTier:s,error:e instanceof Error?e.message:"Unknown error"})}throw a||new z("No available fee tiers for quote",{feeTiers:r,fromToken:e.fromToken,toToken:e.toToken})}catch(e){this.handleGSwapError("Failed to get swap quote for exact input",z,e)}}async getSwapQuoteExactOutput(e){try{if(new u(e.amount).isLessThanOrEqualTo(0))throw new z("Amount must be greater than zero",{amount:e.amount,fromToken:e.fromToken,toToken:e.toToken});if(!this.dexQuoteService)throw new z("DexQuoteService not configured - cannot provide quotes",{fromToken:e.fromToken,toToken:e.toToken});this.logger.debug("Getting swap quote for exact output",{fromToken:e.fromToken,toToken:e.toToken,amount:e.amount});const t=this.tokenConverter.toLaunchpadFormat(e.fromToken),n=this.tokenConverter.toLaunchpadFormat(e.toToken),[o,i]=t<n?[t,n]:[n,t],r=[3e3,500,1e4];let a;for(const s of r)try{const r=await this.dexQuoteService.fetchCompositePoolData({token0:o,token1:i,fee:s,gatewayBaseUrl:this.gatewayBaseUrl}),a=await this.dexQuoteService.calculateDexPoolQuoteExactAmount({compositePoolData:r,fromToken:t,toToken:n,amount:e.amount}),c=new u(a.currentSqrtPrice),l=new u(a.newSqrtPrice),d=c.gt(l)?c.minus(l).dividedBy(c):new u(0),h=r.pool.token0,g="string"==typeof h?h.split("|")[0]:"object"==typeof h&&null!==h&&"tokenName"in h?h.tokenName:String(h),p=n.split("|")[0]===g?a.amount1:a.amount0;return{fromToken:e.fromToken,toToken:e.toToken,inputAmount:p,estimatedOutput:e.amount,feeTier:s,priceImpact:d.toFixed(),executionPrice:this.calculateExecutionPrice(p,e.amount),currentSqrtPrice:a.currentSqrtPrice,newSqrtPrice:a.newSqrtPrice}}catch(e){a=e,this.logger.debug("DexQuoteService failed for fee tier, trying next",{feeTier:s,error:e instanceof Error?e.message:"Unknown error"})}throw a||new z("No available fee tiers for quote",{feeTiers:r,fromToken:e.fromToken,toToken:e.toToken})}catch(e){this.handleGSwapError("Failed to get swap quote for exact output",z,e)}}async executeSwap(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");this.logger.debug("Executing swap",{fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.inputAmount});const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.fromToken,e.toToken),o=function(e,t=.01){const n=yo(e),o=new u(1).minus(t);return n.multipliedBy(o)}(e.estimatedOutput,e.slippageTolerance||.01),i=this.getWalletAddress();if(!i)throw new Error("Wallet address required for swap execution");let r;try{const t=await this.getSwapQuoteExactInput({fromToken:e.fromToken,toToken:e.toToken,amount:e.inputAmount});r=t.currentSqrtPrice,this.logger.debug("Quote refetch successful - extracted sqrtPrices",{currentSqrtPrice:r,newSqrtPrice:t.newSqrtPrice,feeTier:t.feeTier})}catch(t){this.logger.debug("Could not re-fetch quote for sqrtPrice, using default",{fromToken:e.fromToken,toToken:e.toToken,error:t instanceof Error?t.message:String(t)})}const a={fromToken:t,toToken:n,inputAmount:e.inputAmount,minOutput:o.toFixed(),feeTier:e.feeTier,walletAddress:i,slippageTolerance:e.slippageTolerance||.01};void 0!==r&&(a.currentSqrtPrice=r);const s=await this.sendSwapToBundler(a);this.logger.debug("Swap submitted, monitoring transaction",{transactionId:s,fromToken:e.fromToken,toToken:e.toToken}),await this.ensureWebSocketConnected();const c=await this.webSocketService.waitForTransaction(s);return{transactionId:c.transactionId,status:c.status,fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.inputAmount,outputAmount:e.estimatedOutput,feeTier:e.feeTier,slippageTolerance:e.slippageTolerance||.01,timestamp:new Date(c.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(s)}}}catch(e){const t=e;this.handleGSwapError("Failed to execute swap",H,e,{transactionHash:t?.txHash})}}async getUserAssets(e,t=1,n=20){try{this.logger.debug("Fetching user assets",{walletAddress:e,page:t,limit:n});const o=await this.dexBackendClient.getUserAssets(e,n,(t-1)*n);return(Array.isArray(o)?o:o?.tokens||[]).map(e=>{const t=e.symbol||"UNKNOWN";try{return{tokenId:hi(`${t}|Unit|none|none`),symbol:t,balance:(n=e.quantity||e.balance||0,void 0!==o?yo(n).toFixed(o):yo(n).toFixed()),decimals:e.decimals||18}}catch(n){return this.logger.debug(`Skipping asset with processing error: ${t}`,{error:n instanceof Error?n.message:String(n),assetData:{symbol:t,quantity:e.quantity,balance:e.balance,decimals:e.decimals}}),null}var n,o}).filter(e=>null!==e)}catch(o){this.handleGSwapError("Failed to fetch user assets",X,o,{walletAddress:e,page:t,limit:n})}}async getAllUserAssets(e){try{this.logger.debug("Fetching all user assets (auto-paginated)",{walletAddress:e});const t=await li((t,n)=>this.getUserAssets(e,t,n).then(e=>({items:e,page:t,limit:n,total:e.length,totalPages:1,hasNext:e.length===n,hasPrevious:t>1})),{maxPages:1e4,logger:this.logger,pageSize:20});return this.logger.debug("Fetched all user assets",{walletAddress:e,totalAssets:t.items.length}),t.items}catch(t){this.handleGSwapError("Failed to fetch all user assets",X,t,{walletAddress:e})}}async getPoolInfo(e,t){try{this.logger.debug("Fetching pool info",{tokenA:e,tokenB:t});const{gswapToken0:n,gswapToken1:o}=this.convertTokenPair(e,t),i=[500,3e3,1e4];let r=new u(0),a=0;for(const s of i)try{const e="string"==typeof n?hi(n):n,t="string"==typeof o?hi(o):o,i=await this.gatewayClient.getPoolData({token0:e,token1:t,fee:s});i&&(r=r.plus(new u(i.liquidity||0)),a++)}catch{this.logger.debug("Pool not found for fee tier",{tokenA:e,tokenB:t,feeTier:s})}return{tokenA:e,tokenB:t,liquidity:r.toFixed(),feeTiers:i,swapCount:a}}catch(n){this.logger.warn("Failed to fetch pool info",n);const o=n;return this.logger.debug("Pool error details",{error:new Q(`Failed to fetch pool info: ${o?.message||String(n)}`,n,e,t,this.extractGSwapErrorCode(n))}),{tokenA:e,tokenB:t,liquidity:"0",feeTiers:[500,3e3,1e4],swapCount:0}}}chunkArray(e,t){const n=[];for(let o=0;o<e.length;o+=t)n.push(e.slice(o,o+t));return n}async fetchPositionPrices(e){const t=this.pricingConcurrency;if(0===e.length)return new Map;const n=new Map;for(const t of e){const e=`${t.token0}|${t.token1}|${t.feeTier}`;n.has(e)||n.set(e,{token0:t.token0,token1:t.token1,feeTier:t.feeTier})}const o=Array.from(n.values()),i=this.chunkArray(o,t);this.logger.debug("Fetching pricing for positions",{totalPositions:e.length,uniquePoolsToPrice:n.size,chunks:i.length,concurrency:t});const r=new Map;for(let e=0;e<i.length;e++){const t=i[e];(await Promise.allSettled(t.map(async e=>{const t=await this.getSwapQuoteExactInput({fromToken:e.token0,toToken:e.token1,amount:"1"});return{key:`${e.token0}|${e.token1}|${e.feeTier}`,data:{token0:e.token0,token1:e.token1,feeTier:e.feeTier,currentPrice:t.executionPrice,executionPrice:t.executionPrice,priceImpact:t.priceImpact,estimatedOutput:t.estimatedOutput,pricedAt:new Date}}}))).forEach(e=>{"fulfilled"===e.status?r.set(e.value.key,e.value.data):this.logger.warn("Failed to fetch price for pool",{error:e.reason})})}return r}normalizePositionResponse(e,t){const n=e=>{if(!e)return"";if("string"==typeof e)return e;if("object"==typeof e){if(e.type&&"none"!==e.type)return e.type;if(e.collection)return e.collection;if(e.symbol)return e.symbol;if(e.tokenSymbol)return e.tokenSymbol;if(e.name)return e.name}return""};return{positionId:e.positionId||e.id||"",ownerAddress:t||e.ownerAddress||e.owner||"",token0:e.token0Symbol||n(e.token0)||n(e.tokenA)||e.tokenSymbol0||"",token1:e.token1Symbol||n(e.token1)||n(e.tokenB)||e.tokenSymbol1||"",feeTier:e.feeTier||e.fee||e.feeAmount||0,tickLower:e.tickLower||e.lowerTick||0,tickUpper:e.tickUpper||e.upperTick||0,liquidity:String(e.liquidity||e.liquidityAmount||"0"),amount0:String(e.amount0||e.amountA||"0"),amount1:String(e.amount1||e.amountB||"0"),feeAmount0:String(e.feeAmount0||e.feesA||"0"),feeAmount1:String(e.feeAmount1||e.feesB||"0"),...e.createdAt&&{createdAt:new Date(e.createdAt)},...e.updatedAt&&{updatedAt:new Date(e.updatedAt)}}}parseTokenFlexible(e){try{return hi(e)}catch(t){if(t instanceof Error&&t.message?.includes("Plain token string"))return this.logger.debug("Using default TokenClassKey for simple token symbol",{token:e}),{collection:"Token",category:"Unit",type:e,additionalKey:"none"};throw t}}async getUserLiquidityPositions(t,n=10,o,i){try{this.logger.debug("Fetching user liquidity positions",{ownerAddress:t,limit:n,bookmark:o});const r=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetUserPositions`,a={user:t,limit:n,bookmark:o||""};this.logger.debug("Sending position query request",{endpoint:r,payload:a});const s=await e.post(r,a,{headers:{"Content-Type":"application/json",Accept:"application/json"}});if(200!==s.status||1!==s.data?.Status)return this.logger.warn("Unexpected API response status",{httpStatus:s.status,apiStatus:s.data?.Status}),{items:[]};const c=s.data.Data||{},l=c.positions||[],u=c.nextBookMark,d=l.filter(e=>null!=e&&"object"==typeof e&&("positionId"in e||"id"in e)).map(e=>this.normalizePositionResponse(e,t));let h;this.logger.debug("Retrieved liquidity positions",{count:d.length,hasNextBookmark:!!u,nextBookmark:u}),i?.withPrices&&d.length>0&&(h=await this.fetchPositionPrices(d));const g={items:d};return void 0!==u&&(g.nextBookmark=u),void 0!==h&&(g.prices=h),g}catch(e){if(e&&"object"==typeof e&&"response"in e){const n=e;this.logger.error("Position query failed with HTTP error",{status:n.response?.status,statusText:n.response?.statusText,data:n.response?.data,endpoint:this.galaChainBaseUrl,ownerAddress:t})}this.handleGSwapError("Failed to fetch user liquidity positions",Z,e)}}async getAllSwapUserLiquidityPositions(e,t){try{this.logger.debug("Fetching all user liquidity positions (auto-paginated)",{ownerAddress:e});const n=async t=>{const n=await this.getUserLiquidityPositions(e,ki,t,void 0);return{items:n.items,nextBookmark:n.nextBookmark}},o=await async function(e,t={}){const{maxPages:n=1e4,logger:o,pageSize:i=20}=t,r=[];let a,s=0;for(;s<n;){o&&o.debug(`Auto-pagination (bookmark): fetching page ${s+1} with pageSize ${i}`,{bookmark:a});const t=await e(a,i);let n,c,l;if(Array.isArray(t))n=t,c=void 0,l=!1;else{if(!t||"object"!=typeof t||!("items"in t)){o&&o.warn("Auto-pagination (bookmark): received invalid result structure, stopping");break}n=t.items,c=t.nextBookmark,l=!0}if(!Array.isArray(n)){o&&o.warn("Auto-pagination (bookmark): received invalid items array, stopping");break}if(0===n.length){o&&o.debug(`Auto-pagination (bookmark): no items returned on page ${s+1}, exiting loop`);break}r.push(...n),s++,o&&o.debug(`Auto-pagination (bookmark): page ${s} returned ${n.length} items`,{hasNextBookmark:!!c,format:l?"BookmarkPaginationResult":"legacy-array"});const u=n.length<i;if(l&&(""===c||void 0===c)){o&&o.debug("Auto-pagination (bookmark): no nextBookmark returned, reached end of results",{nextBookmark:""===c?"(empty string)":"(undefined)"});break}if(u){o&&o.debug("Auto-pagination (bookmark): received fewer items than limit, reached last page",{received:n.length,pageSize:i,format:l?"BookmarkPaginationResult":"legacy-array"});break}a=c}return s>=n&&o&&o.warn(`Auto-pagination (bookmark): exceeded maxPages limit of ${n}, stopping iteration`),o&&o.debug(`Auto-pagination (bookmark): completed with total items: ${r.length}`,{pageCount:s}),{items:r,total:r.length}}(n,{maxPages:1e4,logger:this.logger,pageSize:ki}),i=o.items;if(this.logger.debug("Fetched all user liquidity positions",{ownerAddress:e,totalPositions:i.length}),t?.withPrices&&i.length>0){return{items:i,prices:await this.fetchPositionPrices(i)}}return i}catch(t){this.handleGSwapError("Failed to fetch all user liquidity positions",Z,t,{ownerAddress:e})}}async getLiquidityPosition(e,t){try{this.logger.debug("Fetching liquidity position",{ownerAddress:e,position:t});const{gswapToken0:n,gswapToken1:o}=this.convertTokenPair(t.token0,t.token1),i=wi(n),r=wi(o),a=(await this.gatewayClient.getPositions({owner:e,tokenA:i,tokenB:r,feeTier:t.fee})).positions.find(e=>e.tickLower===t.tickLower&&e.tickUpper===t.tickUpper);if(!a||"object"!=typeof a||!("positionId"in a)&&!("id"in a))throw new Error("Invalid position data returned from API");const s=this.normalizePositionResponse(a,e);return this.logger.debug("Retrieved liquidity position",{positionId:s.positionId}),s}catch(e){this.handleGSwapError("Failed to fetch liquidity position",Z,e)}}async getLiquidityPositionById(e,t,n,o,i,r,a){try{let s;this.logger.debug("Fetching liquidity position by ID",{ownerAddress:e,positionId:t,hasToken0:!!n,hasToken1:!!o,hasFee:!!i,hasTickLower:void 0!==r,hasTickUpper:void 0!==a});let c=null;const l=5,u=2e3;for(let d=1;d<=l;d++)try{if(n&&o&&void 0!==i&&void 0!==r&&void 0!==a)try{this.logger.debug("Attempting compound key lookup",{ownerAddress:e,token0:n,token1:o,feeTier:i,tickLower:r,tickUpper:a});if(s=(await this.gatewayClient.getPositionById(e,n,o,i,r,a,t)).Data,s&&"object"==typeof s&&("positionId"in s||"id"in s)){this.logger.debug("Successfully fetched position via compound key",{attempt:d,positionId:t});break}throw new Error("Invalid position data from compound key lookup")}catch(e){this.logger.debug("Compound key lookup failed, trying fallback",{attempt:d,error:e instanceof Error?e.message:e})}try{if(s=(await this.gatewayClient.getPositionById(t)).Data,s&&"object"==typeof s&&("positionId"in s||"id"in s)){this.logger.debug("Successfully fetched position on attempt",{attempt:d,positionId:t});break}throw new Error("Invalid position data from direct lookup")}catch(n){this.logger.debug("Direct position lookup failed, trying fallback via GetUserPositions",{attempt:d,positionId:t,error:n instanceof Error?n.message:n});const o=await this.getUserLiquidityPositions(e,10);if(Array.isArray(o.items)&&o.items.length>0){const e=o.items.find(e=>(e.positionId||e.id||"").toLowerCase()===t.toLowerCase());if(e){s=e,this.logger.debug("Found position via fallback (GetUserPositions)",{attempt:d,positionId:t});break}}if(c=new Error("Position not found in owner positions"),d<l){this.logger.warn("Fallback query did not find position, retrying",{attempt:d,positionId:t,ownerAddress:e,foundCount:Array.isArray(o?.items)?o.items.length:0}),await new Promise(e=>setTimeout(e,u));continue}}}catch(e){if(d<l){this.logger.warn("Error fetching position, retrying",{attempt:d,positionId:t,error:e instanceof Error?e.message:e}),await new Promise(e=>setTimeout(e,u));continue}c=e instanceof Error?e:new Error(String(e))}if(!s||"object"!=typeof s||!("positionId"in s)&&!("id"in s))throw this.logger.error("Invalid position data returned from API after retries",{positionId:t,resultType:typeof s,resultKeys:s?Object.keys(s):"null",resultValue:JSON.stringify(s),lastError:c?.message}),c||new Error("Invalid position data returned from API");const d=this.normalizePositionResponse(s,e);return this.logger.debug("Retrieved liquidity position by ID",{positionId:d.positionId}),d}catch(e){this.handleGSwapError("Failed to fetch liquidity position by ID",Z,e)}}async fetchSwapPositionDirect(e){try{this.logger.debug("Fetching swap position via direct compound key",{token0:e.token0,token1:e.token1,fee:e.fee,owner:e.owner});const t="string"==typeof e.token0?this.parseTokenFlexible(e.token0):e.token0,n={token0:t,token1:"string"==typeof e.token1?this.parseTokenFlexible(e.token1):e.token1,fee:e.fee,tickLower:e.tickLower,tickUpper:e.tickUpper,owner:e.owner},o=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetPositions`;this.logger.debug("Calling position endpoint via compound key",{url:o});const i=await fetch(o,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify(n)});if(!i.ok){if(404===i.status)throw new Error("Position not found (HTTP 404): No position exists for this compound key");if(400===i.status)throw new Error("Invalid parameters (HTTP 400): Check token format, fee (500|3000|10000), and tick ranges");throw new Error(`HTTP ${i.status}: ${i.statusText}`)}const r=await i.json();if(1!==r.Status||!r.Data)throw new Error(`Position not found: ${r.Message||"API returned no position data"}`);const a=this.normalizePositionResponse(r.Data,e.owner);return this.logger.debug("Retrieved swap position via compound key",{positionId:a.positionId,token0:a.token0,token1:a.token1}),a}catch(e){this.handleGSwapError("Failed to fetch swap position via compound key",Z,e)}}async estimateRemoveLiquidity(e){try{this.logger.debug("Estimating liquidity removal",{token0:e.token0,token1:e.token1});const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.token0,e.token1),o=wi(t),i=wi(n),r=await this.gatewayClient.getRemoveLiquidityEstimation({tokenA:o,tokenB:i,fee:e.fee,liquidity:e.liquidity,tickLower:e.tickLower,tickUpper:e.tickUpper});return this.logger.debug("Estimated removal",{result:r}),r}catch(e){this.handleGSwapError("Failed to estimate liquidity removal",Z,e)}}async addLiquidityByPrice(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");this.logger.debug("Adding liquidity by price",{token0:e.token0,token1:e.token1,priceRange:`${e.minPrice}-${e.maxPrice}`});const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.token0,e.token1);await this.ensureWebSocketConnected(),this.logger.debug("Converting price range to ticks",{token0:e.token0,token1:e.token1,minPrice:e.minPrice,maxPrice:e.maxPrice,fee:e.fee});const o=hi(t),i=hi(n),r=(await this.gatewayClient.getPoolData({token0:o,token1:i,fee:e.fee})).tickSpacing;this.logger.debug("Retrieved tick spacing from pool",{tickSpacing:r,fee:e.fee});const a=new u(e.minPrice),s=new u(e.maxPrice),c=Math.floor(ko(a)),l=Math.ceil(ko(s)),d=Math.floor(c/r)*r,h=Math.ceil(l/r)*r;this.logger.debug("Converted price range to ticks",{minPrice:e.minPrice,maxPrice:e.maxPrice,tickLower:d,tickUpper:h,tickSpacing:r});const g=this.getWalletAddress();if(!g)throw new Error("GSwapService: No wallet address available - cannot create position");const p="string"==typeof e.token0?hi(e.token0):e.token0,m="string"==typeof e.token1?hi(e.token1):e.token1;this.logger.debug("Sending AddLiquidity by price to bundler",{fee:e.fee,tickRange:`${d}-${h}`,walletAddress:g});const f=await this.sendAddLiquidityToBundler({token0:p,token1:m,fee:e.fee,tickLower:d,tickUpper:h,amount0Desired:e.amount0Desired,amount1Desired:e.amount1Desired,amount0Min:e.amount0Min||"0",amount1Min:e.amount1Min||"0",owner:g}),y={transactionId:f};if(y.positionId&&f){this.logger.debug("Position ID returned directly from backend",{transactionId:f,positionId:y.positionId}),await this.ensureWebSocketConnected();const e=await this.webSocketService.waitForTransaction(f);this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:f,status:e.status});const t=this.getWalletAddress();if(t&&y.positionId)try{const n=await this.getLiquidityPositionById(t,y.positionId),{createdAt:o,updatedAt:i,...r}=n,a=o instanceof Date?o.getTime():"number"==typeof o?o:void 0,s={...y,...r,positionId:y.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}};return void 0!==a&&(s.createdAt=a),s}catch(t){return{...y,positionId:y.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}}}}if(f){this.logger.debug("Monitoring liquidity transaction (discovery mode)",{transactionId:f}),await this.ensureWebSocketConnected();const t=await this.webSocketService.waitForTransaction(f);let n;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:f,status:t.status});let o=null;await new Promise(e=>setTimeout(e,2e3));try{const t=this.getWalletAddress();if(!t)throw new Error("No wallet address available");const i=(await this.getUserLiquidityPositions(t,10)).items;if(i&&i.length>0){const t=e.token0.split("|")[0].toUpperCase(),r=e.token1.split("|")[0].toUpperCase(),a=[];for(const n of i){if(!n||!n.positionId)continue;const o=n.token0?.toUpperCase(),i=n.token1?.toUpperCase();if(!o||!i)continue;const s=o===t&&i===r||o===r&&i===t,c=n.feeTier===e.fee;s&&c&&a.push(n)}a.length>0&&(o=a[a.length-1],n=o.positionId,this.logger.debug("Found newly created position",{positionId:n,expectedTokens:`${e.token0}/${e.token1}`,expectedFee:e.fee,positionCount:i.length}))}}catch(e){this.logger.debug("Error waiting for position indexing",{error:e instanceof Error?e.message:String(e)})}let i=o;if(n)try{i=await this.getLiquidityPositionById(g,n)}catch(e){}return{...y,...i,positionId:n,status:t.status,transactionId:t.transactionId,timestamp:new Date(t.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}}}return this.logger.warn("No transaction ID in liquidity result, cannot confirm position creation"),y}catch(e){Error,this.handleGSwapError("Failed to add liquidity by price",Z,e)}}async addSwapLiquidityByTicks(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");const t=this.getWalletAddress();if(!t)throw new Error("GSwapService: No wallet address available - cannot create position");this.logger.debug("Adding liquidity by ticks with direct bundler",{token0:e.token0,token1:e.token1,fee:e.fee,walletAddress:t,tickRange:`${e.tickLower}-${e.tickUpper}`});const n="string"==typeof e.token0?hi(e.token0):e.token0,o="string"==typeof e.token1?hi(e.token1):e.token1;await this.ensureWebSocketConnected();const i=await this.sendAddLiquidityToBundler({token0:n,token1:o,fee:e.fee,tickLower:e.tickLower,tickUpper:e.tickUpper,amount0Desired:e.amount0Desired,amount1Desired:e.amount1Desired,amount0Min:e.amount0Min||"0",amount1Min:e.amount1Min||"0",owner:t});this.logger.info("Liquidity transaction submitted to bundler",{transactionId:i});const r=this.webSocketService.waitForTransaction(i),a={transactionId:i};if(a.positionId&&i){this.logger.info("Position ID returned directly from backend",{transactionId:i,positionId:a.positionId});const e=await r;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:i,status:e.status});const t=this.getWalletAddress();if(t&&a.positionId)try{this.logger.debug("Fetching full position details",{positionId:a.positionId});const n=await this.getLiquidityPositionById(t,a.positionId);this.logger.debug("Fetched full position data",{positionId:n.positionId,liquidity:n.liquidity,amount0:n.amount0,amount1:n.amount1});const{createdAt:o,updatedAt:r,...s}=n,c=o instanceof Date?o.getTime():"number"==typeof o?o:void 0,l={...a,...s,positionId:a.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}};return void 0!==c&&(l.createdAt=c),l}catch(t){return this.logger.warn("Could not fetch full position details",{positionId:a.positionId,error:t instanceof Error?t.message:String(t)}),{...a,positionId:a.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}}}}if(i){this.logger.debug("Monitoring liquidity transaction (discovery mode)",{transactionId:i});const n=await r;let o;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:i,status:n.status});let s=null;const c="string"==typeof e.token0?e.token0:e.token0?.type??"unknown",l="string"==typeof e.token1?e.token1:e.token1?.type??"unknown";this.logger.debug("Waiting for position indexing after WebSocket confirmation"),this.logger.debug("Looking for matching position",{token0:c,token1:l,fee:e.fee});try{const t=this.getWalletAddress();if(!t)throw new Error("No wallet address available");this.logger.debug("Fetching positions from API",{walletAddress:t,pageSize:ki});const n=3,i=5e3,r=3e3;let a=[];for(let c=1;c<=n;c++){const l=1===c?i:r;this.logger.debug("Position discovery attempt",{attempt:c,maxAttempts:n,delayMs:l}),await new Promise(e=>setTimeout(e,l)),this.logger.debug("Querying positions from API",{attempt:c,pageSize:ki});if(a=(await this.getUserLiquidityPositions(t,ki)).items,this.logger.debug("Got positions from API",{count:a?.length||0}),a&&a.length>0){const t=("string"==typeof e.token0?e.token0.split("|")[0]:e.token0.collection).toUpperCase(),n=("string"==typeof e.token1?e.token1.split("|")[0]:e.token1.collection).toUpperCase(),i=[];for(const o of a){if(!o||!o.positionId)continue;const r=o.token0?.toUpperCase(),a=o.token1?.toUpperCase();if(!r||!a){this.logger.debug("Skipping position with empty tokens",{positionId:o.positionId});continue}const s=r===t&&a===n||r===n&&a===t,c=o.feeTier===e.fee;this.logger.debug("Checking position against target",{positionId:o.positionId,tokens:`${r}/${a}`,tokensMatch:s,fee:o.feeTier,feeMatches:c}),s&&c&&i.push(o)}if(i.length>0){s=i[i.length-1],o=s.positionId,this.logger.info("Found newly created position",{positionId:o,liquidity:s.liquidity,amount0:s.amount0,amount1:s.amount1,fee:s.feeTier}),this.logger.debug("Found newly created position",{positionId:o,expectedTokens:`${e.token0}/${e.token1}`,expectedFee:e.fee,positionCount:a.length});break}this.logger.debug("No matching position found in this attempt")}else this.logger.debug("No positions returned from API in this attempt")}}catch(e){this.logger.error("Error fetching positions during discovery",{error:e instanceof Error?e.message:String(e)}),this.logger.debug("Error waiting for position indexing",{error:e instanceof Error?e.message:String(e)})}this.logger.debug("Position discovery complete",{positionId:o||"not found",found:!!o}),this.logger.debug("Matched position data",{positionId:s?.positionId,liquidity:s?.liquidity,amount0:s?.amount0,amount1:s?.amount1,feeAmount0:s?.feeAmount0,feeAmount1:s?.feeAmount1,token0:s?.token0,token1:s?.token1,feeTier:s?.feeTier});let u=s;if(o)try{this.logger.debug("Fetching full position details",{positionId:o}),u=await this.getLiquidityPositionById(t,o),this.logger.debug("Fetched full position data",{positionId:u.positionId,liquidity:u.liquidity,amount0:u.amount0,amount1:u.amount1,feeAmount0:u.feeAmount0,feeAmount1:u.feeAmount1})}catch(e){this.logger.warn("Could not fetch full position details, using discovered data",{positionId:o,error:e instanceof Error?e.message:String(e)})}return{...a,...u,positionId:o,status:n.status,transactionId:n.transactionId,timestamp:new Date(n.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}}}return this.logger.warn("No transaction ID in liquidity result, cannot confirm position creation"),a}catch(e){this.handleGSwapError("Failed to add liquidity by ticks",Z,e)}}async monitorBundlerTransaction(e,t,n="bundler"){let o;try{const i=await t;o={status:i.status,transactionId:i.transactionId||e,timestamp:i.timestamp||Date.now(),data:i.data},this.logger.debug(`${n} transaction confirmed on-chain`,{transactionId:e,status:o.status})}catch(t){return this.logger.warn(`WebSocket monitoring timeout for ${n} transaction, returning result with transaction ID`,{transactionId:e,error:t instanceof Error?t.message:String(t)}),{transactionId:e,status:"SUBMITTED",timestamp:new Date,wait:async t=>{try{await this.webSocketService.waitForTransaction(e)}catch{this.logger.debug("Explicit wait also timed out",{transactionId:e})}}}}return{transactionId:o.transactionId,status:o.status,timestamp:new Date(o.timestamp),wait:async t=>{await this.webSocketService.waitForTransaction(e)}}}async removeLiquidity(e){try{if(!this.privateKey)throw new Error("Private key not available for bundler-direct operations");this.logger.debug("Removing liquidity via bundler",{token0:e.token0,token1:e.token1,liquidity:e.liquidity});try{const t=parseFloat(e.liquidity);if(isNaN(t))throw new Error(`Invalid liquidity value: "${e.liquidity}". Must be a valid number. Position ID: ${e.positionId||"unknown"}`);if(0===t)throw new Error(`Cannot remove zero liquidity from position. This would waste gas fees without any effect. Position ID: ${e.positionId||"unknown"}`)}catch(e){if(e instanceof Error&&e.message.includes("Cannot remove zero liquidity"))throw e;if(e instanceof Error&&e.message.includes("Invalid liquidity value"))throw e;throw e}const t="string"==typeof e.token0?hi(e.token0):e.token0,n="string"==typeof e.token1?hi(e.token1):e.token1;await this.ensureWebSocketConnected();const o=await this.sendRemoveLiquidityToBundler(e.tickLower,e.tickUpper,e.liquidity,t,n,e.fee,e.amount0Min||"0",e.amount1Min||"0",e.positionId||"");this.logger.debug("Liquidity removal submitted to bundler",{transactionId:o});const i=this.webSocketService.waitForTransaction(o);return this.monitorBundlerTransaction(o,i,"liquidity removal")}catch(e){this.handleGSwapError("Failed to remove liquidity",Z,e)}}async collectPositionFees(e){try{if(!this.privateKey)throw new Error("Private key not available for bundler-direct operations");if(e.ownerAddress&&e.positionId&&!e.token0){this.logger.debug("Fetching position data before collecting fees",{ownerAddress:e.ownerAddress,positionId:e.positionId});const t=await this.getLiquidityPositionById(e.ownerAddress,e.positionId);if(!t)throw new Error(`Position ${e.positionId} not found for owner ${e.ownerAddress}`);if(!t.token0||!t.token1)throw new Error("Position missing token information");return this.collectPositionFees({token0:t.token0,token1:t.token1,fee:t.feeTier,tickLower:t.tickLower,tickUpper:t.tickUpper,amount0Requested:e.amount0Max||e.amount0Requested||"0",amount1Requested:e.amount1Max||e.amount1Requested||"0",positionId:e.positionId})}if(!e.token0||!e.token1||void 0===e.fee||void 0===e.tickLower||void 0===e.tickUpper)throw new Error("Missing required parameters: token0, token1, fee, tickLower, tickUpper are required when not using ownerAddress pattern");this.logger.debug("Collecting position fees via bundler",{token0:"string"==typeof e.token0?e.token0:e.token0?.type??"unknown",token1:"string"==typeof e.token1?e.token1:e.token1?.type??"unknown",tickLower:e.tickLower,tickUpper:e.tickUpper});const t="string"==typeof e.token0?hi(e.token0):e.token0,n="string"==typeof e.token1?hi(e.token1):e.token1;await this.ensureWebSocketConnected();const o=await this.sendCollectPositionFeesToBundler(t,n,e.fee,e.amount0Requested||"0",e.amount1Requested||"0",e.tickLower,e.tickUpper,e.positionId||"");this.logger.debug("Fee collection submitted to bundler",{transactionId:o});const i=this.webSocketService.waitForTransaction(o);return this.monitorBundlerTransaction(o,i,"fee collection")}catch(e){this.handleGSwapError("Failed to collect position fees",Z,e)}}async getPoolData(e,t,n){try{this.logger.debug("Getting pool data",{tokenA:e,tokenB:t,feeTier:n});const{gswapToken0:o,gswapToken1:i}=this.convertTokenPair(e,t),r=hi(o),a=hi(i),s=await this.gatewayClient.getPoolData({token0:r,token1:a,fee:n}),c=this.calculatePriceFromSqrtPriceX96(new u(s.sqrtPrice));return{tokenA:e,tokenB:t,feeTier:n,liquidity:s.liquidity.toString(),sqrtPriceX96:s.sqrtPrice.toString(),tick:s.tick,feeGrowthGlobal0X128:s.feeGrowthGlobal0.toString(),feeGrowthGlobal1X128:s.feeGrowthGlobal1.toString(),currentPrice:c.toFixed()}}catch(e){this.handleGSwapError("Failed to get pool data",Q,e)}}async calculateDexPoolSpotPrice(e,t,n){try{this.logger.debug("Calculating spot price",{tokenA:e,tokenB:t,feeTier:n});const o=await this.getPoolData(e,t,n),i=yo(o.currentPrice);return{tokenA:e,tokenB:t,feeTier:n,price:i.toFixed(),invertedPrice:vo(i,!0),tick:o.tick,liquidity:o.liquidity}}catch(e){this.handleGSwapError("Failed to calculate spot price",Q,e)}}async calculateOptimalPositionSize(e,t,n,o,i,r,a){try{this.logger.debug("Calculating optimal position size",{tokenA:e,tokenB:t,desiredAmount0:o,desiredAmount1:i});const s=(await this.getPoolData(e,t,n)).tick,c=d(r),l=d(s),p=d(a),m=h(new u(o),new u(i),c,l,p),f=g(m,l,c,p),y=f[0],w=f[1],k=new u(o),v=new u(i);return{amount0:y.toFixed(),amount1:w.toFixed(),liquidity:m.toFixed(),ratio:y.dividedBy(w).toFixed(),utilizationPercent:{amount0:y.dividedBy(k).multipliedBy(100).toFixed(2),amount1:w.dividedBy(v).multipliedBy(100).toFixed(2)}}}catch(e){this.handleGSwapError("Failed to calculate optimal position size",Z,e)}}async validatePositionParameters(e,t,n,o,i,r,a){const s=[],c=[];try{this.logger.debug("Validating position parameters",{tokenA:e,tokenB:t,tickLower:o,tickUpper:i});const l=[500,3e3,1e4];l.includes(n)||s.push(`Invalid fee tier: ${n}. Must be one of: ${l.join(", ")}`);const d=this.getTickSpacing(n);let h;o%d!==0&&s.push(`tickLower must be multiple of ${d}`),i%d!==0&&s.push(`tickUpper must be multiple of ${d}`),o>=i&&s.push(`tickLower (${o}) must be less than tickUpper (${i})`);try{h=await this.getPoolData(e,t,n)}catch{return s.push(`Pool not found for ${e}/${t} at fee tier ${n}`),{valid:!1,errors:s,warnings:c,gasEstimate:0}}const g=new u(r),p=new u(a);if(g.isNaN()||p.isNaN())s.push("Amounts must be valid numbers");else try{To(g,p)}catch(e){s.push(`Liquidity amounts must be non-negative: ${e.message}`)}const m=h.tick;(m<o||m>i)&&c.push("Position is out of current price range - will not earn fees until price moves into range");new u(h.liquidity||"0").lt("1000000")&&c.push("Low pool liquidity - consider higher slippage tolerance");const f=0===s.length?35e4:0;return{valid:0===s.length,errors:s,warnings:c,gasEstimate:f,tickSpacing:d,currentTick:m,poolLiquidity:h.liquidity}}catch(e){const t=e;return s.includes(t?.message||"")||s.push(`Validation failed: ${t?.message||String(e)}`),{valid:!1,errors:s,warnings:c,gasEstimate:0}}}async calculateTicksForPrice(e,t,n,o,i){try{this.logger.debug("Calculating ticks for price range",{tokenA:e,tokenB:t,minPrice:n,maxPrice:o});const r=this.getTickSpacing(i),a=new u(n),s=new u(o);if(a.gte(s))throw new Error("minPrice must be less than maxPrice");const c=Math.floor(ko(a)),l=Math.ceil(ko(s)),d=Math.floor(c/r)*r,h=Math.ceil(l/r)*r,g=Math.pow(1.0001,d),p=Math.pow(1.0001,h),m=new u(g),f=new u(p);return{tokenA:e,tokenB:t,feeTier:i,tickLower:d,tickUpper:h,tickSpacing:r,requestedMinPrice:n,requestedMaxPrice:o,actualMinPrice:m.toFixed(8),actualMaxPrice:f.toFixed(8),priceDeviation:{minPriceDeviation:m.minus(a).dividedBy(a).multipliedBy(100).toFixed(4),maxPriceDeviation:f.minus(s).dividedBy(s).multipliedBy(100).toFixed(4)}}}catch(e){this.handleGSwapError("Failed to calculate ticks for price",Z,e)}}async calculatePriceForTicks(e,t,n,o){try{this.logger.debug("Calculating price for ticks",{tokenA:e,tokenB:t,tickLower:n,tickUpper:o});const i=Math.pow(1.0001,n),r=Math.pow(1.0001,o);let a;try{a=(await this.getPoolData(e,t,3e3)).currentPrice}catch{}const s=new u(i),c=new u(r),l={tokenA:e,tokenB:t,tickLower:n,tickUpper:o,minPrice:s.toFixed(8),maxPrice:c.toFixed(8),priceRange:`${s.toFixed(4)} - ${c.toFixed(4)}`,tickSpread:o-n};return void 0!==a&&(l.currentPrice=a),l}catch(e){this.handleGSwapError("Failed to calculate price for ticks",Z,e)}}calculateExecutionPrice(e,t){try{const n=new u(e),o=new u(t);return n.isZero()?"0":o.dividedBy(n).toFixed()}catch{return"0"}}getTickSpacing(e){switch(e){case 500:return 10;case 3e3:return 60;case 1e4:return 200;default:throw new Error(`Invalid fee tier: ${e}`)}}calculatePriceFromSqrtPriceX96(e){try{const t=new u(2).pow(96);return e.dividedBy(t).pow(2)}catch{return new u(0)}}calculatePriceFromSqrtPriceDecimal(e){try{return e.pow(2)}catch{return new u(0)}}async getPoolSlot0(t,n,o){try{this.logger.debug("Fetching pool slot0 data",{token0:t,token1:n,fee:o});const i="string"==typeof t?hi(t):t,r="string"==typeof n?hi(n):n,a=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetSlot0`,s=await e.post(a,{token0:i,token1:r,fee:o},{headers:{"Content-Type":"application/json",Accept:"application/json"}});if(200!==s.status||1!==s.data?.Status)throw this.logger.warn("Unexpected GetSlot0 API response",{httpStatus:s.status,apiStatus:s.data?.Status}),new Q("GetSlot0 API returned unexpected status",{httpStatus:s.status,apiStatus:s.data?.Status});const c=s.data.Data||{},l={sqrtPrice:c.sqrtPrice||"0",tick:c.tick||0,liquidity:c.liquidity||"0",grossPoolLiquidity:c.grossPoolLiquidity||"0"};return this.logger.debug("Retrieved pool slot0 data",{sqrtPrice:l.sqrtPrice,tick:l.tick,liquidity:l.liquidity}),l}catch(e){this.handleGSwapError("Failed to fetch pool slot0 data",Q,e,{token0:t,token1:n,fee:o})}}async getPositionCurrentPrice(e){try{this.logger.debug("Fetching position current price",{token0:e.token0,token1:e.token1,feeTier:e.feeTier});const t=await this.getPoolSlot0(e.token0,e.token1,e.feeTier),n=new u(t.sqrtPrice),o={price:this.calculatePriceFromSqrtPriceDecimal(n).toFixed(18),sqrtPrice:t.sqrtPrice,tick:t.tick,liquidity:t.liquidity};return this.logger.debug("Calculated position current price",{price:o.price,tick:o.tick}),o}catch(t){this.handleGSwapError("Failed to fetch position current price",Q,t,{token0:e.token0,token1:e.token1})}}calculateLiquidityFromAmount0(e,t,n){try{const o=wo(t),i=wo(n);return p(e,o,i)}catch{return new u(0)}}calculateLiquidityFromAmount1(e,t,n){try{const o=wo(t),i=wo(n);return m(e,o,i)}catch{return new u(0)}}calculateAmount0FromLiquidity(e,t,n){try{const o=wo(t),i=wo(n);return f(o,i,e)}catch{return new u(0)}}calculateAmount1FromLiquidity(e,t,n){try{const o=wo(t),i=wo(n);return y(o,i,e)}catch{return new u(0)}}convertTokenPair(e,t){return{gswapToken0:this.tokenConverter.toLaunchpadFormat(e),gswapToken1:this.tokenConverter.toLaunchpadFormat(t)}}async sendAddLiquidityToBundler(n){if(!this.privateKey)throw new Error("GSwapService: AddLiquidity requires wallet (full-access mode)");if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");try{this.logger.debug("Sending AddLiquidity to bundler",{token0:n.token0?.type??"unknown",token1:n.token1?.type??"unknown",fee:n.fee,tickRange:`${n.tickLower}-${n.tickUpper}`});const o=`galaswap - operation - ${N()}-${Date.now()}-${n.owner}`,i={token0:n.token0,token1:n.token1,fee:n.fee,owner:n.owner,tickLower:n.tickLower,tickUpper:n.tickUpper,amount0Desired:n.amount0Desired,amount1Desired:n.amount1Desired,amount0Min:n.amount0Min,amount1Min:n.amount1Min,positionId:"",uniqueKey:o},r=new t.Wallet(this.privateKey),a={AddLiquidity:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"owner",type:"string"},{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"amount0Desired",type:"string"},{name:"amount1Desired",type:"string"},{name:"amount0Min",type:"string"},{name:"amount1Min",type:"string"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},s={name:"ethereum",chainId:1},c=this.calculatePersonalSignPrefix(i),l={...i,prefix:c},u=await r.signTypedData(s,a,l),d={...l,signature:u,types:a,domain:s};this.logger.debug("AddLiquidity DTO signed with manual types",{signature:d.signature?.substring(0,20)+"...",prefix:d.prefix,tickLower:i.tickLower,tickUpper:i.tickUpper});const h=this.buildLiquidityStringsInstructions(n.token0,n.token1,n.fee,n.owner),g=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),p=await g.post("/bundle",{method:"AddLiquidity",signedDto:d,stringsInstructions:h}),m=p.data?.data||p.data?.transactionId||p.data?.id;if(!m)throw this.logger.error("Bundler response structure",{status:p.status,data:p.data,dataType:typeof p.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(p.data)}`);return this.logger.debug("AddLiquidity transaction sent to bundler",{transactionId:m}),m}catch(e){throw this.logger.error("Failed to send AddLiquidity to bundler",e),e}}async sendRemoveLiquidityToBundler(n,o,i,r,a,s,c,l,u){try{if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const d=new t.Wallet(this.privateKey),h=await d.getAddress(),g=`galaswap - operation - ${N()}-${Date.now()}-${h}`,p={tickLower:n,tickUpper:o,amount:i,token0:r,token1:a,fee:s,amount0Min:c,amount1Min:l,positionId:u,uniqueKey:g},m={RemoveLiquidity:[{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"amount",type:"string"},{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount0Min",type:"string"},{name:"amount1Min",type:"string"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},f={name:"ethereum",chainId:1},y=this.calculatePersonalSignPrefix(p),w={...p,prefix:y},k=await d.signTypedData(f,m,w),v={...w,signature:k,types:m,domain:f},b=this.buildLiquidityStringsInstructions(r,a,s,h);this.logger.debug("Submitting RemoveLiquidity to bundler",{tickLower:n,tickUpper:o,amount:i,fee:s,positionId:u,transactionId:g});const T=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),S=await T.post("/bundle",{method:"RemoveLiquidity",signedDto:v,stringsInstructions:b}),A=S.data?.data||S.data?.transactionId||S.data?.id;if(!A)throw this.logger.error("Bundler response structure",{status:S.status,data:S.data,dataType:typeof S.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(S.data)}`);return this.logger.info("RemoveLiquidity transaction sent to bundler",{transactionId:A}),A}catch(e){throw this.logger.error("Failed to send RemoveLiquidity to bundler",e),e}}async sendCollectPositionFeesToBundler(n,o,i,r,a,s,c,l){try{if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const u=new t.Wallet(this.privateKey),d=await u.getAddress(),h=`galaswap - operation - ${N()}-${Date.now()}-${d}`,g={token0:n,token1:o,fee:i,amount0Requested:r,amount1Requested:a,tickLower:s,tickUpper:c,positionId:l,uniqueKey:h},p={CollectPositionFees:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount0Requested",type:"string"},{name:"amount1Requested",type:"string"},{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},m={name:"ethereum",chainId:1},f=this.calculatePersonalSignPrefix(g),y={...g,prefix:f},w=await u.signTypedData(m,p,y),k={...y,signature:w,types:p,domain:m},v=this.buildLiquidityStringsInstructions(n,o,i,d);this.logger.debug("Submitting CollectPositionFees to bundler",{fee:i,amount0Requested:r,amount1Requested:a,tickLower:s,tickUpper:c,positionId:l,transactionId:h});const b=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),T=await b.post("/bundle",{method:"CollectPositionFees",signedDto:k,stringsInstructions:v}),S=T.data?.data||T.data?.transactionId||T.data?.id;if(!S)throw this.logger.error("Bundler response structure",{status:T.status,data:T.data,dataType:typeof T.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(T.data)}`);return this.logger.info("CollectPositionFees transaction sent to bundler",{transactionId:S}),S}catch(e){throw this.logger.error("Failed to send CollectPositionFees to bundler",e),e}}async sendSwapToBundler(n){if(!this.privateKey)throw new Error("GSwapService: Swap requires wallet (full-access mode)");if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const o=[500,3e3,1e4];if(!o.includes(n.feeTier))throw new Error(`GSwapService: Invalid fee tier ${n.feeTier}. Must be one of: ${o.join(", ")} (basis points)`);try{this.logger.debug("Sending Swap to bundler",{fromToken:"string"==typeof n.fromToken?n.fromToken:n.fromToken?.type??"unknown",toToken:"string"==typeof n.toToken?n.toToken:n.toToken?.type??"unknown",inputAmount:n.inputAmount,minOutput:n.minOutput,feeTier:n.feeTier});let o=n.fromToken,i=n.toToken;"string"==typeof o&&(o=hi(o)),"string"==typeof i&&(i=hi(i));const r=`${o.collection}|${o.category}|${o.type}|${o.additionalKey}`,a=`${i.collection}|${i.category}|${i.type}|${i.additionalKey}`,s=r<a?[o,i,r,a]:[i,o,a,r],[c,l,d,h]=s,g="string"==typeof n.fromToken?hi(n.fromToken):n.fromToken,p=`${g.collection}|${g.category}|${g.type}|${g.additionalKey}`,m=p===d,f=`galaswap - operation - ${N()}-${Date.now()}-${n.walletAddress}`;let y;if(!n.currentSqrtPrice)throw new Error("GSwapService: currentSqrtPrice is required for sqrtPriceLimit calculation");const w=new u(n.currentSqrtPrice),k=n.slippageTolerance??.01;if(m){const e=new u(1).minus(k);y=w.multipliedBy(e).toString()}else{const e=new u(1).plus(k);y=w.multipliedBy(e).toString()}this.logger.debug("Calculated sqrtPriceLimit based on slippage tolerance",{currentSqrtPrice:n.currentSqrtPrice,slippageTolerance:100*k+"%",zeroForOne:m,sqrtPriceLimit:y,direction:m?"token0→token1 (downward price movement)":"token1→token0 (upward price movement)",reason:"sqrtPriceLimit sets price boundaries, amountOutMinimum provides volume protection"});const v={token0:c,token1:l,fee:n.feeTier,amount:new u(n.inputAmount).toFixed(),zeroForOne:m,sqrtPriceLimit:y,recipient:n.walletAddress,amountOutMinimum:new u(n.minOutput).multipliedBy(-1).toFixed(),uniqueKey:f};this.logger.info("🔄 SWAP DTO DETAILS (what we're sending to bundler)",{orderedToken0String:d,orderedToken1String:h,fromTokenStr:p,zeroForOne:m?`TRUE (${d} → ${h})`:`FALSE (${h} → ${d})`,inputAmount:n.inputAmount,expectedOutput:n.minOutput,slippageTolerance:100*(n.slippageTolerance||.01)+"%",currentSqrtPrice:n.currentSqrtPrice,swapDto:{amount:v.amount,zeroForOne:v.zeroForOne,sqrtPriceLimit:v.sqrtPriceLimit,amountOutMinimum:v.amountOutMinimum}});const b=new t.Wallet(this.privateKey),T={Swap:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount",type:"string"},{name:"zeroForOne",type:"bool"},{name:"sqrtPriceLimit",type:"string"},{name:"recipient",type:"string"},{name:"amountOutMinimum",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},S={name:"ethereum",chainId:1},A=this.calculatePersonalSignPrefix(v),E={...v,prefix:A},I=await b.signTypedData(S,T,E),P={...E,signature:I,types:T,domain:S};this.logger.debug("Swap DTO signed",{signature:P.signature?.substring(0,20)+"...",prefix:P.prefix,zeroForOne:v.zeroForOne});const D=this.buildLiquidityStringsInstructions(c,l,n.feeTier,n.walletAddress),F=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),C=await F.post("/bundle",{method:"Swap",signedDto:P,stringsInstructions:D}),x=C.data?.data||C.data?.transactionId||C.data?.id;if(!x)throw this.logger.error("Bundler response structure",{status:C.status,data:C.data,dataType:typeof C.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(C.data)}`);return this.logger.debug("Swap transaction sent to bundler",{transactionId:x,inputAmount:n.inputAmount,minOutput:n.minOutput}),x}catch(e){throw this.logger.error("Failed to send Swap to bundler",e),e}}buildLiquidityStringsInstructions(e,t,n,o){const i=`$${e.collection}$${e.category}$${e.type}$${e.additionalKey}`,r=`$${t.collection}$${t.category}$${t.type}$${t.additionalKey}`,a=`$pool${i}${r}$${n}`;return[a,`$userPosition${o}`,`$tokenBalance${i}${o}`,`$tokenBalance${r}${o}`,`$tokenBalance${i}${a}`,`$tokenBalance${r}${a}`]}handleGSwapError(e,t,n,o){this.logger.error(e,n);const i=this.extractGSwapErrorCode(n),r=n,a=[`${e}: ${r?.message||String(n)}`,n];throw o&&("GSwapSwapError"===t.name&&o.transactionHash&&a.push(o.transactionHash),"GSwapPoolError"===t.name&&(o.tokenA&&a.push(o.tokenA),o.tokenB&&a.push(o.tokenB)),"GSwapAssetError"===t.name&&o.walletAddress&&a.push(o.walletAddress)),i&&a.push(i),new t(...a)}extractGSwapErrorCode(e){if(e&&"object"==typeof e){const t=e;if(t.constructor&&"GSwapSDKError"===t.constructor.name)return t.code;if("code"in t&&"string"==typeof t.code)return t.code}}async ensureWebSocketConnected(){this.webSocketService.isConnected()||await this.webSocketService.connect()}calculatePersonalSignPrefix(e){return`Ethereum Signed Message:\n${JSON.stringify(e).length}${JSON.stringify(e)}`}}class bi{}bi.BASE_PRICE=1650667151e-14,bi.PRICE_SCALING_FACTOR=1166069e-12,bi.TRADING_FEE_FACTOR=.001,bi.GAS_FEE="1",bi.MIN_UNBONDING_FEE_FACTOR=0,bi.MAX_UNBONDING_FEE_FACTOR=.5,bi.NET_UNBONDING_FEE_FACTOR=.5,bi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY=1e7;class Ti extends Rn{constructor(e=!1){super(e),this.cache=new Map}getLRUKey(){const e=this.cache.keys().next().value;return void 0!==e?e:null}normalizeTokenName(e){return e.trim().toLowerCase().replace(/\s+/g," ").replace(/[\u0000-\u001F\u007F-\u009F\u200B-\u200D\uFEFF]/g,"")}updateCacheEntry(e,t){const n=this.cache.get(e);if(this.cache.has(e)&&this.cache.delete(e),this.cache.size>=Ti.MAX_CACHE_SIZE){const e=this.getLRUKey();null!==e&&this.cache.delete(e)}this.cache.set(e,{...n||{},...t,lastUpdated:Date.now()})}warmFromPoolData(e,t){const n=this.normalizeTokenName(e);this.updateCacheEntry(n,t)}set(e,t){const n=this.normalizeTokenName(e);this.updateCacheEntry(n,t)}get(e){const t=this.normalizeTokenName(e);return this.cache.get(t)||null}getMaxSupply(e){const t=this.normalizeTokenName(e),n=this.cache.get(t);return n?.maxSupply||bi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY.toString()}has(e){const t=this.normalizeTokenName(e);return this.cache.has(t)}clear(e){if(e){const t=this.normalizeTokenName(e);this.cache.delete(t)}else this.cache.clear()}dump(){const e={};return this.cache.forEach((t,n)=>{e[n]=t}),e}stats(){let e=Date.now(),t=0;return this.cache.forEach((n,o)=>{n.lastUpdated<e&&(e=n.lastUpdated);let i=0;i+=2*o.length,void 0!==n.reverseBondingCurveMinFeeFactor&&(i+=8),void 0!==n.reverseBondingCurveMaxFeeFactor&&(i+=8),void 0!==n.reverseBondingCurveNetFeeFactor&&(i+=8),i+=8,n.vaultAddress&&(i+=2*n.vaultAddress.length),n.maxSupply&&(i+=2*n.maxSupply.length),n.symbol&&(i+=2*n.symbol.length),i+=32,t+=i}),{totalTokens:this.cache.size,cacheSize:t,oldestEntry:this.cache.size>0?e:0}}getByTokenId(e){const t=`token:${e.toLowerCase().trim()}`;return this.cache.get(t)||null}setByTokenId(e,t){const n=`token:${e.toLowerCase().trim()}`;this.updateCacheEntry(n,t)}hasByTokenId(e){const t=`token:${e.toLowerCase().trim()}`;return this.cache.has(t)}}Ti.MAX_CACHE_SIZE=1e4;class Si extends Mn{constructor(e,t,n=void 0,o=5,i=!1){super(e,i),this.pricingConcurrency=5,this.dexBackendBaseUrl=t,this.gswapService=n,this.pricingConcurrency=o}setGSwapService(e){this.gswapService=e}setPricingConcurrency(e){this.pricingConcurrency=Math.max(1,Math.min(e,20))}async enrichPoolsWithPricing(e){if(!this.gswapService)return this.logger.warn("GSwap service not available, skipping pricing enrichment"),e;if(0===e.length)return e;this.logger.debug("Starting pricing enrichment",{poolCount:e.length,concurrency:this.pricingConcurrency});const t=[];for(let n=0;n<e.length;n++){const o=e[n];t.push({poolIndex:n,token:o.token0,isToken0:!0,task:this.gswapService.getSwapQuoteExactInput({fromToken:o.token0,toToken:"GUSDC",amount:"1"}).then(e=>e.estimatedOutput).catch(e=>{this.logger.debug(`Failed to price ${o.token0}`,{error:e.message})})}),t.push({poolIndex:n,token:o.token1,isToken0:!1,task:this.gswapService.getSwapQuoteExactInput({fromToken:o.token1,toToken:"GUSDC",amount:"1"}).then(e=>e.estimatedOutput).catch(e=>{this.logger.debug(`Failed to price ${o.token1}`,{error:e.message})})})}const n=new Map;for(let t=0;t<e.length;t++)n.set(t,{});for(let e=0;e<t.length;e+=this.pricingConcurrency){const o=t.slice(e,e+this.pricingConcurrency),i=await Promise.allSettled(o.map(e=>e.task));for(let e=0;e<o.length;e++){const t=o[e],r=i[e],a=n.get(t.poolIndex)||{};"fulfilled"===r.status&&r.value&&(t.isToken0?a.token0Price=r.value:a.token1Price=r.value),n.set(t.poolIndex,a)}}const o=e.map((e,t)=>{const o=n.get(t)||{},i={...e};return void 0!==o.token0Price&&(i.token0Price=o.token0Price),void 0!==o.token1Price&&(i.token1Price=o.token1Price),i}),i=o.filter(e=>e.token0Price&&e.token1Price).length;return this.logger.debug("Pricing enrichment complete",{total:e.length,successful:i,failed:e.length-i}),o}async fetchDexPools(e={}){const{search:t,sortBy:n="tvl",sortOrder:o="desc",page:i=Bn.DEFAULT_PAGE,limit:r=Bn.DEFAULT_LIMIT,withPrices:a=!1}=e;this.logger.debug("Fetching DEX pools",{search:t,sortBy:n,sortOrder:o,page:i,limit:r,withPrices:a});const s=new URLSearchParams({page:i.toString(),limit:Math.min(r,20).toString(),sortBy:n,sortOrder:o});t&&s.append("search",t);const c=`${this.dexBackendBaseUrl}/explore/pools?${s}`;try{const e=await this.http.get(c);if(!e||!e.data)throw new Error("No response from DEX pool service");let t=e.data.pools;const n=e.data.count,o=Math.min(r,20),s=Math.ceil(n/o);return a&&(t=await this.enrichPoolsWithPricing(t)),this.logger.debug("DEX pools fetched successfully",{poolCount:t.length,total:n,totalPages:s,withPrices:a}),{pools:t,page:i,limit:o,total:n,totalPages:s,hasNext:i<s,hasPrevious:i>1}}catch(e){throw this.logger.error("Failed to fetch DEX pools",{error:e,url:c}),e}}async fetchAllDexPools(e={}){this.logger.debug("Fetching all DEX pools (auto-paginated)",e);const t=await li((t,n)=>this.fetchDexPools({...e,page:t,limit:n}).then(e=>({items:e.pools,page:e.page,limit:e.limit,total:e.total,totalPages:e.totalPages,hasNext:e.hasNext,hasPrevious:e.hasPrevious})),{maxPages:1e4,logger:this.logger,pageSize:20});this.logger.debug("All DEX pools fetched",{totalPoolsFetched:t.items.length,totalCount:t.total,withPrices:e.withPrices});return ui(t.items,t.total,"pools")}}const Ai=5;class Ei extends Mn{constructor(e,t,n=!1,o=3e4){super(e,n),this.compositePoolFetchConcurrency=5,this.galaChainBaseUrl=t,this.networkTimeout=o}validateFetchCompositePoolDataInput(e,t,n){if(!e||"string"!=typeof e)throw new Y("token0 must be a non-empty string",{token0:e});if(!t||"string"!=typeof t)throw new Y("token1 must be a non-empty string",{token1:t});const o=e.split("|"),i=t.split("|");if(4!==o.length)throw new Y("token0 format must be: collection|category|type|additionalKey (4 pipe-separated parts)",{token0:e});if(4!==i.length)throw new Y("token1 format must be: collection|category|type|additionalKey (4 pipe-separated parts)",{token1:t});const r=[500,3e3,1e4];if(!Number.isInteger(n)||!r.includes(n))throw new Y(`fee must be one of: ${r.join(", ")} (got ${n})`,{fee:n})}validateQuoteAmount(e){if(!e||"string"!=typeof e)throw new Y("amount must be a non-empty string",{amount:e});const t=new u(e);try{So(t,"amount","for quote calculation")}catch(t){throw new Y(t.message,{amount:e})}}convertTokenClassKey(e){const t=new I;return t.collection=e.collection,t.category=e.category,t.type=e.type,t.additionalKey=e.additionalKey,t}setCompositePoolFetchConcurrency(e){this.compositePoolFetchConcurrency=Math.max(1,Math.min(e,20)),this.logger.debug(`Composite pool fetch concurrency set to ${this.compositePoolFetchConcurrency}`)}async fetchCompositePoolData(e){const{token0:t,token1:n,fee:o,gatewayBaseUrl:i}=e;this.logger.debug("Fetching composite pool data",{token0:t,token1:n,fee:o}),this.validateFetchCompositePoolDataInput(t,n,o);try{const e=wi(t),r=wi(n),a=this.convertTokenClassKey(e),s=this.convertTokenClassKey(r),c=new b(a,s,o),l=`${i||this.galaChainBaseUrl}/api/asset/dexv3-contract/GetCompositePool`,d=await this.http.post(l,c);if(!d||1!==d.Status)throw new J(`Pool not found: ${t}/${n} with fee ${o}`);const h=function(e){return{pool:e.pool,tickDataMap:e.tickDataMap,token0Balance:e.token0Balance,token1Balance:e.token1Balance,token0Decimals:e.token0Decimals,token1Decimals:e.token1Decimals,compositePoolDto:e}}(function(e){const t=new w(e.pool.token0,e.pool.token1,e.pool.token0ClassKey,e.pool.token1ClassKey,e.pool.fee,new u(e.pool.sqrtPrice),e.pool.protocolFees);t.bitmap=e.pool.bitmap,t.grossPoolLiquidity=new u(e.pool.grossPoolLiquidity),t.liquidity=new u(e.pool.liquidity),t.feeGrowthGlobal0=new u(e.pool.feeGrowthGlobal0),t.feeGrowthGlobal1=new u(e.pool.feeGrowthGlobal1),t.protocolFeesToken0=new u(e.pool.protocolFeesToken0),t.protocolFeesToken1=new u(e.pool.protocolFeesToken1),t.tickSpacing=e.pool.tickSpacing,t.maxLiquidityPerTick=new u(e.pool.maxLiquidityPerTick);const n={};Object.keys(e.tickDataMap).forEach(t=>{const o=e.tickDataMap[t],i=new k(o.poolHash,o.tick);i.initialised=o.initialised,i.liquidityNet=new u(o.liquidityNet),i.liquidityGross=new u(o.liquidityGross),i.feeGrowthOutside0=new u(o.feeGrowthOutside0),i.feeGrowthOutside1=new u(o.feeGrowthOutside1),n[t]=i});const o={...e.token0Balance},i=new P(o);i.quantity=new u(e.token0Balance.quantity);const r={...e.token1Balance},a=new P(r);return a.quantity=new u(e.token1Balance.quantity),new v(t,n,i,a,e.token0Decimals,e.token1Decimals)}(d.Data),d.Data);return this.logger.debug("Composite pool data fetched successfully",{token0:t,token1:n,fee:o,liquidity:h.pool.liquidity.toString()}),h}catch(e){if(e instanceof J)throw e;const i=e instanceof Error?e.message:String(e);throw this.logger.error("Failed to fetch composite pool data",e),new Y(`Failed to fetch composite pool data: ${i}`,{token0:t,token1:n,fee:o})}}async calculateDexPoolQuoteExactAmountLocal(e){const{compositePoolData:t,fromToken:n,toToken:o,amount:i}=e;if(this.logger.debug("Calculating local DEX quote",{fromToken:n,toToken:o,amount:i}),this.validateQuoteAmount(i),!t)throw new Y("compositePoolData is required for local quote calculation",{compositePoolData:t});try{const e=n===t.pool.token0.replace(/\$/g,"|"),r=wi(n),a=wi(o),s=this.convertTokenClassKey(r),c=this.convertTokenClassKey(a),[l,d]=n<o?[s,c]:[c,s],h=new T(l,d,t.pool.fee,new u(i),e,t.compositePoolDto),g=await S(void 0,h);return this.logger.debug("Local quote calculated",{amount0:g.amount0,amount1:g.amount1}),{amount0:g.amount0.toString(),amount1:g.amount1.toString(),currentSqrtPrice:g.currentSqrtPrice.toString(),newSqrtPrice:g.newSqrtPrice.toString()}}catch(e){const t=e instanceof Error?e.message:String(e);throw this.logger.error("Local quote calculation failed",e),new Y(`Local quote calculation failed: ${t}`,{fromToken:n,toToken:o,amount:i})}}async calculateDexPoolQuoteExactAmountExternal(e){const{compositePoolData:t,fromToken:n,toToken:o,amount:i}=e;if(this.logger.debug("Calculating external DEX quote",{fromToken:n,toToken:o,amount:i}),this.validateQuoteAmount(i),!t)throw new Y("compositePoolData is required for external quote calculation (token format info)",{compositePoolData:t});try{const e=n===t.pool.token0.replace(/\$/g,"|"),r=wi(n),a=wi(o),s=this.convertTokenClassKey(r),c=this.convertTokenClassKey(a),l=new T(s,c,t.pool.fee,new u(i),e,void 0),d=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/QuoteExactAmount`,h=await this.http.post(d,l);if(!h||1!==h.Status)throw new Y("External quote failed: "+(h?.Message||"Unknown error"));const g=h.Data;return this.logger.debug("External quote calculated",{amount0:g.amount0,amount1:g.amount1}),{amount0:g.amount0.toString(),amount1:g.amount1.toString(),currentSqrtPrice:g.currentSqrtPrice.toString(),newSqrtPrice:g.newSqrtPrice.toString()}}catch(e){const t=e instanceof Error?e.message:String(e);throw this.logger.error("External quote calculation failed",e),new Y(`External quote calculation failed: ${t}`,{fromToken:n,toToken:o,amount:i})}}async calculateDexPoolQuoteExactAmount(e,t="local"){return"external"===t?this.calculateDexPoolQuoteExactAmountExternal(e):this.calculateDexPoolQuoteExactAmountLocal(e)}}class Ii{constructor(){this.eventLatencies=[],this.maxLatencySamples=1e4,this.eventsProcessed=0,this.eventsDropped=0,this.queueDepth=0,this.maxQueueDepth=0,this.startTime=Date.now(),this.perPoolMetrics=new Map,this.memorySnapshots=[],this.maxMemorySnapshots=100,this.recordMemory()}recordEventLatency(e){this.eventLatencies.push(e),this.eventLatencies.length>this.maxLatencySamples&&this.eventLatencies.shift(),this.eventsProcessed++,this.lastEventTime=new Date}recordEventDropped(){this.eventsDropped++}updateQueueDepth(e){this.queueDepth=e,this.maxQueueDepth=Math.max(this.maxQueueDepth,e)}recordPoolCacheHit(e,t){const n=this.getPoolMetrics(e);n.cacheHits++,n.eventsProcessed++,n.totalLatency+=t,n.lastEventTime=new Date}recordPoolCacheMiss(e,t){const n=this.getPoolMetrics(e);n.cacheMisses++,n.eventsProcessed++,n.totalLatency+=t,n.lastEventTime=new Date}getLatencyPercentiles(){if(0===this.eventLatencies.length)return{p50:0,p95:0,p99:0};const e=[...this.eventLatencies].sort((e,t)=>e-t),t=Math.floor(.5*e.length),n=Math.floor(.95*e.length),o=Math.floor(.99*e.length);return{p50:e[t]??0,p95:e[n]??0,p99:e[o]??0}}getCacheHitRate(){if(0===this.eventsProcessed)return 0;let e=0;for(const t of this.perPoolMetrics.values())e+=t.cacheHits;return e/this.eventsProcessed*100}getThroughputPerSecond(){const e=(Date.now()-this.startTime)/1e3;return 0===e?0:this.eventsProcessed/e}recordMemory(){if("undefined"!=typeof process&&process.memoryUsage){const e=process.memoryUsage().heapUsed/1024/1024;this.memorySnapshots.push(e),this.memorySnapshots.length>this.maxMemorySnapshots&&this.memorySnapshots.shift()}}getMemoryUsedMB(){return"undefined"!=typeof process&&process.memoryUsage?process.memoryUsage().heapUsed/1024/1024:0}getPoolAverageLatency(e){const t=this.perPoolMetrics.get(e);return t&&0!==t.eventsProcessed?t.totalLatency/t.eventsProcessed:0}getPoolCacheHitRate(e){const t=this.perPoolMetrics.get(e);if(!t)return 0;const n=t.cacheHits+t.cacheMisses;return 0===n?0:t.cacheHits/n*100}getHealthMetrics(e,t,n,o,i){const r=this.getLatencyPercentiles(),a=this.getMemoryUsedMB();return{eventProcessing:{queueDepth:this.queueDepth,eventsProcessed:this.eventsProcessed,eventsDropped:this.eventsDropped,throughputPerSecond:this.getThroughputPerSecond()},metrics:{latencyP50:r.p50,latencyP95:r.p95,latencyP99:r.p99,cacheHitRate:this.getCacheHitRate()},memory:{usedMB:Math.round(10*a)/10,maxMB:i,percentUsed:Math.round(a/i*1e3)/10},pools:{totalMonitored:e,hotCacheSize:t,warmCacheSize:n,coldCacheSize:o}}}reset(){this.eventLatencies=[],this.eventsProcessed=0,this.eventsDropped=0,this.queueDepth=0,this.maxQueueDepth=0,this.startTime=Date.now(),this.lastEventTime=void 0,this.perPoolMetrics.clear(),this.memorySnapshots=[]}getSummary(){const e=this.eventLatencies.length>0?this.eventLatencies.reduce((e,t)=>e+t,0)/this.eventLatencies.length:0;return{eventsProcessed:this.eventsProcessed,eventsDropped:this.eventsDropped,cacheHitRate:this.getCacheHitRate(),averageLatency:Math.round(e),memoryUsedMB:Math.round(10*this.getMemoryUsedMB())/10,throughputPerSecond:Math.round(100*this.getThroughputPerSecond())/100}}getPoolMetrics(e){let t=this.perPoolMetrics.get(e);return t||(t={eventsProcessed:0,totalLatency:0,cacheHits:0,cacheMisses:0},this.perPoolMetrics.set(e,t)),t}}class Pi{static createPoolKey(e,t,n){return`${e}/${t}/${n}`}static parsePoolKey(e){if(!e||"string"!=typeof e)return null;const t=e.split("/");if(3!==t.length)return null;const n=t[0]?.trim(),o=t[1]?.trim(),i=t[2]?.trim();if(!n||!o||!i)return null;const r=Number.parseInt(i,10);return Number.isNaN(r)?null:{token0:n,token1:o,feeTier:r}}static isValidPoolKey(e){if("string"!=typeof e)return!1;return null!==this.parsePoolKey(e)}static getToken0(e){const t=this.parsePoolKey(e);return t?.token0??null}static getToken1(e){const t=this.parsePoolKey(e);return t?.token1??null}static getFeeTier(e){const t=this.parsePoolKey(e);return t?.feeTier??null}static containsToken(e,t){const n=this.parsePoolKey(e);return!!n&&(n.token0===t||n.token1===t)}static containsTokenPair(e,t,n){const o=this.parsePoolKey(e);if(!o)return!1;const i=o.token0===t||o.token1===t,r=o.token0===n||o.token1===n;return i&&r&&t!==n}static normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:Number.isInteger(t)&&t>0?t:null}static formatFeeAsPercentage(e){return`${(e/1e4).toFixed(2)}%`}static isValidTokenPair(e,t){return Boolean(e)&&Boolean(t)&&e!==t}}class Ni{constructor(e){this.logger=e||new U({debug:!1,context:"SwapEventExtractor"})}walkPayloadForSwaps(e,t){const n=[],o=new WeakSet,i=(e,r=0)=>{if(r>50)this.logger.debug("Payload nesting exceeded maximum depth of 50");else if(e&&"string"!=typeof e&&"object"==typeof e){if(o.has(e))return;o.add(e);const a=this.extractSwapFromObject(e);a&&!t.has(a.transactionId)&&(n.push(a),t.add(a.transactionId));for(const t of Object.values(e))i(t,r+1)}};return i(e,0),n}extractSwapFromObject(e){const t=this.extractTransactionId(e);if(!t)return null;const n=e.Data,o=n&&"object"==typeof n&&!Array.isArray(n)?n:e,i=this.extractToken(o,"token0","fromToken","source"),r=this.extractToken(o,"token1","toToken","destination");if(!i||!r)return null;const a=this.extractAmount(o,"amount0","amountIn","inputAmount"),s=this.extractAmount(o,"amount1","amountOut","outputAmount");if(!a||!s)return null;const c=this.extractFeeTier(o);if(null===c)return null;const l=this.extractTimestamp(o),u=this.buildPoolKey(i,r,c),d=this.determineDirection(o,i,r),h={transactionId:t,poolKey:u,token0:i,token1:r,amount0:a,amount1:s,feeTier:c,direction:d,timestamp:l,exactInput:this.determineExactInput(o,d)},g=this.extractUser(o);return void 0!==g&&(h.user=g),h}extractTransactionId(e){const t=["transactionId","txId","tx_id","hash","txHash","id"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractToken(e,...t){for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractAmount(e,...t){for(const n of t){const t=e[n];if(null!=t){const e=String(t).trim();if(/^-?\d+(\.\d+)?([eE]-?\d+)?$/.test(e))return e}}return null}extractFeeTier(e){const t=["poolFee","feeTier","fee","feeTierBps","liquidityFeeBps","feeAmount"];for(const n of t){const t=e[n],o=this.normalizeFee(t);if(null!==o)return o}return null}normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:Number.isInteger(t)?t:null}extractTimestamp(e){const t=["timeStamp","timestamp","time","createdAt","date"];for(const n of t){const t=e[n];if("number"==typeof t)return t;if("string"==typeof t){const e=new Date(t).getTime();if(!Number.isNaN(e))return e}}return Date.now()}extractUser(e){const t=["userAddress","user","from","sender","wallet","address"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}}determineDirection(e,t,n){const o=e.zeroForOne||e.direction;if("boolean"==typeof o)return o?"zeroForOne":"oneForZero";if("string"==typeof o){if("zerotoone"===o.toLowerCase()||"0to1"===o)return"zeroForOne";if("onetozero"===o.toLowerCase()||"1to0"===o)return"oneForZero"}if(e.fromToken===t||e.inputToken===t)return"zeroForOne";if(e.fromToken===n||e.inputToken===n)return"oneForZero";const i=this.extractAmount(e,"amount0","amountIn");return i&&Number(i),"zeroForOne"}determineExactInput(e,t){if("boolean"==typeof e.exactInput)return e.exactInput;if("boolean"==typeof e.exactOutput)return!e.exactOutput;const n=void 0!==e.amountIn&&null!==e.amountIn,o=void 0!==e.amountOut&&null!==e.amountOut,i=void 0!==e.inputAmount&&null!==e.inputAmount,r=void 0!==e.outputAmount&&null!==e.outputAmount;return!(!n||o)||!(o&&!n)&&(!(!i||r)||!(r&&!i))}buildPoolKey(e,t,n){return`${e}/${t}/${n}`}}class Di{static getCached(e){const t=e.toString();return this.CACHE.has(t)||this.CACHE.set(t,new u(e)),this.CACHE.get(t)}static clearCache(){this.CACHE.clear()}static getCacheStats(){return{size:this.CACHE.size,entries:Array.from(this.CACHE.keys())}}static trimCache(e=1e3){if(this.CACHE.size>e){const t=this.CACHE.size-e,n=Array.from(this.CACHE.keys());for(let e=0;e<t;e++)this.CACHE.delete(n[e])}}}Di.CACHE=new Map,Di.ZERO=new u(0),Di.ONE=new u(1),Di.FEE_PIPS=new u(1e6),Di.MIN_SQRT_RATIO=new u("4295128739"),Di.MAX_SQRT_RATIO=new u("1461446703485210103287273052203988822378723970342");const Fi={maxIterations:100,enableBigNumberCache:!0,roundingMode:u.ROUND_DOWN,debugLogging:!1};class Ci{static calculateSwapDelta(e,t,n={}){const o=Date.now(),i={...Fi,...n};try{const n=this.initializeSwapState(e,t,i);i.debugLogging&&this.logger.debug("Initialized swap state",{sqrtPrice:n.sqrtPrice.toString(),liquidity:n.liquidity.toString(),tick:n.tick,zeroForOne:t.zeroForOne});const r=this.computeSwapLoop(n,e,t,i);i.debugLogging&&this.logger.debug("Swap loop completed",{stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length,priceHitLimit:r.priceHitLimit});const a=this.createUpdatedPool(e.pool,r.state,t,i),s=this.calculateFinalAmounts(n,r.state,t);let c;if(t.actualSqrtPrice){const e=new u(a.sqrtPrice),n=new u(t.actualSqrtPrice);c=e.minus(n).abs().div(n).times(100).toNumber()}const l=Date.now()-o;i.debugLogging&&this.logger.debug("Swap delta calculated",{calculationTimeMs:l,amount0:s.amount0.toString(),amount1:s.amount1.toString(),driftPercentage:c}),l>100&&this.logger.warn("Swap calculation exceeded 100ms",{calculationTimeMs:l,stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length}),r.priceHitLimit&&this.logger.warn("Swap price hit limit - partially fulfilled",{zeroForOne:t.zeroForOne,stepCount:r.stepCount}),r.stepCount>50&&this.logger.warn("Unusually complex swap detected",{stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length});return{updatedPool:a,updatedTicks:r.updatedTicks,amount0:s.amount0,amount1:s.amount1,feeAmount0:s.feeAmount0,feeAmount1:s.feeAmount1,ticksCrossed:r.ticksCrossed,metadata:{calculationTimeMs:l,swapSteps:r.stepCount,priceHitLimit:r.priceHitLimit,...void 0!==c&&{driftPercentage:c}}}}catch(e){this.logger.error("Swap delta calculation failed",e);const t=e instanceof Error?e.message:String(e);throw new Error(`Swap delta calculation failed: ${t}`)}}static initializeSwapState(e,t,n){const{pool:o}=e;if(!o.sqrtPrice||!o.liquidity)throw new Error("Invalid pool data: missing sqrtPrice or liquidity");const i=n.enableBigNumberCache?Di.getCached.bind(Di):e=>new u(e),r="string"==typeof o.sqrtPrice?o.sqrtPrice:new u(o.sqrtPrice).toFixed(0),a="string"==typeof o.liquidity?o.liquidity:new u(o.liquidity).toFixed(0),s=i(r),c=i(a),l=o.tick??0,d=A(new u(r)),h=Math.abs(d-l);h>100&&this.logger.warn("Significant tick/price mismatch detected in pool state",{poolTick:l,calculatedTick:d,drift:h,threshold:100});const g=i(t.amountSpecified);So(g,"amountSpecified","for swap operation");const p="string"==typeof o.feeGrowthGlobal1?o.feeGrowthGlobal1:new u(o.feeGrowthGlobal1).toFixed(0),m="string"==typeof o.feeGrowthGlobal0?o.feeGrowthGlobal0:new u(o.feeGrowthGlobal0).toFixed(0),f=t.zeroForOne?i(p):i(m);return{sqrtPrice:s,liquidity:c,tick:l,amountSpecifiedRemaining:g,amountCalculated:Di.ZERO,feeGrowthGlobalX:f,protocolFee:Di.ZERO}}static computeSwapLoop(e,t,n,o){const{pool:i,tickDataMap:r}=t,a=[],s={};let c=0;const l=n.zeroForOne?Di.MIN_SQRT_RATIO:Di.MAX_SQRT_RATIO,h=new u("0.000001");for(;e.amountSpecifiedRemaining.gt(h)&&!e.sqrtPrice.eq(l)&&c<o.maxIterations;){c++;const[t,h]=this.findNextInitializedTick(r,e.tick,i.tickSpacing,n.zeroForOne);let g;if(o.debugLogging&&this.logger.debug(`Swap step ${c}`,{currentTick:e.tick,tickNext:t,initialized:h,sqrtPrice:e.sqrtPrice.toString(),liquidity:e.liquidity.toString(),amountRemaining:e.amountSpecifiedRemaining.toString()}),h&&t>=-887272&&t<=887272){const e=d(t);g=e instanceof u?e:new u(String(e))}else g=l;const p=n.zeroForOne?u.max(g,l):u.min(g,l),m=this.executeSwapStep(e.sqrtPrice,p,e.liquidity,e.amountSpecifiedRemaining,i.fee,n.exactInput);if(e.sqrtPrice=m.sqrtPriceNext,n.exactInput){const t=m.amountIn.plus(m.feeAmount);t.lte(0)?e.amountSpecifiedRemaining=Di.ZERO:(e.amountSpecifiedRemaining=e.amountSpecifiedRemaining.minus(t),e.amountSpecifiedRemaining.lt(0)&&(e.amountSpecifiedRemaining=Di.ZERO)),e.amountCalculated=e.amountCalculated.minus(m.amountOut)}else{m.amountOut.lte(0)?e.amountSpecifiedRemaining=Di.ZERO:e.amountSpecifiedRemaining=e.amountSpecifiedRemaining.plus(m.amountOut),e.amountCalculated=e.amountCalculated.plus(m.amountIn.plus(m.feeAmount))}if(e.liquidity.gt(0)){const t=m.feeAmount.div(e.liquidity);e.feeGrowthGlobalX=e.feeGrowthGlobalX.plus(t)}if(e.sqrtPrice.eq(g)&&h){const i=r[t.toString()];if(!i)throw new Error(`Missing tick data for initialized tick ${t}`);const c=n.zeroForOne?new u(i.liquidityNet).negated():new u(i.liquidityNet);if(e.liquidity=e.liquidity.plus(c),e.liquidity.lt(0))throw new Error(`Negative liquidity after crossing tick ${t}: ${e.liquidity.toString()}`);a.push(t),s[t.toString()]=i,o.debugLogging&&this.logger.debug(`Crossed tick ${t}`,{liquidityNet:c.toString(),newLiquidity:e.liquidity.toString()})}if(e.sqrtPrice.eq(g))e.tick=n.zeroForOne?t-1:t;else{const t=A(new u(e.sqrtPrice.toString()));e.tick=t}}if(c>=o.maxIterations)throw new Error(`Swap calculation exceeded maximum iterations (${o.maxIterations}). Possible infinite loop or very complex swap.`);const g=e.sqrtPrice.eq(l);return{state:e,ticksCrossed:a,priceHitLimit:g,stepCount:c,updatedTicks:s}}static createUpdatedPool(e,t,n,o){const i=Object.assign(Object.create(Object.getPrototypeOf(e)),e);if(i.sqrtPrice=t.sqrtPrice.toFixed(0),i.liquidity=t.liquidity.toFixed(0),i.tick=t.tick,n.zeroForOne?i.feeGrowthGlobal1=t.feeGrowthGlobalX.toFixed(0):i.feeGrowthGlobal0=t.feeGrowthGlobalX.toFixed(0),n.zeroForOne){const n=new u(e.protocolFeesToken0);i.protocolFeesToken0=n.plus(t.protocolFee).toFixed(0)}else{const n=new u(e.protocolFeesToken1);i.protocolFeesToken1=n.plus(t.protocolFee).toFixed(0)}return i}static calculateFinalAmounts(e,t,n){let o,i,r,a;if(n.exactInput){const e=new u(n.amountSpecified),s=t.amountCalculated.abs();n.zeroForOne?(o=e.negated(),i=s,r=Di.ZERO,a=Di.ZERO):(o=s,i=e.negated(),r=Di.ZERO,a=Di.ZERO)}else{const e=new u(n.amountSpecified),s=t.amountCalculated.abs();n.zeroForOne?(o=s.negated(),i=e,r=Di.ZERO,a=Di.ZERO):(o=e,i=s.negated(),r=Di.ZERO,a=Di.ZERO)}const s=t.feeGrowthGlobalX.minus(e.feeGrowthGlobalX).times(e.liquidity);return n.zeroForOne?a=s:r=s,{amount0:o,amount1:i,feeAmount0:r,feeAmount1:a}}static findNextInitializedTick(e,t,n,o){const i=Object.keys(e).map(e=>parseInt(e,10)).sort((e,t)=>e-t);if(0===i.length){return[o?-887272:887272,!1]}if(o){const e=i.reverse().find(e=>e<t);return void 0!==e?[e,!0]:[-887272,!1]}{const e=i.find(e=>e>t);return void 0!==e?[e,!0]:[887272,!1]}}static executeSwapStep(e,t,n,o,i,r){To(n),So(o,"amountRemaining","for swap step");const a=[500,3e3,1e4];if(!a.includes(i))throw new Error(`Invalid fee tier: ${i}. Must be one of: ${a.join(", ")}`);const s=E(e,t,n,o,i,t.lt(e)),c=s[0],l=s[1],d=s[2],h=s[3],g=u.isBigNumber(c)?c:new u(String(c)),p=u.isBigNumber(l)?l:new u(String(l)),m=u.isBigNumber(d)?d:new u(String(d)),f=u.isBigNumber(h)?h:new u(String(h));return{sqrtPriceStart:e,tickNext:A(g),sqrtPriceNext:g,initialised:!1,amountIn:p,amountOut:m,feeAmount:f}}}Ci.logger=new U({debug:!1,context:"SwapDeltaCalculator"});class xi{constructor(e,t,n,o){this.cache=new Map,this.tierSizes={hot:50,warm:200,cold:0},this.tierTTLs={hot:1/0,warm:18e5,cold:3e5},this.refetchThresholds={swapCount:50,driftPercent:.05},this.fetchPoolFn=e,this.config=t,this.metrics=n,this.logger=o||new U({debug:!1,context:"PoolCacheManager"}),this.tierSizes.cold=Math.max(0,this.config.maxPools-this.tierSizes.hot-this.tierSizes.warm),this.logger.debug(`Initialized with cache limits: hot=${this.tierSizes.hot}, warm=${this.tierSizes.warm}, cold=${this.tierSizes.cold}, max=${this.config.maxPools}`)}async getPool(e){const t=this.cache.get(e);if(t){if(!(Date.now()>t.expiresAt))return t.lastAccessTime=Date.now(),this.checkRefetchNeeded(e,t),this.metrics.recordPoolCacheHit(e,0),t.poolData;this.cache.delete(e),this.logger.debug(`Cache expired for pool ${e}`)}this.metrics.recordPoolCacheMiss(e,0);try{const t=await this.fetchPoolFn(e),n=this.determineTier(),o={poolData:t,tier:n,lastAccessTime:Date.now(),expiresAt:Date.now()+this.tierTTLs[n],swapsSinceRefetch:0,cumulativeDrift:0,lastDeltaAppliedTime:Date.now()};return this.cache.set(e,o),this.cache.size>this.config.maxPools&&this.evictLRU(),this.logger.debug(`Fetched pool ${e} (tier: ${n})`),t}catch(t){throw this.logger.error(`Failed to fetch pool ${e}:`,t),t}}updatePoolWithSwapDelta(e,t,n,o,i){const r=this.cache.get(e);if(!r)return this.logger.debug(`Pool ${e} not in cache for delta update`),!1;try{if(i){const a="zeroForOne"===t,s=a?n:o,c={transactionId:i.transactionId,timestamp:i.timestamp,amountSpecified:s,zeroForOne:a,exactInput:i.exactInput},l=Ci.calculateSwapDelta(r.poolData,c);r.poolData={...r.poolData,pool:l.updatedPool},r.swapsSinceRefetch++,r.lastDeltaAppliedTime=Date.now(),void 0!==l.metadata.driftPercentage?(r.cumulativeDrift+=l.metadata.driftPercentage,this.logger.debug(`Delta applied for ${e}: drift=${l.metadata.driftPercentage.toFixed(4)}%, cumulative=${r.cumulativeDrift.toFixed(2)}%`)):this.logger.debug(`Delta applied for ${e}: ${l.ticksCrossed.length} ticks crossed`)}else r.swapsSinceRefetch++,r.lastDeltaAppliedTime=Date.now();return this.shouldRefetch(r)&&(this.logger.debug(`Refetch needed for ${e}: swaps=${r.swapsSinceRefetch}, drift=${(100*r.cumulativeDrift).toFixed(2)}%`),r.expiresAt=Date.now()),!0}catch(t){return this.logger.error(`Failed to update pool ${e}:`,t),r.expiresAt=Date.now(),!1}}getStats(){const e={totalCached:this.cache.size,hotCacheSize:0,warmCacheSize:0,coldCacheSize:0,memoryUsedMB:this.metrics.getMemoryUsedMB()};for(const t of this.cache.values())"hot"===t.tier?e.hotCacheSize++:"warm"===t.tier?e.warmCacheSize++:e.coldCacheSize++;return e}getPoolInfo(e){const t=this.cache.get(e);return t?{poolKey:e,tier:t.tier,lastAccessTime:new Date(t.lastAccessTime),expiresAt:new Date(t.expiresAt),swapsSinceRefetch:t.swapsSinceRefetch,cumulativeDrift:t.cumulativeDrift,isExpired:Date.now()>t.expiresAt}:null}async warmCache(e){if(this.cache.has(e))return!0;try{return await this.getPool(e),this.logger.debug(`Cache warmed for ${e}`),!0}catch(t){return this.logger.error(`Failed to warm cache for ${e}:`,t),!1}}async warmCacheBatch(e,t=5){const n={succeeded:0,failed:0,total:e.length};let o=0;const i=new Set;for(;o<e.length||i.size>0;){for(;o<e.length&&i.size<t;){const t=e[o];o++;const r=this.warmCache(t).then(e=>{e?n.succeeded++:n.failed++});i.add(r),r.finally(()=>i.delete(r))}i.size>0&&await Promise.race(i)}return this.logger.debug(`Cache warming complete: ${n.succeeded}/${n.total} succeeded`),n}clear(){this.cache.clear(),this.logger.debug("Cache cleared")}clearExpired(){const e=Date.now();let t=0;for(const[n,o]of this.cache)e>o.expiresAt&&(this.cache.delete(n),t++);t>0&&this.logger.debug(`Cleared ${t} expired entries`)}determineTier(){const e=Array.from(this.cache.values()).filter(e=>"hot"===e.tier).length,t=Array.from(this.cache.values()).filter(e=>"warm"===e.tier).length;return e<this.tierSizes.hot?"hot":t<this.tierSizes.warm?"warm":"cold"}checkRefetchNeeded(e,t){this.shouldRefetch(t)&&(this.logger.debug(`Scheduling refetch for ${e}: swaps=${t.swapsSinceRefetch}, drift=${(100*t.cumulativeDrift).toFixed(2)}%`),t.expiresAt=Date.now())}shouldRefetch(e){return e.swapsSinceRefetch>=this.refetchThresholds.swapCount||e.cumulativeDrift>=this.refetchThresholds.driftPercent}evictLRU(){const e=Array.from(this.cache.entries()).filter(([e,t])=>"cold"===t.tier).sort((e,t)=>e[1].lastAccessTime-t[1].lastAccessTime);if(0===e.length){this.logger.warn("No cold cache entries to evict, trying warm cache");const e=Array.from(this.cache.entries()).filter(([e,t])=>"warm"===t.tier).sort((e,t)=>e[1].lastAccessTime-t[1].lastAccessTime);if(e.length>0){const[t]=e[0];this.cache.delete(t),this.logger.debug(`Evicted warm cache entry: ${t}`)}return}const[t]=e[0];this.cache.delete(t),this.logger.debug(`Evicted cold cache entry: ${t}`)}async refreshWarmAndHotTiers(){const e=Array.from(this.cache.entries()).filter(([e,t])=>"hot"===t.tier||"warm"===t.tier).sort((e,t)=>t[1].lastAccessTime-e[1].lastAccessTime).slice(0,10);if(0===e.length)return;const t=e.map(([e,t])=>this.fetchPoolFn(e).then(n=>{t&&(t.poolData=n,t.lastAccessTime=Date.now(),t.swapsSinceRefetch=0,t.cumulativeDrift=0),this.logger.debug(`Refreshed ${e} during background warming`)}).catch(t=>{this.logger.debug(`Failed to refresh ${e} during background warming:`,t)})),n=new Promise(e=>setTimeout(()=>e(),5e3));await Promise.race([Promise.all(t),n])}}class $i{constructor(e,t,n){this.queue=[],this.isShuttingDown=!1,this.currentConcurrency=0,this.maxConcurrencyReached=0,this.eventsDropped=0,this.totalBatchesProcessed=0,this.totalBatchSize=0,this.eventsProcessedCount=0,this.processor=null,this.processingScheduled=!1,this.scheduleProcessing=e=>{"undefined"!=typeof setImmediate?setImmediate(e):Promise.resolve().then(e)},this.config=e,this.metrics=t,this.logger=n||new U({debug:!1,context:"SwapEventQueue"}),this.logger.debug(`Initialized with maxQueueSize=${this.config.maxQueueSize}, batchSize=${this.config.batchSize}, maxConcurrent=${this.config.maxConcurrent}`)}setProcessor(e){this.processor=e}enqueue(e){return this.isShuttingDown?(this.logger.debug(`Rejecting event (queue shutting down): ${e.transactionId}`),this.metrics.recordEventDropped(),this.eventsDropped++,!1):this.queue.length>=this.config.maxQueueSize?(this.logger.warn(`Queue full (${this.queue.length}/${this.config.maxQueueSize}), dropping event: ${e.transactionId}`),this.metrics.recordEventDropped(),this.eventsDropped++,!1):(this.queue.push(e),this.metrics.updateQueueDepth(this.queue.length),this.processingScheduled||this.isShuttingDown||(this.processingScheduled=!0,this.scheduleProcessing(()=>this.processNextBatch())),!0)}getQueueSize(){return this.queue.length}getStats(){return{queueSize:this.queue.length,eventsProcessed:this.eventsProcessedCount,eventsDropped:this.eventsDropped,currentConcurrent:this.currentConcurrency,maxConcurrentReached:this.maxConcurrencyReached,averageBatchSize:this.totalBatchesProcessed>0?Math.floor(this.totalBatchSize/this.totalBatchesProcessed):0,totalBatchesProcessed:this.totalBatchesProcessed}}async waitForEmpty(e){return new Promise(t=>{const n=()=>{0!==this.queue.length||0!==this.currentConcurrency?setTimeout(n,10):t()};e&&setTimeout(()=>t(),e),n()})}async shutdown(e=3e4){this.isShuttingDown=!0,this.logger.debug("Shutting down queue...");const t=Date.now();for(;this.queue.length>0||this.currentConcurrency>0;){if(Date.now()-t>e){this.logger.warn(`Queue shutdown timeout: ${this.queue.length} events remaining, ${this.currentConcurrency} processing`);break}await new Promise(e=>setTimeout(e,50))}this.logger.debug("Queue shutdown complete")}clear(){const e=this.queue.length;this.queue.length=0,this.eventsDropped+=e,this.metrics.updateQueueDepth(0),this.logger.warn(`Cleared ${e} events from queue`)}async processNextBatch(){if(this.processingScheduled=!1,this.isShuttingDown&&0===this.queue.length)return;if(this.currentConcurrency>=this.config.maxConcurrent)return void setTimeout(()=>{this.processingScheduled||(this.processingScheduled=!0,setImmediate(()=>this.processNextBatch()))},10);const e=Math.min(this.config.batchSize,this.queue.length,this.config.maxConcurrent-this.currentConcurrency);if(0===e)return;const t=this.queue.splice(0,e);this.metrics.updateQueueDepth(this.queue.length),this.currentConcurrency+=t.length,this.currentConcurrency>this.maxConcurrencyReached&&(this.maxConcurrencyReached=this.currentConcurrency),this.totalBatchSize+=t.length,this.totalBatchesProcessed++;try{const e=await Promise.allSettled(t.map(e=>this.processEvent(e)));for(let n=0;n<e.length;n++){const o=e[n];this.eventsProcessedCount++,"rejected"===o.status&&this.logger.error(`Failed to process event ${t[n].transactionId}:`,o.reason)}}finally{this.currentConcurrency-=t.length}this.queue.length>0&&!this.processingScheduled&&(this.processingScheduled=!0,this.scheduleProcessing(()=>this.processNextBatch()))}async processEvent(e){if(!this.processor)return void this.logger.warn("No processor set, discarding event:",e.transactionId);const t=Date.now();try{await this.processor(e);const n=Date.now()-t;this.metrics.recordEventLatency(n)}catch(t){throw this.logger.error(`Event processing failed for ${e.transactionId}:`,t),t}}}class Li{constructor(e,t,n,o={},i){this.socket=null,this.maxSeenTransactions=1e4,this.listeners=[],this.onErrorCallbacks=[],this.isActive=!1,this.listenerRegistered=!1,this.handleSwapEvent=null,this.warmingIntervalHandle=null,this.reconnectAttempts=0,this.maxReconnectAttempts=3,this.reconnectDelayMs=1e3,this.logger=i||new U({debug:!1,context:"MultiPoolStateManager"}),e instanceof Promise?this.socketReady=e.then(e=>(this.socket=e,this.setupConnectionMonitoring(),e)).catch(e=>{throw this.logger.error("Failed to resolve socket promise:",e),e}):(this.socket=e,this.socketReady=Promise.resolve(e),this.setupConnectionMonitoring()),this.metrics=new Ii,this.config=this.applyDefaults(o),this.eventExtractor=new Ni(this.logger),this.quoteService=n,this.cacheManager=new xi(t,this.config,this.metrics,this.logger),this.eventQueue=new $i(this.config,this.metrics,this.logger),this.seenTransactions=new Oi(this.maxSeenTransactions),this.eventQueue.setProcessor(e=>this.processSwapEvent(e)),this.logger.debug("Initialized MultiPoolStateManager")}subscribe(e,t){this.listeners.push(t),e.onError&&this.onErrorCallbacks.push(e.onError),this.isActive||(this.setupWebSocketListener(e),this.isActive=!0);const n=this;return()=>{n.listeners=n.listeners.filter(e=>e!==t),e.onError&&(n.onErrorCallbacks=n.onErrorCallbacks.filter(t=>t!==e.onError)),0===n.listeners.length&&0===n.onErrorCallbacks.length&&n.unsubscribe()}}getHealth(){const e=this.cacheManager.getStats(),t=this.eventQueue.getStats(),n=this.metrics.getHealthMetrics(e.totalCached,e.hotCacheSize,e.warmCacheSize,e.coldCacheSize,this.getMaxMemoryMB()),o=this.determineHealthStatus(t,e),i={connected:this.socket?.connected??!1,reconnectAttempts:this.reconnectAttempts};this.socket?.connected&&(i.lastConnectionTime=new Date);const r=t.queueSize/this.config.maxQueueSize*100,a=e.totalCached/this.config.maxPools*100,s=e.memoryUsedMB/this.getMaxMemoryMB()*100,c=this.generateHealthRecommendations(o,r,a,s,n.metrics.cacheHitRate);return{...n,status:o,websocket:i,recommendations:c,detailedMetrics:{eventQueueUtilization:r,cacheUtilization:a,memoryUtilization:s}}}generateHealthRecommendations(e,t,n,o,i){const r=[];return"failed"===e&&(r.push("🔴 System is in FAILED state - immediate action required"),this.socket?.connected||r.push("Reconnect WebSocket - connection lost"),t>90&&r.push("Reduce incoming event rate or increase maxQueueSize")),"degraded"===e&&(r.push("⚠️ System is DEGRADED - performance may be impacted"),t>75&&r.push(`Queue utilization ${t.toFixed(1)}% - consider increasing maxQueueSize`),o>80&&r.push(`Memory usage ${o.toFixed(1)}% - consider reducing cache size or memory profile`)),"healthy"===e&&(i<50&&r.push(`Cache hit rate ${i.toFixed(1)}% is low - consider warming more pools`),o>50&&r.push("Memory usage is moderate - monitor for growth trends"),t>50&&r.push("Queue utilization is elevated - monitor for bottlenecks")),r}getSummary(){return{...this.metrics.getSummary(),queueStats:this.eventQueue.getStats(),cacheStats:this.cacheManager.getStats()}}startBackgroundWarming(){if(this.warmingIntervalHandle)return;const e=this.config.refreshIntervalMs;this.warmingIntervalHandle=setInterval(()=>{this.performBackgroundWarming().catch(e=>{this.logger.error("Background warming error:",e)})},e),this.logger.debug(`Background warming started (interval: ${e}ms)`)}stopBackgroundWarming(){this.warmingIntervalHandle&&(clearInterval(this.warmingIntervalHandle),this.warmingIntervalHandle=null,this.logger.debug("Background warming stopped"))}async performBackgroundWarming(){const e=this.cacheManager.getStats();if(0!==e.totalCached)try{await this.cacheManager.refreshWarmAndHotTiers(),this.logger.debug(`Background warming completed: ${e.totalCached} pools in cache (hot: ${e.hotCacheSize}, warm: ${e.warmCacheSize})`)}catch(e){this.logger.debug("Background warming encountered an error:",e)}}async shutdown(){this.stopBackgroundWarming(),await this.unsubscribe(),await this.eventQueue.shutdown(),this.cacheManager.clear(),this.metrics.reset()}setupConnectionMonitoring(){this.socket&&(this.socket.on("disconnect",()=>{this.logger.warn("WebSocket disconnected"),this.notifyError(new Error("WebSocket disconnected")),this.config.autoRecover&&this.attemptReconnection().catch(e=>{this.logger.error("Reconnection failed:",e)})}),this.socket.on("connect_error",e=>{this.logger.error("WebSocket connection error:",e),this.notifyError(e instanceof Error?e:new Error(String(e)))}))}async attemptReconnection(){if(this.logger.debug(`Reconnection attempt ${this.reconnectAttempts+1}/${this.maxReconnectAttempts}`),this.reconnectAttempts>=this.maxReconnectAttempts)return this.logger.error(`Max reconnection attempts (${this.maxReconnectAttempts}) exceeded - performing full reset`),void await this.performFullReset();this.reconnectAttempts<2&&(0===this.reconnectAttempts?this.logger.debug("Tier 1: Quick reconnect"):(this.logger.debug(`Tier 2: Exponential backoff (${this.reconnectDelayMs}ms)`),await new Promise(e=>setTimeout(e,this.reconnectDelayMs)),this.reconnectDelayMs=Math.min(2*this.reconnectDelayMs,3e4)));try{this.socket?.disconnect&&(this.socket.disconnect(),this.logger.debug("Disconnected for reconnection")),this.socket?.connect?.(),this.logger.debug("Reconnection initiated"),this.reconnectAttempts++}catch(e){throw this.logger.error("Failed to initiate reconnection:",e),e}}async performFullReset(){this.logger.warn("Performing full system reset due to connection failures"),this.stopBackgroundWarming();const e=this.getHealth();this.logger.debug("System state before reset:",{status:e.status,queueSize:e.eventProcessing.eventsProcessed,cachedPools:e.pools.totalMonitored,memory:`${e.memory.usedMB}MB / ${e.memory.maxMB}MB`,cacheHitRate:`${e.metrics.cacheHitRate.toFixed(2)}%`}),this.cacheManager.clear(),this.metrics.reset(),this.reconnectAttempts=0,this.reconnectDelayMs=1e3,this.notifyError(new Error("System reset: connection lost and recovery failed - please restart monitoring")),this.logger.info("System reset complete - ready for restart")}setupWebSocketListener(e){if(this.listenerRegistered)return void this.logger.debug("WebSocket listener already registered");const t=this;this.handleSwapEvent=(n,...o)=>{try{const n=Date.now(),i=o[0],r=t.eventExtractor.walkPayloadForSwaps(i,t.seenTransactions);if(0===r.length)return;t.logger.debug(`Extracted ${r.length} swaps from payload`);for(const n of r){if(t.filterSwap(n,e)){t.eventQueue.enqueue(n)||t.logger.debug(`Swap dropped due to queue overflow: ${n.transactionId}`)}}const a=Date.now()-n;t.metrics.recordEventLatency(a)}catch(e){t.logger.error("Error processing WebSocket payload:",e),t.notifyError(e instanceof Error?e:new Error(String(e)))}},this.socket?(this.socket.onAny(this.handleSwapEvent),this.listenerRegistered=!0,this.setupConnectionMonitoring()):this.logger.warn("Socket not available for listener registration"),this.startBackgroundWarming(),this.logger.debug("WebSocket listener registered for all events")}async unsubscribe(){this.stopBackgroundWarming(),this.handleSwapEvent&&this.listenerRegistered&&this.socket&&(this.socket.offAny(this.handleSwapEvent),this.listenerRegistered=!1),this.isActive=!1,this.listeners=[],this.onErrorCallbacks=[],this.logger.debug("Unsubscribed from swap events")}filterSwap(e,t){if(t.tokenFilter){if(!Pi.containsToken(e.poolKey,t.tokenFilter))return!1}if(t.pairTokens){const[n,o]=t.pairTokens;if(!Pi.containsTokenPair(e.poolKey,n,o))return!1}if(t.feeTierFilter){if(Pi.normalizeFee(t.feeTierFilter)!==e.feeTier)return!1}return!t.userFilter||e.user===t.userFilter}async processSwapEvent(e){const t=Date.now();try{const n=this.cacheManager.updatePoolWithSwapDelta(e.poolKey,e.direction,e.amount0,e.amount1,e);e.poolStateUpdated=n;const o=Date.now()-t;this.metrics.recordEventLatency(o);for(const t of this.listeners)try{const n=t(e);n instanceof Promise&&await n}catch(t){this.logger.error(`Listener error for swap ${e.transactionId}:`,t)}}catch(t){this.logger.error(`Failed to process swap ${e.transactionId}:`,t),this.notifyError(t instanceof Error?t:new Error(String(t)))}}notifyError(e){for(const t of this.onErrorCallbacks)try{t(e)}catch(e){this.logger.error("Error in error callback:",e)}}determineHealthStatus(e,t){return!this.socket?.connected||e.queueSize>.9*this.config.maxQueueSize?"failed":e.queueSize>.75*this.config.maxQueueSize||t.memoryUsedMB>.9*this.getMaxMemoryMB()?"degraded":"healthy"}getMaxMemoryMB(){switch(this.config.memoryProfile){case"conservative":return 55;case"aggressive":return 530;default:return 250}}applyDefaults(e){return{memoryProfile:e.memoryProfile??"moderate",maxPools:e.maxPools??500,softLimit:e.softLimit??200,preloadTopN:e.preloadTopN??200,warmingTimeoutMs:e.warmingTimeoutMs??3e4,refreshIntervalMs:e.refreshIntervalMs??3e5,maxQueueSize:e.maxQueueSize??1e4,batchSize:e.batchSize??100,maxConcurrent:e.maxConcurrent??10,autoRecover:e.autoRecover??!0,maxParallelRefetch:e.maxParallelRefetch??20,enableDeltaOptimization:e.enableDeltaOptimization??!0,enableOfflineQuotes:e.enableOfflineQuotes??!0,metricsEnabled:e.metricsEnabled??!0,debug:e.debug??!1}}}class Oi{constructor(e){this.map=new Map,this.maxSize=e}has(e){return this.map.has(e)}add(e){if(this.map.has(e))return this.map.delete(e),this.map.set(e,Date.now()),this;if(this.map.size>=this.maxSize){const e=this.map.keys().next().value;this.map.delete(e)}return this.map.set(e,Date.now()),this}delete(e){return this.map.delete(e)}clear(){this.map.clear()}get size(){return this.map.size}}function _i(e){const t=function(e){const t=jt(e);return t.success?[]:t.errors||["Unknown validation error"]}(e);if(t.length>0)throw new Error(`LaunchTokenData validation failed:\n${t.map(e=>`- ${e}`).join("\n")}`)}const Bi="/api/asset/launchpad-contract/CallNativeTokenIn",Ui="/api/asset/launchpad-contract/CallNativeTokenOut",Mi="/api/asset/launchpad-contract/CallMemeTokenIn",Ri="/api/asset/launchpad-contract/CallMemeTokenOut";class qi extends s{constructor(e){super(),this.tokenName=e.tokenName,this.tokenSymbol=e.tokenSymbol,this.tokenDescription=e.tokenDescription,this.tokenImage=e.tokenImage,this.preBuyQuantity=e.preBuyQuantity,e.websiteUrl&&(this.websiteUrl=e.websiteUrl),e.telegramUrl&&(this.telegramUrl=e.telegramUrl),e.twitterUrl&&(this.twitterUrl=e.twitterUrl),this.tokenCategory=e.tokenCategory,this.tokenCollection=e.tokenCollection,this.uniqueKey=e.uniqueKey,e.reverseBondingCurveConfiguration&&(this.reverseBondingCurveConfiguration=e.reverseBondingCurveConfiguration)}}function Ki(e){if(!e||"object"!=typeof e)return!1;const t=e;return"number"==typeof t.Status&&void 0!==t.Data&&"object"==typeof t.Data&&null!==t.Data&&"string"==typeof t.Data.calculatedQuantity&&void 0!==t.Data.extraFees&&"object"==typeof t.Data.extraFees&&null!==t.Data.extraFees&&"string"==typeof t.Data.extraFees.reverseBondingCurve&&"string"==typeof t.Data.extraFees.transactionFees}const Gi={NATIVE:"native",EXACT:"exact"},Wi={RECENT:"recent",POPULAR:"popular"},ji={LOCAL:"local",EXTERNAL:"external"};class Vi{static calculateBuyWithExact(e,t){const n=parseFloat(e),o=parseFloat(t),{BASE_PRICE:i,PRICE_SCALING_FACTOR:r,TRADING_FEE_FACTOR:a,GAS_FEE:s}=bi,c=this.roundUp(i*(Math.exp((o+n)*r)-Math.exp(o*r))/r,8),l=new u(c).multipliedBy(a).toFixed();return{amount:c.toString(),reverseBondingCurveFee:"0",transactionFee:l,gasFee:s}}static calculateBuyWithNative(e,t){const n=parseFloat(e),o=parseFloat(t),{BASE_PRICE:i,PRICE_SCALING_FACTOR:r,TRADING_FEE_FACTOR:a,GAS_FEE:s}=bi,c=Math.log(n*r/i+Math.exp(o*r))/r-o,l=new u(c).multipliedBy(a).toFixed();return{amount:c.toString(),reverseBondingCurveFee:"0",transactionFee:l,gasFee:s}}static calculateSellWithExact(e,t,n,o,i){const r=parseFloat(e),a=parseFloat(t),s=parseFloat(n),{BASE_PRICE:c,PRICE_SCALING_FACTOR:l,TRADING_FEE_FACTOR:d,GAS_FEE:h}=bi,g=c*(Math.exp(a*l)-Math.exp((a-r)*l))/l,p=new u(g),m=o+a/s*(i-o),f=p.multipliedBy(m).toFixed(8,u.ROUND_UP),y=p.multipliedBy(d).toFixed();return{amount:g.toString(),reverseBondingCurveFee:f,transactionFee:y,gasFee:h}}static calculateSellWithNative(e,t,n,o,i){const r=parseFloat(e),a=parseFloat(t),s=parseFloat(n),{BASE_PRICE:c,PRICE_SCALING_FACTOR:l,TRADING_FEE_FACTOR:d,GAS_FEE:h}=bi;if(r>=c*(Math.exp(a*l)-1)/l){const e=new u(r),t=o+a/s*(i-o),n=e.multipliedBy(t).toFixed(8,u.ROUND_UP),c=e.multipliedBy(d).toFixed();return{amount:a.toString(),reverseBondingCurveFee:n,transactionFee:c,gasFee:h}}const g=a-Math.log(Math.exp(a*l)-r*l/c)/l,p=new u(r),m=o+a/s*(i-o),f=p.multipliedBy(m).toFixed(8,u.ROUND_UP),y=p.multipliedBy(d).toFixed();return{amount:g.toString(),reverseBondingCurveFee:f,transactionFee:y,gasFee:h}}static roundUp(e,t){const n=Math.pow(10,t);return Math.ceil(e*n)/n}}class zi{constructor(e,t,n,o,i,r,a="local"){this.http=e,this.tokenResolver=t,this.logger=n,this.bundleHttp=o,this.galaChainHttp=i,this.dexApiHttp=r,this.defaultCalculateAmountMode=a,this.metadataCache=new Ti}addIfDefined(e,t,n){return void 0!==n&&(e[t]=n),e}async uploadImageByTokenName(e){const{tokenName:t,options:n}=e;dn(t);const o=`${t}.png`;lo(n.file,o,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&n.file instanceof File)e.append("image",n.file);else{if(!Buffer.isBuffer(n.file))throw ne("file","a File object (browser) or Buffer (Node.js)");{const o=`${n.tokenName||t}.png`,i=new Blob([n.file],{type:"image/png"});e.append("image",i,o)}}const o=await this.http.request({method:"POST",url:`/launchpad/upload-image?tokenName=${encodeURIComponent(n.tokenName||t)}`,data:e,headers:{}});if(!0===o.error||200!==o.status||!o.data?.imageUrl)throw oe(o.message||"Image upload failed - no URL returned",o.status);return o.data.imageUrl}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw ie("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.","FormData");throw e}}async fetchPoolsFromAPI(e){hn(e),e.tokenName&&dn(e.tokenName);const t={page:e.page.toString(),limit:e.limit.toString()};void 0!==e.type&&(t.type=e.type),void 0!==e.tokenName&&(t.tokenName=e.tokenName),void 0!==e.search&&(t.search=e.search);const n=M(t),o=await this.http.get("/launchpad/fetch-pool",n);if(!0===o.error||200!==o.status||!o.data)throw oe(o.message||"Failed to fetch pools",o.status);let i=[];const r=(await import("bignumber.js")).default;if(o.data.tokens)if(Array.isArray(o.data.tokens))i=o.data.tokens.map(e=>{const t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",o=!new r(t).isZero()||!new r(n).isZero();return{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:o,createdAt:e.created_at||e.createdAt||""}});else{const e=o.data.tokens,t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",a=!new r(t).isZero()||!new r(n).isZero();i=[{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:a,createdAt:e.created_at||e.createdAt||""}]}else o.data.pools&&Array.isArray(o.data.pools)&&(i=o.data.pools.map(e=>{const t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",o=!new r(t).isZero()||!new r(n).isZero();return{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:o,createdAt:e.created_at||e.createdAt||""}}));const{extractMetadataFromPoolData:a,isValidPoolForCaching:s}=await Promise.resolve().then(function(){return Ir});i.forEach(e=>{if(!s(e))return void this.logger.debug("Skipping pool with invalid structure for caching",e);const t=a(e,this.logger);t&&this.warmCacheFromPoolData(e.tokenName,t)});const c=o.data.count??o.data.total??0,l=o.data.page??e.page??1,u=o.data.limit??e.limit??10,d=u>0?Math.ceil(c/u):1;return{pools:i,page:l,limit:u,total:c,totalPages:d,hasNext:l<d,hasPrevious:l>1}}async _getAmount(e){if(pn(e),!this.galaChainHttp)throw ie("GalaChain client not configured. Direct GalaChain calls require galaChainHttp client.","galaChainHttp");const{endpoint:t,body:n}=((e,t,n,o)=>{if("NATIVE"===e&&"IN"===t)return{endpoint:Bi,body:{vaultAddress:n,tokenQuantity:o,IsPreMint:!1}};if("NATIVE"===e&&"OUT"===t)return{endpoint:Ui,body:{vaultAddress:n,tokenQuantity:o,IsPreMint:!1}};if("MEME"===e&&"IN"===t)return{endpoint:Mi,body:{vaultAddress:n,nativeTokenQuantity:o,IsPreMint:!1}};if("MEME"===e&&"OUT"===t)return{endpoint:Ri,body:{vaultAddress:n,nativeTokenQuantity:o,IsPreMint:!1}};throw ne("type-method","one of: NATIVE-IN, NATIVE-OUT, MEME-IN, MEME-OUT")})(e.type,e.method,e.vaultAddress,e.amount);try{const e=await this.galaChainHttp.post(t,n);if(!Ki(e))throw oe("Malformed response data from GalaChain gateway");if(1!==e.Status)throw oe(`GalaChain calculation failed with status ${e.Status}`,e.Status);const{calculatedQuantity:o,extraFees:i}=e.Data;return{amount:o,reverseBondingCurveFee:i.reverseBondingCurve,transactionFee:i.transactionFees,gasFee:"1"}}catch(o){throw this.logger.error(`GalaChain ${e.type}-${e.method} operation failed:`,{endpoint:t,requestBody:n,error:o instanceof Error?o.message:o}),o}}async checkPool(e){gn(e),e.tokenName&&dn(e.tokenName);const t=M(e),n=await this.http.get("/launchpad/check-pool",t);if(!0===n.error||200!==n.status)throw oe(n.message||"Failed to check pool",n.status);const o=n.data;return e.symbol?o?.isSymbolExist??!1:e.tokenName?o?.isNameExist??!1:o?.exists??!1}async fetchVolumeData(e){if(!jn(e))throw new G("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:n,to:o,resolution:i}=e;if(dn(t),!n||!o||!i)throw new G("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const r={tokenName:t,from:n,to:o,resolution:i};mn(r);const a=M(r),s=await this.http.get("/launchpad/get-graph-data",a);if(!0===s.error||200!==s.status||!s.data)throw oe(s.message||"Failed to fetch graph data",s.status);return{dataPoints:s.data}}async fetchPools(e={}){let t;"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const n={page:e.page||1,limit:e.limit||10};return e.search&&(n.search=e.search),e.tokenName&&(n.tokenName=e.tokenName),t&&(n.type=t),this.fetchPoolsFromAPI(n)}async isTokenNameAvailable(e){try{return!await this.checkPool({tokenName:e})}catch{return!1}}async isTokenSymbolAvailable(e){try{return!await this.checkPool({symbol:e})}catch{return!1}}async calculateBuyAmount(e){if(!e||"object"!=typeof e)throw new G("Invalid options provided. Expected an options object.","options","INVALID_OPTIONS");const{tokenName:t,amount:n,type:o,currentSupply:i}=e,r=e.mode??this.defaultCalculateAmountMode;if("local"!==r&&"external"!==r)throw new G(`Invalid calculation mode "${r}". Must be "local" or "external".`,"mode","INVALID_CALCULATION_MODE");if(!t||"string"!=typeof t)throw new G("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Gi.NATIVE&&o!==Gi.EXACT)throw new G('Type must be either "native" or "exact"',"type","INVALID_TYPE");return"external"===r?this.calculateBuyAmountExternal({tokenName:t,amount:n,type:o}):this.calculateBuyAmountLocal(this.addIfDefined({tokenName:t,amount:n,type:o},"currentSupply",i))}async calculateBuyAmountExternal(e){const{tokenName:t,amount:n,type:o}=e,i=await this.tokenResolver.resolveTokenToVault(t);if(!i)throw new G(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return o===Gi.EXACT?this._getAmount({type:"NATIVE",method:"IN",vaultAddress:i,amount:n}):this._getAmount({type:"MEME",method:"OUT",vaultAddress:i,amount:n})}async calculateSellAmount(e){const{tokenName:t,amount:n,type:o,currentSupply:i,maxSupply:r,reverseBondingCurveMaxFeeFactor:a,reverseBondingCurveMinFeeFactor:s}=e,c=e.mode??this.defaultCalculateAmountMode;if("local"!==c&&"external"!==c)throw new G(`Invalid calculation mode "${c}". Must be "local" or "external".`,"mode","INVALID_CALCULATION_MODE");if(!t||"string"!=typeof t)throw new G("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Gi.EXACT&&o!==Gi.NATIVE)throw new G('Type must be either "exact" or "native"',"type","INVALID_TYPE");if("external"===c)return this.calculateSellAmountExternal({tokenName:t,amount:n,type:o});{const e={tokenName:t,amount:n,type:o,...void 0!==i&&{currentSupply:i},...void 0!==r&&{maxSupply:r},...void 0!==a&&{reverseBondingCurveMaxFeeFactor:a},...void 0!==s&&{reverseBondingCurveMinFeeFactor:s}};return this.calculateSellAmountLocal(e)}}async calculateSellAmountExternal(e){const{tokenName:t,amount:n,type:o}=e,i=await this.tokenResolver.resolveTokenToVault(t);if(!i)throw new G(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return o===Gi.EXACT?this._getAmount({type:"NATIVE",method:"OUT",vaultAddress:i,amount:n}):this._getAmount({type:"MEME",method:"IN",vaultAddress:i,amount:n})}async calculateBuyAmountLocal(e){const{tokenName:t,amount:n,type:o,currentSupply:i}=e;if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Gi.NATIVE&&o!==Gi.EXACT)throw new G('Type must be either "native" or "exact"',"type","INVALID_TYPE");void 0!==i&&yn(i,"currentSupply");const r=!i;if(r&&!t)throw new G("Token name is required when currentSupply is not provided","tokenName","MISSING_TOKEN_NAME");t&&dn(t);let a=i;if(r){a=(await this.fetchPoolDetailsForCalculation(t)).currentSupply}return o===Gi.EXACT?Vi.calculateBuyWithExact(n,a):Vi.calculateBuyWithNative(n,a)}async calculateSellAmountLocal(e){const{tokenName:t,amount:n,type:o,currentSupply:i,maxSupply:r,reverseBondingCurveMaxFeeFactor:a,reverseBondingCurveMinFeeFactor:s}=e;if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Gi.EXACT&&o!==Gi.NATIVE)throw new G('Type must be either "exact" or "native"',"type","INVALID_TYPE");void 0!==i&&yn(i,"currentSupply");const c=!i||!r||void 0===a||void 0===s;if(c&&!t)throw new G("Token name is required when currentSupply, maxSupply, or fee factors are not provided","tokenName","MISSING_TOKEN_NAME");t&&dn(t);let l=i,u=r,d=a,h=s;if(c&&t){const e=this.metadataCache.get(t);u=u??this.metadataCache.getMaxSupply(t),d=d??e?.reverseBondingCurveMaxFeeFactor,h=h??e?.reverseBondingCurveMinFeeFactor,l||(l=await this.fetchCurrentSupply(t));if(void 0===d||void 0===h){const e=await this.fetchPoolDetailsForCalculation(t);d=d??e.reverseBondingCurveMaxFeeFactor,h=h??e.reverseBondingCurveMinFeeFactor}}return o===Gi.EXACT?Vi.calculateSellWithExact(n,l,u,h,d):Vi.calculateSellWithNative(n,l,u,h,d)}async calculateBuyAmountForGraduation(e){const t="string"==typeof e?{tokenName:e}:e;if("object"==typeof e&&!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&Wn(t)&&Kn(t,"currentSupply")}(e))throw new Error("Invalid CalculateBuyAmountForGraduationOptions provided");const{tokenName:n,calculateAmountMode:o,currentSupply:i}=t;dn(n);const r=await this.tokenResolver.resolveTokenToVault(n);if(!r)throw new G(K(n),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const a=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:r});if(1!==a.Status)throw oe(`Failed to fetch pool details: Status ${a.Status}`,a.Status);const s=a.Data,c=(await import("bignumber.js")).default,l=i??new c(s.maxSupply).minus(s.sellingTokenQuantity).toFixed(),u=s.sellingTokenQuantity;if("0"===u)throw new G(`Token ${n} is already graduated (no tokens remaining in pool)`,"tokenName","ALREADY_GRADUATED");const d={tokenName:n,amount:u,type:"exact",currentSupply:l,...void 0!==o&&{mode:o}};return await this.calculateBuyAmount(d)}async launchToken(e){if(!this.bundleHttp)throw ie("Bundle backend client not configured. LaunchToken requires bundleHttp client.","bundleHttp");_i(e);const t=e.preBuyQuantity||"0";if(isNaN(Number(t))||Number(t)<0)throw new G("Pre-buy quantity must be a valid non-negative number string","preBuyQuantity","INVALID_PRE_BUY_QUANTITY");if(e.reverseBondingCurveConfiguration){const{minFeePortion:t,maxFeePortion:n}=e.reverseBondingCurveConfiguration,o=Number(t),i=Number(n);if(isNaN(o)||isNaN(i)||o<=0||i<=0||o>=i)throw new G("Reverse bonding curve configuration must have valid min/max fee portions with min < max","reverseBondingCurveConfiguration","INVALID_BONDING_CURVE_CONFIG")}let n="";if(e.tokenImage)if(e.tokenImage instanceof File||Buffer.isBuffer(e.tokenImage)){const t=await this.uploadImageByTokenName({tokenName:e.tokenName,options:{file:e.tokenImage,tokenName:e.tokenName}});if(!t)throw oe("Image upload failed: No URL returned");n=t}else"string"==typeof e.tokenImage&&(n=e.tokenImage);const o=`galaswap - operation - ${N()}-${Date.now()}-${this.http.getAddress()}`,i={tokenName:e.tokenName.trim(),tokenSymbol:e.tokenSymbol.trim().toUpperCase(),tokenDescription:e.tokenDescription.trim(),tokenImage:n.trim(),preBuyQuantity:t.toString(),tokenCategory:e.tokenCategory||"Unit",tokenCollection:e.tokenCollection||"Token",uniqueKey:o};e.websiteUrl?.trim()&&(i.websiteUrl=e.websiteUrl.trim()),e.telegramUrl?.trim()&&(i.telegramUrl=e.telegramUrl.trim()),e.twitterUrl?.trim()&&(i.twitterUrl=e.twitterUrl.trim()),e.reverseBondingCurveConfiguration&&(i.reverseBondingCurveConfiguration={minFeePortion:e.reverseBondingCurveConfiguration.minFeePortion.toString(),maxFeePortion:e.reverseBondingCurveConfiguration.maxFeePortion.toString()});const a=new qi(i),s=await this.http.signWithGalaChain("CreateSale",a,r.SIGN_TYPED_DATA),{signature:c,types:l,domain:u,prefix:d}=s,h={tokenName:a.tokenName,tokenSymbol:a.tokenSymbol,tokenDescription:a.tokenDescription,tokenImage:a.tokenImage,preBuyQuantity:a.preBuyQuantity,...a.websiteUrl&&{websiteUrl:a.websiteUrl},...a.telegramUrl&&{telegramUrl:a.telegramUrl},...a.twitterUrl&&{twitterUrl:a.twitterUrl},tokenCategory:a.tokenCategory,tokenCollection:a.tokenCollection,uniqueKey:a.uniqueKey,signature:c,types:l,domain:u,...d&&{prefix:d},...a.reverseBondingCurveConfiguration&&{reverseBondingCurveConfiguration:a.reverseBondingCurveConfiguration}},g=`${e.tokenName.trim()}$Unit$none$none`,p="GALA$Unit$none$none";let m;if(parseFloat(t)>0){const e=`$service$${g}$launchpad`;m=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${p}$${e}`,`$tokenBalance$${p}$${e}`]}else{const e=`$service$${g}$launchpad`;m=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`]}const f={signedDto:h,stringsInstructions:m,method:"CreateSale"},y=await this.bundleHttp.post("/bundle",f);if(y.error||!y.data)throw oe(y.message||"Token launch failed");return y.data}async fetchTokenDistribution(e){if(!e)throw te("tokenName","Token name");dn(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");this.metadataCache.set(e,{vaultAddress:t});const n=encodeURIComponent(t),o=await this.http.get(`/holders/${n}`);if(!0===o.error||200!==o.status||!o.data)throw oe(o.message||"Failed to fetch token distribution",o.status);const i=o.data;if(!Array.isArray(i))throw oe("Invalid API response: expected array of holders",o.status);for(const e of i){if(!e.owner||"string"!=typeof e.owner)throw oe("Invalid holder data: missing or invalid owner field",o.status);if(!e.quantity||"string"!=typeof e.quantity)throw oe("Invalid holder data: missing or invalid quantity field",o.status);const t=parseFloat(e.quantity);if(isNaN(t)||!isFinite(t))throw oe(`Invalid holder quantity: "${e.quantity}"`,o.status)}const r=i.reduce((e,t)=>e.plus(t.quantity),new u(0));return{holders:i.map(e=>{const t=new u(e.quantity),n=r.isZero()?0:t.dividedBy(r).multipliedBy(100).toNumber();return{address:e.owner,balance:e.quantity,percentage:n}}),totalSupply:r.toFixed(),totalHolders:i.length,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw te("tokenName","Token name");dn(e);const t=await this.http.get("/launchpad/get-badge/",{tokenName:e});if(t.error||!t.data)throw oe(t.message||"Failed to fetch token badges");return{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadgeByTokenName(e){const{tokenName:t,badgeType:n,badgeName:o}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const i=("volume"===n?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===o);return i?.isActive||!1}catch{return!1}}async calculateInitialBuyAmount(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.nativeTokenQuantity&&(void 0===t.vaultAddress||"string"==typeof t.vaultAddress)}(e))throw new G("Invalid pre-mint calculation data","data","INVALID_PRE_MINT_DATA");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not available. Please initialize SDK with galaChainBaseUrl.","galaChainHttp");try{const t={vaultAddress:"service|testToken",nativeTokenQuantity:e.nativeTokenQuantity,IsPreMint:!0},n=await this.galaChainHttp.post("/api/asset/launchpad-contract/CallMemeTokenOut",t);if(!Ki(n))throw oe("Malformed response data from GalaChain gateway");if(1!==n.Status)throw oe(`GalaChain calculation failed with status ${n.Status}`,n.Status);const{calculatedQuantity:o,extraFees:i}=n.Data;return{amount:o,reverseBondingCurveFee:i.reverseBondingCurve,transactionFee:i.transactionFees,gasFee:"1"}}catch(e){if(e instanceof Error){const t=new Error(`Pre-mint calculation failed: ${e.message}`);throw e.stack&&(t.stack=e.stack),t}throw new Error(`Pre-mint calculation failed: ${String(e)}`)}}async fetchPoolDetailsForCalculation(e){const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const n=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:t});if(1!==n.Status)throw oe(`Failed to fetch pool details: Status ${n.Status}`,n.Status);const o=n.Data,i=new(0,(await import("bignumber.js")).default)(o.maxSupply).minus(o.sellingTokenQuantity).toFixed(),r=o.sellingTokenQuantity,a=o.maxSupply;let s=.5,c=0;o.reverseBondingCurveConfiguration?(s=parseFloat(o.reverseBondingCurveConfiguration.maxFeePortion),c=parseFloat(o.reverseBondingCurveConfiguration.minFeePortion)):this.logger.debug(`Pool details missing reverseBondingCurveConfiguration for token ${e}, using defaults (min: 0.0, max: 0.5)`);const l=s-c;return this.metadataCache.set(e,{maxSupply:a,reverseBondingCurveMaxFeeFactor:s,reverseBondingCurveMinFeeFactor:c,reverseBondingCurveNetFeeFactor:l}),{currentSupply:i,remainingTokens:r,maxSupply:a,reverseBondingCurveMaxFeeFactor:s,reverseBondingCurveMinFeeFactor:c,reverseBondingCurveNetFeeFactor:l}}async fetchCurrentSupply(e){dn(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const n=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:t});if(1!==n.Status)throw oe(`Failed to fetch pool details: Status ${n.Status}`,n.Status);const o=n.Data,i=new(0,(await import("bignumber.js")).default)(o.maxSupply).minus(o.sellingTokenQuantity).toFixed(),r=o.maxSupply;return this.metadataCache.set(e,{maxSupply:r}),i}getAddress(){return this.http.getAddress()}formatAddressForBackend(e){return fn(e)}validateTokenName(e){return dn(e)}validatePagination(e){return hn(e)}async fetchTokenPrice(e){if(!this.dexApiHttp)throw ie("DEX API client not configured. Token price fetching requires dexApiHttp client.","dexApiHttp");if(!e||Array.isArray(e)&&0===e.length)throw te("symbols","At least one symbol");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.dexApiHttp.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),n=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&n.push({symbol:e.symbol,price:e.currentPrices.usd})}),n}catch(e){throw oe(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`,void 0,e instanceof Error?e:void 0)}}async fetchLaunchpadTokenSpotPrice(e){const t="string"==typeof e?{tokenName:e}:e;if("object"==typeof e&&!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return qn(t,"tokenName")&&Wn(t)&&Kn(t,"currentSupply")}(e))throw new Error("Invalid FetchLaunchpadTokenSpotPriceOptions provided");const{tokenName:n,calculateAmountMode:o,currentSupply:i}=t;if(!n||"string"!=typeof n)throw te("tokenName","Token name (string)");try{const e={tokenName:n,amount:"1",type:"native",...void 0!==o&&{mode:o},...void 0!==i&&{currentSupply:i}},t=await this.calculateBuyAmount(e),r=(await this.fetchTokenPrice("GALA"))[0];if(!r)throw oe("GALA price not available");const a=Number(t.amount);if(a<=0)throw new G(`Invalid token amount calculation: ${a}`,"amount","INVALID_CALCULATION");const s=r.price/a;return{symbol:n.toUpperCase(),price:s}}catch(e){if(e instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${n}: ${e.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${n}: ${String(e)}`)}}warmCacheFromPoolData(e,t){this.metadataCache.warmFromPoolData(e,t)}getCacheStats(){return this.metadataCache.stats()}clearCache(e){this.metadataCache.clear(e)}}const Hi={PROD:{launchpadBaseUrl:"https://lpad-backend-prod1.defi.gala.com",galaChainBaseUrl:"https://gateway-mainnet.galachain.com",bundleBaseUrl:"https://bundle-backend-prod1.defi.gala.com",webSocketUrl:"https://bundle-backend-prod1.defi.gala.com",dexApiBaseUrl:"https://dex-api-platform-dex-prod-gala.gala.com",dexBackendBaseUrl:"https://dex-backend-prod1.defi.gala.com",launchpadFrontendUrl:"https://lpad-frontend-prod1.defi.gala.com"},STAGE:{launchpadBaseUrl:"https://lpad-backend-dev1.defi.gala.com",galaChainBaseUrl:"https://galachain-gateway-chain-platform-stage-chain-platform-eks.stage.galachain.com",bundleBaseUrl:"https://bundle-backend-dev1.defi.gala.com",webSocketUrl:"https://bundle-backend-dev1.defi.gala.com",dexApiBaseUrl:"https://dex-api-platform-dex-stage-gala.gala.com",dexBackendBaseUrl:"https://dex-backend-dev1.defi.gala.com",launchpadFrontendUrl:"https://lpad-frontend-test1.defi.gala.com"}};function Qi(e){return Hi[e]}class Xi extends Error{constructor(e,t){super(e),this.cause=t,this.name="WebSocketError"}}class Zi extends Xi{constructor(e,t){super(`WebSocket confirmation timeout for transaction ${e} after ${t}ms`),this.name="WebSocketTimeoutError"}}class Yi extends Error{constructor(e,t,n){super(`Transaction ${e} failed with status: ${t}${n?` - ${n}`:""}`),this.transactionId=e,this.status=t,this.name="TransactionFailedError"}}function Ji(e,t){if(!e)throw new Xi(`Invalid WebSocket response received for transaction ${t}: response is null or undefined`);if("object"!=typeof e)throw new Xi(`Invalid WebSocket response received for transaction ${t}: expected object, got ${typeof e}`);if(!Object.prototype.hasOwnProperty.call(e,"status")&&!Object.prototype.hasOwnProperty.call(e,"Status"))throw new Xi(`Invalid WebSocket response received for transaction ${t}: missing status field`)}function er(e,t,n,o){Ji(e,t);const i=e,r=i.data||{};if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(void 0!==t.inputQuantity&&"string"!=typeof t.inputQuantity||void 0!==t.outputQuantity&&"string"!=typeof t.outputQuantity||void 0!==t.totalFees&&"string"!=typeof t.totalFees||void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress)}(r))throw new Xi(`Invalid trade data received for transaction ${t}`);const a={transactionId:t,type:n,method:"native"===o.type?"native":"exact",inputAmount:r.inputQuantity||o.amount,outputAmount:r.outputQuantity||o.expectedAmount||"0",totalFees:r.totalFees||"0",tokenName:o.tokenName,vaultAddress:r.vaultAddress||"",timestamp:Date.now()};return void 0!==i.blockHash&&(a.blockHash=i.blockHash),void 0!==i.gasUsed&&(a.gasUsed=i.gasUsed),void 0!==o.slippageToleranceFactor&&(a.slippageTolerance=o.slippageToleranceFactor),a}const tr="4.0.1-beta.9";class nr{constructor(e){this.logger=e||new U({debug:!1,context:"LiquidityEventExtractor"})}walkPayloadForLiquidityEvents(e,t){const n=[],o=new WeakSet,i=(e,r=0)=>{if(r>50)this.logger.debug("Payload nesting exceeded maximum depth of 50");else if(e&&"string"!=typeof e&&"object"==typeof e){if(o.has(e))return;o.add(e);const a=this.extractLiquidityFromObject(e);a&&!t.has(a.transactionId)&&(n.push(a),t.add(a.transactionId));for(const t of Object.values(e))i(t,r+1)}};return i(e,0),n}extractLiquidityFromObject(e){const t=this.extractTransactionId(e);if(!t)return null;const n=e.Data,o=n&&"object"==typeof n&&!Array.isArray(n)?n:e,i=this.extractPositionId(o),r=this.extractPoolHash(o),a=this.extractAmounts(o),s=this.extractUserAddress(o),c=this.extractPoolFee(o);if(!(i&&r&&a&&s&&null!==c))return null;const l=this.extractPoolAlias(o),u=this.extractUserBalanceDelta(o),d=this.extractTimestamp(o),h=u?.token0Balance?.collection,g=u?.token1Balance?.collection,p={transactionId:t,positionId:i,poolHash:r,poolFee:c,amounts:a,userAddress:s};return void 0!==h&&(p.token0=h),void 0!==g&&(p.token1=g),void 0!==d&&(p.timestamp=d),void 0!==l&&(p.poolAlias=l),void 0!==u&&(p.userBalanceDelta=u),p}extractTransactionId(e){const t=["transactionId","txId","tx_id","hash","txHash","id"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPositionId(e){const t=["positionId","position_id","tokenId","nftId"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolHash(e){const t=["poolHash","pool_hash","poolId","pool"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolAlias(e){const t=["poolAlias","pool_alias"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}}extractAmounts(e){const t=e.amounts;if(Array.isArray(t)&&t.length>=2){const e=String(t[0]).trim(),n=String(t[1]).trim();if(e&&n)return[e,n]}const n=e.amount0||e.amount0Desired,o=e.amount1||e.amount1Desired;return void 0!==n&&void 0!==o?[String(n),String(o)]:null}extractUserAddress(e){const t=["userAddress","user","owner","from","sender","wallet","address"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolFee(e){const t=["poolFee","fee","feeTier","feeTierBps"];for(const n of t){const t=e[n];if("number"==typeof t)return this.normalizeFee(t);if("string"==typeof t){const e=Number(t);if(!Number.isNaN(e))return this.normalizeFee(e)}}return null}normalizeFee(e){return 1===e||1e4===e?1e4:.3===e||3e3===e?3e3:.05===e||500===e?500:Number.isInteger(e)?e:e<1?Math.round(1e4*e):e}extractTimestamp(e){const t=["timeStamp","timestamp","time","createdAt","date"];for(const n of t){const t=e[n];if("number"==typeof t)return t;if("string"==typeof t){const e=new Date(t).getTime();if(!Number.isNaN(e))return e}}}extractUserBalanceDelta(e){const t=e.userBalanceDelta;if(!t||"object"!=typeof t)return;const n=t,o=this.extractBalanceObject(n.token0Balance),i=this.extractBalanceObject(n.token1Balance);if(!o&&!i)return;const r={};return void 0!==o&&(r.token0Balance=o),void 0!==i&&(r.token1Balance=i),r}extractBalanceObject(e){if(!e||"object"!=typeof e)return;const t=e,n=t.collection,o=t.category,i=t.type,r=t.additionalKey,a=t.quantity,s=t.owner;return"string"==typeof n&&"string"==typeof o&&"string"==typeof i&&"string"==typeof r&&"string"==typeof a&&"string"==typeof s?{collection:n,category:o,type:i,additionalKey:r,quantity:a,owner:s}:void 0}}class or{constructor(t){this.wallet=t.wallet;let n=null,o="STAGE";t.env?(o=t.env,n=Qi(t.env)):t.baseUrl?.includes("prod")?(o="PROD",n=Qi("PROD")):(o="STAGE",n=Qi("STAGE")),this.environment=o,this.config={baseUrl:n.launchpadBaseUrl,galaChainBaseUrl:n.galaChainBaseUrl,bundleBaseUrl:n.bundleBaseUrl,webSocketUrl:n.webSocketUrl,dexApiBaseUrl:n.dexApiBaseUrl,dexBackendBaseUrl:n.dexBackendBaseUrl,launchpadFrontendUrl:n.launchpadFrontendUrl,timeout:3e4,debug:!1,...t},this.logger=new U({debug:this.config.debug??!1,context:"LaunchpadSDK"}),this.validateConfiguration(),this.slippageToleranceFactor=void 0===t.slippageToleranceFactor?or.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR:this.parseSlippageToleranceFactor(t.slippageToleranceFactor),this.maxAcceptableReverseBondingCurveFeeSlippageFactor=void 0===t.maxAcceptableReverseBondingCurveFeeSlippageFactor?or.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR:this.parseFeeSlippageFactor(t.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.calculateAmountMode=t.calculateAmountMode||or.DEFAULT_CALCULATE_AMOUNT_MODE,this.pricingConcurrency=t.pricingConcurrency||5,this.auth=new B({wallet:t.wallet,messagePrefix:"Create a GalaChain Wallet"}),this.http=new R(this.auth,this.config),this.galaChainHttp=new R(this.auth,{...this.config,baseUrl:this.config.galaChainBaseUrl}),this.bundleHttp=new R(this.auth,{...this.config,baseUrl:this.config.bundleBaseUrl}),this.dexApiHttp=new R(this.auth,{...this.config,baseUrl:this.config.dexApiBaseUrl}),this.dexBackendHttp=new R(this.auth,{...this.config,baseUrl:this.config.dexBackendBaseUrl}),this.galaChainPublicAxios=e.create({baseURL:this.config.galaChainBaseUrl,timeout:this.config.timeout||3e4,headers:{"Content-Type":"application/json",Accept:"application/json"}}),this.cache=new Ti(t.debug||!1),this.launchpadService=new go(this.http),this.tokenResolverService=new ci(this.launchpadService.poolService),this.launchpadAPI=new zi(this.http,this.tokenResolverService,this.logger,this.bundleHttp,this.galaChainHttp,this.dexApiHttp,this.calculateAmountMode),this.galaChainService=new Wo(this.galaChainHttp,t.wallet,this.tokenResolverService,t.debug||!1,this.galaChainPublicAxios),this.dexService=new jo(this.dexBackendHttp,this.cache,this.galaChainService,t.debug||!1),this.bundleService=new oi(this.bundleHttp,this.tokenResolverService,this.config.debug||!1,t.wallet,t.wallet?this.getAddress():void 0,this.slippageToleranceFactor,this.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.websocketService=new si({url:this.config.webSocketUrl},this.config.debug),this.priceHistoryService=new di(this.dexBackendHttp,this.config.debug||!1,this.tokenResolverService),this.dexQuoteService=new Ei(this.galaChainHttp,this.config.galaChainBaseUrl,t.debug||!1,t.dexQuoteNetworkTimeout||3e4),this.gswapService=new vi({privateKey:t.wallet?.privateKey,getWalletAddress:()=>this.wallet?this.getAddress():void 0,gatewayBaseUrl:this.config.galaChainBaseUrl,bundlerBaseUrl:this.config.bundleBaseUrl,galaChainBaseUrl:this.config.galaChainBaseUrl,dexBackendBaseUrl:this.config.dexBackendBaseUrl,dexBackendHttp:this.dexBackendHttp},this.websocketService,this.dexQuoteService),this.dexPoolService=new Si(this.dexBackendHttp,this.config.dexBackendBaseUrl,this.gswapService,this.pricingConcurrency,t.debug||!1)}createOverrideSdk(e){if(!e||"string"!=typeof e)throw ie("Invalid privateKey: must be a non-empty string","privateKey");if(!e.match(/^0x[a-fA-F0-9]{64}$/))throw ie('Invalid privateKey format: must be "0x" followed by 64 hexadecimal characters',"privateKey");const t=new n(e),o={...this.config,wallet:t};return new or(o)}getAddress(){return this.validateWallet(),this.auth.getAddress()}getEthereumAddress(){return this.validateWallet(),this.wallet.address}validateWallet(){if(!this.wallet)throw new G("Wallet is required for this operation",void 0,"WALLET_REQUIRED");return this.wallet}setWallet(e){if(!e||"object"!=typeof e||!("address"in e))throw new G("Invalid wallet: must be an ethers Wallet instance, received "+typeof e,"wallet","INVALID_WALLET");this.wallet=e,this.auth.setWallet(e)}getWallet(){return this.wallet}hasWallet(){return void 0!==this.wallet}getConfig(){const{wallet:e,...t}=this.config;return{...t,environment:this.environment,slippageToleranceFactor:this.slippageToleranceFactor,maxAcceptableReverseBondingCurveFeeSlippageFactor:this.maxAcceptableReverseBondingCurveFeeSlippageFactor,calculateAmountMode:this.calculateAmountMode}}getVersion(){return tr}getUrlByTokenName(e){const t=this.config.launchpadFrontendUrl;if(!t)throw ie("launchpadFrontendUrl not configured in SDK","launchpadFrontendUrl");return`${t.replace(/\/$/,"")}/buy-sell/${e}`}async fetchPools(e){const t=await this.launchpadService.fetchPools(e||{});return await this.warmCacheFromPools(t.pools),t}async fetchAllPools(e){const t=await this.launchpadService.fetchAllPools(e);return await this.warmCacheFromPools(t.pools),t}async fetchDexPools(e={}){return this.dexPoolService.fetchDexPools(e)}async fetchAllDexPools(e={}){return this.dexPoolService.fetchAllDexPools(e)}async fetchCompositePoolData(e){return this.dexQuoteService.fetchCompositePoolData(e)}async calculateDexPoolQuoteExactAmountLocal(e){return this.dexQuoteService.calculateDexPoolQuoteExactAmountLocal(e)}async calculateDexPoolQuoteExactAmountExternal(e){return this.dexQuoteService.calculateDexPoolQuoteExactAmountExternal(e)}async calculateDexPoolQuoteExactAmount(e,t="local"){return this.dexQuoteService.calculateDexPoolQuoteExactAmount(e,t)}async fetchTokenDistribution(e){return this.launchpadService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.launchpadService.fetchTokenBadges(e)}async fetchTokenPrice(e){const{tokenName:t,tokenId:n,currentSupply:o,calculateAmountMode:i}=e;if(t&&!n){const e={tokenName:t};return o&&(e.currentSupply=o),i&&(e.calculateAmountMode=i),this.fetchLaunchpadTokenSpotPrice(e)}if(n&&!t)try{return await this.dexService.fetchTokenPrice({tokenId:n})}catch(e){const t=function(e){if(no(e)&&e.response)return e.response.status}(e);if(400===t||404===t){this.logger.debug(`DEX spot price not available (HTTP ${t}) for tokenId, attempting launchpad fallback`);try{const t=(await this.fetchTokenDetails(n)).name.trim().toLowerCase();if(!/^[a-z0-9]{3,20}$/.test(t))throw this.logger.error(`Token name extracted from GalaChain doesn't match launchpad format: "${t}"`),e;this.logger.debug(`Falling back to launchpad pricing using extracted token name: "${t}"`);const r={tokenName:t};return o&&(r.currentSupply=o),i&&(r.calculateAmountMode=i),this.fetchLaunchpadTokenSpotPrice(r)}catch(t){throw this.logger.error(`Launchpad fallback failed: ${to(t)}`),e}}throw e}if(!t&&!n)throw te("tokenName or tokenId","Either tokenName (for launchpad tokens) or tokenId (for DEX tokens) is required");throw new G("tokenName and tokenId are mutually exclusive - provide only one","params","INVALID_PARAMS")}async fetchGalaPrice(){return this.fetchTokenPrice({tokenId:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none"}})}async fetchLaunchpadTokenSpotPrice(e){const t="string"==typeof e?{tokenName:e}:e;return this.dexService.fetchLaunchpadTokenSpotPrice(t.tokenName,e=>this.launchpadAPI.calculateBuyAmount(e),e=>this.fetchPoolDetails(e))}async fetchTokenDetails(e){return this.dexService.fetchTokenDetails(e)}async fetchAllDexSeasons(){return this.dexService.fetchAllDexSeasons()}async fetchCurrentDexSeason(){return this.dexService.fetchCurrentDexSeason()}async fetchDexLeaderboardBySeasonId(e){return this.dexService.fetchDexLeaderboardBySeasonId(e)}async fetchCurrentDexLeaderboard(){return this.dexService.fetchCurrentDexLeaderboard()}async fetchDexAggregatedVolumeSummary(){return this.dexService.fetchDexAggregatedVolumeSummary()}async fetchLaunchTokenFee(){return this.galaChainService.fetchLaunchTokenFee()}async fetchTokenClassesWithSupply(e){return this.galaChainService.fetchTokenClassesWithSupply(e)}async fetchPoolDetails(e){const t=await this.resolveVaultAddress(e);if(!t)throw new Error(K(e));const n=(await this.galaChainService.fetchPoolDetails({vaultAddress:t})).Data,o=await this.launchpadAPI.fetchPoolDetailsForCalculation(e);return n.currentSupply=o.currentSupply,n.reverseBondingCurveMaxFeeFactor=o.reverseBondingCurveMaxFeeFactor,n.reverseBondingCurveMinFeeFactor=o.reverseBondingCurveMinFeeFactor,n.reverseBondingCurveNetFeeFactor=o.reverseBondingCurveNetFeeFactor,n.tokenName=e,n}async fetchPoolDetailsForCalculation(e){return this.launchpadAPI.fetchPoolDetailsForCalculation(e)}async isTokenGraduated(e){return(await this.fetchPoolDetails(e)).isGraduated}async fetchVolumeData(e){return this.launchpadService.fetchVolumeData(e)}async fetchTrades(e){return this.launchpadService.fetchTrades(e)}async fetchGalaBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n=t(e)||this.getAddress();return this.galaChainService.fetchGalaBalance({owner:n,collection:"GALA",category:"Unit",additionalKey:"none",type:"none",instance:"0"})}async fetchTokenBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n=t(e.address);if(e.tokenId){const{normalizeToTokenInstanceKey:t}=await Promise.resolve().then(function(){return Bo}),o=t(e.tokenId),{collection:i,category:r,type:a,additionalKey:s}=o;return this.galaChainService.fetchTokenBalance({owner:n,collection:i,category:r,additionalKey:s,type:a,instance:"0"})}if(e.tokenName){const t=(await this.fetchTokensHeld({tokenName:e.tokenName,page:1,limit:1,...n&&{address:n}})).tokens[0];return t?{quantity:t.quantity,collection:t.collection||"Token",category:"Unit",tokenId:`${t.collection||"Token"}|Unit|${t.symbol}|none`,symbol:t.symbol,name:t.name}:null}throw te("tokenId or tokenName","Either tokenId or tokenName")}async fetchComments(e){return this.launchpadService.fetchComments(e)}async calculateBuyAmount(e){return this.launchpadAPI.calculateBuyAmount(e)}async calculateSellAmount(e){return this.launchpadAPI.calculateSellAmount(e)}async calculateBuyAmountLocal(e){return this.launchpadAPI.calculateBuyAmountLocal(e)}async calculateSellAmountLocal(e){return this.launchpadAPI.calculateSellAmountLocal(e)}async calculateBuyAmountExternal(e){return this.launchpadAPI.calculateBuyAmountExternal(e)}async calculateSellAmountExternal(e){return this.launchpadAPI.calculateSellAmountExternal(e)}async calculateBuyAmountForGraduation(e){return this.launchpadAPI.calculateBuyAmountForGraduation(e)}async graduateToken(e){const{tokenName:t,slippageToleranceFactor:n,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,privateKey:i,calculateAmountMode:r,currentSupply:a}=e;let s=t;void 0===r&&void 0===a||(s={tokenName:t,...void 0!==r&&{calculateAmountMode:r},...void 0!==a&&{currentSupply:a}});const c=await this.calculateBuyAmountForGraduation(s),l={tokenName:t,amount:c.amount,type:"exact",expectedAmount:c.amount,maxAcceptableReverseBondingCurveFee:c.reverseBondingCurveFee,slippageToleranceFactor:this.slippageToleranceFactor};return void 0!==n&&(l.slippageToleranceFactor=n),void 0!==o&&(l.maxAcceptableReverseBondingCurveFeeSlippageFactor=o),void 0!==i&&(l.privateKey=i),await this.buy(l)}async calculateInitialBuyAmount(e){const t={nativeTokenQuantity:e};return this.launchpadAPI.calculateInitialBuyAmount(t)}async buy(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.buy(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=(await this.bundleService.buyToken(e)).data,n=t?.transactionId;if(!n)throw re("No transaction ID returned from buy operation");return this.waitForConfirmation(n,t=>er(t,n,"buy",e))}async sell(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.sell(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=(await this.bundleService.sellToken(e)).data,n=t?.transactionId;if(!n)throw re("No transaction ID returned from sell operation");return this.waitForConfirmation(n,t=>er(t,n,"sell",e))}async getBundlerTransactionResult(e){return this.bundleService.getBundlerTransactionResult(e)}async postComment(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.postComment(o)}return this.validateWallet(),this.launchpadService.postComment(e)}async launchToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.launchToken(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=await this.launchpadAPI.launchToken(e);return this.waitForConfirmation(t,n=>{Ji(n,t);const o=n?.data||{};if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress||void 0!==t.tokenStringKey&&"string"!=typeof t.tokenStringKey||void 0!==t.creatorAddress&&"string"!=typeof t.creatorAddress)}(o))throw new Xi(`Invalid launch data received for transaction ${t}`);const i={transactionId:t,vaultAddress:o.vaultAddress||"",tokenStringKey:o.tokenStringKey||"",tokenName:e.tokenName,tokenSymbol:e.tokenSymbol,creatorAddress:o.creatorAddress||this.getAddress(),timestamp:Date.now(),...n.blockHash&&{blockHash:n.blockHash},...n.gasUsed&&{gasUsed:n.gasUsed}};return"string"==typeof e.tokenImage&&(i.tokenImage=e.tokenImage),void 0!==e.preBuyQuantity&&(i.preBuyQuantity=e.preBuyQuantity),i.vaultAddress&&this.tokenResolverService.set(e.tokenName,i.vaultAddress),i})}async uploadTokenImage(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.uploadTokenImage(o)}return this.validateWallet(),this.launchpadService.uploadImageByTokenName(e)}async isTokenNameAvailable(e){return this.launchpadService.isTokenNameAvailable(e)}async isTokenSymbolAvailable(e){return this.launchpadService.isTokenSymbolAvailable(e)}async fetchProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n=t(e)||this.getAddress();return this.launchpadService.fetchProfile(n)}async updateProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={...e,address:t(e.address)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.updateProfile(o)}return this.validateWallet(),this.launchpadService.updateProfile(n)}async uploadProfileImage(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={...e,address:t(e.address)||this.getAddress()};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.uploadProfileImage(o)}return this.validateWallet(),this.launchpadService.uploadProfileImage(n)}async retrieveGalaFromFaucet(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={walletAddress:t(e)||this.getAddress(),amount:"5"};return this.launchpadService.transferFaucets(n)}async fetchTokensHeld(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n=t(e?.address)||this.getAddress(),o={page:e?.page||1,limit:e?.limit||10,address:n};return e?.tokenName&&(o.tokenName=e.tokenName),e?.search&&(o.search=e.search),this.launchpadService.fetchTokensHeld(o)}async fetchTokensCreated(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={type:"DEFI",address:t(e?.address)||this.getAddress(),page:e?.page||1,limit:e?.limit||10};return e?.tokenName&&(n.tokenName=e.tokenName),e?.search&&(n.search=e.search),this.launchpadService.fetchTokenList(n)}async fetchPriceHistory(e){return this.priceHistoryService.fetchPriceHistory(e)}async fetchAllPriceHistory(e){return this.priceHistoryService.fetchAllPriceHistory(e)}async transferGala(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={...e,recipientAddress:t(e.recipientAddress)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.transferGala(o)}return this.validateWallet(),this.galaChainService.transferGala(n)}async transferToken(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn}),n={...e,to:t(e.to)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.transferToken(o)}return this.validateWallet(),this.galaChainService.transferToken(n)}async resolveTokenClassKey(e){return this.galaChainService.resolveTokenClassKey(e)}async lockToken(e){if(e.lockAuthority){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return wn});e={...e,lockAuthority:t(e.lockAuthority)}}if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.lockToken(o)}return this.validateWallet(),this.galaChainService.lockToken(e)}async unlockToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.unlockToken(o)}return this.validateWallet(),this.galaChainService.unlockToken(e)}async resolveVaultAddress(e){return this.tokenResolverService.resolveTokenToVault(e)}getCacheInfo(){return this.launchpadAPI.getCacheStats()}clearCache(e){this.launchpadAPI.clearCache(e)}validateConfiguration(){if(("number"!=typeof this.config.timeout||this.config.timeout<=0||this.config.timeout>3e5)&&(this.logger.warn(`Invalid timeout value: ${this.config.timeout}. Using default 30000ms.`),this.config.timeout=3e4),!this.config.baseUrl)throw ie("baseUrl is required in configuration","baseUrl");if(!this.config.webSocketUrl)throw ie("webSocketUrl is required in configuration","webSocketUrl");try{new URL(this.config.baseUrl)}catch{throw ie(`Invalid baseUrl format: ${this.config.baseUrl}`,"baseUrl")}try{new URL(this.config.webSocketUrl)}catch{throw ie(`Invalid webSocketUrl format: ${this.config.webSocketUrl}`,"webSocketUrl")}if(this.config.galaChainBaseUrl)try{new URL(this.config.galaChainBaseUrl)}catch{throw ie(`Invalid galaChainBaseUrl format: ${this.config.galaChainBaseUrl}`,"galaChainBaseUrl")}if(this.config.bundleBaseUrl)try{new URL(this.config.bundleBaseUrl)}catch{throw ie(`Invalid bundleBaseUrl format: ${this.config.bundleBaseUrl}`,"bundleBaseUrl")}if(this.config.launchpadFrontendUrl)try{new URL(this.config.launchpadFrontendUrl)}catch{throw ie(`Invalid launchpadFrontendUrl format: ${this.config.launchpadFrontendUrl}`,"launchpadFrontendUrl")}}parseSlippageToleranceFactor(e){const t=parseFloat(String(e));return isNaN(t)||t<0||t>1?(this.logger.warn(`Invalid slippage tolerance factor: ${e}, using default: ${or.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR}`),or.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR):t}parseFeeSlippageFactor(e){const t=parseFloat(String(e));return isNaN(t)||t<0||t>1?(this.logger.warn(`Invalid fee slippage factor: ${e}, using default: ${or.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR}`),or.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR):t}async ensureWebSocketConnection(){this.websocketService.isConnected()||(await this.websocketService.connect(),this.logger.debug("WebSocket connection established"))}async waitForConfirmation(e,t){this.logger.debug(`Waiting for confirmation of transaction: ${e}`);try{const n=await this.websocketService.waitForTransaction(e);if("completed"!==n.status)throw new Yi(e,n.status,n.message);let o;try{o=t(n)}catch(t){if(t instanceof Xi)throw t;throw new Xi(`Failed to transform WebSocket response for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}return this.logger.debug(`Transaction confirmed: ${e}`,o),o}catch(t){if(this.logger.error(`Transaction confirmation failed: ${e}`,t),t instanceof Yi||t instanceof Xi)throw t;throw new Xi(`WebSocket confirmation failed for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}}async warmCacheFromPools(e){if(!e||!Array.isArray(e))return;const{extractMetadataFromPoolData:t,isValidPoolForCaching:n}=await Promise.resolve().then(function(){return Ir});e.forEach(e=>{if(!n(e))return;const o=t(e,this.logger);o&&this.launchpadAPI.warmCacheFromPoolData(e.tokenName,o)})}async getSwapQuoteExactInput(e,t,n){return this.gswapService.getSwapQuoteExactInput({fromToken:e,toToken:t,amount:n})}async getSwapQuoteExactOutput(e,t,n){return this.gswapService.getSwapQuoteExactOutput({fromToken:e,toToken:t,amount:n})}async executeSwap(e,t,n,o,i,r=.01){return this.validateWallet(),this.gswapService.executeSwap({fromToken:e,toToken:t,inputAmount:n,estimatedOutput:o,feeTier:i,slippageTolerance:r})}async getSwapUserAssets(e){return this.gswapService.getUserAssets(e)}async getAllSwapUserAssets(e){return this.gswapService.getAllUserAssets(e)}async getSwapPoolInfo(e,t){return this.gswapService.getPoolInfo(e,t)}async getSwapPoolPrice(e,t,n){return this.gswapService.getPositionCurrentPrice({token0:e,token1:t,feeTier:n})}async getSwapUserLiquidityPositions(e,t,n,o){let i,r;"string"==typeof n?(i=n,r=o):"object"==typeof n?r=n:o&&(r=o);return await this.gswapService.getUserLiquidityPositions(e,t,i,r)}async getAllSwapUserLiquidityPositions(e,t){const n=await this.gswapService.getAllSwapUserLiquidityPositions(e,t);if(!t?.withPrices){if(Array.isArray(n))return n;if(n&&"items"in n)return n.items}return n}async getSwapLiquidityPosition(e,t){return this.gswapService.getLiquidityPosition(e,t)}async getSwapLiquidityPositionById(e,t,n,o,i,r,a){return this.gswapService.getLiquidityPositionById(e,t,n,o,i,r,a)}async fetchSwapPositionDirect(e){return this.gswapService.fetchSwapPositionDirect(e)}async getSwapEstimateRemoveLiquidity(e){return this.gswapService.estimateRemoveLiquidity(e)}async addSwapLiquidityByPrice(e){return this.gswapService.addLiquidityByPrice(e)}async addSwapLiquidityByTicks(e){this.validateWallet();const t={...e,fee:e.feeTier??e.fee};return this.gswapService.addSwapLiquidityByTicks(t)}async removeSwapLiquidity(e){return this.validateWallet(),this.gswapService.removeLiquidity(e)}async collectSwapPositionFees(e){return this.validateWallet(),this.gswapService.collectPositionFees(e)}connectWebSocket(){this.websocketService.connect()}disconnectWebSocket(){this.websocketService.disconnect()}isWebSocketConnected(){return this.websocketService.isConnected()}subscribeToEvent(e,t){const n=this.websocketService.getSocket();return n?(n.on(e,t),()=>{n.off(e,t),this.logger.debug(`Unsubscribed from event: "${e}"`)}):(this.logger.warn(`⚠️ WebSocket not connected - subscribing to "${e}" without connection`),()=>{})}onDexPoolCreation(e,t){const n=1e3,o=Math.max(t?.intervalMs??3e4,n);t?.intervalMs&&t.intervalMs<n&&this.logger.warn(`Poll interval ${t.intervalMs}ms is below minimum 1000ms. Using minimum interval instead.`);const i=t?.minTVL,r=t?.tokens,a=new Map;let s=!0,c=null;let l=0;const u=async()=>{if(s){try{const t=await this.fetchDexPools({limit:20});l>0&&(this.logger.debug("Successfully recovered from polling errors"),l=0),t.pools.forEach(t=>{const n=(e=>`${e.token0}-${e.token1}-${e.fee}`)(t);if(!a.has(n)){if((e=>{if(a.set(e,!0),a.size>1e3){const e=a.keys().next().value;void 0!==e&&a.delete(e)}})(n),i){if((t.token0Tvl+t.token1Tvl)/2<i)return}if(r&&r.length>0){if(!(r.includes(t.token0)||r.includes(t.token1)))return}e(t)}})}catch(e){l++;const t=e instanceof Error?e.message:String(e);l>=5?this.logger.error(`Polling for new DEX pools failed ${l} consecutive times. Last error: ${t}. Continuing to retry...`):l>1?this.logger.warn(`Error polling for new DEX pools (attempt ${l}/5): ${t}`):this.logger.debug(`Error polling for new DEX pools: ${t}`)}if(s){const e=Math.min(Math.max(l-1,0),2),t=o*Math.pow(2,e);c=setTimeout(u,t)}}};return u(),()=>{s=!1,c&&clearTimeout(c),this.logger.debug("Stopped watching for DEX pool creation")}}onLaunchpadTokenCreation(e,t){const n=1e3,o=Math.max(t?.intervalMs??3e4,n);t?.intervalMs&&t.intervalMs<n&&this.logger.warn(`Poll interval ${t.intervalMs}ms is below minimum 1000ms. Using minimum interval instead.`);const i=t?.creatorAddress,r=new Map;let a=!0,s=null;let c=0;const l=async()=>{if(a){try{const t=await this.fetchPools({type:"recent",limit:20});c>0&&(this.logger.debug("Successfully recovered from polling errors"),c=0),t.pools.forEach(t=>{r.has(t.tokenName)||((e=>{if(r.set(e,!0),r.size>1e3){const e=r.keys().next().value;void 0!==e&&r.delete(e)}})(t.tokenName),i&&t.creatorAddress!==i||e(t))})}catch(e){c++;const t=e instanceof Error?e.message:String(e);c>=5?this.logger.error(`Polling for new launchpad tokens failed ${c} consecutive times. Last error: ${t}. Continuing to retry...`):c>1?this.logger.warn(`Error polling for new launchpad tokens (attempt ${c}/5): ${t}`):this.logger.debug(`Error polling for new launchpad tokens: ${t}`)}if(a){const e=Math.min(Math.max(c-1,0),2),t=o*Math.pow(2,e);s=setTimeout(l,t)}}};return l(),()=>{a=!1,s&&clearTimeout(s),this.logger.debug("Stopped watching for launchpad token creation")}}normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:!Number.isInteger(t)||500!==t&&3e3!==t&&1e4!==t?null:t}extractField(e,...t){if("object"!=typeof e||null===e)return"";const n=e;for(const e of t)if(n[e])return String(n[e]);return""}looksLikePoolPair(e){if("string"!=typeof e)return null;const t=e.trim();if(!t.includes("/"))return null;const n=t.split("/");return 3!==n.length?null:n[0]&&n[1]&&n[2]?t:null}buildPoolPairFromObject(e){if("object"!=typeof e||null===e)return null;const t=e,n=this.extractField(t,"token0ClassKey","token0Class","token0","token0Symbol")||"",o=this.extractField(t,"token1ClassKey","token1Class","token1","token1Symbol")||"",i=this.normalizeFee(t.feeTier??t.fee??t.feeTierBps??t.liquidityFeeBps??t.feeBps);return n&&o&&null!==i?`${n}/${o}/${i}`:null}parsePoolPairString(e){const t=e.split("/");if(3!==t.length)return null;const n=t[0].split("|")[0],o=t[1].split("|")[0],i=t[2];return n&&o&&i?{token0:n,token1:o,fee:i,poolPair:e}:null}serializeBalanceToken(e){if(!e||"object"!=typeof e)return"";const t=e;return[(t.collection??t.token??"")||"",(t.category??"")||"none",(t.type??"")||"none",(t.additionalKey??"none")||"none"].join("|")}buildPoolPairFromBalances(e){if("object"!=typeof e||null===e)return null;const t=e,n=t.userBalanceDelta??t.balanceDelta??t.delta;if(!n||"object"!=typeof n||null===n)return null;const o=n,i=o.token0Balance??o.token0??o.baseBalance??o.primaryBalance,r=o.token1Balance??o.token1??o.quoteBalance??o.secondaryBalance,a=this.serializeBalanceToken(i),s=this.serializeBalanceToken(r),c=this.normalizeFee(t.poolFee??t.feeTier??t.fee??t.feeTierBps??t.liquidityFeeBps);return a&&s&&null!==c?`${a}/${s}/${c}`:null}extractPoolDataFromPayload(e){if("string"==typeof e){const t=this.looksLikePoolPair(e);return t?this.parsePoolPairString(t):null}if("object"!=typeof e||null===e)return null;const t=e,n=this.looksLikePoolPair(t.poolPair);if(n)return this.parsePoolPairString(n);const o=this.buildPoolPairFromBalances(t);if(o)return this.parsePoolPairString(o);const i=this.buildPoolPairFromObject(t);if(i)return this.parsePoolPairString(i);if(t.pool&&"object"==typeof t.pool&&null!==t.pool){const e=this.extractPoolDataFromPayload(t.pool);if(e)return e}return null}matchesPoolFilter(e,t){if(t?.tokenFilter){if(!(e.token0===t.tokenFilter||e.token1===t.tokenFilter))return!1}if(t?.pairTokens){const[n,o]=t.pairTokens,i=e.token0===n||e.token1===n,r=e.token0===o||e.token1===o;if(!i||!r||n===o)return!1}if(void 0!==t?.feeTierFilter){if(this.normalizeFee(t.feeTierFilter)!==this.normalizeFee(e.fee))return!1}return!0}matchesCreatorFilter(e,t){return!t||e.creatorAddress===t}subscribeToTokenCreations(e,t){if(this.logger.debug("Subscribing to token creation broadcasts"+(t?.creatorFilter?` (filter: ${t.creatorFilter})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=(n,...o)=>{try{if(o.length>0&&"object"==typeof o[0]&&null!==o[0]){const n=o[0].data;if(n&&n.Data&&"object"==typeof n.Data){const o=n.Data;if("CreateSale"===o.functionName){const n={tokenName:o.tokenName||"",symbol:o.symbol||"",creatorAddress:o.creatorAddress||"",description:o.description||"",image:o.image||"",vaultAddress:o.vaultAddress||"",tokenStringKey:o.tokenStringKey||"",preBuyQuantity:o.initialBuyQuantity||"0",websiteUrl:o.websiteUrl||"",telegramUrl:o.telegramUrl||"",twitterUrl:o.twitterUrl||"",isFinalized:o.isFinalized||!1};this.matchesCreatorFilter(n,t?.creatorFilter)&&e(n)}}}}catch(e){this.logger.warn(`Error processing token creation broadcast: ${e instanceof Error?e.message:String(e)}`)}};let r=this.websocketService.getSocket();if(r)r.onAny(i),n=!0,this.logger.debug("Token creation broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const a=()=>{if(r=this.websocketService.getSocket(),!r&&e<or.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>a(),or.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS));if(!r&&e>=or.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${or.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS*or.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("Token creation broadcast subscription timeout:",e.message),void(t?.onError&&t.onError(e))}r&&(r.onAny(i),n=!0,this.logger.debug("Token creation broadcast listener registered"))};a()}return()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared token creation broadcast polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(i),n=!1,this.logger.debug("Stopped listening to token creation broadcasts"))}catch(e){this.logger.warn("Error removing token creation listener:",e)}}}walkPayloadForPools(e,t,n=new WeakSet){const o=[];if("string"==typeof e){const n=this.looksLikePoolPair(e);if(n){const e=this.parsePoolPairString(n);e&&!t.has(e.poolPair)&&(t.add(e.poolPair),o.push(e))}return o}if("object"!=typeof e||null===e)return o;if(n.has(e))return o;n.add(e);const i=this.extractPoolDataFromPayload(e);i&&!t.has(i.poolPair)&&(t.add(i.poolPair),o.push(i));for(const i of Object.values(e)){const e=this.walkPayloadForPools(i,t,n);o.push(...e)}return o}subscribeToDexPoolAdded(e,t){if(this.logger.debug("Subscribing to DEX pool creation broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=new Set,r=(n,...o)=>{try{for(const n of o){const o=this.walkPayloadForPools(n,i);for(const n of o)this.matchesPoolFilter(n,t)&&e(n)}}catch(e){this.logger.warn(`Error processing DEX pool broadcast: ${e instanceof Error?e.message:String(e)}`)}};let a=this.websocketService.getSocket();if(a)a.onAny(r),n=!0,this.logger.debug("DEX pool broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const i=()=>{if(a=this.websocketService.getSocket(),!a&&e<or.DEX_POOL_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>i(),or.DEX_POOL_SOCKET_POLL_INTERVAL_MS));if(!a&&e>=or.DEX_POOL_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${or.DEX_POOL_SOCKET_WAIT_ATTEMPTS*or.DEX_POOL_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("DEX pool subscription timeout:",e.message),void(t?.onError&&t.onError(e))}a&&(a.onAny(r),n=!0,this.logger.debug("DEX pool broadcast listener registered"))};i()}return()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared DEX pool polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(r),n=!1,this.logger.debug("Stopped listening to DEX pool broadcasts"))}catch(e){this.logger.warn("Error removing DEX pool listener:",e)}}}subscribeToDexSwapExecuted(e,t){if(this.logger.debug("Subscribing to DEX swap execution broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=null,o=null,i=null,r=!1;const a=async e=>{const t=Pi.parsePoolKey(e);if(!t)throw new Error(`Invalid pool key format: ${e}`);return await this.dexQuoteService.fetchCompositePoolData({token0:t.token0,token1:t.token1,fee:t.feeTier})},s=()=>{const c=this.websocketService.getSocket();if(!c)return this.logger.debug("WebSocket not yet ready for swap monitoring, polling..."),void(n=setTimeout(()=>s(),100));r=!0,o=new Li(c,a,this.dexQuoteService,t||{},this.logger),i=o.subscribe(t||{},e),this.logger.debug("DEX swap monitoring subscription established")};return this.websocketService.getSocket()||(this.logger.debug("WebSocket not yet connected, initiating connection for swap monitoring..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed for swap monitoring: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)})),s(),()=>{try{n&&clearTimeout(n),i&&r&&i(),o&&o.shutdown().catch(e=>{this.logger.warn("Error shutting down swap monitor:",e)})}catch(e){this.logger.warn("Error cleaning up swap monitor:",e)}}}subscribeToDexLiquidityAdded(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityRemoved(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityChanged(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityEvents(e,t){if(this.logger.debug("Subscribing to DEX liquidity broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=new Set,r=new nr(this.logger),a=(n,...o)=>{try{for(const n of o){const o=r.walkPayloadForLiquidityEvents(n,i);for(const n of o)if(this.matchesLiquidityFilter(n,t))try{const t=e(n);t instanceof Promise&&t.catch(e=>{this.logger.warn(`Error in liquidity event callback: ${e instanceof Error?e.message:String(e)}`)})}catch(e){this.logger.warn(`Error in liquidity event callback: ${e instanceof Error?e.message:String(e)}`)}}}catch(e){this.logger.warn(`Error processing DEX liquidity broadcast: ${e instanceof Error?e.message:String(e)}`)}};let s=this.websocketService.getSocket();if(s)s.onAny(a),n=!0,this.logger.debug("DEX liquidity broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection for liquidity monitoring..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const i=()=>{if(s=this.websocketService.getSocket(),!s&&e<or.DEX_POOL_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>i(),or.DEX_POOL_SOCKET_POLL_INTERVAL_MS));if(!s&&e>=or.DEX_POOL_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${or.DEX_POOL_SOCKET_WAIT_ATTEMPTS*or.DEX_POOL_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("DEX liquidity subscription timeout:",e.message),void(t?.onError&&t.onError(e))}s&&(s.onAny(a),n=!0,this.logger.debug("DEX liquidity broadcast listener registered"))};i()}return async()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared DEX liquidity polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(a),n=!1,this.logger.debug("Stopped listening to DEX liquidity broadcasts"))}catch(e){this.logger.warn("Error removing DEX liquidity listener:",e)}}}matchesLiquidityFilter(e,t){if(!t)return!0;if(t.positionId&&e.positionId!==t.positionId)return!1;if(t.poolHash&&e.poolHash!==t.poolHash)return!1;if(void 0!==t.feeTierFilter){const n=this.normalizeFeeTier(t.feeTierFilter);if(e.poolFee!==n)return!1}if(t.userFilter&&e.userAddress!==t.userFilter)return!1;if(t.tokenFilter){const n=e.token0?.toLowerCase().includes(t.tokenFilter.toLowerCase()),o=e.token1?.toLowerCase().includes(t.tokenFilter.toLowerCase());if(!n&&!o)return!1}if(t.pairTokens){const[n,o]=t.pairTokens.map(e=>e.toLowerCase()),i=e.token0?.toLowerCase()||"",r=e.token1?.toLowerCase()||"",a=i.includes(n)&&r.includes(o),s=i.includes(o)&&r.includes(n);if(!a&&!s)return!1}if(t.minAmount){const n=parseFloat(t.minAmount),o=parseFloat(e.amounts[0])||0,i=parseFloat(e.amounts[1])||0;if(Math.abs(o)<n&&Math.abs(i)<n)return!1}return!0}normalizeFeeTier(e){if("number"==typeof e)return e>=100?e:Math.round(1e4*e);const t=e.replace("%","").trim(),n=parseFloat(t);return n>=100?n:Math.round(1e4*n)}async cleanup(){try{this.logger.debug("Starting cleanup..."),this.http.cleanup(),this.websocketService&&this.websocketService.disconnect(),this.logger.debug("Cleanup completed")}catch(e){this.logger.error("Error during cleanup:",e)}}static cleanupAll(e=!1){const t=new U({debug:e,context:"LaunchpadSDK"});t.debug("Starting global cleanup...");const{WebSocketService:n}=require("./services/WebSocketService");n.cleanupAll(e),t.debug("Global cleanup completed")}}or.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR=.15,or.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR=.01,or.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY=bi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY,or.DEFAULT_CALCULATE_AMOUNT_MODE=ji.LOCAL,or.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS=30,or.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS=100,or.DEX_POOL_SOCKET_WAIT_ATTEMPTS=30,or.DEX_POOL_SOCKET_POLL_INTERVAL_MS=100;class ir{static generateWallet(){try{const e=n.createRandom();if(!e.mnemonic?.phrase)throw new Error("Failed to generate wallet with mnemonic phrase");const t=this.toGalaAddress(e.address);return{privateKey:e.privateKey,address:e.address,galaAddress:t,mnemonic:e.mnemonic.phrase,wallet:new n(e.privateKey)}}catch(e){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const e=`test-wallet-${Date.now()}-${++this.testCounter}`,t="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64),o=new n(t),i=this.toGalaAddress(o.address);return{privateKey:o.privateKey,address:o.address,galaAddress:i,mnemonic:"test test test test test test test test test test test junk",wallet:o}}throw e}}static fromPrivateKey(e){const t=new n(e),o=this.toGalaAddress(t.address);return{privateKey:t.privateKey,address:t.address,galaAddress:o,mnemonic:"",wallet:t}}static fromMnemonic(e,t=0){try{const r=o.fromPhrase(e),a=i.fromMnemonic(r,`m/44'/60'/0'/0/${t}`),s=new n(a.privateKey),c=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:c,mnemonic:e,wallet:s}}catch(o){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const o=`test-mnemonic-index-${t}-${e}`,i="0x"+Buffer.from(o).toString("hex").padStart(64,"1").slice(0,64),r=new n(i),a=this.toGalaAddress(r.address);return{privateKey:r.privateKey,address:r.address,galaAddress:a,mnemonic:e,wallet:r}}throw o}}static toGalaAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new Error(`Invalid Ethereum address format: ${e}`);return`eth|${t}`}static toEthereumAddress(e){if(!e.startsWith("eth|"))throw new Error(`Invalid Gala address format: ${e}. Must start with 'eth|'`);const t=e.slice(4);if(!/^[a-fA-F0-9]{40}$/.test(t))throw new Error(`Invalid address in Gala format: ${e}`);return`0x${t}`}static isValidEthereumAddress(e){try{const t=e.replace(/^0x/i,"");return/^[a-fA-F0-9]{40}$/.test(t)}catch{return!1}}static isValidGalaAddress(e){try{if(!e.startsWith("eth|"))return!1;const t=e.slice(4);return/^[a-fA-F0-9]{40}$/.test(t)}catch{return!1}}static generateMultipleWallets(e=1){if(e<1||e>100)throw new Error("Count must be between 1 and 100");const t=[];if("undefined"!=typeof process&&"test"===process.env.NODE_ENV)for(let n=0;n<e;n++){const e=`test-multi-${n}-${Date.now()}-${++this.testCounter}`,o="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64);t.push(this.fromPrivateKey(o))}else for(let n=0;n<e;n++)t.push(this.generateWallet());return t}static getWalletSummary(e,t=!1){const n=["🔐 Wallet Information","═".repeat(50),`📍 Address: ${e.address}`,`🎮 Gala Address: ${e.galaAddress}`,`🌱 Mnemonic: ${e.mnemonic||"Not available"}`];return t?n.splice(3,0,`🔑 Private Key: ${e.privateKey}`):n.splice(3,0,"🔑 Private Key: [HIDDEN - use includeSensitive=true to show]"),n.push("═".repeat(50)),n.push("💾 IMPORTANT: Save your mnemonic phrase securely!"),n.push("This is your backup to recover the wallet."),n.join("\n")}}function rr(e){if(void 0===e)return ir.generateWallet();const t=e.trim();if(!t)throw new Error("Input cannot be empty string");if(function(e){const t=e.replace(/^0x/i,"");return/^[a-fA-F0-9]{64}$/.test(t)}(t))return ir.fromPrivateKey(t);if(function(e){const t=e.split(/\s+/).filter(e=>e.length>0);if(12!==t.length&&24!==t.length)return!1;return t.every(e=>/^[a-zA-Z]+$/.test(e))}(t))return ir.fromMnemonic(t);throw new Error(`Unable to detect input format. Expected:\n- Private key: 64 hexadecimal characters (with or without 0x prefix)\n- Mnemonic: 12 or 24 space-separated words\nReceived: "${t.slice(0,50)}${t.length>50?"...":""}"`)}function ar(e){e||(e={});const{wallet:t,env:o,config:i={},...r}=e,a={...r,...i},{wallet:s,env:c,config:l,...u}=a;let d;if(t)if("string"==typeof t){d=rr(t).wallet}else{if(!(t instanceof n))throw new Error("Invalid wallet input. Expected string (private key or mnemonic) or Wallet instance.");d=t}else{d=rr().wallet}const h={wallet:d,...o&&{env:o},debug:!1,timeout:3e4,...u};return new or(h)}function sr(){const e=F.join(process.cwd(),"..","..",".env");C.existsSync(e)&&x.config({path:e});const t=F.join(process.cwd(),".env");C.existsSync(t)&&x.config({path:t})}function cr(e,t){const n=process.env[e];if(!n){throw new Error(t?`${e} not set in root or local .env (${t})`:`${e} not set in root or local .env`)}return n}function lr(e,t){return process.env[e]??t}ir.testCounter=0;class ur{static async quickSetup(e={}){const t=e.environment||this.detectEnvironment(),n=this.setupWallet(e.privateKey),o={wallet:n.wallet,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t),...e.config||{}},i=new or(o),r={sdk:i,wallet:n,config:o};if(!1!==e.autoValidate){const e=await this.validateSetup(i,n);return{...r,validation:e}}return r}static async readOnlySetup(e={}){const t=e.environment||this.detectEnvironment(),n={wallet:void 0,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t),...e.config||{}};return{sdk:new or(n),config:n}}static async validateSetup(e,t){const n=[],o=[],i={canTrade:!1,canCreateTokens:!1,hasBalance:!1,connectionHealthy:!1};try{const t=await e.fetchGalaBalance(e.getAddress());if(i.connectionHealthy=!0,t&&t.quantity){const e=parseFloat(t.quantity);i.hasBalance=e>0,i.canTrade=e>=.1,i.canCreateTokens=e>=100,0===e?o.push("Wallet has zero GALA balance - cannot perform transactions"):e<.1?o.push("GALA balance too low for trading (minimum 0.1 GALA)"):e<100&&o.push("GALA balance too low for token creation (minimum 100 GALA)")}else n.push("Failed to fetch GALA balance: No balance returned")}catch(e){n.push(`Balance check error: ${e instanceof Error?e.message:String(e)}`)}try{const t=await e.fetchPools({type:"recent",page:1,limit:1});t.pools&&0!==t.pools.length||o.push("Pool listing not accessible - some features may be limited")}catch(e){o.push(`Pool access test failed: ${e instanceof Error?e.message:String(e)}`)}return{ready:0===n.length&&i.connectionHealthy,sdk:e,wallet:t||ir.generateWallet(),issues:n,warnings:o,capabilities:i}}static getRecommendedConfig(e,t="general"){const n={environment:e,autoValidate:!0};switch(e){case"production":Object.assign(n,{debug:!1,timeout:3e4});break;case"development":Object.assign(n,{debug:!0,timeout:45e3});break;case"testing":Object.assign(n,{debug:!0,timeout:6e4})}switch(t){case"trading":n.timeout=1.5*(n.timeout||3e4);break;case"creation":n.timeout=2*(n.timeout||3e4);break;case"monitoring":n.timeout=.5*(n.timeout||3e4)}return n}static async multiWalletSetup(e,t="development"){const n={};for(const[o,i]of Object.entries(e)){const{sdk:e}=await this.quickSetup({environment:t,privateKey:i,agentId:`multi-wallet-${o}`,autoValidate:!1});n[o]=e}return n}static detectEnvironment(){const e=process.env.NODE_ENV?.toLowerCase();return"development"===e?"development":"test"===e||"testing"===e?"testing":"production"}static setupWallet(e){if(!e){const e=process.env.PRIVATE_KEY;return e?ir.fromPrivateKey(e):ir.generateWallet()}return"generate"===e?ir.generateWallet():ir.fromPrivateKey(e)}static getDefaultBaseUrl(e){return"production"===e?"https://lpad-backend-prod1.defi.gala.com":"https://lpad-backend-dev1.defi.gala.com"}static getDefaultTimeout(e){switch(e){case"production":default:return 3e4;case"development":return 45e3;case"testing":return 6e4}}static getEnvironmentDefaults(e){const t={};if("production"===e)t.bundleBaseUrl="https://bundle-backend-prod1.defi.gala.com",t.galaChainBaseUrl="https://galachain-gateway-chain-platform-prod-chain-platform-eks.prod.galachain.com";else t.bundleBaseUrl="https://bundle-backend-dev1.defi.gala.com",t.galaChainBaseUrl="https://galachain-gateway-chain-platform-stage-chain-platform-eks.stage.galachain.com";return t}}function dr(e){return e.filter(e=>new u(e.liquidity).isGreaterThan(0))}function hr(e,t){const n=new u(t);return e.filter(e=>new u(e.liquidity).isGreaterThanOrEqualTo(n))}function gr(e,t="desc"){return[...e].sort((e,n)=>{const o=new u(e.liquidity),i=new u(n.liquidity);return"desc"===t?i.minus(o).toNumber():o.minus(i).toNumber()})}function pr(e,t,n){const o=t.toUpperCase(),i=n.toUpperCase();return e.filter(e=>{const t=e.token0.toUpperCase(),n=e.token1.toUpperCase();return t===o&&n===i||t===i&&n===o})}function mr(e,t){const n=t.toUpperCase();return e.filter(e=>e.token0.toUpperCase()===n||e.token1.toUpperCase()===n)}function fr(e,t){return e.filter(e=>e.feeTier===t)}function yr(e){const t=new Map;return e.forEach(e=>{const n=`${e.token0.toUpperCase()}/${e.token1.toUpperCase()}`;t.has(n)||t.set(n,[]),t.get(n).push(e)}),t}function wr(e){const t=new Map;return e.forEach(e=>{t.has(e.feeTier)||t.set(e.feeTier,[]),t.get(e.feeTier).push(e)}),t}function kr(e,t,n,o){const i=t.toUpperCase(),r=n.toUpperCase();return e.filter(e=>{const t=e.token0.toUpperCase(),n=e.token1.toUpperCase();return(t===i&&n===r||t===r&&n===i)&&e.feeTier===o})}function vr(e){const t=new Map;return e.forEach(e=>{const n=`${e.token0.toUpperCase()}|${e.token1.toUpperCase()}|${e.feeTier}`;t.has(n)||t.set(n,[]),t.get(n).push(e)}),t}const br={PAGE:"page",LIMIT:"limit",TOKEN_NAME:"tokenName",VAULT_ADDRESS:"vaultAddress",USER_ADDRESS:"userAddress",TRADE_TYPE:"tradeType",POOL_TYPE:"type",SEARCH:"search",SORT_ORDER:"sortOrder",START_DATE:"startDate",END_DATE:"endDate"};class Tr{static fastValidation(e,t,n,o=Tr.DEFAULT_CONFIG){const i=Date.now();let r=!0;try{const a=new u(e.sqrtPrice),s=new u(t.sqrtPrice);n.zeroForOne?s.gte(a)&&(this.logger.error("Fast validation failed: price did not decrease for zeroForOne swap",{originalSqrtPrice:a.toString(),updatedSqrtPrice:s.toString(),zeroForOne:n.zeroForOne}),r=!1):s.lte(a)&&(this.logger.error("Fast validation failed: price did not increase for oneForZero swap",{originalSqrtPrice:a.toString(),updatedSqrtPrice:s.toString(),zeroForOne:n.zeroForOne}),r=!1);const c=new u(e.liquidity),l=new u(t.liquidity);if(!c.isZero()){const e=l.minus(c).abs().div(c);e.gt(o.maxLiquidityChangePct)&&this.logger.warn("Fast validation warning: large liquidity change detected (could be legitimate)",{originalLiquidity:c.toString(),updatedLiquidity:l.toString(),changePct:e.times(100).toFixed(2)})}const d=new u(e.feeGrowthGlobal0),h=new u(t.feeGrowthGlobal0),g=new u(e.feeGrowthGlobal1),p=new u(t.feeGrowthGlobal1);n.zeroForOne?p.lt(g)&&(this.logger.error("Fast validation failed: feeGrowthGlobal1 decreased for zeroForOne",{originalFeeGrowth1:g.toString(),updatedFeeGrowth1:p.toString()}),r=!1):h.lt(d)&&(this.logger.error("Fast validation failed: feeGrowthGlobal0 decreased for oneForZero",{originalFeeGrowth0:d.toString(),updatedFeeGrowth0:h.toString()}),r=!1);const m=new u(e.protocolFeesToken0),f=new u(t.protocolFeesToken0),y=new u(e.protocolFeesToken1),w=new u(t.protocolFeesToken1);f.lt(m)&&(this.logger.error("Fast validation failed: protocolFeesToken0 decreased",{originalProtocolFees0:m.toString(),updatedProtocolFees0:f.toString()}),r=!1),w.lt(y)&&(this.logger.error("Fast validation failed: protocolFeesToken1 decreased",{originalProtocolFees1:y.toString(),updatedProtocolFees1:w.toString()}),r=!1);const k=Date.now()-i;return this.logger.debug("Fast validation completed",{passed:r,elapsedMs:k}),r}catch(e){return this.logger.error("Fast validation exception",e),!1}}static moderateValidation(e,t,n=Tr.DEFAULT_CONFIG){const o=Date.now(),i=[];let r=0;try{if(t.actualSqrtPrice){const o=new u(e.sqrtPrice),a=new u(t.actualSqrtPrice),s=this.calculateDriftPercentage(o,a);r=s,s>100*n.maxDriftThreshold&&i.push(`Price drift detected: ${s.toFixed(4)}% (threshold: ${(100*n.maxDriftThreshold).toFixed(4)}%)`),this.logger.debug("Price drift comparison",{calculatedSqrtPrice:o.toString(),actualSqrtPrice:a.toString(),driftPct:s.toFixed(4)})}const a=new u(e.sqrtPrice),s=new u(2).pow(96),c=a.dividedBy(s),l=A(c),d=e.tick??0,h=Math.abs(l-d);h>n.maxTickDrift&&i.push(`Tick/price mismatch: tick=${d}, calculated=${l}, drift=${h}`);const g=new u(e.feeGrowthGlobal0),p=new u(e.feeGrowthGlobal1),m=new u(e.liquidity);try{To(g,p,m)}catch(e){i.push(e.message)}const f=0===i.length,y=!f||r>100*n.maxDriftThreshold,w=Date.now()-o;return this.logger.debug("Moderate validation completed",{isValid:f,shouldRefetch:y,driftPercentage:r,errorCount:i.length,elapsedMs:w}),this.buildValidationResult(f,r,y,i)}catch(e){this.logger.error("Moderate validation exception",e);const t=e instanceof Error?e.message:String(e);return this.buildValidationResult(!1,0,!0,[`Exception during validation: ${t}`])}}static async fullValidation(e,t,n){const o=Date.now(),i=[];let r=0;try{this.logger.debug("Starting full validation with fresh pool data fetch",{poolKey:e});const a=await n(),s=new u(t.pool.sqrtPrice),c=new u(a.pool.sqrtPrice),l=this.calculateDriftPercentage(s,c);r=Math.max(r,l),l>100*this.DEFAULT_CONFIG.maxPriceDriftPct&&i.push(`Price drift: ${l.toFixed(4)}% (cached: ${s.toString()}, fresh: ${c.toString()})`);const d=new u(t.pool.liquidity),h=new u(a.pool.liquidity),g=this.calculateDriftPercentage(d,h);r=Math.max(r,g),g>100*this.DEFAULT_CONFIG.maxLiquidityDriftPct&&i.push(`Liquidity drift: ${g.toFixed(4)}% (cached: ${d.toString()}, fresh: ${h.toString()})`);const p=Object.keys(t.tickDataMap).length,m=Object.keys(a.tickDataMap).length;if(m>0){const e=Math.abs(m-p)/m;e>this.DEFAULT_CONFIG.maxTickCountDriftPct&&i.push(`Tick data incomplete: cached has ${p} ticks, fresh has ${m} ticks (${(100*e).toFixed(2)}% difference)`)}const f=0===i.length,y=!f,w=Date.now()-o;return this.logger.debug("Full validation completed",{poolKey:e,isValid:f,shouldRefetch:y,maxDriftPercentage:r,priceDrift:l,liquidityDrift:g,cachedTickCount:p,freshTickCount:m,errorCount:i.length,elapsedMs:w}),this.buildValidationResult(f,r,y,i)}catch(t){this.logger.error("Full validation exception",{poolKey:e,error:t});const n=t instanceof Error?t.message:String(t);return this.buildValidationResult(!1,0,!0,[`Exception during full validation: ${n}`])}}static calculateDriftPercentage(e,t){if(t.isZero())return this.logger.warn("Cannot calculate drift: actual value is zero"),1/0;return t.minus(e).abs().div(t).times(100).toNumber()}static buildValidationResult(e,t,n,o=[]){let i;if(n&&o.length>0){const e=o.join(" ").toLowerCase();i=e.includes("drift")?"drift":e.includes("tick")&&e.includes("mismatch")?"tick-mismatch":e.includes("tick")?"missing-tick-data":"manual"}const r={isValid:e,driftPercentage:t,shouldRefetch:n,validationErrors:o};return void 0!==i&&(r.refetchReason=i),r}}Tr.logger=new U({debug:!1,context:"PoolStateValidator"}),Tr.DEFAULT_CONFIG={maxDriftThreshold:.001,maxLiquidityChangePct:.5,maxTickDrift:1,maxTickCountDriftPct:.1,maxPriceDriftPct:.001,maxLiquidityDriftPct:.01};class Sr{static calculatePoolStateHash(e){const t=`${e.sqrtPrice.toString()}|${e.liquidity.toString()}|${e.tick||0}`;return $.createHash("sha256").update(t).digest("hex").substring(0,16)}constructor(e,t){this.logger=new U({debug:t?.debug??!1,context:"PoolStateManager"});const n={maxIterations:t?.maxIterations??100,enableBigNumberCache:t?.enableBigNumberCache??!0,roundingMode:t?.roundingMode??u.ROUND_DOWN,debug:t?.debug??!1,maxSwapsSinceRefetch:t?.maxSwapsSinceRefetch??50,maxCumulativeDrift:t?.maxCumulativeDrift??5,strictValidation:t?.strictValidation??!1,enablePerformanceWarnings:t?.enablePerformanceWarnings??!0,performanceWarningThreshold:t?.performanceWarningThreshold??100};this.config={...n,...t?.onRefetchNeeded?{onRefetchNeeded:t.onRefetchNeeded}:{}},this.validationConfig=Tr.DEFAULT_CONFIG,this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata={lastFullRefetch:Date.now(),swapsSinceRefetch:0,cumulativeDrift:0,processedTransactions:[],stateHash:Sr.calculatePoolStateHash(this.pool)};if(!Tr.fastValidation(this.pool,this.pool,{transactionId:"init",timestamp:Date.now(),amountSpecified:"0",zeroForOne:!1,exactInput:!0},this.validationConfig)&&this.config.strictValidation)throw new Error("Initial pool state validation failed");this.logger.info("PoolStateManager initialized",{pool:{sqrtPrice:this.pool.sqrtPrice.toString(),liquidity:this.pool.liquidity.toString(),tick:this.pool.tick},config:this.config})}async applySwapDelta(e){const t=Date.now();if(this.metadata.processedTransactions.includes(e.transactionId))throw this.logger.warn("Duplicate swap transaction",{transactionId:e.transactionId}),new Error(`Duplicate transaction ID: ${e.transactionId}`);try{const t={pool:this.pool,tickDataMap:this.tickDataMap},n={maxIterations:this.config.maxIterations,enableBigNumberCache:this.config.enableBigNumberCache,roundingMode:this.config.roundingMode,debugLogging:this.config.debug},o=Ci.calculateSwapDelta(t,e,n);this.lastSwapMetrics={calculationTimeMs:o.metadata.calculationTimeMs,swapSteps:o.metadata.swapSteps,timestamp:Date.now()},this.config.enablePerformanceWarnings&&o.metadata.calculationTimeMs>this.config.performanceWarningThreshold&&this.logger.warn("Slow swap calculation",{calculationTimeMs:o.metadata.calculationTimeMs,swapSteps:o.metadata.swapSteps,threshold:this.config.performanceWarningThreshold});if(!Tr.fastValidation(t.pool,o.updatedPool,e,this.validationConfig)&&this.config.strictValidation)throw new Error("Swap validation failed");if(e.actualAmount0&&e.actualAmount1&&e.actualSqrtPrice){const t=new u(e.actualAmount0),n=new u(e.actualAmount1),i=o.amount0.minus(t).abs(),r=o.amount1.minus(n).abs(),a=t.isZero()?0:i.div(t.abs()).times(100).toNumber(),s=n.isZero()?0:r.div(n.abs()).times(100).toNumber(),c=Math.max(a,s);c>1&&(this.logger.warn("Drift detected in swap delta",{driftPercentage:c.toFixed(2),swapId:e.transactionId}),this.metadata.cumulativeDrift+=c)}if((this.metadata.swapsSinceRefetch>this.config.maxSwapsSinceRefetch||this.metadata.cumulativeDrift>this.config.maxCumulativeDrift)&&(this.logger.info("Triggering full refetch due to drift accumulation",{swapsSinceRefetch:this.metadata.swapsSinceRefetch,cumulativeDrift:this.metadata.cumulativeDrift.toFixed(2)}),this.config.onRefetchNeeded)){const e=await this.config.onRefetchNeeded(this.pool,this.tickDataMap);this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata.lastFullRefetch=Date.now(),this.metadata.swapsSinceRefetch=0,this.metadata.cumulativeDrift=0,this.metadata.stateHash=Sr.calculatePoolStateHash(this.pool),this.logger.info("Full refetch completed",{sqrtPrice:this.pool.sqrtPrice.toString()})}return this.pool=o.updatedPool,Object.assign(this.tickDataMap,o.updatedTicks),this.metadata.swapsSinceRefetch++,this.metadata.stateHash=Sr.calculatePoolStateHash(this.pool),this.metadata.processedTransactions.push(e.transactionId),this.metadata.processedTransactions.length>1e3&&(this.metadata.processedTransactions=this.metadata.processedTransactions.slice(-1e3)),this.logger.debug("Swap delta applied",{transactionId:e.transactionId,amount0:o.amount0.toString(),amount1:o.amount1.toString(),sqrtPriceNew:this.pool.sqrtPrice.toString()}),o}catch(n){const o=n instanceof Error?n.message:String(n);if(this.logger.error("Failed to apply swap delta",{transactionId:e.transactionId,error:o}),this.config.strictValidation)throw n;return{updatedPool:this.pool,updatedTicks:{},amount0:new u(0),amount1:new u(0),feeAmount0:new u(0),feeAmount1:new u(0),ticksCrossed:[],metadata:{calculationTimeMs:Date.now()-t,swapSteps:0,priceHitLimit:!1}}}}forceFullRefetch(e){this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata.lastFullRefetch=Date.now(),this.metadata.swapsSinceRefetch=0,this.metadata.cumulativeDrift=0,this.logger.info("Full refetch forced",{sqrtPrice:this.pool.sqrtPrice.toString()})}getPoolState(){return this.pool}getTickDataMap(){return{...this.tickDataMap}}getMetadata(){return{...this.metadata}}getLastSwapMetrics(){if(this.lastSwapMetrics)return{...this.lastSwapMetrics}}isRefetchRecommended(){return this.metadata.swapsSinceRefetch>this.config.maxSwapsSinceRefetch||this.metadata.cumulativeDrift>this.config.maxCumulativeDrift}reset(e){this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata={lastFullRefetch:Date.now(),swapsSinceRefetch:0,cumulativeDrift:0,processedTransactions:[],stateHash:Sr.calculatePoolStateHash(this.pool)},this.logger.info("PoolStateManager reset",{sqrtPrice:this.pool.sqrtPrice.toString()})}}function Ar(e,t){return new Sr(e,t)}class Er extends Rn{constructor(e=!1){super(e),this.cache={},this.cacheExpiry=36e5}async resolveTokenMetadata(e){const t=this.getCacheKey(e),n=this.cache[t];if(n&&!this.isCacheExpired(n.timestamp))return this.logger.debug(`Using cached metadata for token: ${t}`),n.data;const o=this.extractMetadata(e);return this.cache[t]={data:o,timestamp:Date.now()},o}async getTokenSymbol(e){return(await this.resolveTokenMetadata(e)).symbol}async getTokenDecimals(e){return(await this.resolveTokenMetadata(e)).decimals}clearCache(e){e?(delete this.cache[e],this.logger.debug(`Cleared cache for token: ${e}`)):(this.cache={},this.logger.debug("Cleared all token metadata cache"))}getCacheStats(){const e=Object.keys(this.cache);return{size:e.length,entries:e}}getCacheKey(e){if("string"==typeof e)return e.toLowerCase();return`${e.type||e.symbol||"unknown"}|${e.additionalKey||"none"}`.toLowerCase()}extractMetadata(e){let t,n,o="Token",i="Unit",r="none";if("string"==typeof e)if(e.includes("|")){const a=e.split("|");"Token"===a[0]&&a[2]?(o=a[0],i=a[1]||"Unit",t=a[2],r=a[3]||"none",n=t):(t=a[0],n=t,i=a[1]||"Unit",r=a[3]||"none")}else t=e,n=e;else t=e.type||"unknown",o=e.collection||"Token",i=e.category||"Unit",r=e.additionalKey||"none",n=e.symbol||("Token"===o?t:o)||"unknown";const a=this.getDecimalsForToken(n);return{symbol:n.toUpperCase(),decimals:a,collection:o,category:i,type:t.toUpperCase(),additionalKey:r,verified:!1}}getDecimalsForToken(e){return{GALA:8,GUSDC:6,USDC:6,USDT:6,WETH:18,DAI:18}[e.toUpperCase()]??18}isCacheExpired(e){return Date.now()-e>this.cacheExpiry}setCacheExpiry(e){this.cacheExpiry=e,this.logger.debug(`Set token metadata cache expiry to ${e}ms`)}}"undefined"!=typeof process&&process.env&&(process.env.CORE_CHAINCODE_LOGGING_LEVEL=process.env.CORE_CHAINCODE_LOGGING_LEVEL||"ERROR");var Ir=Object.freeze({__proto__:null,extractMetadataFromPoolData:function(e,t){const n={};if(e.vaultAddress&&(n.vaultAddress=e.vaultAddress),void 0!==e.reverseBondingCurveMinFeePortion){const o=parseFloat(e.reverseBondingCurveMinFeePortion);isNaN(o)?t&&t.debug(`Skipping invalid reverseBondingCurveMinFeePortion for ${e.tokenName}: "${e.reverseBondingCurveMinFeePortion}"`):n.reverseBondingCurveMinFeeFactor=o}if(void 0!==e.reverseBondingCurveMaxFeePortion){const o=parseFloat(e.reverseBondingCurveMaxFeePortion);isNaN(o)?t&&t.debug(`Skipping invalid reverseBondingCurveMaxFeePortion for ${e.tokenName}: "${e.reverseBondingCurveMaxFeePortion}"`):n.reverseBondingCurveMaxFeeFactor=o}return void 0!==n.reverseBondingCurveMaxFeeFactor&&void 0!==n.reverseBondingCurveMinFeeFactor&&(n.reverseBondingCurveNetFeeFactor=n.reverseBondingCurveMaxFeeFactor-n.reverseBondingCurveMinFeeFactor),Object.keys(n).length>0?n:null},isValidPoolForCaching:function(e){if(null===e||"object"!=typeof e)return!1;const t=e;return"tokenName"in t&&"string"==typeof t.tokenName&&t.tokenName.length>0}});export{ur as AgentConfig,ji as CALCULATION_MODES,Ai as COMPOSITE_POOL_FETCH_CONCURRENCY,j as ConfigurationError,J as DexPoolNotFoundError,Y as DexQuoteError,co as FileValidationError,Vo as GALA_DECIMALS,Ho as GALA_TOKEN_CLASS_KEY,X as GSwapAssetError,Q as GSwapPoolError,z as GSwapQuoteError,H as GSwapSwapError,Ye as IMAGE_EXTENSIONS,zo as LAUNCHPAD_TOKEN_DECIMALS,or as LaunchpadSDK,Ro as LockError,_o as LockErrorType,_n as MAX_CONCURRENT_POOL_FETCHES,W as NetworkError,Bn as PAGINATION_DEFAULTS,Un as POOL_FETCH_CONFIG,Wi as POOL_TYPES,Sr as PoolStateManager,br as QUERY_FIELD_NAMES,ri as SDKTransactionStatus,tr as SDK_VERSION,Gi as TRADING_TYPES,Er as TokenMetadataService,V as TransactionError,Yi as TransactionFailedError,G as ValidationError,Xi as WebSocketError,Zi as WebSocketTimeoutError,ye as addressFormatSchema,Qe as amountMethodSchema,He as amountTypeSchema,et as browserFileSchema,tt as bufferFileSchema,At as buyTokensDataSchema,Ct as calculatePreMintDataSchema,ze as checkPoolOptionsSchema,Le as commentMessageSchema,ct as commentPaginationSchema,ar as createLaunchpadSDK,Pe as createLimitSchema,gt as createPaginatedResultSchema,Ar as createPoolStateManager,St as createTradeDataSchema,rr as createWallet,we as ethereumAddressSchema,Se as faucetAmountSchema,yt as fetchGalaBalanceOptionsSchema,xt as fetchPoolDetailsDataSchema,vt as fetchTokenBalanceOptionsSchema,Ce as fileSizeSchema,Je as fileUploadSchema,xe as filenameSchema,fr as filterByFeeTier,dr as filterByLiquidity,hr as filterByMinLiquidity,kr as filterByPoolKey,mr as filterByToken,pr as filterByTokenPair,ke as flexibleAddressSchema,nt as flexibleFileSchema,Xo as formatGalaForDTO,Zo as formatLaunchpadTokenForDTO,fe as fullNameSchema,Ft as getAmountOptionsSchema,lr as getEnv,cr as getEnvOrThrow,It as getTradeOptionsSchema,Ze as graduateTokenOptionsSchema,Xe as graphDataOptionsSchema,wr as groupByFeeTier,vr as groupByPoolKey,yr as groupByTokenPair,ot as imageExtensionSchema,it as imageFilenameSchema,$e as imageMimeTypeSchema,je as imageUploadOptionsSchema,Uo as isLockTokenData,Mo as isUnlockTokenData,Oe as isoDateStringSchema,We as launchTokenDataSchema,sr as loadEnvWithFallback,Te as nonNegativeDecimalStringSchema,Ee as optionalUrlSchema,Ie as pageNumberSchema,ht as paginationResultMetaSchema,Ve as poolFetchTypeSchema,lt as poolPaginationSchema,be as positiveDecimalStringSchema,Be as privateKeySchema,To as requireNonNegative,bo as requirePositive,So as requirePositiveWithContext,Ge as reverseBondingCurveConfigSchema,$t as reverseBondingCurveConfigurationSchema,me as searchQuerySchema,Et as sellTokensDataSchema,gr as sortByLiquidity,Ne as standardLimitSchema,rt as standardPaginationSchema,_e as timestampSchema,qe as tokenCategorySchema,Ke as tokenCollectionSchema,ge as tokenDescriptionSchema,mt as tokenListOptionsSchema,de as tokenNameSchema,he as tokenSymbolSchema,Re as tokenUrlsSchema,Dt as tradeCalculationMethodSchema,Nt as tradeCalculationTypeSchema,Fe as tradeLimitSchema,Pt as tradeListParamsSchema,st as tradePaginationSchema,dt as tradePaginationWithFiltersSchema,Tt as tradeTypeBackendSchema,bt as tradeTypeSchema,Ue as transactionIdSchema,ft as transferFaucetsDataSchema,Me as uniqueKeySchema,wt as updateProfileDataSchema,kt as uploadProfileImageOptionsSchema,Ae as urlSchema,De as userLimitSchema,at as userPaginationSchema,pe as userTokenNameSchema,pt as userTokenTypeSchema,ut as userTokensPaginationSchema,Ut as validateAddress,Rt as validateAmountString,nn as validateBuyTokensData,cn as validateCalculatePreMintData,Ht as validateCheckPoolOptions,tn as validateCreateTradeData,qt as validateFaucetAmount,Zt as validateFetchGalaBalanceOptions,ln as validateFetchPoolDetailsData,en as validateFetchTokenBalanceOptions,Kt as validateFullName,sn as validateGetAmountOptions,rn as validateGetTradeOptions,zt as validateImageUploadOptions,jt as validateLaunchTokenData,Gt as validateSearchQuery,on as validateSellTokensData,Bt as validateTokenDescription,Qt as validateTokenListOptions,Ot as validateTokenName,_t as validateTokenSymbol,Vt as validateTokenUrls,an as validateTradeListParams,Xt as validateTransferFaucetsData,Yt as validateUpdateProfileData,Jt as validateUploadProfileImageOptions,Wt as validateUserTokenName,Mt as validateVaultAddress,ve as vaultAddressSchema};
1
+ import e from"axios";import{ethers as t,Wallet as n,Mnemonic as o,HDNodeWallet as i}from"ethers";import{SigningType as r,SigningClient as a,ChainCallDTO as s,calculatePersonalSignPrefix as c}from"@gala-chain/connect";import{z as l}from"zod";import u from"bignumber.js";import{tickToSqrtPrice as d,getLiquidityForAmounts as h,getAmountsForLiquidity as g,liquidity0 as p,liquidity1 as m,getAmount0Delta as f,getAmount1Delta as y,Pool as w,TickData as k,CompositePoolDto as v,GetCompositePoolDto as b,QuoteExactAmountDto as T,quoteExactAmount as S,sqrtPriceToTick as A,computeSwapStep as E}from"@gala-chain/dex";import{TokenClassKey as I,TokenBalance as P}from"@gala-chain/api";import{v4 as N}from"uuid";import{io as D}from"socket.io-client";import*as F from"path";import*as x from"fs";import*as C from"dotenv";import $ from"crypto";if("undefined"==typeof File){const{File:e}=require("web-file-polyfill");global.File=e}var L,B;!function(e){e.WALLET_NOT_CONNECTED="WALLET_NOT_CONNECTED",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.INVALID_ADDRESS="INVALID_ADDRESS",e.MESSAGE_GENERATION_FAILED="MESSAGE_GENERATION_FAILED"}(L||(L={}));class U extends Error{constructor(e,t,n){super(t),this.type=e,this.originalError=n,this.name="AuthError"}}class O{constructor(e){if(this.wallet=e.wallet,this.messagePrefix=e.messagePrefix||"Create a GalaChain Wallet",""===e.messagePrefix)throw new U(L.SIGNATURE_FAILED,"Message prefix cannot be empty")}hasWallet(){return void 0!==this.wallet}setWallet(e){if(void 0!==e){if("object"!=typeof e||!("address"in e))throw new U(L.WALLET_NOT_CONNECTED,"Invalid wallet: must be an ethers Wallet instance or undefined");if(!e.address||"string"!=typeof e.address)throw new U(L.INVALID_ADDRESS,"Wallet address is not available")}this.wallet=e}async generateSignature(){this.validateWallet();try{const e=Date.now(),t=`${this.messagePrefix} ${e}`,n=await this.wallet.signMessage(t);return{message:t,signature:n,address:this.formatAddress(this.wallet.address),timestamp:e}}catch(e){if(e instanceof U)throw e;throw new U(L.SIGNATURE_FAILED,"Failed to generate signature for authentication",e instanceof Error?e:new Error(String(e)))}}getAddress(){return this.validateWallet(),this.formatAddress(this.wallet.address)}getEthereumAddress(){return this.validateWallet(),this.wallet.address}getPrivateKey(){if(this.validateWallet(),!this.wallet.privateKey)throw new U(L.WALLET_NOT_CONNECTED,"Wallet private key not available for @gala-chain signing");return this.wallet.privateKey}formatAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new U(L.INVALID_ADDRESS,`Invalid Ethereum address format: ${e}`);return`eth|${t}`}async signMessage(e){this.validateWallet();try{return{message:e,signature:await this.wallet.signMessage(e),address:this.wallet.address,timestamp:Date.now()}}catch(e){if(e instanceof U)throw e;const t=e instanceof Error?e.message:String(e);throw new U(L.SIGNATURE_FAILED,t,e instanceof Error?e:new Error(String(e)))}}async generateAuthHeaders(e,t){this.validateWallet();try{const n=Date.now(),o=`${this.messagePrefix} ${t.toUpperCase()} ${e} ${n}`,i=await this.wallet.signMessage(o);return{"x-signature":i,"x-address":this.formatAddress(this.wallet.address),"x-message":o,"x-timestamp":n.toString()}}catch(e){if(e instanceof U)throw e;throw new U(L.SIGNATURE_FAILED,"Failed to generate authentication headers",e instanceof Error?e:new Error(String(e)))}}async signTypedData(e,t,n){this.validateWallet();try{return await this.wallet.signTypedData(e,t,n)}catch(e){if(e instanceof U)throw e;throw new U(L.SIGNATURE_FAILED,"Failed to sign typed data",e instanceof Error?e:new Error(String(e)))}}async generateCustomSignature(e){if(!e||"string"!=typeof e||0===e.trim().length)throw new U(L.SIGNATURE_FAILED,"Custom message must be a non-empty string");this.validateWallet();try{const t=await this.wallet.signMessage(e);return{message:e,signature:t,address:this.formatAddress(this.wallet.address),timestamp:Date.now()}}catch(e){if(e instanceof U)throw e;throw new U(L.SIGNATURE_FAILED,"Failed to generate custom message signature",e instanceof Error?e:new Error(String(e)))}}validateWallet(){if(!this.wallet)throw new U(L.WALLET_NOT_CONNECTED,"Wallet is required for authentication");if(!this.wallet.address)throw new U(L.WALLET_NOT_CONNECTED,"Wallet address is not available");if(!this.wallet.privateKey&&!this.wallet.signMessage)throw new U(L.WALLET_NOT_CONNECTED,"Wallet must have a private key for signing messages")}}!function(e){e.DEBUG="DEBUG",e.INFO="INFO",e.WARN="WARN",e.ERROR="ERROR"}(B||(B={}));class _{constructor(e){this.levelPriority={[B.DEBUG]:0,[B.INFO]:1,[B.WARN]:2,[B.ERROR]:3},this.debugEnabled=e.debug,this.context=e.context||"SDK",this.minLevel=e.minLevel||(e.debug?B.DEBUG:B.INFO)}debug(e,t){this.log(B.DEBUG,e,t)}info(e,t){this.log(B.INFO,e,t)}warn(e,t){this.log(B.WARN,e,t)}error(e,t){this.log(B.ERROR,e,t)}log(e,t,n){if(this.levelPriority[e]<this.levelPriority[this.minLevel])return;if(e===B.DEBUG&&!this.debugEnabled)return;const o=`[${(new Date).toISOString()}] [${this.context}] [${e}]`,i=this.getConsoleMethod(e);void 0!==n?n instanceof Error?i(`${o} ${t}`,n.message,n.stack):i(`${o} ${t}`,n):i(`${o} ${t}`)}getConsoleMethod(e){switch(e){case B.DEBUG:return console.debug;case B.INFO:return console.info;case B.WARN:return console.warn;case B.ERROR:return console.error;default:return console.log}}child(e){return new _({debug:this.debugEnabled,context:`${this.context}:${e}`,minLevel:this.minLevel})}isDebugEnabled(){return this.debugEnabled&&this.levelPriority[B.DEBUG]>=this.levelPriority[this.minLevel]}}function M(e){if(!e||"object"!=typeof e)return{};const t={};for(const[n,o]of Object.entries(e))null!=o&&("string"==typeof o?t[n]=o:"number"==typeof o||"boolean"==typeof o?t[n]=o.toString():Array.isArray(o)?t[n]=o.join(","):t[n]="object"==typeof o?JSON.stringify(o):String(o));return t}class R{constructor(t,n={}){this.auth=t,this.debug=n.debug??!1,this.logger=new _({debug:this.debug,context:"HttpClient"}),this.axios=e.create({baseURL:n.baseUrl||"https://lpad-backend-dev1.defi.gala.com",timeout:n.timeout||3e4,headers:{Accept:"application/json",...n.headers}}),this.setupInterceptors()}async request(e){try{const t={method:e.method,url:e.url,data:e.data,...e.params&&{params:M(e.params)},...e.headers&&{headers:e.headers},...e.timeout&&{timeout:e.timeout}};e.headers&&this.logger.debug("Custom headers provided:",e.headers),e.data instanceof FormData&&(t.headers&&t.headers["Content-Type"]&&delete t.headers["Content-Type"],this.logger.debug("FormData detected - removing Content-Type header for multipart upload"));const n=e.data instanceof FormData?"[FormData object - multipart/form-data]":e.data;this.logger.debug("Request:",{method:e.method,url:e.url,fullUrl:`${this.axios.defaults.baseURL}${e.url}`,baseURL:this.axios.defaults.baseURL,params:t.params,data:n,isFormData:e.data instanceof FormData,contentType:t.headers?.["Content-Type"]||"not set"});const o=await this.axios.request(t);return this.logger.debug("Response:",{status:o.status,data:o.data}),o.data}catch(e){throw this.logger.error("Error:",e),e}}async get(e,t,n){return this.request({method:"GET",url:e,...t&&{params:t},...n&&{headers:n}})}async post(e,t,n){return this.request({method:"POST",url:e,data:t,...n&&{headers:n}})}async put(e,t,n){return this.request({method:"PUT",url:e,data:t,...n&&{headers:n}})}async delete(e,t,n){return this.request({method:"DELETE",url:e,...t&&{params:t},...n&&{headers:n}})}async patch(e,t,n){return this.request({method:"PATCH",url:e,data:t,...n&&{headers:n}})}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.auth.getEthereumAddress()}async signMessage(e){return(await this.auth.signMessage(e)).signature}async signTypedData(e,t,n){return await this.auth.signTypedData(e,t,n)}async signCustomMessage(e){try{const t=await this.auth.generateCustomSignature(e);return this.logger.debug("Generated custom signature:",{message:e,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}),{signature:t.signature,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}}catch(e){throw this.logger.error("Custom signature generation failed:",e),new Error(`Failed to generate custom signature for message: ${e instanceof Error?e.message:"Unknown error"}`)}}async signWithGalaChain(e,t,n=r.SIGN_TYPED_DATA){const o=this.auth.getPrivateKey(),i=new a(o);return await i.sign(e,t,n)}setupInterceptors(){this.requestInterceptorId=this.axios.interceptors.request.use(async e=>{try{if(e.headers||(e.headers={}),this.auth.hasWallet()){const t=await this.auth.generateSignature();e.headers.Sign=t.signature,this.logger.debug("Added signature header:",{address:t.address,message:t.message,timestamp:t.timestamp})}else this.logger.debug("No wallet configured - skipping signature header");return e.data instanceof FormData||(e.headers["Content-Type"]="application/json"),this.logger.debug("Final request headers being sent:",e.headers),e}catch(e){throw this.logger.error("Failed to add signature:",e),e}},e=>Promise.reject(e)),this.responseInterceptorId=this.axios.interceptors.response.use(e=>e,e=>{if(e.response){const t={message:e.response.data?.message||e.message,error:e.response.data?.error,statusCode:e.response.status,details:e.response.data?.details,timestamp:e.response.data?.timestamp,path:e.response.data?.path};e.launchpadError=t,this.logger.error("Backend error:",t)}else e.request?this.logger.error("Network error:",e.message):this.logger.error("Request setup error:",e.message);return Promise.reject(e)})}cleanup(){void 0!==this.requestInterceptorId&&(this.axios.interceptors.request.eject(this.requestInterceptorId),this.requestInterceptorId=void 0),void 0!==this.responseInterceptorId&&(this.axios.interceptors.response.eject(this.responseInterceptorId),this.responseInterceptorId=void 0),this.logger.debug("Interceptors cleaned up")}}const q="Token name is required and must be a string",K=e=>`Could not find vault address for token: ${e}`;class G extends Error{constructor(e,t,n){super(e),this.field=t,this.code=n,this.name="ValidationError"}}class W extends Error{constructor(e,t,n){super(e),this.statusCode=t,this.originalError=n,this.name="NetworkError"}}class j extends Error{constructor(e,t){super(e),this.field=t,this.name="ConfigurationError"}}class H extends Error{constructor(e,t,n){super(e),this.transactionId=t,this.code=n,this.name="TransactionError"}}class z extends Error{constructor(e,t,n){super(e),this.originalError=t,this.code=n,this.name="GSwapQuoteError"}}class V extends Error{constructor(e,t,n,o){super(e),this.originalError=t,this.transactionHash=n,this.code=o,this.name="GSwapSwapError"}}class Q extends Error{constructor(e,t,n,o,i){super(e),this.originalError=t,this.tokenA=n,this.tokenB=o,this.code=i,this.name="GSwapPoolError"}}class X extends Error{constructor(e,t,n,o){super(e),this.originalError=t,this.walletAddress=n,this.code=o,this.name="GSwapAssetError"}}class Z extends Error{constructor(e,t,n){super(e),this.originalError=t,this.code=n,this.name="GSwapPositionError"}}class Y extends G{constructor(e,t){super(e,"dexQuote","DEX_QUOTE_ERROR"),this.context=t,this.name="DexQuoteError"}}class J extends G{constructor(e){super(e,"dexPool","DEX_POOL_NOT_FOUND"),this.name="DexPoolNotFoundError"}}function ee(e,t){return new G(`Token "${e}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND")}function te(e,t){const n=t||e.charAt(0).toUpperCase()+e.slice(1);return new G(`${n} is required`,e,"REQUIRED_FIELD")}function ne(e,t,n){const o=n||e.charAt(0).toUpperCase()+e.slice(1);return new G(`${o} must be ${t}`,e,"INVALID_FORMAT")}function oe(e,t,n){return new W(e,t,n)}function ie(e,t){return new j(e,t)}function re(e,t,n){return new H(e,t,n)}function ae(e,t,n){const{MIN_PAGE:o,MAX_PAGE:i,MIN_LIMIT:r,MAX_LIMIT:a}=n.PAGINATION;if("number"!=typeof e||e<o||e>i)throw new G(`Page must be a number between ${o} and ${i}`,"page","INVALID_PAGE");if("number"!=typeof t||t<r||t>a)throw new G(`Limit must be a number between ${r} and ${a}`,"limit","INVALID_LIMIT")}const se={ETH_ADDRESS:/^0x[0-9a-fA-F]{40}$/,BACKEND_ADDRESS:/^eth\|[0-9a-fA-F]{40}$/};function ce(e){return"string"==typeof e&&e.trim().length>0}function le(e){return!(!e||"string"!=typeof e)&&(se.ETH_ADDRESS.test(e)||se.BACKEND_ADDRESS.test(e))}function ue(e){return e.startsWith("0x")?`eth|${e.slice(2)}`:e}const de=l.string().min(3,"Token name must be at least 3 characters").max(20,"Token name must be at most 20 characters").regex(/^[a-zA-Z0-9]{3,20}$/,"Token name can only contain letters and numbers"),he=l.string().min(1,"Token symbol must be at least 1 character").max(8,"Token symbol must be at most 8 characters").regex(/^[A-Z]{1,8}$/,"Token symbol must be uppercase letters only"),ge=l.string().min(1,"Token description is required").max(500,"Token description must be at most 500 characters"),pe=l.string().min(1,"Token name must be at least 1 character").max(50,"Token name must be at most 50 characters"),me=l.string().min(1,"Search query must be at least 1 character").max(100,"Search query must be at most 100 characters"),fe=l.string().min(1,"Full name is required").max(100,"Full name must be at most 100 characters").regex(/^[a-zA-Z\s]+$/,"Full name can only contain letters and spaces"),ye=l.string().regex(se.BACKEND_ADDRESS,"Address must be in format: eth|[40-hex-chars]"),we=l.string().regex(se.ETH_ADDRESS,"Invalid Ethereum address format"),ke=l.string().refine(e=>se.BACKEND_ADDRESS.test(e)||se.ETH_ADDRESS.test(e),"Address must be either eth|[40-hex-chars] or 0x[40-hex-chars] format").transform(e=>e.startsWith("0x")?`eth|${e.slice(2)}`:e),ve=l.string().refine(e=>se.BACKEND_ADDRESS.test(e)||/^service\|Token\$Unit\$[A-Z0-9]+\$eth:[0-9a-fA-F]{40}\$launchpad$/.test(e),"Invalid vault address format"),be=l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>0,"Amount must be greater than zero"),Te=l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>=0,"Amount must be zero or greater"),Se=l.string().url("Must be a valid URL").regex(/^https?:\/\//,"URL must start with http:// or https://"),Ae=l.string().optional().refine(e=>!e||/^https?:\/\/.+\..+/.test(e),"Must be a valid URL if provided"),Ee=l.number().int("Page must be an integer").min(1,"Page must be at least 1").max(1e3,"Page must be at most 1000").default(1);function Ie(e=100){return l.number().int("Limit must be an integer").min(1,"Limit must be at least 1").max(e,`Limit must be at most ${e}`).default(10)}const Pe=Ie(100),Ne=Ie(20),De=Ie(20),Fe=l.number().int("File size must be an integer").min(1,"File must be at least 1 byte").max(10485760,"File must be at most 10MB"),xe=l.string().max(255,"Filename must be at most 255 characters"),Ce=l.enum(["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"]),$e=l.string().datetime("Must be a valid ISO 8601 date string"),Le=l.number().int("Timestamp must be an integer").min(0,"Timestamp must be non-negative"),Be=l.string().regex(/^0x[a-fA-F0-9]{64}$/,"Private key must be format: 0x + 64 hex characters"),Ue=l.string().regex(/^[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$/,"Transaction ID must be in UUID format"),Oe=l.string().regex(/^galaconnect-operation-[a-z0-9-]+$/,"Unique key must be format: galaconnect-operation-{unique-id}"),_e=l.object({websiteUrl:Ae,telegramUrl:Ae,twitterUrl:Ae}).refine(e=>e.websiteUrl||e.telegramUrl||e.twitterUrl,"At least one social URL (website, telegram, or twitter) is required"),Me=l.string().min(1,"Token category must not be empty").default("Unit"),Re=l.string().min(1,"Token collection must not be empty").default("Token"),qe=l.object({minFeePortion:l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal string").refine(e=>parseFloat(e)>=.1,"Minimum fee must be >= 0.1").refine(e=>parseFloat(e)<=.5,"Minimum fee must be <= 0.5"),maxFeePortion:l.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal string").refine(e=>parseFloat(e)>=.1,"Maximum fee must be >= 0.1").refine(e=>parseFloat(e)<=.5,"Maximum fee must be <= 0.5")}).refine(e=>parseFloat(e.maxFeePortion)>=parseFloat(e.minFeePortion),{message:"Maximum fee must be >= minimum fee",path:["maxFeePortion"]}),Ke=l.object({tokenName:de,tokenSymbol:he,tokenDescription:ge,tokenImage:l.union([l.instanceof(File),l.instanceof(Buffer),l.string().url("Token image must be a valid URL")]).optional(),preBuyQuantity:Te.default("0"),websiteUrl:Ae,telegramUrl:Ae,twitterUrl:Ae,tokenCategory:Me,tokenCollection:Re,reverseBondingCurveConfiguration:qe.optional(),privateKey:Be.optional()}),Ge=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),tokenName:de}),We=l.enum(["recent","popular"]),je=l.object({tokenName:de.optional(),symbol:he.optional()}).refine(e=>e.tokenName||e.symbol,"At least one of tokenName or symbol is required"),He=l.enum(["NATIVE","MEME"]),ze=l.enum(["IN","OUT"]),Ve=l.object({from:l.number().int("From timestamp must be an integer").min(173e6,"From timestamp must be at least 173000000"),to:l.number().int("To timestamp must be an integer").min(173e6,"To timestamp must be at least 173000000"),resolution:l.number().int("Resolution must be an integer").min(1,"Resolution must be at least 1"),tokenName:de}),Qe=l.object({tokenName:de,slippageToleranceFactor:l.number().min(0).max(1).optional(),maxAcceptableReverseBondingCurveFeeSlippageFactor:l.number().min(0).max(1).optional(),privateKey:Be.optional()}),Xe=[".png",".jpg",".jpeg",".gif",".webp",".svg"],Ze=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),name:xe,size:Fe,type:Ce}),Ye=l.instanceof(File).refine(e=>e.size>=1&&e.size<=10485760,"File size must be between 1 byte and 10MB").refine(e=>["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"].includes(e.type),"File must be a valid image type (PNG, JPG, JPEG, GIF, WebP, or SVG)").refine(e=>e.name.length<=255,"Filename must be at most 255 characters"),Je=l.instanceof(Buffer).refine(e=>e.length>=1&&e.length<=10485760,"Buffer size must be between 1 byte and 10MB"),et=l.union([Ye,Je]),tt=l.enum([".png",".jpg",".jpeg",".gif",".webp",".svg"]),nt=xe.refine(e=>{const t=e.slice(e.lastIndexOf(".")).toLowerCase();return Xe.includes(t)},`Filename must end with one of: ${Xe.join(", ")}`),ot=l.object({page:Ee,limit:Pe}),it=l.object({page:Ee,limit:Ne}),rt=l.object({page:Ee,limit:De}),at=ot.extend({type:l.enum(["recent","popular"]).optional(),tokenName:l.string().min(1).max(50).optional(),search:l.string().min(1).max(100).optional()}),st=it.extend({tokenName:l.string().min(1).max(50).optional(),search:l.string().min(1).max(100).optional()}),ct=rt.extend({tradeType:l.enum(["BUY","SELL"]).optional(),tokenName:l.string().min(1).max(50).optional(),userAddress:l.string().regex(/^(0x[a-fA-F0-9]{40}|eth\|[a-fA-F0-9]{40})$/).optional(),startDate:l.string().datetime().optional(),endDate:l.string().datetime().optional(),sortOrder:l.enum(["ASC","DESC"]).default("DESC")}),lt=l.object({page:l.number().int().min(1),limit:l.number().int().min(1),total:l.number().int().min(0),totalPages:l.number().int().min(0),hasNext:l.boolean(),hasPrevious:l.boolean()});function ut(e){return l.object({data:l.array(e),page:l.number().int().min(1),limit:l.number().int().min(1),total:l.number().int().min(0),totalPages:l.number().int().min(0),hasNext:l.boolean(),hasPrevious:l.boolean()})}const dt=l.enum(["all","DEFI","ASSET"]),ht=it.extend({type:dt.optional(),address:ke.optional(),search:me.optional(),tokenName:pe.optional()}),gt=l.object({address:ke.optional(),refresh:l.boolean().optional()}),pt=l.object({profileImage:l.string(),fullName:fe,address:ke,privateKey:Be.optional()}),mt=l.object({file:l.union([l.instanceof(File),l.instanceof(Buffer)]),address:ke.optional(),privateKey:Be.optional()}),ft=l.object({created:l.number(),createdBy:l.string(),expires:l.number(),instanceId:l.string(),lockAuthority:l.string(),name:l.string(),quantity:l.string(),vestingPeriodStart:l.number()}),yt=l.object({address:ke,tokenId:l.union([l.string(),l.object({collection:l.string(),category:l.string(),type:l.string(),additionalKey:l.string()}),l.object({collection:l.string(),category:l.string(),type:l.string(),additionalKey:l.string(),instance:l.string()})]).optional(),tokenName:pe.optional(),withExpired:l.boolean().optional()}).refine(e=>void 0!==e.tokenId||void 0!==e.tokenName,"At least one token identifier (tokenId or tokenName) is required"),wt=l.enum(["buy","sell"]),kt=l.enum(["BUY","SELL"]),vt=l.object({tradeType:wt,tokenAmount:be,vaultAddress:ve,userAddress:ke,slippageTolerance:be.optional(),deadline:l.number().int().positive().optional()}),bt=l.object({tokenSymbol:he,nativeTokenQuantity:be,expectedToken:be,maxAcceptableReverseBondingCurveFee:Te.default("0").optional()}),Tt=l.object({tokenSymbol:he,tokenQuantity:be,expectedNativeToken:be,maxAcceptableReverseBondingCurveFee:Te.default("0").optional()}),St=rt.extend({tokenName:pe.optional()}),At=l.object({page:l.number().int().min(1).max(1e3).default(1).optional(),limit:l.number().int().min(1).max(20).default(10).optional()}),Et=l.enum(["NATIVE","MEME"]),It=l.enum(["IN","OUT"]),Pt=l.object({type:Et,method:It,vaultAddress:ve,amount:be}),Nt=l.object({nativeTokenQuantity:be}),Dt=l.object({vaultAddress:ve}),Ft=l.object({minFeePortion:be,maxFeePortion:be});function xt(e){return t=>{const n=e.safeParse(t);return{success:n.success,data:n.success?n.data:void 0,errors:n.success?void 0:n.error.errors.map(e=>e.message)}}}const Ct=xt(de),$t=xt(he),Lt=xt(ge),Bt=xt(ke),Ut=xt(ve),Ot=xt(be),_t=xt(fe),Mt=xt(me),Rt=xt(pe),qt=xt(Ke),Kt=xt(_e),Gt=xt(Ge),Wt=xt(je),jt=xt(ht),Ht=xt(gt),zt=xt(pt),Vt=xt(mt),Qt=xt(yt),Xt=xt(vt),Zt=xt(bt),Yt=xt(Tt),Jt=xt(St),en=xt(At),tn=xt(Pt),nn=xt(Nt),on=xt(Dt);function rn(e,t){throw new G(e.join("; "),t,"VALIDATION_ERROR")}function an(e){const t=Ct(e);!t.success&&t.errors&&rn(t.errors,"tokenName")}function sn(e){const t=at.safeParse(e);t.success||rn(t.error.errors.map(e=>e.message),"pagination")}function cn(e){const t=Wt(e);!t.success&&t.errors&&rn(t.errors,"options")}function ln(e){const t=tn(e);!t.success&&t.errors&&rn(t.errors,"options")}function un(e){const t=Ve.safeParse(e);t.success||rn(t.error.errors.map(e=>e.message),"options")}function dn(e){const t=ke.safeParse(e);if(!t.success)throw new G("Ethereum address must be 40 hex characters (with or without 0x prefix)","ethereumAddress","INVALID_FORMAT");return t.data}function hn(e,t,n=!0){if(!e||""===e.trim())throw new G(`${t} cannot be empty or whitespace-only. Provide a valid numeric string or omit the parameter to auto-fetch.`,t,"INVALID_NUMERIC_STRING");if(/[eE]/.test(e))throw new G(`${t} cannot use scientific notation. Use standard decimal format (e.g., "1000" instead of "1e3").`,t,"INVALID_NUMERIC_STRING");const o=parseFloat(e);if(isNaN(o))throw new G(`${t} must be a valid numeric string. Received: "${e}"`,t,"INVALID_NUMERIC_STRING");if(!isFinite(o))throw new G(`${t} must be a finite number. Cannot be Infinity or -Infinity.`,t,"INVALID_NUMERIC_STRING");if(o<0)throw new G(`${t} must be non-negative. Received: "${e}"`,t,"INVALID_NUMERIC_STRING");if(!n&&0===o)throw new G(`${t} must be greater than zero. Received: "${e}"`,t,"INVALID_NUMERIC_STRING")}var gn=Object.freeze({__proto__:null,normalizeAddressInput:function(e){if(!e)return;const t=ke.safeParse(e);if(!t.success)throw new G(`Invalid address format: ${e}. Must be either "0x..." (Ethereum) or "eth|..." (GalaChain) format`,"address","INVALID_FORMAT");return t.data},toBackendAddressFormat:dn,validateCheckPoolOptions:cn,validateGetAmountOptions:ln,validateGetGraphOptions:un,validateNumericString:hn,validatePagination:sn,validateTokenName:an});function pn(e,t){const n=e,o=Number(n.page)||t.page,i=Number(n.limit)||t.limit,r=n.data,a=Number(n.total)||Number(r?.count)||0;return{page:o,limit:i,total:a,totalPages:Math.ceil(a/i)}}function mn(e,t){return{hasNext:e<t,hasPrevious:e>1}}function fn(e,t,n=!1){const o=!0===e.error||200!==e.status,i=n&&!e.data;if(o||i)throw new Error(e.message||t)}const yn="/launchpad/upload-image",wn="/launchpad/fetch-pool",kn="/launchpad/check-pool",vn="/launchpad/get-graph-data",bn="/holders",Tn="/launchpad/get-badge/",Sn="/trade/",An="/user/profile",En="/user/profile",In="/user/token-list",Pn="/user/token-hold",Nn=5,Dn={DEFAULT_PAGE:1,DEFAULT_LIMIT:10,BACKEND_MAX_PAGE_SIZE:20,SAFETY_MAX_PAGES:100},Fn={MAX_CONCURRENT_FETCHES:5,BACKEND_PAGE_SIZE:20};class xn{constructor(e,t=!1){this.http=e,this.logger=new _({debug:t,context:this.constructor.name})}}class Cn{constructor(e=!1){this.logger=new _({debug:e,context:this.constructor.name})}}function $n(e,t){return"string"==typeof e[t]}function Ln(e,t){return void 0===e[t]||"string"==typeof e[t]}function Bn(e,t){return void 0===e[t]||"number"==typeof e[t]}function Un(e){return void 0===e.calculateAmountMode||"local"===e.calculateAmountMode||"external"===e.calculateAmountMode}function On(e){if(!e||"object"!=typeof e)return!1;const t=e;return $n(t,"tokenName")&&Bn(t,"from")&&Bn(t,"to")&&Bn(t,"resolution")}class _n extends xn{constructor(e,t=!1){super(e,t)}async fetchSinglePage(e){const t={page:e.page.toString(),limit:e.limit.toString()};void 0!==e.type&&(t.type=e.type),void 0!==e.tokenName&&(t.tokenName=e.tokenName),void 0!==e.search&&(t.search=e.search);const n=M(t),o=await this.http.get(wn,n);if(!o)throw oe("No response from pool service",500);fn(o,"Failed to fetch pools",!0);const i=function(e){if(!e)return[];let t=[];if(e.tokens)if(Array.isArray(e.tokens))t=e.tokens.map(e=>({...e,createdAt:e.created_at||e.createdAt||""}));else{const n=e.tokens;t=[{...n,createdAt:n.created_at||n.createdAt||""}]}else e.pools&&Array.isArray(e.pools)&&(t=e.pools.map(e=>({...e,createdAt:e.created_at||e.createdAt||""})));return t}(o.data),r=o.data.count??o.data.total??0;return{pools:i,total:r,totalPages:e.limit>0?Math.ceil(r/e.limit):1}}async fetchMultiplePages(e){const{startPage:t,totalPages:n,pageSize:o,...i}=e,r=[];if(n&&n>=t){const e=[];for(let o=t;o<=n;o++)e.push(o);for(let t=0;t<e.length;t+=5){const n=e.slice(t,t+5).map(e=>this.fetchSinglePage({...i,page:e,limit:o}).catch(e=>{if(400===e?.launchpadError?.statusCode)return{pools:[],total:0,totalPages:0};throw e}));(await Promise.all(n)).forEach(e=>{r.push(...e.pools)})}return r}let a=t,s=!0;for(;s;){const e=[];for(let t=0;t<5&&s;t++)e.push(a+t);const t=e.map(e=>this.fetchSinglePage({...i,page:e,limit:o}).catch(e=>{if(400===e?.launchpadError?.statusCode)return{pools:[],total:0,totalPages:0};throw e})),n=await Promise.all(t);for(const e of n){if(0===e.pools.length){s=!1;break}r.push(...e.pools)}a+=e.length,a>Dn.SAFETY_MAX_PAGES&&(s=!1)}return r}async fetchPools(e={}){const t=e.page||Dn.DEFAULT_PAGE,n=e.limit??Dn.DEFAULT_LIMIT;let o;if(0!==n&&sn({page:t,limit:n}),e.tokenName&&an(e.tokenName),"recent"===e.type?o="RECENT":"popular"===e.type&&(o="POPULAR"),n>0&&n<=20){const i=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:t,limit:n});return{pools:i.pools,page:t,limit:n,total:i.total,totalPages:i.totalPages,hasNext:t<i.totalPages,hasPrevious:t>1}}if(0===n){const t=Dn.BACKEND_MAX_PAGE_SIZE,n=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:1,limit:t}),i=[...n.pools];if(n.pools.length<t)return{pools:i,page:1,limit:i.length,total:i.length,totalPages:1,hasNext:!1,hasPrevious:!1};if(n.pools.length>0){const r=await this.fetchMultiplePages({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},startPage:2,totalPages:n.totalPages>1?n.totalPages:null,pageSize:t});i.push(...r)}return{pools:i,page:1,limit:i.length,total:n.total||i.length,totalPages:1,hasNext:!1,hasPrevious:!1}}const i=Dn.BACKEND_MAX_PAGE_SIZE,r=Math.ceil(n/i),a=await this.fetchSinglePage({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},page:t,limit:i}),s=[...a.pools],c=Math.min(r,a.totalPages-t+1);if(c>1){const n=t+c-1,r=await this.fetchMultiplePages({...e.search&&{search:e.search},...e.tokenName&&{tokenName:e.tokenName},...o&&{type:o},startPage:t+1,totalPages:n,pageSize:i});s.push(...r)}const l=s.slice(0,n);return{pools:l,page:t,limit:n,total:a.total,totalPages:a.totalPages,hasNext:t<a.totalPages&&l.length<a.total,hasPrevious:t>1}}async fetchAllPools(e){return this.fetchPools({...e,limit:0})}async checkPool(e){cn(e),e.tokenName&&an(e.tokenName);const t=M(e),n=await this.http.get(kn,t);if(!n)throw oe("No response from pool service",500);fn(n,"Failed to check pool");const o=n.data;return e.symbol?o?.isSymbolExist??!1:e.tokenName?o?.isNameExist??!1:o?.exists??!1}async isTokenNameAvailable(e){try{return!await this.checkPool({tokenName:e})}catch{return!1}}async isTokenSymbolAvailable(e){try{return!await this.checkPool({symbol:e})}catch{return!1}}async fetchVolumeData(e){if(!On(e))throw new G("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:n,to:o,resolution:i}=e;if(an(t),!n||!o||!i)throw new G("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const r={tokenName:t,from:n,to:o,resolution:i};un(r);const a=M(r),s=await this.http.get(vn,a);if(!s)throw oe("No response from pool service",500);return fn(s,"Failed to fetch graph data",!0),{dataPoints:s.data}}async fetchTokenDistribution(e){if(!e)throw te("tokenName","Token name");an(e);const t=await this.resolveTokenNameToVault(e);if(!t)throw ee(e);const n=encodeURIComponent(t);let o;try{o=await this.http.get(`${bn}/${n}`)}catch(t){if(t&&"object"==typeof t&&"response"in t){const n=t;if(500===n.response?.status)throw oe(`Token distribution data temporarily unavailable for ${e}. This is a backend issue - please try again later.`,500)}throw t}if(!o)throw oe("No response from pool service",500);fn(o,"Failed to fetch token distribution",!0);const i=o.data;if(!Array.isArray(i))throw oe("Invalid API response: expected array of holders",o.status);for(const e of i){if(!e.owner||"string"!=typeof e.owner)throw oe("Invalid holder data: missing or invalid owner field",o.status);if(!e.quantity||"string"!=typeof e.quantity)throw oe("Invalid holder data: missing or invalid quantity field",o.status);const t=parseFloat(e.quantity);if(isNaN(t)||!isFinite(t))throw oe(`Invalid holder quantity: "${e.quantity}"`,o.status)}const r=i.reduce((e,t)=>e.plus(t.quantity),new u(0));return{holders:i.map(e=>{const t=new u(e.quantity),n=r.isZero()?0:t.dividedBy(r).multipliedBy(100).toNumber();return{address:e.owner,balance:e.quantity,percentage:n}}),totalSupply:r.toFixed(),totalHolders:i.length,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw te("tokenName","Token name");an(e);const t=await this.http.get(Tn,{tokenName:e});if(!t)throw oe("No response from pool service",500);return fn(t,"Failed to fetch token badges",!0),{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadge(e){const{tokenName:t,badgeType:n,badgeName:o}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const i=("volume"===n?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===o);return i?.isActive||!1}catch{return!1}}async resolveTokenNameToVault(e){try{const t=await this.fetchPools({tokenName:e});if(t.pools&&Array.isArray(t.pools)&&t.pools.length>0)return t.pools[0].vaultAddress||null;if(t.pools&&"object"==typeof t.pools){const e=t.pools.tokens;return e?.vaultAddress||null}return null}catch{return null}}}function Mn(e,t={}){const{stringifyFields:n=[],optionalFields:o=[],fieldMappings:i={}}=t,r={};for(const[t,a]of Object.entries(e)){const e=t;if(o.includes(e)&&void 0===a)continue;if(o.includes(e)&&"string"==typeof a&&0===a.trim().length)continue;const s=i[e],c=s?String(s):t;n.includes(e)?r[c]=String(a):r[c]=a}return M(r)}const Rn={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20}};class qn extends xn{constructor(e,t=!1){super(e,t)}async fetchTrades(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return $n(t,"tokenName")&&(void 0===t.tradeType||"buy"===t.tradeType||"sell"===t.tradeType)&&Ln(t,"userAddress")&&Bn(t,"page")&&Bn(t,"limit")}(e))throw new G("Invalid options provided. Expected { tokenName: string, tradeType?: string, userAddress?: string, page?: number, limit?: number, startDate?: Date, endDate?: Date, sortOrder?: string }","options","INVALID_OPTIONS");const{tokenName:t,tradeType:n,userAddress:o,page:i=Dn.DEFAULT_PAGE,limit:r=Dn.DEFAULT_LIMIT,startDate:a,endDate:s,sortOrder:c}=e;if(!ce(t))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");ae(i,r,Rn);const l=function(e,t,n){return Mn({tokenName:e,page:t,limit:n},{stringifyFields:["page","limit"]})}(t,i,r),u=await this.http.get(Sn,l);if(!u)throw new G("No response from trade service","response","NO_RESPONSE");const d=(h=u.data)?Array.isArray(h)?h:h.trades:[];var h;const g=pn(u,{page:i,limit:r}),p=mn(g.page,g.totalPages);return{trades:d,...g,...p}}}function Kn(e){return e instanceof Error}function Gn(e){return Kn(e)||function(e){return"object"==typeof e&&null!==e&&"message"in e&&"string"==typeof e.message}(e)?e.message:"string"==typeof e?e:String(e)}function Wn(e){return"object"==typeof e&&null!==e&&"message"in e&&("response"in e||"request"in e||"config"in e)}function jn(e,t="image",n){const o=new FormData;if("undefined"!=typeof File&&e instanceof File)o.append(t,e);else{if(!Buffer.isBuffer(e))throw ne("file","a File object (browser) or Buffer (Node.js)");{const i=new Blob([e],{type:"image/png"});o.append(t,i,n)}}return o}const Hn={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20},USER_ADDRESS:{PATTERN:/^eth\|[0-9a-fA-F]{40}$/},TOKEN_NAME:{MIN_LENGTH:1,MAX_LENGTH:50},SEARCH:{MIN_LENGTH:1,MAX_LENGTH:100},PROFILE:{FULL_NAME:{MIN_LENGTH:1,MAX_LENGTH:100,ALPHABETS_ONLY_PATTERN:/^[a-zA-Z]+(?:\s[a-zA-Z]+)?$/}}};function zn(e){return!(!e||"string"!=typeof e)&&Hn.USER_ADDRESS.PATTERN.test(e)}const Vn="Update profile",Qn="Upload profile image";class Xn extends xn{constructor(e,t=!1){super(e,t)}async fetchProfile(e){const t=e??this.http.getAddress();if(!t||!zn(t))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");const n={userAddress:t},o=await this.http.get(An,n);if(!o)throw new G("No response from user service","response","NO_RESPONSE");return o}async updateProfile(e){this.validateUpdateProfileData(e);let t=e.profileImage;if(!t||""===t.trim())try{const n=await this.fetchProfile(e.address);t=n.data?.profileImage||""}catch{t=""}const n={profileImage:t,fullName:e.fullName,userAddress:e.address},o=await this.http.signCustomMessage(Vn);if(!o)throw new G("Failed to generate signature - wallet not configured","signature","NO_SIGNATURE");const i={address:o.address,message:Vn,publickey:o.ethereumAddress,sign:o.signature},r=await this.http.put(En,n,i);if(!r)throw new G("No response from user service","response","NO_RESPONSE");fn(r,"Profile update failed")}async uploadProfileImage(e){this.validateUploadProfileImageOptions(e);const t=e.address??this.http.getAddress();if(!t)throw new G("Wallet address not available - wallet not configured","address","NO_WALLET");try{const n=`profile-image-${t}.png`,o=jn(e.file,"image",n),i=await this.http.signCustomMessage(Qn);if(!i)throw new G("Failed to generate signature - wallet not configured","signature","NO_SIGNATURE");const r={address:i.address,message:Qn,publickey:i.ethereumAddress,sign:i.signature},a=await this.http.request({method:"POST",url:`${yn}?tokenName=${encodeURIComponent(t)}`,data:o,headers:r});if(!a)throw new G("No response from user service","response","NO_RESPONSE");return fn(a,"Image upload failed"),"string"==typeof a.data?a.data:""}catch(e){if(e instanceof G)throw e;throw new G(`Profile image upload failed: ${Gn(e)}`,"file","UPLOAD_FAILED")}}async fetchTokenList(e){return this.buildFetchRequest(In,e,{includeType:!0,errorMessage:"Failed to fetch token list"})}async fetchTokensHeld(e){return this.buildFetchRequest(Pn,e,{includeType:!1,errorMessage:"Failed to fetch tokens held"})}async fetchTokensCreated(e={}){const{page:t=Dn.DEFAULT_PAGE,limit:n=Dn.DEFAULT_LIMIT,search:o,tokenName:i}=e,r=this.http.getAddress();if(!r)throw new G("Wallet address not available - wallet not configured","address","NO_WALLET");const a={type:"DEFI",address:r,page:t,limit:n};return void 0!==o&&(a.search=o),void 0!==i&&(a.tokenName=i),this.fetchTokenList(a)}async buildFetchRequest(e,t,n){this.validateGetTokenListOptions(t);const o={page:t.page,limit:t.limit,address:t.address,search:t.search,tokenName:t.tokenName},i=Mn(n.includeType?{...o,type:"all"!==t.type&&t.type?t.type:"DEFI"}:o,{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),r=await this.http.get(e,i);if(!r)throw new G("No response from user service","response","NO_RESPONSE");fn(r,n.errorMessage,!0);const a=(s=r.data)?Array.isArray(s)?s:s.token:[];var s;const c=pn(r,{page:t.page||Dn.DEFAULT_PAGE,limit:t.limit||Dn.DEFAULT_LIMIT}),l=mn(c.page,c.totalPages);return{tokens:a,...c,...l}}validateGetTokenListOptions(e){if(ae(e.page,e.limit,Hn),void 0!==e.address&&!zn(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(void 0!==e.search&&e.search.trim().length>0&&!((t=e.search)&&"string"==typeof t&&t.length>=Hn.SEARCH.MIN_LENGTH&&t.length<=Hn.SEARCH.MAX_LENGTH))throw new G(`Search query must be between ${Hn.SEARCH.MIN_LENGTH} and ${Hn.SEARCH.MAX_LENGTH} characters`,"search","INVALID_SEARCH");var t,n;if(void 0!==e.tokenName&&e.tokenName.trim().length>0&&!((n=e.tokenName)&&"string"==typeof n&&n.length>=Hn.TOKEN_NAME.MIN_LENGTH&&n.length<=Hn.TOKEN_NAME.MAX_LENGTH))throw new G(`Token name must be between ${Hn.TOKEN_NAME.MIN_LENGTH} and ${Hn.TOKEN_NAME.MAX_LENGTH} characters`,"tokenName","INVALID_TOKEN_NAME")}validateUpdateProfileData(e){if(!zn(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(!((t=e.fullName)&&"string"==typeof t&&t.length>=Hn.PROFILE.FULL_NAME.MIN_LENGTH&&t.length<=Hn.PROFILE.FULL_NAME.MAX_LENGTH&&Hn.PROFILE.FULL_NAME.ALPHABETS_ONLY_PATTERN.test(t)))throw new G(`Full name must be between ${Hn.PROFILE.FULL_NAME.MIN_LENGTH} and ${Hn.PROFILE.FULL_NAME.MAX_LENGTH} characters`,"fullName","INVALID_FULL_NAME");var t}validateUploadProfileImageOptions(e){if(e.address&&!zn(e.address))throw new G("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS")}}class Zn extends Error{constructor(e,t,n){super(e),this.filename=t,this.mimeType=n,this.name="FileValidationError"}}function Yn(e,t,n){if(!e)throw new Zn("File is required",t,n);if("undefined"!=typeof File&&e instanceof File){const t=Ye.safeParse(e);if(!t.success){const n=t.error.errors.map(e=>e.message).join("; ");throw new Zn(n,e.name,e.type)}return}if(Buffer.isBuffer(e)){if(!t)throw new Zn("Filename is required when uploading Buffer objects",t,n);const o=Je.safeParse(e);if(!o.success){const e=o.error.errors.map(e=>e.message).join("; ");throw new Zn(e,t,n)}if(t.length>255)throw new Zn(`Filename length ${t.length} exceeds maximum allowed length of 255 characters`,t,n);const i=["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"];if(!i.includes(n))throw new Zn(`Invalid file type "${n}" is not allowed. Allowed types: ${i.join(", ")}`,t,n);const r=function(e){if(!e)return"";const t=e.lastIndexOf(".");if(-1===t)return"";return e.substring(t).toLowerCase()}(t),a=[".png",".jpg",".jpeg",".gif",".webp",".svg"];if(!a.includes(r))throw new Zn(`File extension "${r}" is not allowed. Allowed extensions: ${a.join(", ")}`,t,n);const s=function(e){switch(e.toLowerCase()){case".png":return"image/png";case".jpg":case".jpeg":return"image/jpeg";case".gif":return"image/gif";case".webp":return"image/webp";case".svg":return"image/svg+xml";default:return"application/octet-stream"}}(r);if(s!==n&&"application/octet-stream"!==s)throw new Zn(`File extension "${r}" does not match MIME type "${n}"`,t,n);return}throw new Zn("File must be a File object (browser) or Buffer (Node.js)",t,n)}const Jn="Upload token image";class eo extends xn{constructor(e,t=!1){super(e,t)}async uploadImageByTokenName(e){const{tokenName:t,options:n}=e;an(t);const o=`${t}.png`;Yn(n.file,o,"image/png");try{const e=`${n.tokenName??t}.png`,o=jn(n.file,"image",e),i=await this.http.signCustomMessage(Jn);if(!i)throw ie("Failed to generate signature - wallet not configured","signature");const r={address:i.address,message:Jn,publickey:i.ethereumAddress,sign:i.signature},a=await this.http.request({method:"POST",url:`${yn}?tokenName=${encodeURIComponent(n.tokenName??t)}`,data:o,headers:r});if(!a)throw ie("No response from image upload service","response");return fn(a,"Image upload failed"),"string"==typeof a.data?a.data:""}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw ie("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.","FormData");throw e}}}class to{constructor(e,t=!1){this.http=e,this.poolService=new _n(e,t),this.tradeService=new qn(e,t),this.userService=new Xn(e,t),this.imageService=new eo(e,t)}async uploadImageByTokenName(e){return this.imageService.uploadImageByTokenName(e)}async fetchPools(e={}){return this.poolService.fetchPools(e)}async fetchAllPools(e){return this.poolService.fetchAllPools(e)}async checkPool(e){return this.poolService.checkPool(e)}async isTokenNameAvailable(e){return this.poolService.isTokenNameAvailable(e)}async isTokenSymbolAvailable(e){return this.poolService.isTokenSymbolAvailable(e)}async fetchVolumeData(e){return this.poolService.fetchVolumeData(e)}async fetchTokenDistribution(e){return this.poolService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.poolService.fetchTokenBadges(e)}async hasTokenBadge(e){return this.poolService.hasTokenBadge(e)}async fetchTrades(e){return this.tradeService.fetchTrades(e)}async fetchProfile(e){return this.userService.fetchProfile(e)}async updateProfile(e){return this.userService.updateProfile(e)}async uploadProfileImage(e){return this.userService.uploadProfileImage(e)}async fetchTokenList(e){return this.userService.fetchTokenList(e)}async fetchTokensHeld(e){return this.userService.fetchTokensHeld(e)}async fetchTokensCreated(e={}){return this.userService.fetchTokensCreated(e)}getAddress(){return this.http.getAddress()}validateTokenName(e){return an(e)}}const no={MAX_UNIQUE_KEY_LENGTH:64,UNIQUE_KEY_PATTERN:/^(galaswap-operation-|galaconnect-operation-)/,TOKEN_NAME_PATTERN:/^[a-zA-Z0-9]+$/};var oo;!function(e){e.INVALID_RECIPIENT="INVALID_RECIPIENT",e.INVALID_AMOUNT="INVALID_AMOUNT",e.INSUFFICIENT_BALANCE="INSUFFICIENT_BALANCE",e.TOKEN_NOT_FOUND="TOKEN_NOT_FOUND",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.NETWORK_ERROR="NETWORK_ERROR",e.DUPLICATE_TRANSFER="DUPLICATE_TRANSFER",e.TRANSFER_LIMIT_EXCEEDED="TRANSFER_LIMIT_EXCEEDED",e.WALLET_REQUIRED="WALLET_REQUIRED"}(oo||(oo={}));class io extends Error{constructor(e,t,n){super(e),this.type=t,this.details=n,this.name="TransferError"}}function ro(e,t="0"){return new u(null==e||""===e?t:e)}function ao(e){return d(e)}function so(e){const t=ro(e),n=Math.log(1.0001),o=t.toNumber();return Math.log(o)/n}function co(e,t=!1){const n=ro(e),o=new u(1).dividedBy(n);return t?o.toFixed():o}function lo(...e){e.forEach((e,t)=>{if(e.isNaN())throw new Error(`Value at index ${t} must be a valid number, got: NaN`);if(!e.isFinite())throw new Error(`Value at index ${t} must be finite, got: ${e.toString()}`);if(e.isLessThanOrEqualTo(0))throw new Error(`Value at index ${t} must be positive, got: ${e.toString()}`)})}function uo(...e){e.forEach((e,t)=>{if(e.isNaN())throw new Error(`Value at index ${t} must be a valid number, got: NaN`);if(!e.isFinite())throw new Error(`Value at index ${t} must be finite, got: ${e.toString()}`);if(e.isLessThan(0))throw new Error(`Value at index ${t} must be non-negative, got: ${e.toString()}`)})}function ho(e,t,n){if(e.isNaN())throw new Error(`${t} must be a valid number, got: NaN`);if(!e.isFinite()){const o=n?`${t} must be finite ${n}, got: ${e.toString()}`:`${t} must be finite, got: ${e.toString()}`;throw new Error(o)}if(e.isLessThanOrEqualTo(0)){const o=n?`${t} must be positive ${n}, got: ${e.toString()}`:`${t} must be positive, got: ${e.toString()}`;throw new Error(o)}}class go{static validateAmount(e){const t=new u(e);try{ho(t,"amount","for transfer")}catch(t){throw new io(t.message,oo.INVALID_AMOUNT,{amount:e})}}static validateUniqueKey(e){if(e){if(e.length>no.MAX_UNIQUE_KEY_LENGTH)throw new G(`Unique key too long. Maximum length: ${no.MAX_UNIQUE_KEY_LENGTH}`);if(!no.UNIQUE_KEY_PATTERN.test(e))throw new io('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',oo.INVALID_AMOUNT,{uniqueKey:e})}}}class po{toLaunchpadFormat(e){if(!e)return"unknown|Unit|none|none";if("string"==typeof e)return e.includes("|")?e:`${e}|Unit|none|none`;return`${e.collection||e.symbol||"unknown"}|${e.category||"Unit"}|${e.type||"none"}|${e.additionalKey||"none"}`}toTokenClass(e){if("object"==typeof e&&null!==e)return{collection:e.collection||"Token",category:e.category||"Unit",type:e.type||e.symbol||"unknown",additionalKey:e.additionalKey||"none"};if("string"!=typeof e)throw new Error("Invalid token format: expected string or object, got "+typeof e);const t=e.split("|");if(t.length<3)throw new Error(`Invalid token format: ${e}`);return{collection:t[0]||"Token",category:t[1]||"Unit",type:t[2]||"none",additionalKey:t[3]||"none"}}isGSwapFormat(e){return"string"==typeof e&&e.includes("|")}normalize(e){return"string"==typeof e&&this.isGSwapFormat(e)?e:this.toLaunchpadFormat(e)}}function mo(e){if(!e||"string"!=typeof e)throw new Error("Invalid token format: token must be a non-empty string");return e.replace(/\|/g,"$")}function fo(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("$");if(t.length<4)throw new Error(`Invalid dollar-delimited token format. Expected at least 4 parts separated by $, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("$");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){throw new G(`Invalid dollar-delimited token: "${e}". Expected format: "Token$Unit$SYMBOL$additionalKey". Error: ${t instanceof Error?t.message:String(t)}`,"dollarToken","INVALID_DOLLAR_DELIMITED_TOKEN_FORMAT")}}class yo extends s{constructor(e){super(),this.from=e.from,this.to=e.to,this.quantity=e.quantity,this.tokenInstance=e.tokenInstance,this.uniqueKey=e.uniqueKey,e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o,i){let r;if("string"==typeof o){r={...fo(o),instance:"0"}}else r={collection:o.collection,category:o.category,type:o.type,additionalKey:o.additionalKey,instance:"0"};return new yo({from:e,to:t,quantity:n,tokenInstance:r,uniqueKey:i||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n,o){return new yo({from:e,to:t,quantity:n,tokenInstance:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"},uniqueKey:o||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){return`${this.tokenInstance.collection}$${this.tokenInstance.category}$${this.tokenInstance.type}$${this.tokenInstance.additionalKey}`}toSigningPayload(){return{from:this.from,to:this.to,quantity:this.quantity,tokenInstance:this.tokenInstance,uniqueKey:this.uniqueKey}}}class wo extends s{constructor(e){super(),this.lockAuthority=e.lockAuthority,this.tokenInstances=e.tokenInstances,this.uniqueKey=e.uniqueKey,void 0!==e.expires&&(this.expires=e.expires),void 0!==e.name&&(this.name=e.name),e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o,i){let r;if("string"==typeof o){r={...fo(o),instance:"0"}}else r={collection:o.collection,category:o.category,type:o.type,additionalKey:o.additionalKey,instance:"0"};return new wo({lockAuthority:t||e,tokenInstances:[{owner:e,quantity:n,tokenInstanceKey:r}],...void 0!==i?.expires&&{expires:i.expires},...void 0!==i?.name&&{name:i.name},uniqueKey:i?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n,o){return new wo({lockAuthority:t||e,tokenInstances:[{owner:e,quantity:n,tokenInstanceKey:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}],...void 0!==o?.expires&&{expires:o.expires},...void 0!==o?.name&&{name:o.name},uniqueKey:o?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){if(0===this.tokenInstances.length)return;const e=this.tokenInstances[0].tokenInstanceKey;return`${e.collection}$${e.category}$${e.type}$${e.additionalKey}`}toSigningPayload(){return{lockAuthority:this.lockAuthority,tokenInstances:this.tokenInstances,...void 0!==this.expires&&{expires:this.expires},...void 0!==this.name&&{name:this.name},uniqueKey:this.uniqueKey}}}class ko extends s{constructor(e){super(),this.tokenInstances=e.tokenInstances,this.uniqueKey=e.uniqueKey,void 0!==e.name&&(this.name=e.name),e.signedPayload&&(this.signature=e.signedPayload.signature,this.domain=e.signedPayload.domain,this.types=e.signedPayload.types,e.signedPayload.prefix&&(this.prefix=e.signedPayload.prefix))}static fromTokenClassKey(e,t,n,o){let i;if("string"==typeof n){i={...fo(n),instance:"0"}}else i={collection:n.collection,category:n.category,type:n.type,additionalKey:n.additionalKey,instance:"0"};return new ko({tokenInstances:[{owner:e,quantity:t,tokenInstanceKey:i}],...void 0!==o?.name&&{name:o.name},uniqueKey:o?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,n){return new ko({tokenInstances:[{owner:e,quantity:t,tokenInstanceKey:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}],...void 0!==n?.name&&{name:n.name},uniqueKey:n?.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}getTokenClassKey(){if(0===this.tokenInstances.length)return;const e=this.tokenInstances[0].tokenInstanceKey;return`${e.collection}$${e.category}$${e.type}$${e.additionalKey}`}toSigningPayload(){return{tokenInstances:this.tokenInstances,...void 0!==this.name&&{name:this.name},uniqueKey:this.uniqueKey}}}class vo{constructor(e){this.wallet=e}static generateUniqueKey(){return`${Date.now()}_${Math.random().toString(36).substring(2,8)}`}async signTransferToken(e){const t={name:"GalaChain",chainId:1},n={TransferToken:[{name:"from",type:"string"},{name:"to",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstance",type:"TokenInstance"},{name:"uniqueKey",type:"string"}],TokenInstance:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]};return{signature:await this.wallet.signTypedData(t,n,e),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}async signLockToken(e){const t={name:"GalaChain",chainId:1},n={LockToken:[{name:"lockAuthority",type:"string"},{name:"tokenInstances",type:"TokenInstanceQuantity[]"},{name:"uniqueKey",type:"string"},...void 0!==e.expires?[{name:"expires",type:"uint256"}]:[],...void 0!==e.name?[{name:"name",type:"string"}]:[]],TokenInstanceQuantity:[{name:"owner",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstanceKey",type:"TokenInstanceKey"}],TokenInstanceKey:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]},o={lockAuthority:e.lockAuthority,tokenInstances:e.tokenInstances,uniqueKey:e.uniqueKey};void 0!==e.expires&&(o.expires=e.expires),void 0!==e.name&&(o.name=e.name);return{signature:await this.wallet.signTypedData(t,n,o),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}async signUnlockToken(e){const t={name:"GalaChain",chainId:1},n={UnlockToken:[{name:"tokenInstances",type:"TokenInstanceQuantity[]"},{name:"uniqueKey",type:"string"},...void 0!==e.name?[{name:"name",type:"string"}]:[]],TokenInstanceQuantity:[{name:"owner",type:"string"},{name:"quantity",type:"string"},{name:"tokenInstanceKey",type:"TokenInstanceKey"}],TokenInstanceKey:[{name:"collection",type:"string"},{name:"category",type:"string"},{name:"type",type:"string"},{name:"additionalKey",type:"string"},{name:"instance",type:"string"}]},o={tokenInstances:e.tokenInstances,uniqueKey:e.uniqueKey};void 0!==e.name&&(o.name=e.name);return{signature:await this.wallet.signTypedData(t,n,o),domain:t,types:n,signerPublicKey:this.wallet.signingKey.publicKey}}static toGalaChainAddress(e){const n=e.replace("0x","");return`eth|${t.getAddress(`0x${n}`).replace("0x","")}`}static fromGalaChainAddress(e){return e.startsWith("eth|")?e.substring(4):e}static createGALATokenInstance(){return{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"}}static createTokenInstanceFromClassKey(e){return{...fo(e),instance:"0"}}}function bo(e){if("string"==typeof e){const t=e.split("|");if(t.length<4)throw new G(`Invalid tokenId string format: "${e}". Expected format: "collection|category|type|additionalKey" or "collection|category|type|additionalKey|instance"`,"tokenId","INVALID_TOKEN_ID_FORMAT");if(!(t[0]&&t[1]&&t[2]&&t[3]))throw new G(`Invalid tokenId string format: "${e}". All components (collection, category, type, additionalKey) must be non-empty`,"tokenId","INVALID_TOKEN_ID_FORMAT");return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3],instance:t[4]||"0"}}if("object"==typeof e&&null!==e){if(!(e.collection&&e.category&&e.type&&e.additionalKey))throw new G("Invalid tokenId object format. All fields (collection, category, type, additionalKey) are required","tokenId","INVALID_TOKEN_ID_FORMAT");return"instance"in e&&void 0!==e.instance?e:{...e,instance:"0"}}throw new G(`Invalid tokenId type: ${typeof e}. Expected string, TokenClassKey, or TokenInstanceKey`,"tokenId","INVALID_TOKEN_ID_TYPE")}function To(e){try{const[t,n]=e.split("|");if(!n)throw new Error("Missing token part after service");const o=n.split("$");if(o.length<4)throw new Error(`Invalid vault address format. Expected at least 4 parts separated by $, got ${o.length}`);if(!(o[0]&&o[1]&&o[2]&&o[3]))throw new Error("All vault address components (collection, category, type, additionalKey) must be non-empty");const i=o.slice(3,-1),r=i.length>0?i.join("$"):o[3];return{collection:o[0],category:o[1],type:o[2],additionalKey:r}}catch(t){throw new G(`Invalid vault address: "${e}". Expected format: "service|Token$Unit$SYMBOL$additionalKey$launchpad". Error: ${t instanceof Error?t.message:String(t)}`,"vaultAddress","INVALID_VAULT_ADDRESS_FORMAT")}}function So(e){return{...To(e),instance:"0"}}function Ao(e){return To(e).type}var Eo,Io=Object.freeze({__proto__:null,extractTokenSymbolFromVault:Ao,normalizeToTokenInstanceKey:bo,parseVaultAddressToTokenClassKey:To,parseVaultAddressToTokenInstance:So});function Po(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(!("string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0))||void 0!==t.lockAuthority&&"string"!=typeof t.lockAuthority||void 0!==t.expires&&"number"!=typeof t.expires||void 0!==t.name&&"string"!=typeof t.name||void 0!==t.uniqueKey&&"string"!=typeof t.uniqueKey)}function No(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0)&&(void 0===t.name||"string"==typeof t.name)&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}!function(e){e.TOKEN_NOT_FOUND="TOKEN_NOT_FOUND",e.INVALID_AMOUNT="INVALID_AMOUNT",e.INSUFFICIENT_BALANCE="INSUFFICIENT_BALANCE",e.SIGNATURE_FAILED="SIGNATURE_FAILED",e.NETWORK_ERROR="NETWORK_ERROR",e.WALLET_REQUIRED="WALLET_REQUIRED",e.LOCK_NOT_FOUND="LOCK_NOT_FOUND",e.LOCK_EXPIRED="LOCK_EXPIRED",e.INSUFFICIENT_LOCKED_BALANCE="INSUFFICIENT_LOCKED_BALANCE",e.NOT_LOCK_AUTHORITY="NOT_LOCK_AUTHORITY",e.LOCK_NAME_MISMATCH="LOCK_NAME_MISMATCH"}(Eo||(Eo={}));class Do extends Error{constructor(e,t,n){super(e),this.type=t,this.details=n,this.name="LockError"}}const Fo="gala-transfer-successful",xo="token-transfer-successful",Co="token-locked-successfully",$o="token-unlocked-successfully",Lo="transfer-successful-no-id";function Bo(e){return e&&0!==e.length?e.reduce((e,t)=>u(e).plus(t.quantity).toString(),"0"):"0"}function Uo(e,t){if(!e||0===e.length)return[];if(t)return e;const n=Date.now();return e.filter(e=>0===e.expires||e.expires>n)}class Oo extends xn{constructor(e,t,n,o=!1,i){super(e,o),this.wallet=t,this.tokenResolver=n,this.publicAxios=i,this.signatureHelper=t?new vo(t):void 0}async fetchPoolDetails(e){this.validateFetchPoolDetailsData(e);const t=await this.http.post("/api/asset/launchpad-contract/FetchSaleDetails",e);if(!t)throw oe("No response from GalaChain service",500);if(1!==t.Status)throw oe(`Failed to fetch pool details: Status ${t.Status}`,t.Status);const n=t.Data.reverseBondingCurveConfiguration,o=n?.minFeePortion??"0",i=n?.maxFeePortion??"0",r=(await import("bignumber.js")).default,a=!new r(o).isZero()||!new r(i).isZero(),s=t.Data;return s.reverseBondingCurveMinFeePortion=o,s.reverseBondingCurveMaxFeePortion=i,s.hasReverseBondingCurveFee=a,s.isGraduated="Finished"===t.Data.saleStatus,delete s.reverseBondingCurveConfiguration,t}async fetchLaunchTokenFee(){const e=await this.http.post("/api/asset/launchpad-contract/FetchLaunchpadFeeAmount",{});if(!e)throw oe("No response from GalaChain service",500);if(1!==e.Status)throw oe(`Failed to fetch launch token fee: Status ${e.Status}`,e.Status);return e.Data.feeAmount}validateFetchPoolDetailsData(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.vaultAddress)throw new G("Invalid fetch pool details data: missing required fields","data","INVALID_TYPE");var t;if(!e.vaultAddress||"string"!=typeof e.vaultAddress)throw new G("Vault address is required and must be a string","vaultAddress","INVALID_VAULT_ADDRESS");if(!e.vaultAddress.startsWith("service|Token$Unit$"))throw new G("Vault address must be in service format: service|Token$Unit$...","vaultAddress","INVALID_VAULT_ADDRESS")}async fetchGalaBalance(e){return this.fetchTokenBalance(e)}async fetchTokenBalance(e,t=!1){try{const n=await this.http.post("/api/asset/token-contract/FetchBalances",e);if(!n)return null;if(1!==n.Status||!n.Data||0===n.Data.length)return null;const o=n.Data.find(t=>t.collection===e.collection&&t.category===e.category&&t.additionalKey===e.additionalKey&&t.type===e.type);if(!o||"0"===o.quantity)return null;const i=`${o.collection}|${o.category}|${o.type}|${o.additionalKey}`,r={quantity:o.quantity,collection:o.collection,category:o.category,tokenId:i};if(o.inUseHolds?.length){const e=Uo(o.inUseHolds??[],t);e.length>0&&(r.inUseHolds=e,r.inUseQuantity=Bo(e))}if(o.lockedHolds?.length){const e=Uo(o.lockedHolds??[],t);e.length>0&&(r.lockedHolds=e,r.lockedQuantity=Bo(e))}return(r.lockedQuantity||r.inUseQuantity)&&(r.availableQuantity=function(e,t="0",n="0"){return u(e).minus(t).minus(n).toString()}(r.quantity,r.lockedQuantity,r.inUseQuantity)),r}catch(e){throw oe(`Failed to fetch token balance from GalaChain: ${Gn(e)}`,void 0,Kn(e)?e:void 0)}}async fetchTokenClassFromChain(e){try{const t="string"==typeof e?bo(e):e,n={tokenClasses:[{collection:t.collection,category:t.category,type:t.type,additionalKey:t.additionalKey}]},o=(await this.publicAxios.post("/api/asset/token-contract/FetchTokenClasses",n)).data;if(1!==o.Status)throw oe(`Failed to fetch token class from GalaChain: Status ${o.Status}${o.Message?` - ${o.Message}`:""}`,o.Status);if(!o.Data||0===o.Data.length)throw oe(`Token not found on GalaChain: ${t.collection}|${t.category}|${t.type}|${t.additionalKey}`,404);return o.Data[0]}catch(e){if(Wn(e)&&404===e.response?.status)throw oe("Token not found on GalaChain",404,Kn(e)?e:void 0);const t=Gn(e);if(t.includes("Token not found")||t.includes("Status"))throw e;throw oe(`Failed to fetch token class from GalaChain: ${t}`,void 0,Kn(e)?e:void 0)}}async fetchTokenClassesWithSupply(e){try{if(!e||0===e.length)throw new G("tokenClasses array must not be empty");const t={tokenClasses:e},n=(await this.publicAxios.post("/api/asset/token-contract/FetchTokenClassesWithSupply",t)).data;if(1!==n.Status)throw oe(`Failed to fetch token classes with supply from GalaChain: Status ${n.Status}${n.Message?` - ${n.Message}`:""}`,n.Status);if(!n.Data||0===n.Data.length)throw oe("No token supply data found for requested token classes",404);return n.Data}catch(e){if(Wn(e)&&404===e.response?.status)throw oe("Token supply data not found on GalaChain",404,Kn(e)?e:void 0);const t=Gn(e);if(t.includes("Token")||t.includes("Status")||t.includes("tokenClasses"))throw e;throw oe(`Failed to fetch token classes with supply from GalaChain: ${t}`,void 0,Kn(e)?e:void 0)}}async transferGala(e){if(this.validateTransferGalaData(e),!this.wallet||!this.signatureHelper)throw new io("Wallet required for GALA transfer operations",oo.WALLET_REQUIRED);try{const t=ue(e.recipientAddress),n=ue(this.wallet.address),o=yo.forGALA(n,t,e.amount,e.uniqueKey),i=await this.signatureHelper.signTransferToken(o.toSigningPayload()),r=new yo({...o.toSigningPayload(),signedPayload:i});this.logger.debug("[DEBUG] Full GALA Transfer Request Payload:",JSON.stringify(r,null,2));const a=await this.http.post("/api/asset/token-contract/TransferToken",r);if(!a)throw new io("No response from GalaChain transfer service",oo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Transfer response:",JSON.stringify(a,null,2)),this.extractTransactionIdFromResponse(a,"gala")}catch(t){throw this.handleTransferError(t,"GALA transfer failed",e)}}async transferToken(e){if(this.validateTransferTokenData(e),!this.wallet||!this.signatureHelper)throw new io("Wallet required for token transfer operations",oo.WALLET_REQUIRED);try{const t=ue(e.to),n=ue(this.wallet.address);let o;if(e.tokenId)o=bo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId),this.logger.debug("[DEBUG] Normalized Token Instance:",JSON.stringify(o,null,2));else{if(!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);o=await this.resolveTokenInstance(e.tokenName)}const i=new yo({from:n,to:t,quantity:e.amount,tokenInstance:o,uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),r=await this.signatureHelper.signTransferToken(i.toSigningPayload()),a=new yo({...i.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full Transfer Request Payload:",JSON.stringify(a,null,2));const s=await this.http.post("/api/asset/token-contract/TransferToken",a);if(!s)throw new io("No response from GalaChain transfer service",oo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token transfer response:",JSON.stringify(s,null,2)),this.extractTransactionIdFromResponse(s,"token")}catch(t){throw this.handleTransferError(t,"Token transfer failed",e)}}async resolveTokenClassKey(e){try{const t=await this.tokenResolver.resolveTokenClassKey(e);return this.logger.debug(`[DEBUG] Token class key resolution for '${e}':`,JSON.stringify(t,null,2)),t}catch(t){if(t instanceof io)throw t;throw new io(`Failed to resolve token class key for '${e}': ${Gn(t)}`,oo.TOKEN_NOT_FOUND,{tokenName:e})}}async lockToken(e){if(this.validateLockTokenData(e),!this.wallet||!this.signatureHelper)throw new io("Wallet required for token lock operations",oo.WALLET_REQUIRED);try{const t=ue(this.wallet.address),n=e.lockAuthority?ue(e.lockAuthority):t;let o;if(e.tokenId)o=bo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId);else{if(!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);o=await this.resolveTokenInstance(e.tokenName)}const i=new wo({lockAuthority:n,tokenInstances:[{owner:t,quantity:e.amount,tokenInstanceKey:o}],...void 0!==e.expires&&{expires:e.expires},...void 0!==e.name&&{name:e.name},uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),r=await this.signatureHelper.signLockToken(i.toSigningPayload()),a=new wo({...i.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full Lock Request Payload:",JSON.stringify(a,null,2));const s=await this.http.post("/api/asset/token-contract/LockTokens",a);if(!s)throw new io("No response from GalaChain lock service",oo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token lock response:",JSON.stringify(s,null,2)),this.extractTransactionIdFromResponse(s,"lock")}catch(t){throw this.handleTransferError(t,"Token lock failed",e)}}async unlockToken(e){if(this.validateUnlockTokenData(e),!this.wallet||!this.signatureHelper)throw new io("Wallet required for token unlock operations",oo.WALLET_REQUIRED);try{const t=ue(this.wallet.address);let n;if(e.tokenId)n=bo(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId);else{if(!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);n=await this.resolveTokenInstance(e.tokenName)}const o=new ko({tokenInstances:[{owner:t,quantity:e.amount,tokenInstanceKey:n}],...void 0!==e.name&&{name:e.name},uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),i=await this.signatureHelper.signUnlockToken(o.toSigningPayload()),r=new ko({...o.toSigningPayload(),signedPayload:i});this.logger.debug("[DEBUG] Full Unlock Request Payload:",JSON.stringify(r,null,2));const a=await this.http.post("/api/asset/token-contract/UnlockTokens",r);if(!a)throw new io("No response from GalaChain unlock service",oo.NETWORK_ERROR);return this.logger.debug("[DEBUG] Token unlock response:",JSON.stringify(a,null,2)),this.extractTransactionIdFromResponse(a,"unlock")}catch(t){throw this.handleTransferError(t,"Token unlock failed",e)}}validateTransferGalaData(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.recipientAddress&&t.recipientAddress.trim().length>0&&"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}(e))throw new G("Invalid GALA transfer data: missing required fields");if(!le(e.recipientAddress))throw new io("Invalid recipient address format",oo.INVALID_RECIPIENT,{recipientAddress:e.recipientAddress});go.validateAmount(e.amount),go.validateUniqueKey(e.uniqueKey)}validateTransferTokenData(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.to&&t.to.trim().length>0&&"string"==typeof t.amount&&t.amount.trim().length>0&&(void 0!==t.tokenId||"string"==typeof t.tokenName&&t.tokenName.trim().length>0)&&(void 0===t.uniqueKey||"string"==typeof t.uniqueKey)}(e))throw new G("Invalid token transfer data: missing required fields");if(!le(e.to))throw new io("Invalid recipient address format",oo.INVALID_RECIPIENT,{recipientAddress:e.to});if(!e.tokenId&&!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);if(e.tokenName&&!no.TOKEN_NAME_PATTERN.test(e.tokenName))throw new io("Invalid token name format",oo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});go.validateAmount(e.amount),go.validateUniqueKey(e.uniqueKey)}validateLockTokenData(e){if(!Po(e))throw new G("Invalid lock token data: missing required fields");if(e.lockAuthority&&!le(e.lockAuthority))throw new io("Invalid lock authority address format",oo.INVALID_RECIPIENT,{recipientAddress:e.lockAuthority});if(!e.tokenId&&!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);if(e.tokenName&&!no.TOKEN_NAME_PATTERN.test(e.tokenName))throw new io("Invalid token name format",oo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});if(void 0!==e.expires&&(e.expires<=0||!Number.isInteger(e.expires)))throw new G("Expires must be a positive integer (epoch milliseconds)");go.validateAmount(e.amount),go.validateUniqueKey(e.uniqueKey)}validateUnlockTokenData(e){if(!No(e))throw new G("Invalid unlock token data: missing required fields");if(!e.tokenId&&!e.tokenName)throw new io("Must provide either tokenId or tokenName for token identification",oo.TOKEN_NOT_FOUND);if(e.tokenName&&!no.TOKEN_NAME_PATTERN.test(e.tokenName))throw new io("Invalid token name format",oo.TOKEN_NOT_FOUND,{tokenName:e.tokenName});go.validateAmount(e.amount),go.validateUniqueKey(e.uniqueKey)}async resolveTokenInstance(e){try{const t=await this.tokenResolver.resolveTokenToVault(e);if(t){const n=this.resolveTokenInstanceFromVaultAddress(t);return this.logger.debug(`[DEBUG] Token resolution for '${e}' (launchpad):\n Vault Address: ${t}\n Token Instance: ${JSON.stringify(n,null,2)}`),n}const n={collection:e.trim().toUpperCase(),category:"Unit",type:"none",additionalKey:"none",instance:"0"};return this.logger.debug(`[DEBUG] Token resolution for '${e}' (standard format):\n Token Instance: ${JSON.stringify(n,null,2)}`),n}catch(t){if(t instanceof io)throw t;throw new io(`Failed to resolve token '${e}': ${Gn(t)}`,oo.TOKEN_NOT_FOUND,{tokenName:e})}}resolveTokenInstanceFromVaultAddress(e){try{return So(e)}catch(e){if(e instanceof G)throw new io(`Invalid vault address format: ${e.message}`,oo.TOKEN_NOT_FOUND);throw new io(`Failed to parse vault address: ${e instanceof Error?e.message:String(e)}`,oo.TOKEN_NOT_FOUND)}}extractTransactionIdFromResponse(e,t){if(e&&"object"==typeof e){if("Status"in e&&1===e.Status&&"Data"in e){const n=e;if(Array.isArray(n.Data)&&n.Data.length>0)switch(t){case"gala":return Fo;case"token":return xo;case"lock":return Co;case"unlock":return $o}return Lo}if("transactionId"in e&&"string"==typeof e.transactionId&&e.transactionId)return e.transactionId}throw new io("Operation succeeded but transaction ID could not be extracted",oo.NETWORK_ERROR)}handleTransferError(e,t,n){if(e instanceof io)return e;if(e instanceof G)return new io(e.message,oo.INVALID_AMOUNT);if(Wn(e)&&e.response){const t=e.response.status,o=e.response.data;if(400===t)return new io(("string"==typeof o?.message?o.message:void 0)||"Invalid transfer request",oo.INVALID_AMOUNT);if(403===t)return new io("Insufficient balance for transfer",oo.INSUFFICIENT_BALANCE);if(404===t){const e={};return"tokenName"in n&&(e.tokenName=n.tokenName),new io("Token not found",oo.TOKEN_NOT_FOUND,e)}}if("object"==typeof e&&null!==e&&"code"in e&&("ECONNABORTED"===e.code||"ETIMEDOUT"===e.code))return new io("Transfer request timed out",oo.NETWORK_ERROR);const o=Gn(e);return new io(o||t,oo.NETWORK_ERROR)}}class _o{constructor(e,t,n,o=!1){this.dexBackendHttp=e,this.cache=t,this.galaChainService=n,this.logger=new _({debug:o,context:"DexService"})}async fetchTokenPrice(e){const{tokenName:t,tokenId:n}=e;if(!t&&!n)throw te("tokenName or tokenId","Either tokenName (for launchpad tokens) or tokenId (for DEX tokens) is required");if(t&&n)throw new G("tokenName and tokenId are mutually exclusive - provide only one","params","INVALID_PARAMS");if(n)return this.logger.debug(`Fetching spot price by tokenId: ${n}`),this._fetchDexTokenSpotPrice(n);throw new G("tokenName parameter requires LaunchpadSDK routing - call LaunchpadSDK.fetchTokenPrice({tokenName}) instead","tokenName","INVALID_PARAMS")}async _fetchDexTokenSpotPrice(e){if(!e)throw te("tokenId","Token ID");try{const t=bo(e),n=mo(`${t.collection}|${t.category}|${t.type}|${t.additionalKey}`);if(this.logger.debug(`Fetching DEX spot price for token: ${n}`),!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const o=await this.dexBackendHttp.request({method:"GET",url:"/v1/trade/price",params:{token:n}});if(!o.data||"string"!=typeof o.data)throw new G("Invalid price response: data must be a string, got "+typeof o.data,"data","INVALID_RESPONSE");const i=parseFloat(o.data);if(isNaN(i))throw new G(`Invalid price value: could not parse "${o.data}" as number`,"data","INVALID_CALCULATION");const r=`${t.collection}|${t.category}|${t.type}|${t.additionalKey}`;let a;try{if(this.cache){const e=this.cache.getByTokenId(r);if(e?.symbol)return a=e.symbol,this.logger.debug(`DEX spot price for ${a} (cached): $${i}`),{symbol:a,price:i}}this.logger.debug(`Symbol cache miss for ${r}, fetching from API`);a=(await this.fetchTokenDetails(e)).symbol,this.cache&&(this.cache.setByTokenId(r,{symbol:a}),this.logger.debug(`Cached symbol for ${r}: ${a}`)),this.logger.debug(`DEX spot price for ${a}: $${i}`)}catch(e){this.logger.debug(`Could not fetch token details for symbol, falling back to token format parsing: ${e instanceof Error?e.message:String(e)}`),a=("Token"===t.collection?t.type:t.collection).toUpperCase(),this.logger.debug(`DEX spot price for ${a} (fallback): $${i}`)}return{symbol:a,price:i}}catch(e){if(e instanceof G)throw e;throw oe(`Failed to fetch DEX spot price: ${Gn(e)}`)}}async fetchLaunchpadTokenSpotPrice(e,t,n){if(!e||"string"!=typeof e)throw new Error(q);try{if(n)try{this.logger.debug(`Checking graduation status for token: ${e}`);const t=await n(e);if(t&&t.isGraduated){this.logger.debug(`Token ${e} is graduated, using DEX spot price`);const n=`${t.sellingToken.collection}|${t.sellingToken.category}|${t.sellingToken.type}|${t.sellingToken.additionalKey}`;return this._fetchDexTokenSpotPrice(n)}}catch(t){this.logger.debug(`Could not determine graduation status for ${e}, falling back to bonding curve: ${Gn(t)}`)}this.logger.debug(`Using bonding curve calculation for token: ${e}`);const o=await t({tokenName:e,amount:"1",type:"native"}),i=await this._fetchDexTokenSpotPrice({collection:"GALA",category:"Unit",type:"none",additionalKey:"none"});if(!i)throw oe("GALA price not available");const r=Number(o.amount)/1e18;if(r<=0)throw new G(`Invalid token amount calculation: ${r}`,"amount","INVALID_CALCULATION");const a=i.price/r;return{symbol:e.toUpperCase(),price:a}}catch(t){if(t instanceof Error)throw oe(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw oe(`Failed to calculate launchpad token spot price for ${e}: ${Gn(t)}`)}}async fetchTokenDetails(e){this.logger.debug("Fetching token details from GalaChain for tokenId:",e);try{if(!this.galaChainService)throw oe("GalaChainService not available for token metadata fetch",500);const t=await this.galaChainService.fetchTokenClassFromChain(e),n={collection:t.collection,category:t.category,type:t.type,additionalKey:t.additionalKey,symbol:t.symbol,decimals:t.decimals,name:t.name,image:t.image,description:t.description,network:t.network,...void 0!==t.contractAddress&&{contractAddress:t.contractAddress}};return this.logger.debug(`Fetched token details for ${t.symbol} from GalaChain`),n}catch(t){if((t instanceof G||t instanceof Error)&&("NetworkError"===t.name||t.message.includes("Token not found")))throw t;throw oe(`Failed to fetch token details from GalaChain for ${e}: ${Gn(t)}`,500)}}async fetchAllDexSeasons(){try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const e=await this.dexBackendHttp.request({method:"GET",url:"/leaderboard/seasons"});let t;if(Array.isArray(e))t=e;else{if(!e||"object"!=typeof e)return this.logger.warn("Seasons endpoint returned invalid data:",e),[];if(Array.isArray(e.data))t=e.data;else if(e.data&&Array.isArray(e.data.seasons))t=e.data.seasons;else{if(!Array.isArray(e.seasons))return this.logger.warn("Seasons endpoint returned unexpected structure:",e),[];t=e.seasons}}const n=t.map(e=>({id:e?.id??0,name:e?.name??"",start:e?.start?new Date(e.start):new Date,end:e?.end?new Date(e.end):new Date,rulesId:e?.rules_id??0}));return this.logger.debug(`Fetched ${n.length} DEX seasons`),n}catch(e){if(e instanceof Error&&e.message.includes("not configured"))throw e;if(e&&"object"==typeof e&&"response"in e){const t=e;if(404===t.response?.status)return this.logger.warn("Seasons endpoint not available"),[]}throw oe(`Failed to fetch DEX seasons: ${Gn(e)}`)}}async fetchCurrentDexSeason(){const e=await this.fetchAllDexSeasons(),t=new Date,n=e.find(e=>t>=e.start&&t<=e.end);return n?this.logger.debug(`Current DEX season: ${n.name} (ID: ${n.id})`):this.logger.debug("No active DEX season found"),n||null}async fetchDexLeaderboardBySeasonId(e){if(!e||"number"!=typeof e||e<1)throw te("seasonId","Season ID must be a positive number");try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const t=await this.dexBackendHttp.request({method:"GET",url:"/leaderboard",params:{seasonId:e.toString()}});let n;if(Array.isArray(t))n=t;else{if(!t||"object"!=typeof t)return this.logger.warn("Leaderboard endpoint returned invalid data:",t),{entries:[],seasonId:e,totalEntries:0};if(t.data&&Array.isArray(t.data.leaderboard))n=t.data.leaderboard;else if(Array.isArray(t.leaderboard))n=t.leaderboard;else{if(!t.data||!Array.isArray(t.data))return this.logger.warn("Leaderboard endpoint returned unexpected structure:",t),{entries:[],seasonId:e,totalEntries:0};n=t.data}}const o=n.map(e=>({wallet:e?.wallet??"",rank:e?.rank??0,totalXp:e?.total_xp??0,distributionPercent:e?.distribution_percent??0,liquidityXp:e?.liquidity_xp??0,tradingXp:e?.trading_xp??0,masteryTitles:(e?.mastery_titles??[]).map(e=>({name:e?.name??"",type:e?.type??"trade",order:e?.order??0}))}));return this.logger.debug(`Fetched leaderboard for season ${e} with ${o.length} entries`),{entries:o,seasonId:e,totalEntries:o.length}}catch(t){if(t instanceof Error&&t.message.includes("must be a positive number"))throw t;throw oe(`Failed to fetch DEX leaderboard for season ${e}: ${Gn(t)}`)}}async fetchCurrentDexLeaderboard(){const e=await this.fetchCurrentDexSeason();return e?this.fetchDexLeaderboardBySeasonId(e.id):(this.logger.debug("Cannot fetch current leaderboard - no active season"),null)}async fetchDexAggregatedVolumeSummary(){try{if(!this.dexBackendHttp)throw oe("DEX Backend API client not configured");const e=await this.dexBackendHttp.request({method:"GET",url:"/explore/volume"}),t={volume1d:e.data.volume1d,volume1dDelta:e.data.volume1dDelta,volume7d:e.data.volume7d,volume7dDelta:e.data.volume7dDelta,volume30d:e.data.volume30d,volume30dDelta:e.data.volume30dDelta};return this.logger.debug(`Fetched DEX volume summary: $${t.volume1d.toFixed(2)} (1d)`),t}catch(e){throw oe(`Failed to fetch DEX volume summary: ${Gn(e)}`)}}}const Mo=8,Ro=18,qo={collection:"GALA",category:"Unit",type:"none",additionalKey:"none"};function Ko(e,t=18){const n=parseFloat(e);if(0===n)return"0";return n.toFixed(t).replace(/\.?0+$/,"")}function Go(e){return Ko(e,8)}function Wo(e){return Ko(e,18)}function jo(e){return{maxAcceptableReverseBondingCurveFee:Go(e.maxAcceptableReverseBondingCurveFee)}}new _({debug:!1,context:"NumberUtils"});const Ho={BuyNativeDto:class extends s{constructor(e,t,n="0",o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=Go(t),this.expectedToken=Wo(n),this.extraFees=jo(o)}},BuyExactDto:class extends s{constructor(e,t,n,o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Wo(t),this.expectedNativeToken=Go(n),this.extraFees=jo(o)}},SellExactDto:class extends s{constructor(e,t,n="0",o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Wo(t),this.expectedNativeToken=Go(n),this.extraFees=jo(o)}},SellNativeDto:class extends s{constructor(e,t,n,o={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=Go(t),this.expectedToken=Wo(n),this.extraFees=jo(o)}}};class zo extends Cn{constructor(e,t=!1){super(t),this.walletProvider=e}async signDTO(e,t,n){try{this.logger.debug("🔐 Signing DTO:",{methodName:t,dtoKeys:Object.keys(e)});const n=this.generateEIP712Types(t,e),o=c(e),i={...e,prefix:o},{signature:r,domain:a}=await this.signWithEthersWallet(n,i),s={...e,signature:r,types:n,domain:a};return this.logger.debug("✅ DTO signed successfully:",{payloadKeys:Object.keys(s),signatureLength:r.length}),s}catch(e){this.logger.error("❌ Signature generation failed:",e);throw re(`Failed to sign DTO: ${Gn(e)}`)}}async signWithEthersWallet(e,t){try{let n,o;if(this.walletProvider.signTypedData&&!this.walletProvider.getNetwork)n={name:"ethereum",chainId:1},o=await this.walletProvider.signTypedData(n,e,t);else{if(!this.walletProvider.getNetwork||!this.walletProvider.signTypedData)throw ie("Wallet provider does not support typed data signing","walletProvider");{const i=await this.walletProvider.getNetwork();n={name:i.name,chainId:Number(i.chainId)},o=await this.walletProvider.signTypedData(n,e,t)}}return{signature:o,domain:n}}catch(e){throw re(`Ethers.js signing failed: ${Gn(e)}`)}}generateEIP712Types(e,t){const n={};n[e]=[];const o=Object.fromEntries(Object.entries(t).filter(([e,t])=>void 0!==t)),i=(e,t,o,r=!1)=>{if(void 0!==t){if(Array.isArray(t)){if(0===t.length)return;const a=i(e,t[0],o,!0);return r||n[o].push({name:e,type:(a??e)+"[]"}),a?a+"[]":void 0}if("object"==typeof t&&null!==t){if(n[e])throw new G(`Type name collision not supported: ${e}`,"fieldValue","TYPE_COLLISION");return n[e]=[],Object.entries(t).forEach(([t,n])=>{i(t,n,e)}),r||n[o].push({name:e,type:e}),e}{let i;switch(typeof t){case"string":i="string";break;case"number":i="uint256";break;case"boolean":i="bool";break;default:throw new G(`Unsupported type for field "${e}": ${typeof t} (value: ${JSON.stringify(t)})`,"fieldValue","UNSUPPORTED_TYPE")}return r||n[o].push({name:e,type:i}),i}}};return Object.entries(o).forEach(([t,n])=>{i(t,n,e)}),this.logger.debug("📝 Generated EIP-712 types:",n),n}}class Vo extends Cn{constructor(e=!1){super(e)}generateStringsInstructions(e){try{this.logger.debug("🔧 Generating stringsInstructions for:",e);const t=this.extractTokenSymbolFromVault(e),n=this.createTokenInstance(t),o=this.createGalaInstance(),i=`$service$${n.toStringKey()}$launchpad`,r=`$tokenBalance$${n.toStringKey()}$${e}`,a=`$tokenBalance$${n.toStringKey()}$${e}`,s=`$tokenBalance$${o.toStringKey()}$${e}`,c=[i,r,a,s,`$tokenBalance$${o.toStringKey()}$${e}`];return this.logger.debug("✅ Generated stringsInstructions:",c),c}catch(e){this.logger.error("❌ Failed to generate stringsInstructions:",e);const t=Gn(e);throw new G(`Failed to generate stringsInstructions: ${t}`,"vaultAddress","INVALID_VAULT_ADDRESS")}}createTokenInstance(e){const t=new I;return t.collection=e.toLowerCase(),t.category="Unit",t.type="none",t.additionalKey="none",this.logger.debug("🪙 Created token instance:",{symbol:e,lowercaseCollection:e.toLowerCase(),stringKey:t.toStringKey()}),t}createGalaInstance(){const e=new I;return e.collection="GALA",e.category="Unit",e.type="none",e.additionalKey="none",this.logger.debug("🟡 Created GALA instance:",{stringKey:e.toStringKey()}),e}extractTokenSymbolFromVault(e){if(!e||"string"!=typeof e)throw te("vaultAddress","Vault address");try{const t=Ao(e);return this.logger.debug("🔍 Extracted token symbol:",{vaultAddress:e,tokenSymbol:t}),t}catch(e){if(e instanceof G)throw ne("vaultAddress","format: service|Token$Unit$SYMBOL$eth:address$launchpad");throw e}}validateVaultAddress(e){if(!e||"string"!=typeof e)throw te("vaultAddress","Vault address");if(!e.startsWith("service|Token$Unit$"))throw ne("vaultAddress",'starting with "service|Token$Unit$"');if(!e.endsWith("$launchpad"))throw ne("vaultAddress",'ending with "$launchpad"');const t=e.split("$");if(t.length<5)throw ne("vaultAddress",'having at least 5 parts separated by "$"');const n=t[2];if(!n||!/^[A-Za-z]{1,10}$/.test(n))throw ne("vaultAddress","containing a 1-10 letter token symbol (case insensitive)");return this.logger.debug("✅ Vault address validation passed:",e),!0}generateTokenClassKeyString(e,t,n,o){return`${e}$${t}$${n}$${o}`}parseTokenClassKeyString(e){try{return fo(e)}catch(e){if(e instanceof G)throw ne("stringKey","format: collection$category$type$additionalKey (4 parts)");throw e}}}function Qo(e,t,n){if(t<0||t>1)throw new Error(`Invalid slippage tolerance factor: ${t}. Must be between 0 and 1 (e.g., 0.05 for 5%)`);const o=new u(e);if(o.isNaN())throw new Error(`Invalid expected amount: ${e}. Must be a valid number`);if(0===t)return e;const i=o.multipliedBy(t);let r;switch(n){case"buy-native":case"sell-exact":r=o.minus(i);break;case"buy-exact":case"sell-native":r=o.plus(i);break;default:throw new Error(`Unknown operation type: ${n}`)}return r.isLessThan(0)&&(r=new u(0)),r.toFixed()}class Xo extends xn{constructor(e,t,n=!1,o,i,r=.05,a=.01){super(e,n),this.tokenResolver=t,this.walletProvider=o,this.userAddress=i,this.defaultSlippageToleranceFactor=r,this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor=a,this.bundleEndpoint="/bundle",o&&i&&(this.signatureService=new zo(o,n),this.tokenKeyService=new Vo(n))}async submitTransaction(e){try{this.logger.debug("📦 Submitting bundle transaction:",{method:e.method,stringsInstructionsCount:e.stringsInstructions.length,signedDtoKeys:Object.keys(e.signedDto)}),this.validateBundleData(e);const t=this.formatBundleRequest(e);this.logger.debug("🚀 Bundle request payload:",{...t,signedDto:"[REDACTED - Contains signature]"});const n=await this.http.post(this.bundleEndpoint,t);return n?(this.logger.debug("📥 Bundle API response:",{success:n.success,hasData:!!n.data,error:n.error}),this.handleBundleResponse(n)):{success:!1,error:"No response from bundle API"}}catch(e){return this.logger.error("❌ Bundle transaction submission failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}validateBundleData(e){if(!e)throw te("bundleData","Bundle data");if(!e.signedDto)throw te("signedDto","Signed DTO");if(!e.method||"string"!=typeof e.method)throw te("method","Method name");if(!Array.isArray(e.stringsInstructions))throw ne("stringsInstructions","an array of resource tracking strings");if(0===e.stringsInstructions.length)throw new G("stringsInstructions cannot be empty","stringsInstructions","EMPTY_ARRAY");const t=["BuyWithNative","BuyExactToken","SellExactToken","SellWithNative"];if(!t.includes(e.method))throw ne("method",`one of: ${t.join(", ")}`);e.stringsInstructions.forEach((e,t)=>{if("string"!=typeof e||0===e.length)throw new G(`stringsInstructions[${t}] must be a non-empty string`,`stringsInstructions[${t}]`,"INVALID_INSTRUCTION");if(!e.startsWith("$"))throw new G(`stringsInstructions[${t}] must start with '$': ${e}`,`stringsInstructions[${t}]`,"INVALID_INSTRUCTION_FORMAT")}),this.logger.debug("✅ Bundle data validation passed")}formatBundleRequest(e){return{signedDto:e.signedDto,stringsInstructions:e.stringsInstructions,method:e.method}}handleBundleResponse(e){if(e.data&&!1===e.error)return this.logger.debug("✅ Bundle transaction successful:",e.data),{success:!0,data:e.data};const t=e.error||e.message||"Bundle transaction failed";return this.logger.debug("❌ Bundle transaction failed:",t),{success:!1,error:t}}formatErrorMessage(e){if("string"==typeof e)return e;if(Wn(e)&&e.response){const t=e.response.data;if(t&&"object"==typeof t){const e=t;if(e.error)return String(e.error);if(e.message)return String(e.message)}}return Gn(e)||"Unknown bundle transaction error"}async getBundlerTransactionResult(e){try{if(!e||"string"!=typeof e)throw te("transactionId","Transaction ID");this.logger.debug("🔍 Checking bundler transaction result:",e);const t=await this.http.get(`${this.bundleEndpoint}?id=${e}`);return t?(this.logger.debug("📊 Bundler transaction result:",t),{success:!0,data:t}):{success:!1,error:"No response from bundler transaction query"}}catch(e){return this.logger.error("❌ Failed to get bundler transaction result:",e),{success:!1,error:this.formatErrorMessage(e)}}}async cancelTransaction(e){try{if(!e||"string"!=typeof e)throw te("transactionId","Transaction ID");this.logger.debug("🚫 Cancelling transaction:",e);const t=await this.http.delete(`${this.bundleEndpoint}/${e}`);return t?(this.logger.debug("🗑️ Transaction cancellation response:",t),{success:!0,data:t}):{success:!1,error:"No response from transaction cancellation"}}catch(e){return this.logger.error("❌ Failed to cancel transaction:",e),{success:!1,error:this.formatErrorMessage(e)}}}async getHealthStatus(){try{this.logger.debug("🏥 Checking bundle service health");const e=await this.http.get(`${this.bundleEndpoint}/health`);return e?(this.logger.debug("💚 Bundle service health:",e),{success:!0,data:e}):{success:!1,error:"No response from bundle service health check"}}catch(e){return this.logger.error("❌ Bundle service health check failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}async buyToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:n,type:o,expectedAmount:i}=e,{effectiveSlippageFactor:r,effectiveMaxFee:a,vaultAddress:s}=await this.prepareTradingOperation(t,e.maxAcceptableReverseBondingCurveFee,e.maxAcceptableReverseBondingCurveFeeSlippageFactor,e.slippageToleranceFactor);if("native"===o){if(!i)throw new G("expectedAmount is required for native buy operations. Use getBuyTokenAmount() first to calculate expected tokens.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Qo(i,r,"buy-native");this.logger.debug("BuyNative slippage applied:",{originalExpectedTokens:i,slippageFactor:r,adjustedMinTokens:e});const t=new Ho.BuyNativeDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"BuyWithNative",s)}{if(!i)throw new G("expectedAmount is required for exact buy operations. Use getBuyTokenAmount() first to calculate expected GALA cost.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Qo(i,r,"buy-exact");this.logger.debug("BuyExact slippage applied:",{originalExpectedGalaCost:i,slippageFactor:r,adjustedMaxGalaCost:e});const t=new Ho.BuyExactDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"BuyExactToken",s)}}async sellToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:n,type:o,expectedAmount:i}=e,{effectiveSlippageFactor:r,effectiveMaxFee:a,vaultAddress:s}=await this.prepareTradingOperation(t,e.maxAcceptableReverseBondingCurveFee,e.maxAcceptableReverseBondingCurveFeeSlippageFactor,e.slippageToleranceFactor);if("exact"===o){if(!i)throw new G("expectedAmount is required for exact sell operations. Use getSellTokenAmount() first to calculate expected GALA.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Qo(i,r,"sell-exact");this.logger.debug("SellExact slippage applied:",{originalExpectedGala:i,slippageFactor:r,adjustedMinGala:e});const t=new Ho.SellExactDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"SellExactToken",s)}{if(!i)throw new G("expectedAmount is required for native sell operations. Use getSellTokenAmount() first to calculate tokens to sell.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Qo(i,r,"sell-native");this.logger.debug("SellNative slippage applied:",{originalExpectedTokensToSell:i,slippageFactor:r,adjustedMaxTokensToSell:e});const t=new Ho.SellNativeDto(s,n,e,{maxAcceptableReverseBondingCurveFee:a});return await this.executeBundleTransaction(t,"SellWithNative",s)}}async prepareTradingOperation(e,t,n,o){const{effectiveSlippageFactor:i,effectiveMaxFee:r}=this.calculateEffectiveSlippage(t,n,o),a=await this.resolveTokenNameToVault(e);if(!a)throw ee(e);return{effectiveSlippageFactor:i,effectiveMaxFee:r,vaultAddress:a}}calculateEffectiveSlippage(e,t,n){const o=n??this.defaultSlippageToleranceFactor,i=t??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let r=e||"0";return e&&(r=Qo(e,i,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:e,slippageFactor:i,adjustedMaxFee:r})),{effectiveSlippageFactor:o,effectiveFeeSlippageFactor:i,effectiveMaxFee:r}}ensureTradingServicesAvailable(){if(!this.signatureService||!this.tokenKeyService)throw ie("Trading services not available. BundleService requires walletProvider and userAddress for trading operations.","walletProvider");if(!this.userAddress)throw te("userAddress","User address")}async executeBundleTransaction(e,t,n){this.ensureTradingServicesAvailable();try{e.uniqueKey=`galaswap - operation - ${N()}-${Date.now()}-${this.userAddress}`;const o=await this.signatureService.signDTO(e,t,this.userAddress),i=this.tokenKeyService.generateStringsInstructions(n),r={stringsInstructions:i,method:t,signedDto:o};this.logger.debug("📦 Bundle transaction data:",{method:t,stringsInstructions:i,dtoKeys:Object.keys(o)});const a=await this.submitTransaction(r);if(a.success&&a.data)return this.logger.debug("✅ Bundle transaction submitted:",a.data),{success:!0,data:{transactionId:a.data,message:"Transaction submitted successfully. Monitor WebSocket for completion."}};throw new Error(String(a.error||"Bundle transaction failed"))}catch(e){throw this.logger.error("❌ Bundle transaction error:",e),e}}async resolveTokenNameToVault(e){return await this.tokenResolver.resolveTokenToVault(e)}}var Zo,Yo;!function(e){e.PROCESSED="PROCESSED",e.COMPLETED="COMPLETED",e.SUCCESS="SUCCESS",e.FAILED="FAILED",e.ERROR="ERROR",e.PROCESSING="PROCESSING",e.PENDING="PENDING"}(Zo||(Zo={})),function(e){e.PENDING="pending",e.PROCESSING="processing",e.COMPLETED="completed",e.FAILED="failed",e.TIMEOUT="timeout"}(Yo||(Yo={}));const Jo={[Zo.PROCESSED]:Yo.COMPLETED,[Zo.COMPLETED]:Yo.COMPLETED,[Zo.SUCCESS]:Yo.COMPLETED,[Zo.FAILED]:Yo.FAILED,[Zo.ERROR]:Yo.FAILED,[Zo.PROCESSING]:Yo.PROCESSING,[Zo.PENDING]:Yo.PENDING};class ei extends Cn{constructor(e,t=!1){super(t),this.socket=null,this.listeners=new Map,this.timeouts=new Map,this.reconnectCount=0,this.hasOnAnyListener=!1,this.eventBuffer=new Map,this.eventBufferTimeouts=new Map,this.MAX_BUFFER_SIZE=1e3,this.config={reconnectAttempts:5,reconnectDelay:2e3,timeout:3e5,...e},this.debug=t,this.isSocketIOAvailable=this.checkSocketIOAvailability()}checkSocketIOAvailability(){try{return"function"==typeof D||(this.logger.warn('⚠️ Socket.IO client not available. Install "socket.io-client" package.'),!1)}catch(e){return this.logger.warn("⚠️ Socket.IO availability check failed:",e),!1}}async connect(){return new Promise((e,t)=>{try{if(!this.isSocketIOAvailable){const e=new Error('Socket.IO not available in current environment. Install "socket.io-client" package.');return this.logger.error("❌ Socket.IO connection failed:",e.message),void t(e)}this.logger.debug("🔌 Connecting to Socket.IO server:",this.config.url),this.socket=D(this.config.url,{transports:["websocket"],reconnection:!0,reconnectionAttempts:this.config.reconnectAttempts||5,reconnectionDelay:this.config.reconnectDelay||2e3}),this.socket.on("connect",()=>{this.logger.debug("✅ Socket.IO connected successfully:",this.socket?.id),this.logger.debug("📡 Connected to bundle backend WebSocket:",this.config.url),this.logger.debug("🔗 Ready to monitor transaction updates"),this.reconnectCount=0,e()}),this.socket.on("connect_error",e=>{this.logger.error("❌ Socket.IO connection error:",e),t(e)}),this.socket.on("disconnect",e=>{this.logger.debug(`🔌 Socket.IO disconnected: ${e}`),this.handleReconnect()}),this.socket.on("error",e=>{this.logger.error("❌ Socket.IO error:",e)}),this.socket.onAny((e,...t)=>{if(e&&t.length>0&&"object"==typeof t[0]&&null!==t[0]){const n=t[0],o=n.status||n.Status;if(o&&"string"==typeof o){if(this.logger.debug(`📡 [Event Buffer] Buffering event for ${e}: ${o}`),this.eventBuffer.size>=this.MAX_BUFFER_SIZE){const e=this.eventBuffer.keys().next().value;if(e){const t=this.eventBufferTimeouts.get(e);t&&(clearTimeout(t),this.eventBufferTimeouts.delete(e)),this.eventBuffer.delete(e),this.logger.warn(`📡 [Event Buffer] Buffer full (${this.MAX_BUFFER_SIZE}), dropped oldest: ${e}`)}}this.eventBuffer.set(e,n);const t=this.eventBufferTimeouts.get(e);t&&clearTimeout(t);const i=setTimeout(()=>{this.eventBuffer.has(e)&&(this.logger.debug(`📡 [Event Buffer] Cleaning up orphaned event for ${e}`),this.eventBuffer.delete(e),this.eventBufferTimeouts.delete(e))},3e4);this.eventBufferTimeouts.set(e,i)}}this.debug&&this.logger.debug(`📡 [WebSocket Event] "${e}":`,JSON.stringify(t,null,2))}),this.hasOnAnyListener=!0}catch(e){t(e)}})}async monitorTransaction(e,t){this.listeners.set(e,t),this.logger.debug(`📡 Starting to monitor transaction: ${e}`),this.logger.debug(`📡 WebSocket connected: ${!!this.socket&&this.socket.connected}`);const n=this.eventBuffer.get(e);if(n){this.logger.debug(`📡 [Event Buffer] Found buffered event for ${e}, delivering immediately`),setImmediate(()=>{this.processTransactionEvent(e,n,t)});const o=this.eventBufferTimeouts.get(e);o&&(clearTimeout(o),this.eventBufferTimeouts.delete(e)),this.eventBuffer.delete(e)}const o=this.config.timeout||3e5,i=setTimeout(()=>{if(this.listeners.has(e)){const n=Math.round(o/1e3),i={transactionId:e,status:Yo.TIMEOUT,message:`Transaction monitoring timeout - no response after ${n} seconds`,timestamp:Date.now()};this.logger.debug(`📡 Transaction timeout for ${e} (${n}s)`),t(i),this.listeners.delete(e),this.timeouts.delete(e),this.socket?.off(e)}},o);if(this.timeouts.set(e,i),this.socket&&this.socket.connected)this.socket.off(e),this.logger.debug(`📡 Listening for transaction updates: ${e}`),this.logger.debug(`📡 WebSocket connection ID: ${this.socket.id}`),this.logger.debug(`📡 WebSocket URL: ${this.config.url}`),this.socket.on(e,n=>{this.processTransactionEvent(e,n,t)});else{const n={transactionId:e,status:Yo.FAILED,message:"WebSocket not connected - cannot monitor transaction",timestamp:Date.now()};t(n),this.listeners.delete(e),this.timeouts.delete(e)}}processTransactionEvent(e,t,n){this.logger.debug(`📡 Socket.IO transaction update for ${e}:`,JSON.stringify(t,null,2));const o=t,i=o?.data,r=o?.status||o?.Status||i?.status||i?.Status;let a=o?.message||o?.Message||i?.message||i?.Message||o?.error||i?.error;a&&"string"==typeof a||(a=r===Zo.FAILED||r===Zo.ERROR?"Transaction failed - check transaction details":r===Zo.COMPLETED||r===Zo.PROCESSED||r===Zo.SUCCESS?"Transaction completed successfully":r?`Transaction status: ${r}`:"Unknown transaction status");const s=o?.blockHash||i?.blockHash,c=o?.gasUsed||i?.gasUsed,l=o?.Data||i?.Data,u={transactionId:e,status:this.mapSocketStatus(r),message:"string"==typeof a?a:"Transaction update received",timestamp:Date.now(),...s&&{blockHash:s},...c&&{gasUsed:c},...l&&{data:l}};if(this.logger.debug(`📡 Mapped status for ${e}: ${r} -> ${u.status}`),this.logger.debug(`📡 Final message: "${a}"`),n(u),u.status===Yo.COMPLETED||u.status===Yo.FAILED){this.listeners.delete(e);const t=this.timeouts.get(e);t&&(clearTimeout(t),this.timeouts.delete(e)),this.socket?.off(e),this.logger.debug(`📡 Cleaned up listener for ${e} (${u.status})`)}}async waitForTransaction(e){return new Promise((t,n)=>{this.monitorTransaction(e,e=>{e.status===Yo.COMPLETED?t(e):e.status!==Yo.FAILED&&e.status!==Yo.TIMEOUT||n(new Error(`Transaction ${e.status}: ${e.message}`))})})}mapSocketStatus(e){const t=e?.toUpperCase();return Jo[t]||Yo.PENDING}async handleReconnect(){this.reconnectCount<this.config.reconnectAttempts?(this.reconnectCount++,this.logger.debug(`🔄 Attempting Socket.IO reconnect ${this.reconnectCount}/${this.config.reconnectAttempts}`),setTimeout(()=>{this.socket&&!this.socket.connected&&this.socket.connect()},this.config.reconnectDelay)):this.logger.error("❌ Socket.IO max reconnection attempts reached")}disconnect(){this.socket&&(this.listeners.forEach((e,t)=>{this.socket?.off(t)}),this.listeners.clear(),this.timeouts.forEach(e=>{clearTimeout(e)}),this.timeouts.clear(),this.eventBuffer.clear(),this.eventBufferTimeouts.forEach(e=>{clearTimeout(e)}),this.eventBufferTimeouts.clear(),this.logger.debug("🧹 Cleared event buffer and timeouts"),this.hasOnAnyListener&&(this.socket.offAny(),this.hasOnAnyListener=!1,this.logger.debug("🧹 Removed onAny debug listener")),this.socket.disconnect(),this.socket=null,this.logger.debug("🔌 Socket.IO disconnected"))}isConnected(){return this.socket?.connected||!1}getSocket(){return this.socket}}class ti extends Cn{constructor(e,t=!1){super(t),this.poolService=e,this.cache=new Map}async resolveTokenToVault(e){if(!ce(e))throw new G("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");const t=e.trim().toLowerCase(),n=this.get(t);if(n)return n;try{const n=await this.poolService.resolveTokenNameToVault(e);return n&&this.set(t,n),n}catch{return null}}async resolveTokenClassKey(e){const t=await this.resolveTokenToVault(e);if(!t)throw ee(e);return this.parseVaultAddressToTokenClassKey(t)}get(e){return this.cache.get(e.toLowerCase())||null}set(e,t){this.cache.set(e.toLowerCase(),t)}clear(){this.cache.clear()}getStats(){return{size:this.cache.size,keys:Array.from(this.cache.keys())}}preWarm(e){for(const{tokenName:t,vaultAddress:n}of e)this.set(t,n)}parseVaultAddressToTokenClassKey(e){try{return To(e)}catch(e){if(e instanceof G)throw ne("vaultAddress","format: service|Token$Unit$SYMBOL$eth:address$launchpad","Vault address");throw e}}}async function ni(e,t={}){const{maxPages:n=1e4,logger:o,pageSize:i=20}=t,r=[];let a=1,s=!0,c=0;for(;s&&a<=n;){o&&o.debug(`Auto-pagination: fetching page ${a} with limit ${i}`);const t=await e(a,i);if(!t||!Array.isArray(t.items)){o&&o.warn("Auto-pagination: received invalid result structure, stopping");break}if(r.push(...t.items),c=t.total,o&&o.debug(`Auto-pagination: page ${a} returned ${t.items.length} items, hasNext: ${t.hasNext}`),0===t.items.length){o&&o.debug(`Auto-pagination: no items returned on page ${a}, exiting loop`);break}s=t.hasNext,a++}return a>n&&o&&o.warn(`Auto-pagination: exceeded maxPages limit of ${n}, stopping iteration`),o&&o.debug(`Auto-pagination: completed with total items: ${r.length}, total count: ${c}`),{items:r,total:c}}function oi(e,t=e.length,n="items"){return{...{page:1,limit:e.length||0,total:t,totalPages:Math.ceil(t/(e.length||1))||1,hasNext:!1,hasPrevious:!1},[n]:e}}class ii extends xn{constructor(e,t=!1,n){super(e,t),this.tokenResolverService=n}async fetchTokenClassKeyByTokenName(e){if(!this.tokenResolverService)throw ie("TokenResolverService is required for token name resolution. Ensure it is passed to PriceHistoryService constructor.","tokenResolverService");if(!e||"string"!=typeof e||e.length<3||e.length>20)throw ie("Token name must be a string between 3 and 20 characters","tokenName");this.logger.debug(`Resolving token name '${e}' to token class key`);try{const t=await this.tokenResolverService.resolveTokenToVault(e);if(!t)throw ie(`Token '${e}' not found or could not be resolved to vault address`,"tokenName");this.logger.debug(`Resolved '${e}' to vault address: ${t}`);const n=To(t),o=`${n.collection}|${n.category}|${n.type}|${n.additionalKey}`;return this.logger.debug(`Extracted token class key: ${o}`),o}catch(t){if(t instanceof Error&&t.message.includes("ConfigurationError"))throw t;throw oe(`Failed to resolve token name '${e}': ${Gn(t)}`,500)}}async fetchPriceHistory(e){this.logger.debug("Fetching price history from DEX Backend API with options:",e),this.validateOptions(e);try{let t=e.tokenId;if(e.tokenName){this.logger.debug(`Resolving token name '${e.tokenName}' to token ID`);const n=await this.fetchTokenClassKeyByTokenName(e.tokenName);t=n,this.logger.debug(`Resolved to token ID: ${n}`)}if(!t)throw ie("Token ID is required but was not provided or resolved","tokenId");const{normalizeToTokenInstanceKey:n}=await Promise.resolve().then(function(){return Io}),o=n(t),i=mo(`${o.collection}|${o.category}|${o.type}|${o.additionalKey}`),{from:r,to:a,sortOrder:s="DESC",page:c=1,limit:l=10}=e,u={token:i,page:String(c),limit:String(l)};r&&(u.from=r.toISOString()),a&&(u.to=a.toISOString());const d=function(e){if(e)return e.toLowerCase()}(s);d&&(u.order=d),this.logger.debug(`Querying price snapshots for token ${i}, page ${c}, limit ${l}`);const h=await this.http.get("/price-oracle/fetch-price",u);if(!h)throw oe("No response from price history service",500);const g=this.transformApiResponseToPriceHistory(h);return this.logger.debug(`Found ${g.snapshots.length} price snapshots, total ${g.total}`),g}catch(e){if(e instanceof Error&&(e.message.includes("ConfigurationError")||e.message.includes("NetworkError")))throw e;throw oe(`Failed to fetch price history: ${Gn(e)}`,500)}}transformApiResponseToPriceHistory(e){if(!e?.data?.data||!Array.isArray(e.data.data))throw oe("Invalid API response: missing or invalid data.data array",500);if(!e?.data?.meta)throw oe("Invalid API response: missing data.meta pagination info",500);const t=e.data.data.map(e=>({price:e.price,timestamp:new Date(e.createdAt),tokenId:`${e.collection}|${e.category}|${e.type}|${e.additionalKey}`})),n=e.data.meta,o=n.currentPage??1,i=n.totalPages??1;return{snapshots:t,page:o,limit:n.pageSize??50,total:n.totalItems??0,totalPages:i,hasNext:o<i,hasPrevious:o>1}}async fetchAllPriceHistory(e){this.logger.debug("Fetching all price history with options:",e);const t=await ni((t,n)=>this.fetchPriceHistory({...e,page:t,limit:n}).then(e=>({items:e.snapshots,page:e.page,limit:e.limit,total:e.total,totalPages:e.totalPages,hasNext:e.hasNext,hasPrevious:e.hasPrevious})),{maxPages:1e4,logger:this.logger,pageSize:50});return oi(t.items,t.total,"snapshots")}validateOptions(e){const{from:t,to:n,sortOrder:o,page:i=1,limit:r=10}=e;if(function(e,t,n,o={}){const{description:i="parameter",treatEmptyAsNull:r=!0}=o,a=e[t],s=e[n],c=null!=a&&(!r||""!==a),l=null!=s&&(!r||""!==s);if(!c&&!l)throw ie(`Either ${t} or ${n} must be provided (${i})`,n);if(c&&l)throw ie(`Cannot provide both ${t} and ${n}. Provide exactly one (${i}).`,n)}(e,"tokenName","tokenId",{description:"token identifier"}),t&&!(t instanceof Date)&&isNaN(new Date(t).getTime()))throw ie("from must be a valid Date","from");if(n&&!(n instanceof Date)&&isNaN(new Date(n).getTime()))throw ie("to must be a valid Date","to");if(o&&"ASC"!==o&&"DESC"!==o)throw ie('sortOrder must be either "ASC" or "DESC"',"sortOrder");if(!Number.isInteger(i)||i<1)throw ie("page must be a positive integer","page");if(!Number.isInteger(r)||r<1||r>50)throw ie("limit must be between 1 and 50","limit")}}function ri(e){if("object"==typeof e&&null!==e)return function(e){if(!e||"object"!=typeof e)throw new G("Token object must be a non-null object, got "+typeof e,"token","INVALID_TOKEN_OBJECT");const{collection:t,category:n,type:o,additionalKey:i}=e;if(!t||"string"!=typeof t)throw new G("Token.collection must be a non-empty string, got "+typeof t,"token.collection","MISSING_OR_INVALID_COLLECTION");if(!n||"string"!=typeof n)throw new G("Token.category must be a non-empty string, got "+typeof n,"token.category","MISSING_OR_INVALID_CATEGORY");if(!o||"string"!=typeof o)throw new G("Token.type must be a non-empty string, got "+typeof o,"token.type","MISSING_OR_INVALID_TYPE");if(!i||"string"!=typeof i)throw new G("Token.additionalKey must be a non-empty string, got "+typeof i,"token.additionalKey","MISSING_OR_INVALID_ADDITIONAL_KEY");return{collection:t,category:n,type:o,additionalKey:i}}(e);if(!e)throw new G(`Token cannot be null, undefined, or empty. Received: ${JSON.stringify(e)}`,"token","EMPTY_TOKEN");if("string"!=typeof e)throw new G("Token must be a string or TokenClassKey object, got "+typeof e,"token","INVALID_TOKEN_TYPE");if(e.includes("|"))return function(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("|");if(t.length<4)throw new Error(`Invalid pipe-delimited token format. Expected 4+ parts separated by |, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("|");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){const n=e.split("|");throw new G(`Invalid pipe-delimited token: "${e}" (${n.length} parts). Expected format: "collection|category|type|additionalKey" (4 parts minimum). Received: [${n.map(e=>`"${e}"`).join(", ")}]. Error: ${t instanceof Error?t.message:String(t)}`,"token","INVALID_PIPE_DELIMITED_TOKEN")}}(e);if(e.includes("$"))return function(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("$");if(t.length<4)throw new Error(`Invalid dollar-delimited token format. Expected 4+ parts separated by $, got ${t.length}`);const n=t[t.length-1],o=t[t.length-2],i=t[t.length-3],r=t.slice(0,t.length-3).join("$");if(!(r&&i&&o&&n))throw new Error("All components (collection, category, type, additionalKey) must be non-empty");return{collection:r,category:i,type:o,additionalKey:n}}catch(t){const n=e.split("$");throw new G(`Invalid dollar-delimited token: "${e}" (${n.length} parts). Expected format: "collection$category$type$additionalKey" (4 parts minimum). Received: [${n.map(e=>`"${e}"`).join(", ")}]. Error: ${t instanceof Error?t.message:String(t)}`,"token","INVALID_DOLLAR_DELIMITED_TOKEN")}}(e);throw new G(`Plain token string "${e}" (length: ${e.length}) is not allowed - tokens must be delimited with | or $. Expected format: "GALA|Unit|none|none" or "GALA$Unit$none$none". Input: "${e}"`,"token","PLAIN_STRING_NOT_ALLOWED")}class ai extends Error{constructor(e,t,n){super(`API Error [${e}]: ${t}`),this.status=e,this.message=t,this.details=n,this.name="ApiError"}}function si(e){return"object"==typeof e&&null!==e&&"collection"in e&&"category"in e&&"type"in e&&"additionalKey"in e&&"string"==typeof e.collection&&"string"==typeof e.category&&"string"==typeof e.type&&"string"==typeof e.additionalKey}function ci(e){if("object"!=typeof e||null===e)return!1;const t=e;return"string"==typeof t.positionId&&si(t.token0ClassKey)&&si(t.token1ClassKey)&&"number"==typeof t.fee&&"number"==typeof t.tickLower&&"number"==typeof t.tickUpper&&"string"==typeof t.liquidity&&"string"==typeof t.feeGrowthInside0Last&&"string"==typeof t.feeGrowthInside1Last&&"string"==typeof t.tokensOwed0&&"string"==typeof t.tokensOwed1}class li{constructor(t){this.client=e.create({baseURL:t.baseUrl,timeout:t.timeout??3e4})}async getPoolData(e){try{if("string"==typeof e.token0||"string"==typeof e.token1)throw new Error(`GalaChain API getPoolData requires TokenClassKey objects, not strings. Received: token0="${"string"==typeof e.token0?e.token0:"[object]"}", token1="${"string"==typeof e.token1?e.token1:"[object]"}". Convert pipe-delimited tokens using parseToken() before calling getPoolData(). Example: parseToken("GALA|Unit|none|none") → { collection: "GALA", category: "Unit", type: "none", additionalKey: "none" }`);const t=await this.client.post("/api/asset/dexv3-contract/GetPoolData",e);this.validateResponse(t.data);const n=t.data.Data;if(!function(e){if("object"!=typeof e||null===e)return!1;const t=e;return"string"==typeof t.token0&&"string"==typeof t.token1&&si(t.token0ClassKey)&&si(t.token1ClassKey)&&"number"==typeof t.fee&&"number"==typeof t.tickSpacing&&"string"==typeof t.liquidity&&"string"==typeof t.sqrtPrice&&"number"==typeof t.tick&&"string"==typeof t.feeGrowthGlobal0&&"string"==typeof t.feeGrowthGlobal1}(n))throw new ai(t.status,"Invalid pool data response format",n);return n}catch(e){throw this.handleError(e,"getPoolData")}}async getPositions(e){try{const t=await this.client.post("/api/asset/dexv3-contract/GetPositions",e);this.validateResponse(t.data);const n=t.data.Data;let o;o=n&&"object"==typeof n&&"positions"in n&&Array.isArray(n.positions)?n.positions:n&&"object"==typeof n&&"positionId"in n?[n]:Array.isArray(n)?n:[];for(const e of o)if(!ci(e))throw new ai(t.status,"Invalid position in response",e);return{positions:o,count:o.length}}catch(e){throw this.handleError(e,"getPositions")}}async getPositionById(e,t,n,o,i,r,a){try{let s,c;if(void 0!==t&&void 0!==n&&void 0!==o&&void 0!==i&&void 0!==r){s={owner:e,token0:"string"==typeof t?{collection:t,category:"Unit",type:"none",additionalKey:"none"}:t,token1:"string"==typeof n?{collection:n,category:"Unit",type:"none",additionalKey:"none"}:n,fee:o,tickLower:i,tickUpper:r},a&&(s.positionId=a),c=`${e}/${t}/${n}/${o}`}else s={positionId:e},c=e;const l=await this.client.post("/api/asset/dexv3-contract/GetPositions",s);this.validateResponse(l.data);const u=l.data.Data;let d;if(u&&"object"==typeof u&&"positionId"in u&&!("positions"in u))d=u;else{if(!(u&&Array.isArray(u.positions)&&u.positions.length>0))throw new ai(404,`Position not found: ${c}`);d=u.positions[0]}const h={Data:d,Status:l.status};return void 0!==l.data.Message&&(h.Message=l.data.Message),h}catch(t){throw this.handleError(t,`getPositionById(${e})`)}}async getRemoveLiquidityEstimation(e){try{const t=await this.client.post("/api/asset/dexv3-contract/GetRemoveLiquidityEstimation",e);this.validateResponse(t.data);const n=t.data.Data;if("string"!=typeof n.amount0||"string"!=typeof n.amount1)throw new ai(t.status,"Invalid removal estimation response format",n);return n}catch(e){throw this.handleError(e,"getRemoveLiquidityEstimation")}}validateResponse(e){if(!e||"object"!=typeof e)throw new ai(500,"Invalid response format: not an object");if(!("Data"in e)||!("Status"in e))throw new ai(500,"Invalid response format: missing Data or Status field");if(e.Status>=400)throw new ai(e.Status,e.Message??"Gateway error",e.Data)}handleError(t,n){if(t instanceof ai)return t;if(e.isAxiosError(t)){const e=t.response?.status??500,o=t.response?.data?.Message??t.message,i=t.response?.data?.Data??void 0;return new ai(e,`${n}: ${o}`,i)}return new ai(500,`${n}: ${t instanceof Error?t.message:String(t)}`)}}class ui{constructor(e){this.http=e}async getUserAssets(e,t=20,n=0){try{if(!e||"string"!=typeof e)throw new ai(400,"Invalid wallet address");const o=Math.max(1,Math.floor(n/t)+1),i={};i.address=e,i.page=o,i.limit=t;const r=await this.http.get("/user/assets",i);if(!r||"object"!=typeof r)throw new ai(500,"Invalid response format: not an object");const a=r.data;if(!a||"object"!=typeof a)throw new ai(500,"Invalid response format: missing data wrapper");const s=a.token;if(!Array.isArray(s))throw new ai(500,"Invalid response format: token array must be an array");const c=[];for(const e of s){if("object"!=typeof e||null===e)throw new ai(500,"Invalid asset in response: asset must be an object");const t=e;if("string"!=typeof t.symbol||"string"!=typeof t.name)throw new ai(500,"Invalid asset in response: missing symbol or name",t);const n="number"==typeof t.decimals?t.decimals:"string"==typeof t.decimals?parseInt(t.decimals,10):void 0;if("number"!=typeof n||isNaN(n))throw new ai(500,"Invalid asset in response: decimals must be a number",t);const o={tokenId:t.compositeKey||`${t.symbol}$Unit$none$none`,symbol:t.symbol,name:t.name,decimals:n,balance:t.quantity||"0"};t.image&&(o.imageUrl=t.image),t.verify&&(o.verified=t.verify),c.push(o)}const l={tokens:c,count:a.count??c.length};return void 0!==a.totalValue&&(l.totalValue=a.totalValue),l}catch(t){throw this.handleError(t,`getUserAssets(${e})`)}}handleError(e,t){if(e instanceof ai)return e;if(e instanceof Error&&e instanceof Error&&"response"in e){const n=e,o=n.response?.status??500,i=n.response?.data?.message??n.response?.data?.Message??n.message,r=n.response?.data?.Data??n.response?.data?.data??void 0;return n.response,new ai(o,`${t}: ${i}`,r)}return new ai(500,`${t}: ${e instanceof Error?e.message:String(e)}`)}}function di(e){try{if(!e||"string"!=typeof e)throw new Error("Token must be a non-empty string");const t=e.split("|");if(t.length<4)throw new Error(`Invalid pipe-delimited token format. Expected at least 4 parts separated by |, got ${t.length}`);const[n,o,i,...r]=t;if(!n||!o||!i)throw new Error("Collection, category, and type must be non-empty");const a=r.join("|");if(!a)throw new Error("AdditionalKey must be non-empty");return{collection:n,category:o,type:i,additionalKey:a}}catch(t){throw new G(`Invalid pipe-delimited token: "${e}". Expected format: "collection|category|type|additionalKey". Error: ${t instanceof Error?t.message:String(t)}`,"pipeDelimitedToken","INVALID_PIPE_DELIMITED_TOKEN_FORMAT")}}const hi=10;class gi extends Cn{constructor(e,t,n){if(super(!1),this.pricingConcurrency=5,this.tokenConverter=new po,this.webSocketService=t,this.dexQuoteService=n,this.getWalletAddress=e.getWalletAddress,this.galaChainBaseUrl=e.galaChainBaseUrl,this.bundlerBaseUrl=e.bundlerBaseUrl,this.gatewayBaseUrl=e.gatewayBaseUrl,this.privateKey=e.privateKey,!(e.gatewayBaseUrl&&e.bundlerBaseUrl&&e.dexBackendBaseUrl&&e.dexBackendHttp))throw new Error("GSwapService requires explicit gatewayBaseUrl, bundlerBaseUrl, dexBackendBaseUrl, and dexBackendHttp configuration. These must be provided by LaunchpadSDK to ensure environment alignment.");try{this.gatewayClient=new li({baseUrl:e.gatewayBaseUrl,timeout:3e4}),this.dexBackendClient=new ui(e.dexBackendHttp),this.logger.debug("HTTP clients initialized successfully",{gatewayUrl:e.gatewayBaseUrl,dexBackendUrl:e.dexBackendBaseUrl})}catch(e){throw this.logger.error("Failed to initialize HTTP clients",e),new Error("Failed to initialize GSwapService HTTP clients")}}setPricingConcurrency(e){if(e<1)throw new Error("Pricing concurrency must be at least 1");e>100&&this.logger.warn("Pricing concurrency > 100 may cause performance issues",{concurrency:e}),this.pricingConcurrency=e,this.logger.debug("Updated pricing concurrency",{concurrency:this.pricingConcurrency})}async getSwapQuoteExactInput(e){try{if(new u(e.amount).isLessThanOrEqualTo(0))throw new z("Amount must be greater than zero",{amount:e.amount,fromToken:e.fromToken,toToken:e.toToken});if(!this.dexQuoteService)throw new z("DexQuoteService not configured - cannot provide quotes",{fromToken:e.fromToken,toToken:e.toToken});this.logger.debug("Getting swap quote for exact input",{fromToken:e.fromToken,toToken:e.toToken,amount:e.amount});const t=this.tokenConverter.toLaunchpadFormat(e.fromToken),n=this.tokenConverter.toLaunchpadFormat(e.toToken),[o,i]=t<n?[t,n]:[n,t],r=[3e3,500,1e4];let a;for(const s of r)try{const r=await this.dexQuoteService.fetchCompositePoolData({token0:o,token1:i,fee:s,gatewayBaseUrl:this.gatewayBaseUrl}),a=await this.dexQuoteService.calculateDexPoolQuoteExactAmount({compositePoolData:r,fromToken:t,toToken:n,amount:e.amount}),c=new u(a.currentSqrtPrice),l=new u(a.newSqrtPrice),d=c.gt(l)?c.minus(l).dividedBy(c):new u(0),h=new u(a.amount0),g=new u(a.amount1),p=h.isNegative(),m=g.isNegative();this.logger.debug("=== AMOUNT SELECTION RAW DATA ===",{"quoteResult.amount0":a.amount0,"quoteResult.amount1":a.amount1,"amount0BN.isNegative()":p,"amount1BN.isNegative()":m});const f=p?h:g;this.logger.debug("=== AMOUNT SELECTION RESULT ===",{selectedFromAmount0:p,selectedAmount:f.toFixed(),selectedAmountAbs:f.absoluteValue().toFixed()});const y=f.absoluteValue().toFixed();return{fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.amount,estimatedOutput:y,feeTier:s,priceImpact:d.toFixed(),executionPrice:this.calculateExecutionPrice(e.amount,y),currentSqrtPrice:a.currentSqrtPrice,newSqrtPrice:a.newSqrtPrice}}catch(e){a=e,this.logger.debug("DexQuoteService failed for fee tier, trying next",{feeTier:s,error:e instanceof Error?e.message:"Unknown error"})}throw a||new z("No available fee tiers for quote",{feeTiers:r,fromToken:e.fromToken,toToken:e.toToken})}catch(e){this.handleGSwapError("Failed to get swap quote for exact input",z,e)}}async getSwapQuoteExactOutput(e){try{if(new u(e.amount).isLessThanOrEqualTo(0))throw new z("Amount must be greater than zero",{amount:e.amount,fromToken:e.fromToken,toToken:e.toToken});if(!this.dexQuoteService)throw new z("DexQuoteService not configured - cannot provide quotes",{fromToken:e.fromToken,toToken:e.toToken});this.logger.debug("Getting swap quote for exact output",{fromToken:e.fromToken,toToken:e.toToken,amount:e.amount});const t=this.tokenConverter.toLaunchpadFormat(e.fromToken),n=this.tokenConverter.toLaunchpadFormat(e.toToken),[o,i]=t<n?[t,n]:[n,t],r=[3e3,500,1e4];let a;for(const s of r)try{const r=await this.dexQuoteService.fetchCompositePoolData({token0:o,token1:i,fee:s,gatewayBaseUrl:this.gatewayBaseUrl}),a=await this.dexQuoteService.calculateDexPoolQuoteExactAmount({compositePoolData:r,fromToken:t,toToken:n,amount:e.amount}),c=new u(a.currentSqrtPrice),l=new u(a.newSqrtPrice),d=c.gt(l)?c.minus(l).dividedBy(c):new u(0),h=r.pool.token0,g="string"==typeof h?h.split("|")[0]:"object"==typeof h&&null!==h&&"tokenName"in h?h.tokenName:String(h),p=n.split("|")[0]===g?a.amount1:a.amount0;return{fromToken:e.fromToken,toToken:e.toToken,inputAmount:p,estimatedOutput:e.amount,feeTier:s,priceImpact:d.toFixed(),executionPrice:this.calculateExecutionPrice(p,e.amount),currentSqrtPrice:a.currentSqrtPrice,newSqrtPrice:a.newSqrtPrice}}catch(e){a=e,this.logger.debug("DexQuoteService failed for fee tier, trying next",{feeTier:s,error:e instanceof Error?e.message:"Unknown error"})}throw a||new z("No available fee tiers for quote",{feeTiers:r,fromToken:e.fromToken,toToken:e.toToken})}catch(e){this.handleGSwapError("Failed to get swap quote for exact output",z,e)}}async executeSwap(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");this.logger.debug("Executing swap",{fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.inputAmount});const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.fromToken,e.toToken),o=function(e,t=.01){const n=ro(e),o=new u(1).minus(t);return n.multipliedBy(o)}(e.estimatedOutput,e.slippageTolerance||.01),i=this.getWalletAddress();if(!i)throw new Error("Wallet address required for swap execution");let r;try{const t=await this.getSwapQuoteExactInput({fromToken:e.fromToken,toToken:e.toToken,amount:e.inputAmount});r=t.currentSqrtPrice,this.logger.debug("Quote refetch successful - extracted sqrtPrices",{currentSqrtPrice:r,newSqrtPrice:t.newSqrtPrice,feeTier:t.feeTier})}catch(t){this.logger.debug("Could not re-fetch quote for sqrtPrice, using default",{fromToken:e.fromToken,toToken:e.toToken,error:t instanceof Error?t.message:String(t)})}const a={fromToken:t,toToken:n,inputAmount:e.inputAmount,minOutput:o.toFixed(),feeTier:e.feeTier,walletAddress:i,slippageTolerance:e.slippageTolerance||.01};void 0!==r&&(a.currentSqrtPrice=r);const s=await this.sendSwapToBundler(a);this.logger.debug("Swap submitted, monitoring transaction",{transactionId:s,fromToken:e.fromToken,toToken:e.toToken}),await this.ensureWebSocketConnected();const c=await this.webSocketService.waitForTransaction(s);return{transactionId:c.transactionId,status:c.status,fromToken:e.fromToken,toToken:e.toToken,inputAmount:e.inputAmount,outputAmount:e.estimatedOutput,feeTier:e.feeTier,slippageTolerance:e.slippageTolerance||.01,timestamp:new Date(c.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(s)}}}catch(e){const t=e;this.handleGSwapError("Failed to execute swap",V,e,{transactionHash:t?.txHash})}}async getUserAssets(e,t=1,n=20){try{this.logger.debug("Fetching user assets",{walletAddress:e,page:t,limit:n});const o=await this.dexBackendClient.getUserAssets(e,n,(t-1)*n);return(Array.isArray(o)?o:o?.tokens||[]).map(e=>{const t=e.symbol||"UNKNOWN";try{return{tokenId:ri(`${t}|Unit|none|none`),symbol:t,balance:(n=e.quantity||e.balance||0,void 0!==o?ro(n).toFixed(o):ro(n).toFixed()),decimals:e.decimals||18}}catch(n){return this.logger.debug(`Skipping asset with processing error: ${t}`,{error:n instanceof Error?n.message:String(n),assetData:{symbol:t,quantity:e.quantity,balance:e.balance,decimals:e.decimals}}),null}var n,o}).filter(e=>null!==e)}catch(o){this.handleGSwapError("Failed to fetch user assets",X,o,{walletAddress:e,page:t,limit:n})}}async getAllUserAssets(e){try{this.logger.debug("Fetching all user assets (auto-paginated)",{walletAddress:e});const t=await ni((t,n)=>this.getUserAssets(e,t,n).then(e=>({items:e,page:t,limit:n,total:e.length,totalPages:1,hasNext:e.length===n,hasPrevious:t>1})),{maxPages:1e4,logger:this.logger,pageSize:20});return this.logger.debug("Fetched all user assets",{walletAddress:e,totalAssets:t.items.length}),t.items}catch(t){this.handleGSwapError("Failed to fetch all user assets",X,t,{walletAddress:e})}}async getPoolInfo(e,t){try{this.logger.debug("Fetching pool info",{tokenA:e,tokenB:t});const{gswapToken0:n,gswapToken1:o}=this.convertTokenPair(e,t),i=[500,3e3,1e4];let r=new u(0),a=0;for(const s of i)try{const e="string"==typeof n?ri(n):n,t="string"==typeof o?ri(o):o,i=await this.gatewayClient.getPoolData({token0:e,token1:t,fee:s});i&&(r=r.plus(new u(i.liquidity||0)),a++)}catch{this.logger.debug("Pool not found for fee tier",{tokenA:e,tokenB:t,feeTier:s})}return{tokenA:e,tokenB:t,liquidity:r.toFixed(),feeTiers:i,swapCount:a}}catch(n){this.logger.warn("Failed to fetch pool info",n);const o=n;return this.logger.debug("Pool error details",{error:new Q(`Failed to fetch pool info: ${o?.message||String(n)}`,n,e,t,this.extractGSwapErrorCode(n))}),{tokenA:e,tokenB:t,liquidity:"0",feeTiers:[500,3e3,1e4],swapCount:0}}}chunkArray(e,t){const n=[];for(let o=0;o<e.length;o+=t)n.push(e.slice(o,o+t));return n}async fetchPositionPrices(e){const t=this.pricingConcurrency;if(0===e.length)return new Map;const n=new Map;for(const t of e){const e=`${t.token0}|${t.token1}|${t.feeTier}`;n.has(e)||n.set(e,{token0:t.token0,token1:t.token1,feeTier:t.feeTier})}const o=Array.from(n.values()),i=this.chunkArray(o,t);this.logger.debug("Fetching pricing for positions",{totalPositions:e.length,uniquePoolsToPrice:n.size,chunks:i.length,concurrency:t});const r=new Map;for(let e=0;e<i.length;e++){const t=i[e];(await Promise.allSettled(t.map(async e=>{const t=await this.getSwapQuoteExactInput({fromToken:e.token0,toToken:e.token1,amount:"1"});return{key:`${e.token0}|${e.token1}|${e.feeTier}`,data:{token0:e.token0,token1:e.token1,feeTier:e.feeTier,currentPrice:t.executionPrice,executionPrice:t.executionPrice,priceImpact:t.priceImpact,estimatedOutput:t.estimatedOutput,pricedAt:new Date}}}))).forEach(e=>{"fulfilled"===e.status?r.set(e.value.key,e.value.data):this.logger.warn("Failed to fetch price for pool",{error:e.reason})})}return r}normalizePositionResponse(e,t){const n=e=>{if(!e)return"";if("string"==typeof e)return e;if("object"==typeof e){if(e.type&&"none"!==e.type)return e.type;if(e.collection)return e.collection;if(e.symbol)return e.symbol;if(e.tokenSymbol)return e.tokenSymbol;if(e.name)return e.name}return""},o=e.token0Symbol||n(e.token0)||n(e.tokenA)||e.tokenSymbol0||"",i=e.token1Symbol||n(e.token1)||n(e.tokenB)||e.tokenSymbol1||"",r=o?this.tokenConverter.toLaunchpadFormat(o):"",a=i?this.tokenConverter.toLaunchpadFormat(i):"";return{positionId:e.positionId||e.id||"",ownerAddress:t||e.ownerAddress||e.owner||"",token0:r,token1:a,feeTier:e.feeTier||e.fee||e.feeAmount||0,tickLower:e.tickLower||e.lowerTick||0,tickUpper:e.tickUpper||e.upperTick||0,liquidity:String(e.liquidity||e.liquidityAmount||"0"),amount0:String(e.amount0||e.amountA||"0"),amount1:String(e.amount1||e.amountB||"0"),feeAmount0:String(e.feeAmount0||e.feesA||"0"),feeAmount1:String(e.feeAmount1||e.feesB||"0"),...e.createdAt&&{createdAt:new Date(e.createdAt)},...e.updatedAt&&{updatedAt:new Date(e.updatedAt)}}}parseTokenFlexible(e){try{return ri(e)}catch(t){if(t instanceof Error&&t.message?.includes("Plain token string"))return this.logger.debug("Using default TokenClassKey for simple token symbol",{token:e}),{collection:"Token",category:"Unit",type:e,additionalKey:"none"};throw t}}async getUserLiquidityPositions(t,n=10,o,i){try{this.logger.debug("Fetching user liquidity positions",{ownerAddress:t,limit:n,bookmark:o});const r=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetUserPositions`,a={user:t,limit:n,bookmark:o||""};this.logger.debug("Sending position query request",{endpoint:r,payload:a});const s=await e.post(r,a,{headers:{"Content-Type":"application/json",Accept:"application/json"}});if(200!==s.status||1!==s.data?.Status)return this.logger.warn("Unexpected API response status",{httpStatus:s.status,apiStatus:s.data?.Status}),{items:[]};const c=s.data.Data||{},l=c.positions||[],u=c.nextBookMark,d=l.filter(e=>null!=e&&"object"==typeof e&&("positionId"in e||"id"in e)).map(e=>this.normalizePositionResponse(e,t));let h;this.logger.debug("Retrieved liquidity positions",{count:d.length,hasNextBookmark:!!u,nextBookmark:u}),i?.withPrices&&d.length>0&&(h=await this.fetchPositionPrices(d));const g={items:d};return void 0!==u&&(g.nextBookmark=u),void 0!==h&&(g.prices=h),g}catch(e){if(e&&"object"==typeof e&&"response"in e){const n=e;this.logger.error("Position query failed with HTTP error",{status:n.response?.status,statusText:n.response?.statusText,data:n.response?.data,endpoint:this.galaChainBaseUrl,ownerAddress:t})}this.handleGSwapError("Failed to fetch user liquidity positions",Z,e)}}async getAllSwapUserLiquidityPositions(e,t){try{this.logger.debug("Fetching all user liquidity positions (auto-paginated)",{ownerAddress:e});const n=async t=>{const n=await this.getUserLiquidityPositions(e,hi,t,void 0);return{items:n.items,nextBookmark:n.nextBookmark}},o=await async function(e,t={}){const{maxPages:n=1e4,logger:o,pageSize:i=20}=t,r=[];let a,s=0;for(;s<n;){o&&o.debug(`Auto-pagination (bookmark): fetching page ${s+1} with pageSize ${i}`,{bookmark:a});const t=await e(a,i);let n,c,l;if(Array.isArray(t))n=t,c=void 0,l=!1;else{if(!t||"object"!=typeof t||!("items"in t)){o&&o.warn("Auto-pagination (bookmark): received invalid result structure, stopping");break}n=t.items,c=t.nextBookmark,l=!0}if(!Array.isArray(n)){o&&o.warn("Auto-pagination (bookmark): received invalid items array, stopping");break}if(0===n.length){o&&o.debug(`Auto-pagination (bookmark): no items returned on page ${s+1}, exiting loop`);break}r.push(...n),s++,o&&o.debug(`Auto-pagination (bookmark): page ${s} returned ${n.length} items`,{hasNextBookmark:!!c,format:l?"BookmarkPaginationResult":"legacy-array"});const u=n.length<i;if(l&&(""===c||void 0===c)){o&&o.debug("Auto-pagination (bookmark): no nextBookmark returned, reached end of results",{nextBookmark:""===c?"(empty string)":"(undefined)"});break}if(u){o&&o.debug("Auto-pagination (bookmark): received fewer items than limit, reached last page",{received:n.length,pageSize:i,format:l?"BookmarkPaginationResult":"legacy-array"});break}a=c}return s>=n&&o&&o.warn(`Auto-pagination (bookmark): exceeded maxPages limit of ${n}, stopping iteration`),o&&o.debug(`Auto-pagination (bookmark): completed with total items: ${r.length}`,{pageCount:s}),{items:r,total:r.length}}(n,{maxPages:1e4,logger:this.logger,pageSize:hi}),i=o.items;if(this.logger.debug("Fetched all user liquidity positions",{ownerAddress:e,totalPositions:i.length}),t?.withPrices&&i.length>0){return{items:i,prices:await this.fetchPositionPrices(i)}}return i}catch(t){this.handleGSwapError("Failed to fetch all user liquidity positions",Z,t,{ownerAddress:e})}}async getLiquidityPosition(e,t){try{this.logger.debug("Fetching liquidity position",{ownerAddress:e,position:t}),this.validateTickSpacing(t.tickLower,t.tickUpper,t.fee);const{gswapToken0:n,gswapToken1:o}=this.convertTokenPair(t.token0,t.token1),i=di(n),r=di(o),a=(await this.gatewayClient.getPositions({owner:e,token0:i,token1:r,fee:t.fee,tickLower:t.tickLower,tickUpper:t.tickUpper})).positions.find(e=>e.tickLower===t.tickLower&&e.tickUpper===t.tickUpper);if(!a||"object"!=typeof a||!("positionId"in a)&&!("id"in a))throw new Error("Invalid position data returned from API");const s=this.normalizePositionResponse(a,e);return this.logger.debug("Retrieved liquidity position",{positionId:s.positionId}),s}catch(e){this.handleGSwapError("Failed to fetch liquidity position",Z,e)}}async getLiquidityPositionById(e,t,n,o,i,r,a){try{let s;this.logger.debug("Fetching liquidity position by ID",{ownerAddress:e,positionId:t,hasToken0:!!n,hasToken1:!!o,hasFee:!!i,hasTickLower:void 0!==r,hasTickUpper:void 0!==a});let c=null;const l=5,u=2e3;for(let d=1;d<=l;d++)try{if(n&&o&&void 0!==i&&void 0!==r&&void 0!==a)try{this.logger.debug("Attempting compound key lookup",{ownerAddress:e,token0:n,token1:o,feeTier:i,tickLower:r,tickUpper:a});if(s=(await this.gatewayClient.getPositionById(e,n,o,i,r,a,t)).Data,s&&"object"==typeof s&&("positionId"in s||"id"in s)){this.logger.debug("Successfully fetched position via compound key",{attempt:d,positionId:t});break}throw new Error("Invalid position data from compound key lookup")}catch(e){this.logger.debug("Compound key lookup failed, trying fallback",{attempt:d,error:e instanceof Error?e.message:e})}try{if(s=(await this.gatewayClient.getPositionById(t)).Data,s&&"object"==typeof s&&("positionId"in s||"id"in s)){this.logger.debug("Successfully fetched position on attempt",{attempt:d,positionId:t});break}throw new Error("Invalid position data from direct lookup")}catch(n){this.logger.debug("Direct position lookup failed, trying fallback via GetUserPositions",{attempt:d,positionId:t,error:n instanceof Error?n.message:n});const o=await this.getUserLiquidityPositions(e,10);if(Array.isArray(o.items)&&o.items.length>0){const e=o.items.find(e=>(e.positionId||e.id||"").toLowerCase()===t.toLowerCase());if(e){s=e,this.logger.debug("Found position via fallback (GetUserPositions)",{attempt:d,positionId:t});break}}if(c=new Error("Position not found in owner positions"),d<l){this.logger.warn("Fallback query did not find position, retrying",{attempt:d,positionId:t,ownerAddress:e,foundCount:Array.isArray(o?.items)?o.items.length:0}),await new Promise(e=>setTimeout(e,u));continue}}}catch(e){if(d<l){this.logger.warn("Error fetching position, retrying",{attempt:d,positionId:t,error:e instanceof Error?e.message:e}),await new Promise(e=>setTimeout(e,u));continue}c=e instanceof Error?e:new Error(String(e))}if(!s||"object"!=typeof s||!("positionId"in s)&&!("id"in s))throw this.logger.error("Invalid position data returned from API after retries",{positionId:t,resultType:typeof s,resultKeys:s?Object.keys(s):"null",resultValue:JSON.stringify(s),lastError:c?.message}),c||new Error("Invalid position data returned from API");const d=this.normalizePositionResponse(s,e);return this.logger.debug("Retrieved liquidity position by ID",{positionId:d.positionId}),d}catch(e){this.handleGSwapError("Failed to fetch liquidity position by ID",Z,e)}}async fetchSwapPositionDirect(e){try{this.logger.debug("Fetching swap position via direct compound key",{token0:e.token0,token1:e.token1,fee:e.fee,owner:e.owner});const t="string"==typeof e.token0?this.parseTokenFlexible(e.token0):e.token0,n={token0:t,token1:"string"==typeof e.token1?this.parseTokenFlexible(e.token1):e.token1,fee:e.fee,tickLower:e.tickLower,tickUpper:e.tickUpper,owner:e.owner},o=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetPositions`;this.logger.debug("Calling position endpoint via compound key",{url:o});const i=await fetch(o,{method:"POST",headers:{"Content-Type":"application/json",Accept:"application/json"},body:JSON.stringify(n)});if(!i.ok){if(404===i.status)throw new Error("Position not found (HTTP 404): No position exists for this compound key");if(400===i.status)throw new Error("Invalid parameters (HTTP 400): Check token format, fee (500|3000|10000), and tick ranges");throw new Error(`HTTP ${i.status}: ${i.statusText}`)}const r=await i.json();if(1!==r.Status||!r.Data)throw new Error(`Position not found: ${r.Message||"API returned no position data"}`);const a=this.normalizePositionResponse(r.Data,e.owner);return this.logger.debug("Retrieved swap position via compound key",{positionId:a.positionId,token0:a.token0,token1:a.token1}),a}catch(e){this.handleGSwapError("Failed to fetch swap position via compound key",Z,e)}}async estimateRemoveLiquidity(e){try{this.logger.debug("Estimating liquidity removal",{token0:e.token0,token1:e.token1,owner:e.owner}),this.validateTickSpacing(e.tickLower,e.tickUpper,e.fee);const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.token0,e.token1),o=di(t),i=di(n),r=await this.gatewayClient.getRemoveLiquidityEstimation({token0:o,token1:i,fee:e.fee,amount:e.liquidity,tickLower:e.tickLower,tickUpper:e.tickUpper,owner:e.owner});return this.logger.debug("Estimated removal",{result:r}),r}catch(e){this.handleGSwapError("Failed to estimate liquidity removal",Z,e)}}async addLiquidityByPrice(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");this.logger.debug("Adding liquidity by price",{token0:e.token0,token1:e.token1,priceRange:`${e.minPrice}-${e.maxPrice}`});const{gswapToken0:t,gswapToken1:n}=this.convertTokenPair(e.token0,e.token1);await this.ensureWebSocketConnected(),this.logger.debug("Converting price range to ticks",{token0:e.token0,token1:e.token1,minPrice:e.minPrice,maxPrice:e.maxPrice,fee:e.fee});const o=ri(t),i=ri(n),r=(await this.gatewayClient.getPoolData({token0:o,token1:i,fee:e.fee})).tickSpacing;this.logger.debug("Retrieved tick spacing from pool",{tickSpacing:r,fee:e.fee});const a=new u(e.minPrice),s=new u(e.maxPrice),c=Math.floor(so(a)),l=Math.ceil(so(s)),d=Math.floor(c/r)*r,h=Math.ceil(l/r)*r;this.logger.debug("Converted price range to ticks",{minPrice:e.minPrice,maxPrice:e.maxPrice,tickLower:d,tickUpper:h,tickSpacing:r});const g=this.getWalletAddress();if(!g)throw new Error("GSwapService: No wallet address available - cannot create position");const p="string"==typeof e.token0?ri(e.token0):e.token0,m="string"==typeof e.token1?ri(e.token1):e.token1;this.logger.debug("Sending AddLiquidity by price to bundler",{fee:e.fee,tickRange:`${d}-${h}`,walletAddress:g});const f=await this.sendAddLiquidityToBundler({token0:p,token1:m,fee:e.fee,tickLower:d,tickUpper:h,amount0Desired:e.amount0Desired,amount1Desired:e.amount1Desired,amount0Min:e.amount0Min||"0",amount1Min:e.amount1Min||"0",owner:g}),y={transactionId:f};if(y.positionId&&f){this.logger.debug("Position ID returned directly from backend",{transactionId:f,positionId:y.positionId}),await this.ensureWebSocketConnected();const e=await this.webSocketService.waitForTransaction(f);this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:f,status:e.status});const t=this.getWalletAddress();if(t&&y.positionId)try{const n=await this.getLiquidityPositionById(t,y.positionId),{createdAt:o,updatedAt:i,...r}=n,a=o instanceof Date?o.getTime():"number"==typeof o?o:void 0,s={...y,...r,positionId:y.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}};return void 0!==a&&(s.createdAt=a),s}catch(t){return{...y,positionId:y.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}}}}if(f){this.logger.debug("Monitoring liquidity transaction (discovery mode)",{transactionId:f}),await this.ensureWebSocketConnected();const t=await this.webSocketService.waitForTransaction(f);let n;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:f,status:t.status});let o=null;await new Promise(e=>setTimeout(e,2e3));try{const t=this.getWalletAddress();if(!t)throw new Error("No wallet address available");const i=(await this.getUserLiquidityPositions(t,10)).items;if(i&&i.length>0){const t=e.token0.split("|")[0].toUpperCase(),r=e.token1.split("|")[0].toUpperCase(),a=[];for(const n of i){if(!n||!n.positionId)continue;const o=n.token0?.toUpperCase(),i=n.token1?.toUpperCase();if(!o||!i)continue;const s=o===t&&i===r||o===r&&i===t,c=n.feeTier===e.fee;s&&c&&a.push(n)}a.length>0&&(o=a[a.length-1],n=o.positionId,this.logger.debug("Found newly created position",{positionId:n,expectedTokens:`${e.token0}/${e.token1}`,expectedFee:e.fee,positionCount:i.length}))}}catch(e){this.logger.debug("Error waiting for position indexing",{error:e instanceof Error?e.message:String(e)})}let i=o;if(n)try{i=await this.getLiquidityPositionById(g,n)}catch(e){}return{...y,...i,positionId:n,status:t.status,transactionId:t.transactionId,timestamp:new Date(t.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(f)}}}return this.logger.warn("No transaction ID in liquidity result, cannot confirm position creation"),y}catch(e){Error,this.handleGSwapError("Failed to add liquidity by price",Z,e)}}async addSwapLiquidityByTicks(e){try{if(!this.privateKey)throw new Error("GSwapService not initialized with signing capability (privateKey required)");const t=this.getWalletAddress();if(!t)throw new Error("GSwapService: No wallet address available - cannot create position");this.logger.debug("Adding liquidity by ticks with direct bundler",{token0:e.token0,token1:e.token1,fee:e.fee,walletAddress:t,tickRange:`${e.tickLower}-${e.tickUpper}`});const n="string"==typeof e.token0?ri(e.token0):e.token0,o="string"==typeof e.token1?ri(e.token1):e.token1;await this.ensureWebSocketConnected();const i=await this.sendAddLiquidityToBundler({token0:n,token1:o,fee:e.fee,tickLower:e.tickLower,tickUpper:e.tickUpper,amount0Desired:e.amount0Desired,amount1Desired:e.amount1Desired,amount0Min:e.amount0Min||"0",amount1Min:e.amount1Min||"0",owner:t});this.logger.info("Liquidity transaction submitted to bundler",{transactionId:i});const r=this.webSocketService.waitForTransaction(i),a={transactionId:i};if(a.positionId&&i){this.logger.info("Position ID returned directly from backend",{transactionId:i,positionId:a.positionId});const e=await r;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:i,status:e.status});const t=this.getWalletAddress();if(t&&a.positionId)try{this.logger.debug("Fetching full position details",{positionId:a.positionId});const n=await this.getLiquidityPositionById(t,a.positionId);this.logger.debug("Fetched full position data",{positionId:n.positionId,liquidity:n.liquidity,amount0:n.amount0,amount1:n.amount1});const{createdAt:o,updatedAt:r,...s}=n,c=o instanceof Date?o.getTime():"number"==typeof o?o:void 0,l={...a,...s,positionId:a.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}};return void 0!==c&&(l.createdAt=c),l}catch(t){return this.logger.warn("Could not fetch full position details",{positionId:a.positionId,error:t instanceof Error?t.message:String(t)}),{...a,positionId:a.positionId,status:e.status,transactionId:e.transactionId,timestamp:new Date(e.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}}}}if(i){this.logger.debug("Monitoring liquidity transaction (discovery mode)",{transactionId:i});const n=await r;let o;this.logger.debug("Liquidity transaction confirmed on-chain",{transactionId:i,status:n.status});let s=null;const c="string"==typeof e.token0?e.token0:e.token0?.type??"unknown",l="string"==typeof e.token1?e.token1:e.token1?.type??"unknown";this.logger.debug("Waiting for position indexing after WebSocket confirmation"),this.logger.debug("Looking for matching position",{token0:c,token1:l,fee:e.fee});try{const t=this.getWalletAddress();if(!t)throw new Error("No wallet address available");this.logger.debug("Fetching positions from API",{walletAddress:t,pageSize:hi});const n=3,i=5e3,r=3e3;let a=[];for(let c=1;c<=n;c++){const l=1===c?i:r;this.logger.debug("Position discovery attempt",{attempt:c,maxAttempts:n,delayMs:l}),await new Promise(e=>setTimeout(e,l)),this.logger.debug("Querying positions from API",{attempt:c,pageSize:hi});if(a=(await this.getUserLiquidityPositions(t,hi)).items,this.logger.debug("Got positions from API",{count:a?.length||0}),a&&a.length>0){const t=("string"==typeof e.token0?e.token0.split("|")[0]:e.token0.collection).toUpperCase(),n=("string"==typeof e.token1?e.token1.split("|")[0]:e.token1.collection).toUpperCase(),i=[];for(const o of a){if(!o||!o.positionId)continue;const r=o.token0?.toUpperCase(),a=o.token1?.toUpperCase();if(!r||!a){this.logger.debug("Skipping position with empty tokens",{positionId:o.positionId});continue}const s=r===t&&a===n||r===n&&a===t,c=o.feeTier===e.fee;this.logger.debug("Checking position against target",{positionId:o.positionId,tokens:`${r}/${a}`,tokensMatch:s,fee:o.feeTier,feeMatches:c}),s&&c&&i.push(o)}if(i.length>0){s=i[i.length-1],o=s.positionId,this.logger.info("Found newly created position",{positionId:o,liquidity:s.liquidity,amount0:s.amount0,amount1:s.amount1,fee:s.feeTier}),this.logger.debug("Found newly created position",{positionId:o,expectedTokens:`${e.token0}/${e.token1}`,expectedFee:e.fee,positionCount:a.length});break}this.logger.debug("No matching position found in this attempt")}else this.logger.debug("No positions returned from API in this attempt")}}catch(e){this.logger.error("Error fetching positions during discovery",{error:e instanceof Error?e.message:String(e)}),this.logger.debug("Error waiting for position indexing",{error:e instanceof Error?e.message:String(e)})}this.logger.debug("Position discovery complete",{positionId:o||"not found",found:!!o}),this.logger.debug("Matched position data",{positionId:s?.positionId,liquidity:s?.liquidity,amount0:s?.amount0,amount1:s?.amount1,feeAmount0:s?.feeAmount0,feeAmount1:s?.feeAmount1,token0:s?.token0,token1:s?.token1,feeTier:s?.feeTier});let u=s;if(o)try{this.logger.debug("Fetching full position details",{positionId:o}),u=await this.getLiquidityPositionById(t,o),this.logger.debug("Fetched full position data",{positionId:u.positionId,liquidity:u.liquidity,amount0:u.amount0,amount1:u.amount1,feeAmount0:u.feeAmount0,feeAmount1:u.feeAmount1})}catch(e){this.logger.warn("Could not fetch full position details, using discovered data",{positionId:o,error:e instanceof Error?e.message:String(e)})}return{...a,...u,positionId:o,status:n.status,transactionId:n.transactionId,timestamp:new Date(n.timestamp),wait:async e=>{await this.webSocketService.waitForTransaction(i)}}}return this.logger.warn("No transaction ID in liquidity result, cannot confirm position creation"),a}catch(e){this.handleGSwapError("Failed to add liquidity by ticks",Z,e)}}async monitorBundlerTransaction(e,t,n="bundler"){let o;try{const i=await t;o={status:i.status,transactionId:i.transactionId||e,timestamp:i.timestamp||Date.now(),data:i.data},this.logger.debug(`${n} transaction confirmed on-chain`,{transactionId:e,status:o.status})}catch(t){return this.logger.warn(`WebSocket monitoring timeout for ${n} transaction, returning result with transaction ID`,{transactionId:e,error:t instanceof Error?t.message:String(t)}),{transactionId:e,status:"SUBMITTED",timestamp:new Date,wait:async t=>{try{await this.webSocketService.waitForTransaction(e)}catch{this.logger.debug("Explicit wait also timed out",{transactionId:e})}}}}return{transactionId:o.transactionId,status:o.status,timestamp:new Date(o.timestamp),wait:async t=>{await this.webSocketService.waitForTransaction(e)}}}async removeLiquidity(e){try{if(!this.privateKey)throw new Error("Private key not available for bundler-direct operations");this.logger.debug("Removing liquidity via bundler",{token0:e.token0,token1:e.token1,liquidity:e.liquidity});try{const t=parseFloat(e.liquidity);if(isNaN(t))throw new Error(`Invalid liquidity value: "${e.liquidity}". Must be a valid number. Position ID: ${e.positionId||"unknown"}`);if(0===t)throw new Error(`Cannot remove zero liquidity from position. This would waste gas fees without any effect. Position ID: ${e.positionId||"unknown"}`)}catch(e){if(e instanceof Error&&e.message.includes("Cannot remove zero liquidity"))throw e;if(e instanceof Error&&e.message.includes("Invalid liquidity value"))throw e;throw e}const t="string"==typeof e.token0?ri(e.token0):e.token0,n="string"==typeof e.token1?ri(e.token1):e.token1;await this.ensureWebSocketConnected();const o=await this.sendRemoveLiquidityToBundler(e.tickLower,e.tickUpper,e.liquidity,t,n,e.fee,e.amount0Min||"0",e.amount1Min||"0",e.positionId||"");this.logger.debug("Liquidity removal submitted to bundler",{transactionId:o});const i=this.webSocketService.waitForTransaction(o);return this.monitorBundlerTransaction(o,i,"liquidity removal")}catch(e){this.handleGSwapError("Failed to remove liquidity",Z,e)}}async collectPositionFees(e){try{if(!this.privateKey)throw new Error("Private key not available for bundler-direct operations");if(e.ownerAddress&&e.positionId&&!e.token0){this.logger.debug("Fetching position data before collecting fees",{ownerAddress:e.ownerAddress,positionId:e.positionId});const t=await this.getLiquidityPositionById(e.ownerAddress,e.positionId);if(!t)throw new Error(`Position ${e.positionId} not found for owner ${e.ownerAddress}`);if(!t.token0||!t.token1)throw new Error("Position missing token information");const{gswapToken0:n,gswapToken1:o}=this.convertTokenPair(t.token0,t.token1);return this.collectPositionFees({token0:n,token1:o,fee:t.feeTier,tickLower:t.tickLower,tickUpper:t.tickUpper,amount0Requested:e.amount0Max||e.amount0Requested||"0",amount1Requested:e.amount1Max||e.amount1Requested||"0",positionId:e.positionId})}if(!e.token0||!e.token1||void 0===e.fee||void 0===e.tickLower||void 0===e.tickUpper)throw new Error("Missing required parameters: token0, token1, fee, tickLower, tickUpper are required when not using ownerAddress pattern");this.logger.debug("Collecting position fees via bundler",{token0:"string"==typeof e.token0?e.token0:e.token0?.type??"unknown",token1:"string"==typeof e.token1?e.token1:e.token1?.type??"unknown",tickLower:e.tickLower,tickUpper:e.tickUpper});const t="string"==typeof e.token0?ri(e.token0):e.token0,n="string"==typeof e.token1?ri(e.token1):e.token1;await this.ensureWebSocketConnected();const o=await this.sendCollectPositionFeesToBundler(t,n,e.fee,e.amount0Requested||"0",e.amount1Requested||"0",e.tickLower,e.tickUpper,e.positionId||"");this.logger.debug("Fee collection submitted to bundler",{transactionId:o});const i=this.webSocketService.waitForTransaction(o);return this.monitorBundlerTransaction(o,i,"fee collection")}catch(e){this.handleGSwapError("Failed to collect position fees",Z,e)}}async getPoolData(e,t,n){try{this.logger.debug("Getting pool data",{tokenA:e,tokenB:t,feeTier:n});const{gswapToken0:o,gswapToken1:i}=this.convertTokenPair(e,t),r=ri(o),a=ri(i),s=await this.gatewayClient.getPoolData({token0:r,token1:a,fee:n}),c=this.calculatePriceFromSqrtPriceX96(new u(s.sqrtPrice));return{tokenA:e,tokenB:t,feeTier:n,liquidity:s.liquidity.toString(),sqrtPriceX96:s.sqrtPrice.toString(),tick:s.tick,feeGrowthGlobal0X128:s.feeGrowthGlobal0.toString(),feeGrowthGlobal1X128:s.feeGrowthGlobal1.toString(),currentPrice:c.toFixed()}}catch(e){this.handleGSwapError("Failed to get pool data",Q,e)}}async calculateDexPoolSpotPrice(e,t,n){try{this.logger.debug("Calculating spot price",{tokenA:e,tokenB:t,feeTier:n});const o=await this.getPoolData(e,t,n),i=ro(o.currentPrice);return{tokenA:e,tokenB:t,feeTier:n,price:i.toFixed(),invertedPrice:co(i,!0),tick:o.tick,liquidity:o.liquidity}}catch(e){this.handleGSwapError("Failed to calculate spot price",Q,e)}}async calculateOptimalPositionSize(e,t,n,o,i,r,a){try{this.logger.debug("Calculating optimal position size",{tokenA:e,tokenB:t,desiredAmount0:o,desiredAmount1:i});const s=(await this.getPoolData(e,t,n)).tick,c=d(r),l=d(s),p=d(a),m=h(new u(o),new u(i),c,l,p),f=g(m,l,c,p),y=f[0],w=f[1],k=new u(o),v=new u(i);return{amount0:y.toFixed(),amount1:w.toFixed(),liquidity:m.toFixed(),ratio:y.dividedBy(w).toFixed(),utilizationPercent:{amount0:y.dividedBy(k).multipliedBy(100).toFixed(2),amount1:w.dividedBy(v).multipliedBy(100).toFixed(2)}}}catch(e){this.handleGSwapError("Failed to calculate optimal position size",Z,e)}}async validatePositionParameters(e,t,n,o,i,r,a){const s=[],c=[];try{this.logger.debug("Validating position parameters",{tokenA:e,tokenB:t,tickLower:o,tickUpper:i});const l=[500,3e3,1e4];l.includes(n)||s.push(`Invalid fee tier: ${n}. Must be one of: ${l.join(", ")}`);const d=this.getTickSpacing(n);let h;o%d!==0&&s.push(`tickLower must be multiple of ${d}`),i%d!==0&&s.push(`tickUpper must be multiple of ${d}`),o>=i&&s.push(`tickLower (${o}) must be less than tickUpper (${i})`);try{h=await this.getPoolData(e,t,n)}catch{return s.push(`Pool not found for ${e}/${t} at fee tier ${n}`),{valid:!1,errors:s,warnings:c,gasEstimate:0}}const g=new u(r),p=new u(a);if(g.isNaN()||p.isNaN())s.push("Amounts must be valid numbers");else try{uo(g,p)}catch(e){s.push(`Liquidity amounts must be non-negative: ${e.message}`)}const m=h.tick;(m<o||m>i)&&c.push("Position is out of current price range - will not earn fees until price moves into range");new u(h.liquidity||"0").lt("1000000")&&c.push("Low pool liquidity - consider higher slippage tolerance");const f=0===s.length?35e4:0;return{valid:0===s.length,errors:s,warnings:c,gasEstimate:f,tickSpacing:d,currentTick:m,poolLiquidity:h.liquidity}}catch(e){const t=e;return s.includes(t?.message||"")||s.push(`Validation failed: ${t?.message||String(e)}`),{valid:!1,errors:s,warnings:c,gasEstimate:0}}}async calculateTicksForPrice(e,t,n,o,i){try{this.logger.debug("Calculating ticks for price range",{tokenA:e,tokenB:t,minPrice:n,maxPrice:o});const r=this.getTickSpacing(i),a=new u(n),s=new u(o);if(a.gte(s))throw new Error("minPrice must be less than maxPrice");const c=Math.floor(so(a)),l=Math.ceil(so(s)),d=Math.floor(c/r)*r,h=Math.ceil(l/r)*r,g=Math.pow(1.0001,d),p=Math.pow(1.0001,h),m=new u(g),f=new u(p);return{tokenA:e,tokenB:t,feeTier:i,tickLower:d,tickUpper:h,tickSpacing:r,requestedMinPrice:n,requestedMaxPrice:o,actualMinPrice:m.toFixed(8),actualMaxPrice:f.toFixed(8),priceDeviation:{minPriceDeviation:m.minus(a).dividedBy(a).multipliedBy(100).toFixed(4),maxPriceDeviation:f.minus(s).dividedBy(s).multipliedBy(100).toFixed(4)}}}catch(e){this.handleGSwapError("Failed to calculate ticks for price",Z,e)}}async calculatePriceForTicks(e,t,n,o){try{this.logger.debug("Calculating price for ticks",{tokenA:e,tokenB:t,tickLower:n,tickUpper:o});const i=Math.pow(1.0001,n),r=Math.pow(1.0001,o);let a;try{a=(await this.getPoolData(e,t,3e3)).currentPrice}catch{}const s=new u(i),c=new u(r),l={tokenA:e,tokenB:t,tickLower:n,tickUpper:o,minPrice:s.toFixed(8),maxPrice:c.toFixed(8),priceRange:`${s.toFixed(4)} - ${c.toFixed(4)}`,tickSpread:o-n};return void 0!==a&&(l.currentPrice=a),l}catch(e){this.handleGSwapError("Failed to calculate price for ticks",Z,e)}}calculateExecutionPrice(e,t){try{const n=new u(e),o=new u(t);return n.isZero()?"0":o.dividedBy(n).toFixed()}catch{return"0"}}getTickSpacing(e){switch(e){case 500:return 10;case 3e3:return 60;case 1e4:return 200;default:throw new Error(`Invalid fee tier: ${e}`)}}validateTickSpacing(e,t,n){const o=this.getTickSpacing(n);if(e%o!==0)throw new Error(`Invalid tickLower: ${e} must be a multiple of ${o} for fee tier ${n}. Tip: Use getAllSwapUserLiquidityPositions() to discover valid positions with correct tick spacing.`);if(t%o!==0)throw new Error(`Invalid tickUpper: ${t} must be a multiple of ${o} for fee tier ${n}. Tip: Use getAllSwapUserLiquidityPositions() to discover valid positions with correct tick spacing.`)}calculatePriceFromSqrtPriceX96(e){try{const t=new u(2).pow(96);return e.dividedBy(t).pow(2)}catch{return new u(0)}}calculatePriceFromSqrtPriceDecimal(e){try{return e.pow(2)}catch{return new u(0)}}async getPoolSlot0(t,n,o){try{this.logger.debug("Fetching pool slot0 data",{token0:t,token1:n,fee:o});const i="string"==typeof t?ri(t):t,r="string"==typeof n?ri(n):n,a=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/GetSlot0`,s=await e.post(a,{token0:i,token1:r,fee:o},{headers:{"Content-Type":"application/json",Accept:"application/json"}});if(200!==s.status||1!==s.data?.Status)throw this.logger.warn("Unexpected GetSlot0 API response",{httpStatus:s.status,apiStatus:s.data?.Status}),new Q("GetSlot0 API returned unexpected status",{httpStatus:s.status,apiStatus:s.data?.Status});const c=s.data.Data||{},l={sqrtPrice:c.sqrtPrice||"0",tick:c.tick||0,liquidity:c.liquidity||"0",grossPoolLiquidity:c.grossPoolLiquidity||"0"};return this.logger.debug("Retrieved pool slot0 data",{sqrtPrice:l.sqrtPrice,tick:l.tick,liquidity:l.liquidity}),l}catch(e){this.handleGSwapError("Failed to fetch pool slot0 data",Q,e,{token0:t,token1:n,fee:o})}}async getPositionCurrentPrice(e){try{this.logger.debug("Fetching position current price",{token0:e.token0,token1:e.token1,feeTier:e.feeTier});const t=await this.getPoolSlot0(e.token0,e.token1,e.feeTier),n=new u(t.sqrtPrice),o={price:this.calculatePriceFromSqrtPriceDecimal(n).toFixed(18),sqrtPrice:t.sqrtPrice,tick:t.tick,liquidity:t.liquidity};return this.logger.debug("Calculated position current price",{price:o.price,tick:o.tick}),o}catch(t){this.handleGSwapError("Failed to fetch position current price",Q,t,{token0:e.token0,token1:e.token1})}}calculateLiquidityFromAmount0(e,t,n){try{const o=ao(t),i=ao(n);return p(e,o,i)}catch{return new u(0)}}calculateLiquidityFromAmount1(e,t,n){try{const o=ao(t),i=ao(n);return m(e,o,i)}catch{return new u(0)}}calculateAmount0FromLiquidity(e,t,n){try{const o=ao(t),i=ao(n);return f(o,i,e)}catch{return new u(0)}}calculateAmount1FromLiquidity(e,t,n){try{const o=ao(t),i=ao(n);return y(o,i,e)}catch{return new u(0)}}convertTokenPair(e,t){return{gswapToken0:this.tokenConverter.toLaunchpadFormat(e),gswapToken1:this.tokenConverter.toLaunchpadFormat(t)}}async sendAddLiquidityToBundler(n){if(!this.privateKey)throw new Error("GSwapService: AddLiquidity requires wallet (full-access mode)");if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");try{this.logger.debug("Sending AddLiquidity to bundler",{token0:n.token0?.type??"unknown",token1:n.token1?.type??"unknown",fee:n.fee,tickRange:`${n.tickLower}-${n.tickUpper}`});const o=`galaswap - operation - ${N()}-${Date.now()}-${n.owner}`,i={token0:n.token0,token1:n.token1,fee:n.fee,owner:n.owner,tickLower:n.tickLower,tickUpper:n.tickUpper,amount0Desired:n.amount0Desired,amount1Desired:n.amount1Desired,amount0Min:n.amount0Min,amount1Min:n.amount1Min,positionId:"",uniqueKey:o},r=new t.Wallet(this.privateKey),a={AddLiquidity:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"owner",type:"string"},{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"amount0Desired",type:"string"},{name:"amount1Desired",type:"string"},{name:"amount0Min",type:"string"},{name:"amount1Min",type:"string"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},s={name:"ethereum",chainId:1},c=this.calculatePersonalSignPrefix(i),l={...i,prefix:c},u=await r.signTypedData(s,a,l),d={...l,signature:u,types:a,domain:s};this.logger.debug("AddLiquidity DTO signed with manual types",{signature:d.signature?.substring(0,20)+"...",prefix:d.prefix,tickLower:i.tickLower,tickUpper:i.tickUpper});const h=this.buildLiquidityStringsInstructions(n.token0,n.token1,n.fee,n.owner),g=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),p=await g.post("/bundle",{method:"AddLiquidity",signedDto:d,stringsInstructions:h}),m=p.data?.data||p.data?.transactionId||p.data?.id;if(!m)throw this.logger.error("Bundler response structure",{status:p.status,data:p.data,dataType:typeof p.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(p.data)}`);return this.logger.debug("AddLiquidity transaction sent to bundler",{transactionId:m}),m}catch(e){throw this.logger.error("Failed to send AddLiquidity to bundler",e),e}}async sendRemoveLiquidityToBundler(n,o,i,r,a,s,c,l,u){try{if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const d=new t.Wallet(this.privateKey),h=await d.getAddress(),g=`galaswap - operation - ${N()}-${Date.now()}-${h}`,p={tickLower:n,tickUpper:o,amount:i,token0:r,token1:a,fee:s,amount0Min:c,amount1Min:l,positionId:u,uniqueKey:g},m={RemoveLiquidity:[{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"amount",type:"string"},{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount0Min",type:"string"},{name:"amount1Min",type:"string"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},f={name:"ethereum",chainId:1},y=this.calculatePersonalSignPrefix(p),w={...p,prefix:y},k=await d.signTypedData(f,m,w),v={...w,signature:k,types:m,domain:f},b=this.buildLiquidityStringsInstructions(r,a,s,h);this.logger.debug("Submitting RemoveLiquidity to bundler",{tickLower:n,tickUpper:o,amount:i,fee:s,positionId:u,transactionId:g});const T=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),S=await T.post("/bundle",{method:"RemoveLiquidity",signedDto:v,stringsInstructions:b}),A=S.data?.data||S.data?.transactionId||S.data?.id;if(!A)throw this.logger.error("Bundler response structure",{status:S.status,data:S.data,dataType:typeof S.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(S.data)}`);return this.logger.info("RemoveLiquidity transaction sent to bundler",{transactionId:A}),A}catch(e){throw this.logger.error("Failed to send RemoveLiquidity to bundler",e),e}}async sendCollectPositionFeesToBundler(n,o,i,r,a,s,c,l){try{if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const u=new t.Wallet(this.privateKey),d=await u.getAddress(),h=`galaswap - operation - ${N()}-${Date.now()}-${d}`,g={token0:n,token1:o,fee:i,amount0Requested:r,amount1Requested:a,tickLower:s,tickUpper:c,positionId:l,uniqueKey:h},p={CollectPositionFees:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount0Requested",type:"string"},{name:"amount1Requested",type:"string"},{name:"tickLower",type:"int256"},{name:"tickUpper",type:"int256"},{name:"positionId",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},m={name:"ethereum",chainId:1},f=this.calculatePersonalSignPrefix(g),y={...g,prefix:f},w=await u.signTypedData(m,p,y),k={...y,signature:w,types:p,domain:m},v=this.buildLiquidityStringsInstructions(n,o,i,d);this.logger.debug("Submitting CollectPositionFees to bundler",{fee:i,amount0Requested:r,amount1Requested:a,tickLower:s,tickUpper:c,positionId:l,transactionId:h});const b=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),T=await b.post("/bundle",{method:"CollectPositionFees",signedDto:k,stringsInstructions:v}),S=T.data?.data||T.data?.transactionId||T.data?.id;if(!S)throw this.logger.error("Bundler response structure",{status:T.status,data:T.data,dataType:typeof T.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(T.data)}`);return this.logger.info("CollectPositionFees transaction sent to bundler",{transactionId:S}),S}catch(e){throw this.logger.error("Failed to send CollectPositionFees to bundler",e),e}}async sendSwapToBundler(n){if(!this.privateKey)throw new Error("GSwapService: Swap requires wallet (full-access mode)");if(!this.bundlerBaseUrl)throw new Error("GSwapService: Bundler URL not configured");const o=[500,3e3,1e4];if(!o.includes(n.feeTier))throw new Error(`GSwapService: Invalid fee tier ${n.feeTier}. Must be one of: ${o.join(", ")} (basis points)`);try{this.logger.debug("Sending Swap to bundler",{fromToken:"string"==typeof n.fromToken?n.fromToken:n.fromToken?.type??"unknown",toToken:"string"==typeof n.toToken?n.toToken:n.toToken?.type??"unknown",inputAmount:n.inputAmount,minOutput:n.minOutput,feeTier:n.feeTier});let o=n.fromToken,i=n.toToken;"string"==typeof o&&(o=ri(o)),"string"==typeof i&&(i=ri(i));const r=`${o.collection}|${o.category}|${o.type}|${o.additionalKey}`,a=`${i.collection}|${i.category}|${i.type}|${i.additionalKey}`,s=r<a?[o,i,r,a]:[i,o,a,r],[c,l,d,h]=s,g="string"==typeof n.fromToken?ri(n.fromToken):n.fromToken,p=`${g.collection}|${g.category}|${g.type}|${g.additionalKey}`,m=p===d,f=`galaswap - operation - ${N()}-${Date.now()}-${n.walletAddress}`;let y;if(!n.currentSqrtPrice)throw new Error("GSwapService: currentSqrtPrice is required for sqrtPriceLimit calculation");const w=new u(n.currentSqrtPrice),k=n.slippageTolerance??.01;if(m){const e=new u(1).minus(k);y=w.multipliedBy(e).toString()}else{const e=new u(1).plus(k);y=w.multipliedBy(e).toString()}this.logger.debug("Calculated sqrtPriceLimit based on slippage tolerance",{currentSqrtPrice:n.currentSqrtPrice,slippageTolerance:100*k+"%",zeroForOne:m,sqrtPriceLimit:y,direction:m?"token0→token1 (downward price movement)":"token1→token0 (upward price movement)",reason:"sqrtPriceLimit sets price boundaries, amountOutMinimum provides volume protection"});const v={token0:c,token1:l,fee:n.feeTier,amount:new u(n.inputAmount).toFixed(),zeroForOne:m,sqrtPriceLimit:y,recipient:n.walletAddress,amountOutMinimum:new u(n.minOutput).multipliedBy(-1).toFixed(),uniqueKey:f};this.logger.info("🔄 SWAP DTO DETAILS (what we're sending to bundler)",{orderedToken0String:d,orderedToken1String:h,fromTokenStr:p,zeroForOne:m?`TRUE (${d} → ${h})`:`FALSE (${h} → ${d})`,inputAmount:n.inputAmount,expectedOutput:n.minOutput,slippageTolerance:100*(n.slippageTolerance||.01)+"%",currentSqrtPrice:n.currentSqrtPrice,swapDto:{amount:v.amount,zeroForOne:v.zeroForOne,sqrtPriceLimit:v.sqrtPriceLimit,amountOutMinimum:v.amountOutMinimum}});const b=new t.Wallet(this.privateKey),T={Swap:[{name:"token0",type:"token0"},{name:"token1",type:"token1"},{name:"fee",type:"int256"},{name:"amount",type:"string"},{name:"zeroForOne",type:"bool"},{name:"sqrtPriceLimit",type:"string"},{name:"recipient",type:"string"},{name:"amountOutMinimum",type:"string"},{name:"uniqueKey",type:"string"}],token0:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}],token1:[{name:"additionalKey",type:"string"},{name:"category",type:"string"},{name:"collection",type:"string"},{name:"type",type:"string"}]},S={name:"ethereum",chainId:1},A=this.calculatePersonalSignPrefix(v),E={...v,prefix:A},I=await b.signTypedData(S,T,E),P={...E,signature:I,types:T,domain:S};this.logger.debug("Swap DTO signed",{signature:P.signature?.substring(0,20)+"...",prefix:P.prefix,zeroForOne:v.zeroForOne});const D=this.buildLiquidityStringsInstructions(c,l,n.feeTier,n.walletAddress),F=e.create({baseURL:this.bundlerBaseUrl,timeout:3e4}),x=await F.post("/bundle",{method:"Swap",signedDto:P,stringsInstructions:D}),C=x.data?.data||x.data?.transactionId||x.data?.id;if(!C)throw this.logger.error("Bundler response structure",{status:x.status,data:x.data,dataType:typeof x.data}),new Error(`Bundler response does not contain transaction ID. Response: ${JSON.stringify(x.data)}`);return this.logger.debug("Swap transaction sent to bundler",{transactionId:C,inputAmount:n.inputAmount,minOutput:n.minOutput}),C}catch(e){throw this.logger.error("Failed to send Swap to bundler",e),e}}buildLiquidityStringsInstructions(e,t,n,o){const i=`$${e.collection}$${e.category}$${e.type}$${e.additionalKey}`,r=`$${t.collection}$${t.category}$${t.type}$${t.additionalKey}`,a=`$pool${i}${r}$${n}`;return[a,`$userPosition${o}`,`$tokenBalance${i}${o}`,`$tokenBalance${r}${o}`,`$tokenBalance${i}${a}`,`$tokenBalance${r}${a}`]}handleGSwapError(e,t,n,o){this.logger.error(e,n);const i=this.extractGSwapErrorCode(n),r=n,a=[`${e}: ${r?.message||String(n)}`,n];throw o&&("GSwapSwapError"===t.name&&o.transactionHash&&a.push(o.transactionHash),"GSwapPoolError"===t.name&&(o.tokenA&&a.push(o.tokenA),o.tokenB&&a.push(o.tokenB)),"GSwapAssetError"===t.name&&o.walletAddress&&a.push(o.walletAddress)),i&&a.push(i),new t(...a)}extractGSwapErrorCode(e){if(e&&"object"==typeof e){const t=e;if(t.constructor&&"GSwapSDKError"===t.constructor.name)return t.code;if("code"in t&&"string"==typeof t.code)return t.code}}async ensureWebSocketConnected(){this.webSocketService.isConnected()||await this.webSocketService.connect()}calculatePersonalSignPrefix(e){return`Ethereum Signed Message:\n${JSON.stringify(e).length}${JSON.stringify(e)}`}}class pi{}pi.BASE_PRICE=1650667151e-14,pi.PRICE_SCALING_FACTOR=1166069e-12,pi.TRADING_FEE_FACTOR=.001,pi.GAS_FEE="1",pi.MIN_UNBONDING_FEE_FACTOR=0,pi.MAX_UNBONDING_FEE_FACTOR=.5,pi.NET_UNBONDING_FEE_FACTOR=.5,pi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY=1e7;class mi extends Cn{constructor(e=!1){super(e),this.cache=new Map}getLRUKey(){const e=this.cache.keys().next().value;return void 0!==e?e:null}normalizeTokenName(e){return e.trim().toLowerCase().replace(/\s+/g," ").replace(/[\u0000-\u001F\u007F-\u009F\u200B-\u200D\uFEFF]/g,"")}updateCacheEntry(e,t){const n=this.cache.get(e);if(this.cache.has(e)&&this.cache.delete(e),this.cache.size>=mi.MAX_CACHE_SIZE){const e=this.getLRUKey();null!==e&&this.cache.delete(e)}this.cache.set(e,{...n||{},...t,lastUpdated:Date.now()})}warmFromPoolData(e,t){const n=this.normalizeTokenName(e);this.updateCacheEntry(n,t)}set(e,t){const n=this.normalizeTokenName(e);this.updateCacheEntry(n,t)}get(e){const t=this.normalizeTokenName(e);return this.cache.get(t)||null}getMaxSupply(e){const t=this.normalizeTokenName(e),n=this.cache.get(t);return n?.maxSupply||pi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY.toString()}has(e){const t=this.normalizeTokenName(e);return this.cache.has(t)}clear(e){if(e){const t=this.normalizeTokenName(e);this.cache.delete(t)}else this.cache.clear()}dump(){const e={};return this.cache.forEach((t,n)=>{e[n]=t}),e}stats(){let e=Date.now(),t=0;return this.cache.forEach((n,o)=>{n.lastUpdated<e&&(e=n.lastUpdated);let i=0;i+=2*o.length,void 0!==n.reverseBondingCurveMinFeeFactor&&(i+=8),void 0!==n.reverseBondingCurveMaxFeeFactor&&(i+=8),void 0!==n.reverseBondingCurveNetFeeFactor&&(i+=8),i+=8,n.vaultAddress&&(i+=2*n.vaultAddress.length),n.maxSupply&&(i+=2*n.maxSupply.length),n.symbol&&(i+=2*n.symbol.length),i+=32,t+=i}),{totalTokens:this.cache.size,cacheSize:t,oldestEntry:this.cache.size>0?e:0}}getByTokenId(e){const t=`token:${e.toLowerCase().trim()}`;return this.cache.get(t)||null}setByTokenId(e,t){const n=`token:${e.toLowerCase().trim()}`;this.updateCacheEntry(n,t)}hasByTokenId(e){const t=`token:${e.toLowerCase().trim()}`;return this.cache.has(t)}}mi.MAX_CACHE_SIZE=1e4;class fi extends xn{constructor(e,t,n=void 0,o=5,i=!1){super(e,i),this.pricingConcurrency=5,this.dexBackendBaseUrl=t,this.gswapService=n,this.pricingConcurrency=o}setGSwapService(e){this.gswapService=e}setPricingConcurrency(e){this.pricingConcurrency=Math.max(1,Math.min(e,20))}async enrichPoolsWithPricing(e){if(!this.gswapService)return this.logger.warn("GSwap service not available, skipping pricing enrichment"),e;if(0===e.length)return e;this.logger.debug("Starting pricing enrichment",{poolCount:e.length,concurrency:this.pricingConcurrency});const t=[];for(let n=0;n<e.length;n++){const o=e[n];t.push({poolIndex:n,token:o.token0,isToken0:!0,task:this.gswapService.getSwapQuoteExactInput({fromToken:o.token0,toToken:"GUSDC",amount:"1"}).then(e=>e.estimatedOutput).catch(e=>{this.logger.debug(`Failed to price ${o.token0}`,{error:e.message})})}),t.push({poolIndex:n,token:o.token1,isToken0:!1,task:this.gswapService.getSwapQuoteExactInput({fromToken:o.token1,toToken:"GUSDC",amount:"1"}).then(e=>e.estimatedOutput).catch(e=>{this.logger.debug(`Failed to price ${o.token1}`,{error:e.message})})})}const n=new Map;for(let t=0;t<e.length;t++)n.set(t,{});for(let e=0;e<t.length;e+=this.pricingConcurrency){const o=t.slice(e,e+this.pricingConcurrency),i=await Promise.allSettled(o.map(e=>e.task));for(let e=0;e<o.length;e++){const t=o[e],r=i[e],a=n.get(t.poolIndex)||{};"fulfilled"===r.status&&r.value&&(t.isToken0?a.token0Price=r.value:a.token1Price=r.value),n.set(t.poolIndex,a)}}const o=e.map((e,t)=>{const o=n.get(t)||{},i={...e};return void 0!==o.token0Price&&(i.token0Price=o.token0Price),void 0!==o.token1Price&&(i.token1Price=o.token1Price),i}),i=o.filter(e=>e.token0Price&&e.token1Price).length;return this.logger.debug("Pricing enrichment complete",{total:e.length,successful:i,failed:e.length-i}),o}async fetchDexPools(e={}){const{search:t,sortBy:n="tvl",sortOrder:o="desc",page:i=Dn.DEFAULT_PAGE,limit:r=Dn.DEFAULT_LIMIT,withPrices:a=!1}=e;this.logger.debug("Fetching DEX pools",{search:t,sortBy:n,sortOrder:o,page:i,limit:r,withPrices:a});const s=new URLSearchParams({page:i.toString(),limit:Math.min(r,20).toString(),sortBy:n,sortOrder:o});t&&s.append("search",t);const c=`${this.dexBackendBaseUrl}/explore/pools?${s}`;try{const e=await this.http.get(c);if(!e||!e.data)throw new Error("No response from DEX pool service");let t=e.data.pools;const n=e.data.count,o=Math.min(r,20),s=Math.ceil(n/o);return a&&(t=await this.enrichPoolsWithPricing(t)),this.logger.debug("DEX pools fetched successfully",{poolCount:t.length,total:n,totalPages:s,withPrices:a}),{pools:t,page:i,limit:o,total:n,totalPages:s,hasNext:i<s,hasPrevious:i>1}}catch(e){throw this.logger.error("Failed to fetch DEX pools",{error:e,url:c}),e}}async fetchAllDexPools(e={}){this.logger.debug("Fetching all DEX pools (auto-paginated)",e);const t=await ni((t,n)=>this.fetchDexPools({...e,page:t,limit:n}).then(e=>({items:e.pools,page:e.page,limit:e.limit,total:e.total,totalPages:e.totalPages,hasNext:e.hasNext,hasPrevious:e.hasPrevious})),{maxPages:1e4,logger:this.logger,pageSize:20});this.logger.debug("All DEX pools fetched",{totalPoolsFetched:t.items.length,totalCount:t.total,withPrices:e.withPrices});return oi(t.items,t.total,"pools")}}const yi=5;class wi extends xn{constructor(e,t,n=!1,o=3e4){super(e,n),this.compositePoolFetchConcurrency=5,this.galaChainBaseUrl=t,this.networkTimeout=o}validateFetchCompositePoolDataInput(e,t,n){if(!e||"string"!=typeof e)throw new Y("token0 must be a non-empty string",{token0:e});if(!t||"string"!=typeof t)throw new Y("token1 must be a non-empty string",{token1:t});const o=e.split("|"),i=t.split("|");if(4!==o.length)throw new Y("token0 format must be: collection|category|type|additionalKey (4 pipe-separated parts)",{token0:e});if(4!==i.length)throw new Y("token1 format must be: collection|category|type|additionalKey (4 pipe-separated parts)",{token1:t});const r=[500,3e3,1e4];if(!Number.isInteger(n)||!r.includes(n))throw new Y(`fee must be one of: ${r.join(", ")} (got ${n})`,{fee:n})}validateQuoteAmount(e){if(!e||"string"!=typeof e)throw new Y("amount must be a non-empty string",{amount:e});const t=new u(e);try{ho(t,"amount","for quote calculation")}catch(t){throw new Y(t.message,{amount:e})}}convertTokenClassKey(e){const t=new I;return t.collection=e.collection,t.category=e.category,t.type=e.type,t.additionalKey=e.additionalKey,t}setCompositePoolFetchConcurrency(e){this.compositePoolFetchConcurrency=Math.max(1,Math.min(e,20)),this.logger.debug(`Composite pool fetch concurrency set to ${this.compositePoolFetchConcurrency}`)}async fetchCompositePoolData(e){const{token0:t,token1:n,fee:o,gatewayBaseUrl:i}=e;this.logger.debug("Fetching composite pool data",{token0:t,token1:n,fee:o}),this.validateFetchCompositePoolDataInput(t,n,o);try{const e=di(t),r=di(n),a=this.convertTokenClassKey(e),s=this.convertTokenClassKey(r),c=new b(a,s,o),l=`${i||this.galaChainBaseUrl}/api/asset/dexv3-contract/GetCompositePool`,d=await this.http.post(l,c);if(!d||1!==d.Status)throw new J(`Pool not found: ${t}/${n} with fee ${o}`);const h=function(e){return{pool:e.pool,tickDataMap:e.tickDataMap,token0Balance:e.token0Balance,token1Balance:e.token1Balance,token0Decimals:e.token0Decimals,token1Decimals:e.token1Decimals,compositePoolDto:e}}(function(e){const t=new w(e.pool.token0,e.pool.token1,e.pool.token0ClassKey,e.pool.token1ClassKey,e.pool.fee,new u(e.pool.sqrtPrice),e.pool.protocolFees);t.bitmap=e.pool.bitmap,t.grossPoolLiquidity=new u(e.pool.grossPoolLiquidity),t.liquidity=new u(e.pool.liquidity),t.feeGrowthGlobal0=new u(e.pool.feeGrowthGlobal0),t.feeGrowthGlobal1=new u(e.pool.feeGrowthGlobal1),t.protocolFeesToken0=new u(e.pool.protocolFeesToken0),t.protocolFeesToken1=new u(e.pool.protocolFeesToken1),t.tickSpacing=e.pool.tickSpacing,t.maxLiquidityPerTick=new u(e.pool.maxLiquidityPerTick);const n={};Object.keys(e.tickDataMap).forEach(t=>{const o=e.tickDataMap[t],i=new k(o.poolHash,o.tick);i.initialised=o.initialised,i.liquidityNet=new u(o.liquidityNet),i.liquidityGross=new u(o.liquidityGross),i.feeGrowthOutside0=new u(o.feeGrowthOutside0),i.feeGrowthOutside1=new u(o.feeGrowthOutside1),n[t]=i});const o={...e.token0Balance},i=new P(o);i.quantity=new u(e.token0Balance.quantity);const r={...e.token1Balance},a=new P(r);return a.quantity=new u(e.token1Balance.quantity),new v(t,n,i,a,e.token0Decimals,e.token1Decimals)}(d.Data),d.Data);return this.logger.debug("Composite pool data fetched successfully",{token0:t,token1:n,fee:o,liquidity:h.pool.liquidity.toString()}),h}catch(e){if(e instanceof J)throw e;const i=e instanceof Error?e.message:String(e);throw this.logger.error("Failed to fetch composite pool data",e),new Y(`Failed to fetch composite pool data: ${i}`,{token0:t,token1:n,fee:o})}}async calculateDexPoolQuoteExactAmountLocal(e){const{compositePoolData:t,fromToken:n,toToken:o,amount:i}=e;if(this.logger.debug("Calculating local DEX quote",{fromToken:n,toToken:o,amount:i}),this.validateQuoteAmount(i),!t)throw new Y("compositePoolData is required for local quote calculation",{compositePoolData:t});try{const e=n===t.pool.token0.replace(/\$/g,"|"),r=di(n),a=di(o),s=this.convertTokenClassKey(r),c=this.convertTokenClassKey(a),[l,d]=n<o?[s,c]:[c,s],h=new T(l,d,t.pool.fee,new u(i),e,t.compositePoolDto),g=await S(void 0,h);return this.logger.debug("Local quote calculated",{amount0:g.amount0,amount1:g.amount1}),{amount0:g.amount0.toString(),amount1:g.amount1.toString(),currentSqrtPrice:g.currentSqrtPrice.toString(),newSqrtPrice:g.newSqrtPrice.toString()}}catch(e){const t=e instanceof Error?e.message:String(e);throw this.logger.error("Local quote calculation failed",e),new Y(`Local quote calculation failed: ${t}`,{fromToken:n,toToken:o,amount:i})}}async calculateDexPoolQuoteExactAmountExternal(e){const{compositePoolData:t,fromToken:n,toToken:o,amount:i}=e;if(this.logger.debug("Calculating external DEX quote",{fromToken:n,toToken:o,amount:i}),this.validateQuoteAmount(i),!t)throw new Y("compositePoolData is required for external quote calculation (token format info)",{compositePoolData:t});try{const e=n===t.pool.token0.replace(/\$/g,"|"),r=di(n),a=di(o),s=this.convertTokenClassKey(r),c=this.convertTokenClassKey(a),l=new T(s,c,t.pool.fee,new u(i),e,void 0),d=`${this.galaChainBaseUrl}/api/asset/dexv3-contract/QuoteExactAmount`,h=await this.http.post(d,l);if(!h||1!==h.Status)throw new Y("External quote failed: "+(h?.Message||"Unknown error"));const g=h.Data;return this.logger.debug("External quote calculated",{amount0:g.amount0,amount1:g.amount1}),{amount0:g.amount0.toString(),amount1:g.amount1.toString(),currentSqrtPrice:g.currentSqrtPrice.toString(),newSqrtPrice:g.newSqrtPrice.toString()}}catch(e){const t=e instanceof Error?e.message:String(e);throw this.logger.error("External quote calculation failed",e),new Y(`External quote calculation failed: ${t}`,{fromToken:n,toToken:o,amount:i})}}async calculateDexPoolQuoteExactAmount(e,t="local"){return"external"===t?this.calculateDexPoolQuoteExactAmountExternal(e):this.calculateDexPoolQuoteExactAmountLocal(e)}}class ki{constructor(){this.eventLatencies=[],this.maxLatencySamples=1e4,this.eventsProcessed=0,this.eventsDropped=0,this.queueDepth=0,this.maxQueueDepth=0,this.startTime=Date.now(),this.perPoolMetrics=new Map,this.memorySnapshots=[],this.maxMemorySnapshots=100,this.recordMemory()}recordEventLatency(e){this.eventLatencies.push(e),this.eventLatencies.length>this.maxLatencySamples&&this.eventLatencies.shift(),this.eventsProcessed++,this.lastEventTime=new Date}recordEventDropped(){this.eventsDropped++}updateQueueDepth(e){this.queueDepth=e,this.maxQueueDepth=Math.max(this.maxQueueDepth,e)}recordPoolCacheHit(e,t){const n=this.getPoolMetrics(e);n.cacheHits++,n.eventsProcessed++,n.totalLatency+=t,n.lastEventTime=new Date}recordPoolCacheMiss(e,t){const n=this.getPoolMetrics(e);n.cacheMisses++,n.eventsProcessed++,n.totalLatency+=t,n.lastEventTime=new Date}getLatencyPercentiles(){if(0===this.eventLatencies.length)return{p50:0,p95:0,p99:0};const e=[...this.eventLatencies].sort((e,t)=>e-t),t=Math.floor(.5*e.length),n=Math.floor(.95*e.length),o=Math.floor(.99*e.length);return{p50:e[t]??0,p95:e[n]??0,p99:e[o]??0}}getCacheHitRate(){if(0===this.eventsProcessed)return 0;let e=0;for(const t of this.perPoolMetrics.values())e+=t.cacheHits;return e/this.eventsProcessed*100}getThroughputPerSecond(){const e=(Date.now()-this.startTime)/1e3;return 0===e?0:this.eventsProcessed/e}recordMemory(){if("undefined"!=typeof process&&process.memoryUsage){const e=process.memoryUsage().heapUsed/1024/1024;this.memorySnapshots.push(e),this.memorySnapshots.length>this.maxMemorySnapshots&&this.memorySnapshots.shift()}}getMemoryUsedMB(){return"undefined"!=typeof process&&process.memoryUsage?process.memoryUsage().heapUsed/1024/1024:0}getPoolAverageLatency(e){const t=this.perPoolMetrics.get(e);return t&&0!==t.eventsProcessed?t.totalLatency/t.eventsProcessed:0}getPoolCacheHitRate(e){const t=this.perPoolMetrics.get(e);if(!t)return 0;const n=t.cacheHits+t.cacheMisses;return 0===n?0:t.cacheHits/n*100}getHealthMetrics(e,t,n,o,i){const r=this.getLatencyPercentiles(),a=this.getMemoryUsedMB();return{eventProcessing:{queueDepth:this.queueDepth,eventsProcessed:this.eventsProcessed,eventsDropped:this.eventsDropped,throughputPerSecond:this.getThroughputPerSecond()},metrics:{latencyP50:r.p50,latencyP95:r.p95,latencyP99:r.p99,cacheHitRate:this.getCacheHitRate()},memory:{usedMB:Math.round(10*a)/10,maxMB:i,percentUsed:Math.round(a/i*1e3)/10},pools:{totalMonitored:e,hotCacheSize:t,warmCacheSize:n,coldCacheSize:o}}}reset(){this.eventLatencies=[],this.eventsProcessed=0,this.eventsDropped=0,this.queueDepth=0,this.maxQueueDepth=0,this.startTime=Date.now(),this.lastEventTime=void 0,this.perPoolMetrics.clear(),this.memorySnapshots=[]}getSummary(){const e=this.eventLatencies.length>0?this.eventLatencies.reduce((e,t)=>e+t,0)/this.eventLatencies.length:0;return{eventsProcessed:this.eventsProcessed,eventsDropped:this.eventsDropped,cacheHitRate:this.getCacheHitRate(),averageLatency:Math.round(e),memoryUsedMB:Math.round(10*this.getMemoryUsedMB())/10,throughputPerSecond:Math.round(100*this.getThroughputPerSecond())/100}}getPoolMetrics(e){let t=this.perPoolMetrics.get(e);return t||(t={eventsProcessed:0,totalLatency:0,cacheHits:0,cacheMisses:0},this.perPoolMetrics.set(e,t)),t}}class vi{static createPoolKey(e,t,n){return`${e}/${t}/${n}`}static parsePoolKey(e){if(!e||"string"!=typeof e)return null;const t=e.split("/");if(3!==t.length)return null;const n=t[0]?.trim(),o=t[1]?.trim(),i=t[2]?.trim();if(!n||!o||!i)return null;const r=Number.parseInt(i,10);return Number.isNaN(r)?null:{token0:n,token1:o,feeTier:r}}static isValidPoolKey(e){if("string"!=typeof e)return!1;return null!==this.parsePoolKey(e)}static getToken0(e){const t=this.parsePoolKey(e);return t?.token0??null}static getToken1(e){const t=this.parsePoolKey(e);return t?.token1??null}static getFeeTier(e){const t=this.parsePoolKey(e);return t?.feeTier??null}static containsToken(e,t){const n=this.parsePoolKey(e);return!!n&&(n.token0===t||n.token1===t)}static containsTokenPair(e,t,n){const o=this.parsePoolKey(e);if(!o)return!1;const i=o.token0===t||o.token1===t,r=o.token0===n||o.token1===n;return i&&r&&t!==n}static normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:Number.isInteger(t)&&t>0?t:null}static formatFeeAsPercentage(e){return`${(e/1e4).toFixed(2)}%`}static isValidTokenPair(e,t){return Boolean(e)&&Boolean(t)&&e!==t}}class bi{constructor(e){this.logger=e||new _({debug:!1,context:"SwapEventExtractor"})}walkPayloadForSwaps(e,t){const n=[],o=new WeakSet,i=(e,r=0)=>{if(r>50)this.logger.debug("Payload nesting exceeded maximum depth of 50");else if(e&&"string"!=typeof e&&"object"==typeof e){if(o.has(e))return;o.add(e);const a=this.extractSwapFromObject(e);a&&!t.has(a.transactionId)&&(n.push(a),t.add(a.transactionId));for(const t of Object.values(e))i(t,r+1)}};return i(e,0),n}extractSwapFromObject(e){const t=this.extractTransactionId(e);if(!t)return null;const n=e.Data,o=n&&"object"==typeof n&&!Array.isArray(n)?n:e,i=this.extractToken(o,"token0","fromToken","source"),r=this.extractToken(o,"token1","toToken","destination");if(!i||!r)return null;const a=this.extractAmount(o,"amount0","amountIn","inputAmount"),s=this.extractAmount(o,"amount1","amountOut","outputAmount");if(!a||!s)return null;const c=this.extractFeeTier(o);if(null===c)return null;const l=this.extractTimestamp(o),u=this.buildPoolKey(i,r,c),d=this.determineDirection(o,i,r),h={transactionId:t,poolKey:u,token0:i,token1:r,amount0:a,amount1:s,feeTier:c,direction:d,timestamp:l,exactInput:this.determineExactInput(o,d)},g=this.extractUser(o);return void 0!==g&&(h.user=g),h}extractTransactionId(e){const t=["transactionId","txId","tx_id","hash","txHash","id"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractToken(e,...t){for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractAmount(e,...t){for(const n of t){const t=e[n];if(null!=t){const e=String(t).trim();if(/^-?\d+(\.\d+)?([eE]-?\d+)?$/.test(e))return e}}return null}extractFeeTier(e){const t=["poolFee","feeTier","fee","feeTierBps","liquidityFeeBps","feeAmount"];for(const n of t){const t=e[n],o=this.normalizeFee(t);if(null!==o)return o}return null}normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:Number.isInteger(t)?t:null}extractTimestamp(e){const t=["timeStamp","timestamp","time","createdAt","date"];for(const n of t){const t=e[n];if("number"==typeof t)return t;if("string"==typeof t){const e=new Date(t).getTime();if(!Number.isNaN(e))return e}}return Date.now()}extractUser(e){const t=["userAddress","user","from","sender","wallet","address"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}}determineDirection(e,t,n){const o=e.zeroForOne||e.direction;if("boolean"==typeof o)return o?"zeroForOne":"oneForZero";if("string"==typeof o){if("zerotoone"===o.toLowerCase()||"0to1"===o)return"zeroForOne";if("onetozero"===o.toLowerCase()||"1to0"===o)return"oneForZero"}if(e.fromToken===t||e.inputToken===t)return"zeroForOne";if(e.fromToken===n||e.inputToken===n)return"oneForZero";const i=this.extractAmount(e,"amount0","amountIn");return i&&Number(i),"zeroForOne"}determineExactInput(e,t){if("boolean"==typeof e.exactInput)return e.exactInput;if("boolean"==typeof e.exactOutput)return!e.exactOutput;const n=void 0!==e.amountIn&&null!==e.amountIn,o=void 0!==e.amountOut&&null!==e.amountOut,i=void 0!==e.inputAmount&&null!==e.inputAmount,r=void 0!==e.outputAmount&&null!==e.outputAmount;return!(!n||o)||!(o&&!n)&&(!(!i||r)||!(r&&!i))}buildPoolKey(e,t,n){return`${e}/${t}/${n}`}}class Ti{static getCached(e){const t=e.toString();return this.CACHE.has(t)||this.CACHE.set(t,new u(e)),this.CACHE.get(t)}static clearCache(){this.CACHE.clear()}static getCacheStats(){return{size:this.CACHE.size,entries:Array.from(this.CACHE.keys())}}static trimCache(e=1e3){if(this.CACHE.size>e){const t=this.CACHE.size-e,n=Array.from(this.CACHE.keys());for(let e=0;e<t;e++)this.CACHE.delete(n[e])}}}Ti.CACHE=new Map,Ti.ZERO=new u(0),Ti.ONE=new u(1),Ti.FEE_PIPS=new u(1e6),Ti.MIN_SQRT_RATIO=new u("4295128739"),Ti.MAX_SQRT_RATIO=new u("1461446703485210103287273052203988822378723970342");const Si={maxIterations:100,enableBigNumberCache:!0,roundingMode:u.ROUND_DOWN,debugLogging:!1};class Ai{static calculateSwapDelta(e,t,n={}){const o=Date.now(),i={...Si,...n};try{const n=this.initializeSwapState(e,t,i);i.debugLogging&&this.logger.debug("Initialized swap state",{sqrtPrice:n.sqrtPrice.toString(),liquidity:n.liquidity.toString(),tick:n.tick,zeroForOne:t.zeroForOne});const r=this.computeSwapLoop(n,e,t,i);i.debugLogging&&this.logger.debug("Swap loop completed",{stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length,priceHitLimit:r.priceHitLimit});const a=this.createUpdatedPool(e.pool,r.state,t,i),s=this.calculateFinalAmounts(n,r.state,t);let c;if(t.actualSqrtPrice){const e=new u(a.sqrtPrice),n=new u(t.actualSqrtPrice);c=e.minus(n).abs().div(n).times(100).toNumber()}const l=Date.now()-o;i.debugLogging&&this.logger.debug("Swap delta calculated",{calculationTimeMs:l,amount0:s.amount0.toString(),amount1:s.amount1.toString(),driftPercentage:c}),l>100&&this.logger.warn("Swap calculation exceeded 100ms",{calculationTimeMs:l,stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length}),r.priceHitLimit&&this.logger.warn("Swap price hit limit - partially fulfilled",{zeroForOne:t.zeroForOne,stepCount:r.stepCount}),r.stepCount>50&&this.logger.warn("Unusually complex swap detected",{stepCount:r.stepCount,ticksCrossed:r.ticksCrossed.length});return{updatedPool:a,updatedTicks:r.updatedTicks,amount0:s.amount0,amount1:s.amount1,feeAmount0:s.feeAmount0,feeAmount1:s.feeAmount1,ticksCrossed:r.ticksCrossed,metadata:{calculationTimeMs:l,swapSteps:r.stepCount,priceHitLimit:r.priceHitLimit,...void 0!==c&&{driftPercentage:c}}}}catch(e){this.logger.error("Swap delta calculation failed",e);const t=e instanceof Error?e.message:String(e);throw new Error(`Swap delta calculation failed: ${t}`)}}static initializeSwapState(e,t,n){const{pool:o}=e;if(!o.sqrtPrice||!o.liquidity)throw new Error("Invalid pool data: missing sqrtPrice or liquidity");const i=n.enableBigNumberCache?Ti.getCached.bind(Ti):e=>new u(e),r="string"==typeof o.sqrtPrice?o.sqrtPrice:new u(o.sqrtPrice).toFixed(0),a="string"==typeof o.liquidity?o.liquidity:new u(o.liquidity).toFixed(0),s=i(r),c=i(a),l=o.tick??0,d=A(new u(r)),h=Math.abs(d-l);h>100&&this.logger.warn("Significant tick/price mismatch detected in pool state",{poolTick:l,calculatedTick:d,drift:h,threshold:100});const g=i(t.amountSpecified);ho(g,"amountSpecified","for swap operation");const p="string"==typeof o.feeGrowthGlobal1?o.feeGrowthGlobal1:new u(o.feeGrowthGlobal1).toFixed(0),m="string"==typeof o.feeGrowthGlobal0?o.feeGrowthGlobal0:new u(o.feeGrowthGlobal0).toFixed(0),f=t.zeroForOne?i(p):i(m);return{sqrtPrice:s,liquidity:c,tick:l,amountSpecifiedRemaining:g,amountCalculated:Ti.ZERO,feeGrowthGlobalX:f,protocolFee:Ti.ZERO}}static computeSwapLoop(e,t,n,o){const{pool:i,tickDataMap:r}=t,a=[],s={};let c=0;const l=n.zeroForOne?Ti.MIN_SQRT_RATIO:Ti.MAX_SQRT_RATIO,h=new u("0.000001");for(;e.amountSpecifiedRemaining.gt(h)&&!e.sqrtPrice.eq(l)&&c<o.maxIterations;){c++;const[t,h]=this.findNextInitializedTick(r,e.tick,i.tickSpacing,n.zeroForOne);let g;if(o.debugLogging&&this.logger.debug(`Swap step ${c}`,{currentTick:e.tick,tickNext:t,initialized:h,sqrtPrice:e.sqrtPrice.toString(),liquidity:e.liquidity.toString(),amountRemaining:e.amountSpecifiedRemaining.toString()}),h&&t>=-887272&&t<=887272){const e=d(t);g=e instanceof u?e:new u(String(e))}else g=l;const p=n.zeroForOne?u.max(g,l):u.min(g,l),m=this.executeSwapStep(e.sqrtPrice,p,e.liquidity,e.amountSpecifiedRemaining,i.fee,n.exactInput);if(e.sqrtPrice=m.sqrtPriceNext,n.exactInput){const t=m.amountIn.plus(m.feeAmount);t.lte(0)?e.amountSpecifiedRemaining=Ti.ZERO:(e.amountSpecifiedRemaining=e.amountSpecifiedRemaining.minus(t),e.amountSpecifiedRemaining.lt(0)&&(e.amountSpecifiedRemaining=Ti.ZERO)),e.amountCalculated=e.amountCalculated.minus(m.amountOut)}else{m.amountOut.lte(0)?e.amountSpecifiedRemaining=Ti.ZERO:e.amountSpecifiedRemaining=e.amountSpecifiedRemaining.plus(m.amountOut),e.amountCalculated=e.amountCalculated.plus(m.amountIn.plus(m.feeAmount))}if(e.liquidity.gt(0)){const t=m.feeAmount.div(e.liquidity);e.feeGrowthGlobalX=e.feeGrowthGlobalX.plus(t)}if(e.sqrtPrice.eq(g)&&h){const i=r[t.toString()];if(!i)throw new Error(`Missing tick data for initialized tick ${t}`);const c=n.zeroForOne?new u(i.liquidityNet).negated():new u(i.liquidityNet);if(e.liquidity=e.liquidity.plus(c),e.liquidity.lt(0))throw new Error(`Negative liquidity after crossing tick ${t}: ${e.liquidity.toString()}`);a.push(t),s[t.toString()]=i,o.debugLogging&&this.logger.debug(`Crossed tick ${t}`,{liquidityNet:c.toString(),newLiquidity:e.liquidity.toString()})}if(e.sqrtPrice.eq(g))e.tick=n.zeroForOne?t-1:t;else{const t=A(new u(e.sqrtPrice.toString()));e.tick=t}}if(c>=o.maxIterations)throw new Error(`Swap calculation exceeded maximum iterations (${o.maxIterations}). Possible infinite loop or very complex swap.`);const g=e.sqrtPrice.eq(l);return{state:e,ticksCrossed:a,priceHitLimit:g,stepCount:c,updatedTicks:s}}static createUpdatedPool(e,t,n,o){const i=Object.assign(Object.create(Object.getPrototypeOf(e)),e);if(i.sqrtPrice=t.sqrtPrice.toFixed(0),i.liquidity=t.liquidity.toFixed(0),i.tick=t.tick,n.zeroForOne?i.feeGrowthGlobal1=t.feeGrowthGlobalX.toFixed(0):i.feeGrowthGlobal0=t.feeGrowthGlobalX.toFixed(0),n.zeroForOne){const n=new u(e.protocolFeesToken0);i.protocolFeesToken0=n.plus(t.protocolFee).toFixed(0)}else{const n=new u(e.protocolFeesToken1);i.protocolFeesToken1=n.plus(t.protocolFee).toFixed(0)}return i}static calculateFinalAmounts(e,t,n){let o,i,r,a;if(n.exactInput){const e=new u(n.amountSpecified),s=t.amountCalculated.abs();n.zeroForOne?(o=e.negated(),i=s,r=Ti.ZERO,a=Ti.ZERO):(o=s,i=e.negated(),r=Ti.ZERO,a=Ti.ZERO)}else{const e=new u(n.amountSpecified),s=t.amountCalculated.abs();n.zeroForOne?(o=s.negated(),i=e,r=Ti.ZERO,a=Ti.ZERO):(o=e,i=s.negated(),r=Ti.ZERO,a=Ti.ZERO)}const s=t.feeGrowthGlobalX.minus(e.feeGrowthGlobalX).times(e.liquidity);return n.zeroForOne?a=s:r=s,{amount0:o,amount1:i,feeAmount0:r,feeAmount1:a}}static findNextInitializedTick(e,t,n,o){const i=Object.keys(e).map(e=>parseInt(e,10)).sort((e,t)=>e-t);if(0===i.length){return[o?-887272:887272,!1]}if(o){const e=i.reverse().find(e=>e<t);return void 0!==e?[e,!0]:[-887272,!1]}{const e=i.find(e=>e>t);return void 0!==e?[e,!0]:[887272,!1]}}static executeSwapStep(e,t,n,o,i,r){uo(n),ho(o,"amountRemaining","for swap step");const a=[500,3e3,1e4];if(!a.includes(i))throw new Error(`Invalid fee tier: ${i}. Must be one of: ${a.join(", ")}`);const s=E(e,t,n,o,i,t.lt(e)),c=s[0],l=s[1],d=s[2],h=s[3],g=u.isBigNumber(c)?c:new u(String(c)),p=u.isBigNumber(l)?l:new u(String(l)),m=u.isBigNumber(d)?d:new u(String(d)),f=u.isBigNumber(h)?h:new u(String(h));return{sqrtPriceStart:e,tickNext:A(g),sqrtPriceNext:g,initialised:!1,amountIn:p,amountOut:m,feeAmount:f}}}Ai.logger=new _({debug:!1,context:"SwapDeltaCalculator"});class Ei{constructor(e,t,n,o){this.cache=new Map,this.tierSizes={hot:50,warm:200,cold:0},this.tierTTLs={hot:1/0,warm:18e5,cold:3e5},this.refetchThresholds={swapCount:50,driftPercent:.05},this.fetchPoolFn=e,this.config=t,this.metrics=n,this.logger=o||new _({debug:!1,context:"PoolCacheManager"}),this.tierSizes.cold=Math.max(0,this.config.maxPools-this.tierSizes.hot-this.tierSizes.warm),this.logger.debug(`Initialized with cache limits: hot=${this.tierSizes.hot}, warm=${this.tierSizes.warm}, cold=${this.tierSizes.cold}, max=${this.config.maxPools}`)}async getPool(e){const t=this.cache.get(e);if(t){if(!(Date.now()>t.expiresAt))return t.lastAccessTime=Date.now(),this.checkRefetchNeeded(e,t),this.metrics.recordPoolCacheHit(e,0),t.poolData;this.cache.delete(e),this.logger.debug(`Cache expired for pool ${e}`)}this.metrics.recordPoolCacheMiss(e,0);try{const t=await this.fetchPoolFn(e),n=this.determineTier(),o={poolData:t,tier:n,lastAccessTime:Date.now(),expiresAt:Date.now()+this.tierTTLs[n],swapsSinceRefetch:0,cumulativeDrift:0,lastDeltaAppliedTime:Date.now()};return this.cache.set(e,o),this.cache.size>this.config.maxPools&&this.evictLRU(),this.logger.debug(`Fetched pool ${e} (tier: ${n})`),t}catch(t){throw this.logger.error(`Failed to fetch pool ${e}:`,t),t}}updatePoolWithSwapDelta(e,t,n,o,i){const r=this.cache.get(e);if(!r)return this.logger.debug(`Pool ${e} not in cache for delta update`),!1;try{if(i){const a="zeroForOne"===t,s=a?n:o,c={transactionId:i.transactionId,timestamp:i.timestamp,amountSpecified:s,zeroForOne:a,exactInput:i.exactInput},l=Ai.calculateSwapDelta(r.poolData,c);r.poolData={...r.poolData,pool:l.updatedPool},r.swapsSinceRefetch++,r.lastDeltaAppliedTime=Date.now(),void 0!==l.metadata.driftPercentage?(r.cumulativeDrift+=l.metadata.driftPercentage,this.logger.debug(`Delta applied for ${e}: drift=${l.metadata.driftPercentage.toFixed(4)}%, cumulative=${r.cumulativeDrift.toFixed(2)}%`)):this.logger.debug(`Delta applied for ${e}: ${l.ticksCrossed.length} ticks crossed`)}else r.swapsSinceRefetch++,r.lastDeltaAppliedTime=Date.now();return this.shouldRefetch(r)&&(this.logger.debug(`Refetch needed for ${e}: swaps=${r.swapsSinceRefetch}, drift=${(100*r.cumulativeDrift).toFixed(2)}%`),r.expiresAt=Date.now()),!0}catch(t){return this.logger.error(`Failed to update pool ${e}:`,t),r.expiresAt=Date.now(),!1}}getStats(){const e={totalCached:this.cache.size,hotCacheSize:0,warmCacheSize:0,coldCacheSize:0,memoryUsedMB:this.metrics.getMemoryUsedMB()};for(const t of this.cache.values())"hot"===t.tier?e.hotCacheSize++:"warm"===t.tier?e.warmCacheSize++:e.coldCacheSize++;return e}getPoolInfo(e){const t=this.cache.get(e);return t?{poolKey:e,tier:t.tier,lastAccessTime:new Date(t.lastAccessTime),expiresAt:new Date(t.expiresAt),swapsSinceRefetch:t.swapsSinceRefetch,cumulativeDrift:t.cumulativeDrift,isExpired:Date.now()>t.expiresAt}:null}async warmCache(e){if(this.cache.has(e))return!0;try{return await this.getPool(e),this.logger.debug(`Cache warmed for ${e}`),!0}catch(t){return this.logger.error(`Failed to warm cache for ${e}:`,t),!1}}async warmCacheBatch(e,t=5){const n={succeeded:0,failed:0,total:e.length};let o=0;const i=new Set;for(;o<e.length||i.size>0;){for(;o<e.length&&i.size<t;){const t=e[o];o++;const r=this.warmCache(t).then(e=>{e?n.succeeded++:n.failed++});i.add(r),r.finally(()=>i.delete(r))}i.size>0&&await Promise.race(i)}return this.logger.debug(`Cache warming complete: ${n.succeeded}/${n.total} succeeded`),n}clear(){this.cache.clear(),this.logger.debug("Cache cleared")}clearExpired(){const e=Date.now();let t=0;for(const[n,o]of this.cache)e>o.expiresAt&&(this.cache.delete(n),t++);t>0&&this.logger.debug(`Cleared ${t} expired entries`)}determineTier(){const e=Array.from(this.cache.values()).filter(e=>"hot"===e.tier).length,t=Array.from(this.cache.values()).filter(e=>"warm"===e.tier).length;return e<this.tierSizes.hot?"hot":t<this.tierSizes.warm?"warm":"cold"}checkRefetchNeeded(e,t){this.shouldRefetch(t)&&(this.logger.debug(`Scheduling refetch for ${e}: swaps=${t.swapsSinceRefetch}, drift=${(100*t.cumulativeDrift).toFixed(2)}%`),t.expiresAt=Date.now())}shouldRefetch(e){return e.swapsSinceRefetch>=this.refetchThresholds.swapCount||e.cumulativeDrift>=this.refetchThresholds.driftPercent}evictLRU(){const e=Array.from(this.cache.entries()).filter(([e,t])=>"cold"===t.tier).sort((e,t)=>e[1].lastAccessTime-t[1].lastAccessTime);if(0===e.length){this.logger.warn("No cold cache entries to evict, trying warm cache");const e=Array.from(this.cache.entries()).filter(([e,t])=>"warm"===t.tier).sort((e,t)=>e[1].lastAccessTime-t[1].lastAccessTime);if(e.length>0){const[t]=e[0];this.cache.delete(t),this.logger.debug(`Evicted warm cache entry: ${t}`)}return}const[t]=e[0];this.cache.delete(t),this.logger.debug(`Evicted cold cache entry: ${t}`)}async refreshWarmAndHotTiers(){const e=Array.from(this.cache.entries()).filter(([e,t])=>"hot"===t.tier||"warm"===t.tier).sort((e,t)=>t[1].lastAccessTime-e[1].lastAccessTime).slice(0,10);if(0===e.length)return;const t=e.map(([e,t])=>this.fetchPoolFn(e).then(n=>{t&&(t.poolData=n,t.lastAccessTime=Date.now(),t.swapsSinceRefetch=0,t.cumulativeDrift=0),this.logger.debug(`Refreshed ${e} during background warming`)}).catch(t=>{this.logger.debug(`Failed to refresh ${e} during background warming:`,t)})),n=new Promise(e=>setTimeout(()=>e(),5e3));await Promise.race([Promise.all(t),n])}}class Ii{constructor(e,t,n){this.queue=[],this.isShuttingDown=!1,this.currentConcurrency=0,this.maxConcurrencyReached=0,this.eventsDropped=0,this.totalBatchesProcessed=0,this.totalBatchSize=0,this.eventsProcessedCount=0,this.processor=null,this.processingScheduled=!1,this.scheduleProcessing=e=>{"undefined"!=typeof setImmediate?setImmediate(e):Promise.resolve().then(e)},this.config=e,this.metrics=t,this.logger=n||new _({debug:!1,context:"SwapEventQueue"}),this.logger.debug(`Initialized with maxQueueSize=${this.config.maxQueueSize}, batchSize=${this.config.batchSize}, maxConcurrent=${this.config.maxConcurrent}`)}setProcessor(e){this.processor=e}enqueue(e){return this.isShuttingDown?(this.logger.debug(`Rejecting event (queue shutting down): ${e.transactionId}`),this.metrics.recordEventDropped(),this.eventsDropped++,!1):this.queue.length>=this.config.maxQueueSize?(this.logger.warn(`Queue full (${this.queue.length}/${this.config.maxQueueSize}), dropping event: ${e.transactionId}`),this.metrics.recordEventDropped(),this.eventsDropped++,!1):(this.queue.push(e),this.metrics.updateQueueDepth(this.queue.length),this.processingScheduled||this.isShuttingDown||(this.processingScheduled=!0,this.scheduleProcessing(()=>this.processNextBatch())),!0)}getQueueSize(){return this.queue.length}getStats(){return{queueSize:this.queue.length,eventsProcessed:this.eventsProcessedCount,eventsDropped:this.eventsDropped,currentConcurrent:this.currentConcurrency,maxConcurrentReached:this.maxConcurrencyReached,averageBatchSize:this.totalBatchesProcessed>0?Math.floor(this.totalBatchSize/this.totalBatchesProcessed):0,totalBatchesProcessed:this.totalBatchesProcessed}}async waitForEmpty(e){return new Promise(t=>{const n=()=>{0!==this.queue.length||0!==this.currentConcurrency?setTimeout(n,10):t()};e&&setTimeout(()=>t(),e),n()})}async shutdown(e=3e4){this.isShuttingDown=!0,this.logger.debug("Shutting down queue...");const t=Date.now();for(;this.queue.length>0||this.currentConcurrency>0;){if(Date.now()-t>e){this.logger.warn(`Queue shutdown timeout: ${this.queue.length} events remaining, ${this.currentConcurrency} processing`);break}await new Promise(e=>setTimeout(e,50))}this.logger.debug("Queue shutdown complete")}clear(){const e=this.queue.length;this.queue.length=0,this.eventsDropped+=e,this.metrics.updateQueueDepth(0),this.logger.warn(`Cleared ${e} events from queue`)}async processNextBatch(){if(this.processingScheduled=!1,this.isShuttingDown&&0===this.queue.length)return;if(this.currentConcurrency>=this.config.maxConcurrent)return void setTimeout(()=>{this.processingScheduled||(this.processingScheduled=!0,setImmediate(()=>this.processNextBatch()))},10);const e=Math.min(this.config.batchSize,this.queue.length,this.config.maxConcurrent-this.currentConcurrency);if(0===e)return;const t=this.queue.splice(0,e);this.metrics.updateQueueDepth(this.queue.length),this.currentConcurrency+=t.length,this.currentConcurrency>this.maxConcurrencyReached&&(this.maxConcurrencyReached=this.currentConcurrency),this.totalBatchSize+=t.length,this.totalBatchesProcessed++;try{const e=await Promise.allSettled(t.map(e=>this.processEvent(e)));for(let n=0;n<e.length;n++){const o=e[n];this.eventsProcessedCount++,"rejected"===o.status&&this.logger.error(`Failed to process event ${t[n].transactionId}:`,o.reason)}}finally{this.currentConcurrency-=t.length}this.queue.length>0&&!this.processingScheduled&&(this.processingScheduled=!0,this.scheduleProcessing(()=>this.processNextBatch()))}async processEvent(e){if(!this.processor)return void this.logger.warn("No processor set, discarding event:",e.transactionId);const t=Date.now();try{await this.processor(e);const n=Date.now()-t;this.metrics.recordEventLatency(n)}catch(t){throw this.logger.error(`Event processing failed for ${e.transactionId}:`,t),t}}}class Pi{constructor(e,t,n,o={},i){this.socket=null,this.maxSeenTransactions=1e4,this.listeners=[],this.onErrorCallbacks=[],this.isActive=!1,this.listenerRegistered=!1,this.handleSwapEvent=null,this.warmingIntervalHandle=null,this.reconnectAttempts=0,this.maxReconnectAttempts=3,this.reconnectDelayMs=1e3,this.logger=i||new _({debug:!1,context:"MultiPoolStateManager"}),e instanceof Promise?this.socketReady=e.then(e=>(this.socket=e,this.setupConnectionMonitoring(),e)).catch(e=>{throw this.logger.error("Failed to resolve socket promise:",e),e}):(this.socket=e,this.socketReady=Promise.resolve(e),this.setupConnectionMonitoring()),this.metrics=new ki,this.config=this.applyDefaults(o),this.eventExtractor=new bi(this.logger),this.quoteService=n,this.cacheManager=new Ei(t,this.config,this.metrics,this.logger),this.eventQueue=new Ii(this.config,this.metrics,this.logger),this.seenTransactions=new Ni(this.maxSeenTransactions),this.eventQueue.setProcessor(e=>this.processSwapEvent(e)),this.logger.debug("Initialized MultiPoolStateManager")}subscribe(e,t){this.listeners.push(t),e.onError&&this.onErrorCallbacks.push(e.onError),this.isActive||(this.setupWebSocketListener(e),this.isActive=!0);const n=this;return()=>{n.listeners=n.listeners.filter(e=>e!==t),e.onError&&(n.onErrorCallbacks=n.onErrorCallbacks.filter(t=>t!==e.onError)),0===n.listeners.length&&0===n.onErrorCallbacks.length&&n.unsubscribe()}}getHealth(){const e=this.cacheManager.getStats(),t=this.eventQueue.getStats(),n=this.metrics.getHealthMetrics(e.totalCached,e.hotCacheSize,e.warmCacheSize,e.coldCacheSize,this.getMaxMemoryMB()),o=this.determineHealthStatus(t,e),i={connected:this.socket?.connected??!1,reconnectAttempts:this.reconnectAttempts};this.socket?.connected&&(i.lastConnectionTime=new Date);const r=t.queueSize/this.config.maxQueueSize*100,a=e.totalCached/this.config.maxPools*100,s=e.memoryUsedMB/this.getMaxMemoryMB()*100,c=this.generateHealthRecommendations(o,r,a,s,n.metrics.cacheHitRate);return{...n,status:o,websocket:i,recommendations:c,detailedMetrics:{eventQueueUtilization:r,cacheUtilization:a,memoryUtilization:s}}}generateHealthRecommendations(e,t,n,o,i){const r=[];return"failed"===e&&(r.push("🔴 System is in FAILED state - immediate action required"),this.socket?.connected||r.push("Reconnect WebSocket - connection lost"),t>90&&r.push("Reduce incoming event rate or increase maxQueueSize")),"degraded"===e&&(r.push("⚠️ System is DEGRADED - performance may be impacted"),t>75&&r.push(`Queue utilization ${t.toFixed(1)}% - consider increasing maxQueueSize`),o>80&&r.push(`Memory usage ${o.toFixed(1)}% - consider reducing cache size or memory profile`)),"healthy"===e&&(i<50&&r.push(`Cache hit rate ${i.toFixed(1)}% is low - consider warming more pools`),o>50&&r.push("Memory usage is moderate - monitor for growth trends"),t>50&&r.push("Queue utilization is elevated - monitor for bottlenecks")),r}getSummary(){return{...this.metrics.getSummary(),queueStats:this.eventQueue.getStats(),cacheStats:this.cacheManager.getStats()}}startBackgroundWarming(){if(this.warmingIntervalHandle)return;const e=this.config.refreshIntervalMs;this.warmingIntervalHandle=setInterval(()=>{this.performBackgroundWarming().catch(e=>{this.logger.error("Background warming error:",e)})},e),this.logger.debug(`Background warming started (interval: ${e}ms)`)}stopBackgroundWarming(){this.warmingIntervalHandle&&(clearInterval(this.warmingIntervalHandle),this.warmingIntervalHandle=null,this.logger.debug("Background warming stopped"))}async performBackgroundWarming(){const e=this.cacheManager.getStats();if(0!==e.totalCached)try{await this.cacheManager.refreshWarmAndHotTiers(),this.logger.debug(`Background warming completed: ${e.totalCached} pools in cache (hot: ${e.hotCacheSize}, warm: ${e.warmCacheSize})`)}catch(e){this.logger.debug("Background warming encountered an error:",e)}}async shutdown(){this.stopBackgroundWarming(),await this.unsubscribe(),await this.eventQueue.shutdown(),this.cacheManager.clear(),this.metrics.reset()}setupConnectionMonitoring(){this.socket&&(this.socket.on("disconnect",()=>{this.logger.warn("WebSocket disconnected"),this.notifyError(new Error("WebSocket disconnected")),this.config.autoRecover&&this.attemptReconnection().catch(e=>{this.logger.error("Reconnection failed:",e)})}),this.socket.on("connect_error",e=>{this.logger.error("WebSocket connection error:",e),this.notifyError(e instanceof Error?e:new Error(String(e)))}))}async attemptReconnection(){if(this.logger.debug(`Reconnection attempt ${this.reconnectAttempts+1}/${this.maxReconnectAttempts}`),this.reconnectAttempts>=this.maxReconnectAttempts)return this.logger.error(`Max reconnection attempts (${this.maxReconnectAttempts}) exceeded - performing full reset`),void await this.performFullReset();this.reconnectAttempts<2&&(0===this.reconnectAttempts?this.logger.debug("Tier 1: Quick reconnect"):(this.logger.debug(`Tier 2: Exponential backoff (${this.reconnectDelayMs}ms)`),await new Promise(e=>setTimeout(e,this.reconnectDelayMs)),this.reconnectDelayMs=Math.min(2*this.reconnectDelayMs,3e4)));try{this.socket?.disconnect&&(this.socket.disconnect(),this.logger.debug("Disconnected for reconnection")),this.socket?.connect?.(),this.logger.debug("Reconnection initiated"),this.reconnectAttempts++}catch(e){throw this.logger.error("Failed to initiate reconnection:",e),e}}async performFullReset(){this.logger.warn("Performing full system reset due to connection failures"),this.stopBackgroundWarming();const e=this.getHealth();this.logger.debug("System state before reset:",{status:e.status,queueSize:e.eventProcessing.eventsProcessed,cachedPools:e.pools.totalMonitored,memory:`${e.memory.usedMB}MB / ${e.memory.maxMB}MB`,cacheHitRate:`${e.metrics.cacheHitRate.toFixed(2)}%`}),this.cacheManager.clear(),this.metrics.reset(),this.reconnectAttempts=0,this.reconnectDelayMs=1e3,this.notifyError(new Error("System reset: connection lost and recovery failed - please restart monitoring")),this.logger.info("System reset complete - ready for restart")}setupWebSocketListener(e){if(this.listenerRegistered)return void this.logger.debug("WebSocket listener already registered");const t=this;this.handleSwapEvent=(n,...o)=>{try{const n=Date.now(),i=o[0],r=t.eventExtractor.walkPayloadForSwaps(i,t.seenTransactions);if(0===r.length)return;t.logger.debug(`Extracted ${r.length} swaps from payload`);for(const n of r){if(t.filterSwap(n,e)){t.eventQueue.enqueue(n)||t.logger.debug(`Swap dropped due to queue overflow: ${n.transactionId}`)}}const a=Date.now()-n;t.metrics.recordEventLatency(a)}catch(e){t.logger.error("Error processing WebSocket payload:",e),t.notifyError(e instanceof Error?e:new Error(String(e)))}},this.socket?(this.socket.onAny(this.handleSwapEvent),this.listenerRegistered=!0,this.setupConnectionMonitoring()):this.logger.warn("Socket not available for listener registration"),this.startBackgroundWarming(),this.logger.debug("WebSocket listener registered for all events")}async unsubscribe(){this.stopBackgroundWarming(),this.handleSwapEvent&&this.listenerRegistered&&this.socket&&(this.socket.offAny(this.handleSwapEvent),this.listenerRegistered=!1),this.isActive=!1,this.listeners=[],this.onErrorCallbacks=[],this.logger.debug("Unsubscribed from swap events")}filterSwap(e,t){if(t.tokenFilter){if(!vi.containsToken(e.poolKey,t.tokenFilter))return!1}if(t.pairTokens){const[n,o]=t.pairTokens;if(!vi.containsTokenPair(e.poolKey,n,o))return!1}if(t.feeTierFilter){if(vi.normalizeFee(t.feeTierFilter)!==e.feeTier)return!1}return!t.userFilter||e.user===t.userFilter}async processSwapEvent(e){const t=Date.now();try{const n=this.cacheManager.updatePoolWithSwapDelta(e.poolKey,e.direction,e.amount0,e.amount1,e);e.poolStateUpdated=n;const o=Date.now()-t;this.metrics.recordEventLatency(o);for(const t of this.listeners)try{const n=t(e);n instanceof Promise&&await n}catch(t){this.logger.error(`Listener error for swap ${e.transactionId}:`,t)}}catch(t){this.logger.error(`Failed to process swap ${e.transactionId}:`,t),this.notifyError(t instanceof Error?t:new Error(String(t)))}}notifyError(e){for(const t of this.onErrorCallbacks)try{t(e)}catch(e){this.logger.error("Error in error callback:",e)}}determineHealthStatus(e,t){return!this.socket?.connected||e.queueSize>.9*this.config.maxQueueSize?"failed":e.queueSize>.75*this.config.maxQueueSize||t.memoryUsedMB>.9*this.getMaxMemoryMB()?"degraded":"healthy"}getMaxMemoryMB(){switch(this.config.memoryProfile){case"conservative":return 55;case"aggressive":return 530;default:return 250}}applyDefaults(e){return{memoryProfile:e.memoryProfile??"moderate",maxPools:e.maxPools??500,softLimit:e.softLimit??200,preloadTopN:e.preloadTopN??200,warmingTimeoutMs:e.warmingTimeoutMs??3e4,refreshIntervalMs:e.refreshIntervalMs??3e5,maxQueueSize:e.maxQueueSize??1e4,batchSize:e.batchSize??100,maxConcurrent:e.maxConcurrent??10,autoRecover:e.autoRecover??!0,maxParallelRefetch:e.maxParallelRefetch??20,enableDeltaOptimization:e.enableDeltaOptimization??!0,enableOfflineQuotes:e.enableOfflineQuotes??!0,metricsEnabled:e.metricsEnabled??!0,debug:e.debug??!1}}}class Ni{constructor(e){this.map=new Map,this.maxSize=e}has(e){return this.map.has(e)}add(e){if(this.map.has(e))return this.map.delete(e),this.map.set(e,Date.now()),this;if(this.map.size>=this.maxSize){const e=this.map.keys().next().value;this.map.delete(e)}return this.map.set(e,Date.now()),this}delete(e){return this.map.delete(e)}clear(){this.map.clear()}get size(){return this.map.size}}function Di(e){const t=function(e){const t=qt(e);return t.success?[]:t.errors||["Unknown validation error"]}(e);if(t.length>0)throw new Error(`LaunchTokenData validation failed:\n${t.map(e=>`- ${e}`).join("\n")}`)}const Fi="/api/asset/launchpad-contract/CallNativeTokenIn",xi="/api/asset/launchpad-contract/CallNativeTokenOut",Ci="/api/asset/launchpad-contract/CallMemeTokenIn",$i="/api/asset/launchpad-contract/CallMemeTokenOut";class Li extends s{constructor(e){super(),this.tokenName=e.tokenName,this.tokenSymbol=e.tokenSymbol,this.tokenDescription=e.tokenDescription,this.tokenImage=e.tokenImage,this.preBuyQuantity=e.preBuyQuantity,e.websiteUrl&&(this.websiteUrl=e.websiteUrl),e.telegramUrl&&(this.telegramUrl=e.telegramUrl),e.twitterUrl&&(this.twitterUrl=e.twitterUrl),this.tokenCategory=e.tokenCategory,this.tokenCollection=e.tokenCollection,this.uniqueKey=e.uniqueKey,e.reverseBondingCurveConfiguration&&(this.reverseBondingCurveConfiguration=e.reverseBondingCurveConfiguration)}}function Bi(e){if(!e||"object"!=typeof e)return!1;const t=e;return"number"==typeof t.Status&&void 0!==t.Data&&"object"==typeof t.Data&&null!==t.Data&&"string"==typeof t.Data.calculatedQuantity&&void 0!==t.Data.extraFees&&"object"==typeof t.Data.extraFees&&null!==t.Data.extraFees&&"string"==typeof t.Data.extraFees.reverseBondingCurve&&"string"==typeof t.Data.extraFees.transactionFees}const Ui={NATIVE:"native",EXACT:"exact"},Oi={RECENT:"recent",POPULAR:"popular"},_i={LOCAL:"local",EXTERNAL:"external"};class Mi{static calculateBuyWithExact(e,t){const n=parseFloat(e),o=parseFloat(t),{BASE_PRICE:i,PRICE_SCALING_FACTOR:r,TRADING_FEE_FACTOR:a,GAS_FEE:s}=pi,c=this.roundUp(i*(Math.exp((o+n)*r)-Math.exp(o*r))/r,8),l=new u(c).multipliedBy(a).toFixed();return{amount:c.toString(),reverseBondingCurveFee:"0",transactionFee:l,gasFee:s}}static calculateBuyWithNative(e,t){const n=parseFloat(e),o=parseFloat(t),{BASE_PRICE:i,PRICE_SCALING_FACTOR:r,TRADING_FEE_FACTOR:a,GAS_FEE:s}=pi,c=Math.log(n*r/i+Math.exp(o*r))/r-o,l=new u(c).multipliedBy(a).toFixed();return{amount:c.toString(),reverseBondingCurveFee:"0",transactionFee:l,gasFee:s}}static calculateSellWithExact(e,t,n,o,i){const r=parseFloat(e),a=parseFloat(t),s=parseFloat(n),{BASE_PRICE:c,PRICE_SCALING_FACTOR:l,TRADING_FEE_FACTOR:d,GAS_FEE:h}=pi,g=c*(Math.exp(a*l)-Math.exp((a-r)*l))/l,p=new u(g),m=o+a/s*(i-o),f=p.multipliedBy(m).toFixed(8,u.ROUND_UP),y=p.multipliedBy(d).toFixed();return{amount:g.toString(),reverseBondingCurveFee:f,transactionFee:y,gasFee:h}}static calculateSellWithNative(e,t,n,o,i){const r=parseFloat(e),a=parseFloat(t),s=parseFloat(n),{BASE_PRICE:c,PRICE_SCALING_FACTOR:l,TRADING_FEE_FACTOR:d,GAS_FEE:h}=pi;if(r>=c*(Math.exp(a*l)-1)/l){const e=new u(r),t=o+a/s*(i-o),n=e.multipliedBy(t).toFixed(8,u.ROUND_UP),c=e.multipliedBy(d).toFixed();return{amount:a.toString(),reverseBondingCurveFee:n,transactionFee:c,gasFee:h}}const g=a-Math.log(Math.exp(a*l)-r*l/c)/l,p=new u(r),m=o+a/s*(i-o),f=p.multipliedBy(m).toFixed(8,u.ROUND_UP),y=p.multipliedBy(d).toFixed();return{amount:g.toString(),reverseBondingCurveFee:f,transactionFee:y,gasFee:h}}static roundUp(e,t){const n=Math.pow(10,t);return Math.ceil(e*n)/n}}class Ri{constructor(e,t,n,o,i,r,a="local"){this.http=e,this.tokenResolver=t,this.logger=n,this.bundleHttp=o,this.galaChainHttp=i,this.dexApiHttp=r,this.defaultCalculateAmountMode=a,this.metadataCache=new mi}addIfDefined(e,t,n){return void 0!==n&&(e[t]=n),e}async uploadImageByTokenName(e){const{tokenName:t,options:n}=e;an(t);const o=`${t}.png`;Yn(n.file,o,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&n.file instanceof File)e.append("image",n.file);else{if(!Buffer.isBuffer(n.file))throw ne("file","a File object (browser) or Buffer (Node.js)");{const o=`${n.tokenName||t}.png`,i=new Blob([n.file],{type:"image/png"});e.append("image",i,o)}}const o=await this.http.request({method:"POST",url:`/launchpad/upload-image?tokenName=${encodeURIComponent(n.tokenName||t)}`,data:e,headers:{}});if(!0===o.error||200!==o.status||!o.data?.imageUrl)throw oe(o.message||"Image upload failed - no URL returned",o.status);return o.data.imageUrl}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw ie("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.","FormData");throw e}}async fetchPoolsFromAPI(e){sn(e),e.tokenName&&an(e.tokenName);const t={page:e.page.toString(),limit:e.limit.toString()};void 0!==e.type&&(t.type=e.type),void 0!==e.tokenName&&(t.tokenName=e.tokenName),void 0!==e.search&&(t.search=e.search);const n=M(t),o=await this.http.get("/launchpad/fetch-pool",n);if(!0===o.error||200!==o.status||!o.data)throw oe(o.message||"Failed to fetch pools",o.status);let i=[];const r=(await import("bignumber.js")).default;if(o.data.tokens)if(Array.isArray(o.data.tokens))i=o.data.tokens.map(e=>{const t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",o=!new r(t).isZero()||!new r(n).isZero();return{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:o,createdAt:e.created_at||e.createdAt||""}});else{const e=o.data.tokens,t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",a=!new r(t).isZero()||!new r(n).isZero();i=[{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:a,createdAt:e.created_at||e.createdAt||""}]}else o.data.pools&&Array.isArray(o.data.pools)&&(i=o.data.pools.map(e=>{const t=e.reverseBondingCurveMinFeePortion??"0",n=e.reverseBondingCurveMaxFeePortion??"0",o=!new r(t).isZero()||!new r(n).isZero();return{...e,reverseBondingCurveMinFeePortion:t,reverseBondingCurveMaxFeePortion:n,hasReverseBondingCurveFee:o,createdAt:e.created_at||e.createdAt||""}}));const{extractMetadataFromPoolData:a,isValidPoolForCaching:s}=await Promise.resolve().then(function(){return kr});i.forEach(e=>{if(!s(e))return void this.logger.debug("Skipping pool with invalid structure for caching",e);const t=a(e,this.logger);t&&this.warmCacheFromPoolData(e.tokenName,t)});const c=o.data.count??o.data.total??0,l=o.data.page??e.page??1,u=o.data.limit??e.limit??10,d=u>0?Math.ceil(c/u):1;return{pools:i,page:l,limit:u,total:c,totalPages:d,hasNext:l<d,hasPrevious:l>1}}async _getAmount(e){if(ln(e),!this.galaChainHttp)throw ie("GalaChain client not configured. Direct GalaChain calls require galaChainHttp client.","galaChainHttp");const{endpoint:t,body:n}=((e,t,n,o)=>{if("NATIVE"===e&&"IN"===t)return{endpoint:Fi,body:{vaultAddress:n,tokenQuantity:o,IsPreMint:!1}};if("NATIVE"===e&&"OUT"===t)return{endpoint:xi,body:{vaultAddress:n,tokenQuantity:o,IsPreMint:!1}};if("MEME"===e&&"IN"===t)return{endpoint:Ci,body:{vaultAddress:n,nativeTokenQuantity:o,IsPreMint:!1}};if("MEME"===e&&"OUT"===t)return{endpoint:$i,body:{vaultAddress:n,nativeTokenQuantity:o,IsPreMint:!1}};throw ne("type-method","one of: NATIVE-IN, NATIVE-OUT, MEME-IN, MEME-OUT")})(e.type,e.method,e.vaultAddress,e.amount);try{const e=await this.galaChainHttp.post(t,n);if(!Bi(e))throw oe("Malformed response data from GalaChain gateway");if(1!==e.Status)throw oe(`GalaChain calculation failed with status ${e.Status}`,e.Status);const{calculatedQuantity:o,extraFees:i}=e.Data;return{amount:o,reverseBondingCurveFee:i.reverseBondingCurve,transactionFee:i.transactionFees,gasFee:"1"}}catch(o){throw this.logger.error(`GalaChain ${e.type}-${e.method} operation failed:`,{endpoint:t,requestBody:n,error:o instanceof Error?o.message:o}),o}}async checkPool(e){cn(e),e.tokenName&&an(e.tokenName);const t=M(e),n=await this.http.get("/launchpad/check-pool",t);if(!0===n.error||200!==n.status)throw oe(n.message||"Failed to check pool",n.status);const o=n.data;return e.symbol?o?.isSymbolExist??!1:e.tokenName?o?.isNameExist??!1:o?.exists??!1}async fetchVolumeData(e){if(!On(e))throw new G("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:n,to:o,resolution:i}=e;if(an(t),!n||!o||!i)throw new G("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const r={tokenName:t,from:n,to:o,resolution:i};un(r);const a=M(r),s=await this.http.get("/launchpad/get-graph-data",a);if(!0===s.error||200!==s.status||!s.data)throw oe(s.message||"Failed to fetch graph data",s.status);return{dataPoints:s.data}}async fetchPools(e={}){let t;"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const n={page:e.page||1,limit:e.limit||10};return e.search&&(n.search=e.search),e.tokenName&&(n.tokenName=e.tokenName),t&&(n.type=t),this.fetchPoolsFromAPI(n)}async isTokenNameAvailable(e){try{return!await this.checkPool({tokenName:e})}catch{return!1}}async isTokenSymbolAvailable(e){try{return!await this.checkPool({symbol:e})}catch{return!1}}async calculateBuyAmount(e){if(!e||"object"!=typeof e)throw new G("Invalid options provided. Expected an options object.","options","INVALID_OPTIONS");const{tokenName:t,amount:n,type:o,currentSupply:i}=e,r=e.mode??this.defaultCalculateAmountMode;if("local"!==r&&"external"!==r)throw new G(`Invalid calculation mode "${r}". Must be "local" or "external".`,"mode","INVALID_CALCULATION_MODE");if(!t||"string"!=typeof t)throw new G("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Ui.NATIVE&&o!==Ui.EXACT)throw new G('Type must be either "native" or "exact"',"type","INVALID_TYPE");return"external"===r?this.calculateBuyAmountExternal({tokenName:t,amount:n,type:o}):this.calculateBuyAmountLocal(this.addIfDefined({tokenName:t,amount:n,type:o},"currentSupply",i))}async calculateBuyAmountExternal(e){const{tokenName:t,amount:n,type:o}=e,i=await this.tokenResolver.resolveTokenToVault(t);if(!i)throw new G(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return o===Ui.EXACT?this._getAmount({type:"NATIVE",method:"IN",vaultAddress:i,amount:n}):this._getAmount({type:"MEME",method:"OUT",vaultAddress:i,amount:n})}async calculateSellAmount(e){const{tokenName:t,amount:n,type:o,currentSupply:i,maxSupply:r,reverseBondingCurveMaxFeeFactor:a,reverseBondingCurveMinFeeFactor:s}=e,c=e.mode??this.defaultCalculateAmountMode;if("local"!==c&&"external"!==c)throw new G(`Invalid calculation mode "${c}". Must be "local" or "external".`,"mode","INVALID_CALCULATION_MODE");if(!t||"string"!=typeof t)throw new G("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Ui.EXACT&&o!==Ui.NATIVE)throw new G('Type must be either "exact" or "native"',"type","INVALID_TYPE");if("external"===c)return this.calculateSellAmountExternal({tokenName:t,amount:n,type:o});{const e={tokenName:t,amount:n,type:o,...void 0!==i&&{currentSupply:i},...void 0!==r&&{maxSupply:r},...void 0!==a&&{reverseBondingCurveMaxFeeFactor:a},...void 0!==s&&{reverseBondingCurveMinFeeFactor:s}};return this.calculateSellAmountLocal(e)}}async calculateSellAmountExternal(e){const{tokenName:t,amount:n,type:o}=e,i=await this.tokenResolver.resolveTokenToVault(t);if(!i)throw new G(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return o===Ui.EXACT?this._getAmount({type:"NATIVE",method:"OUT",vaultAddress:i,amount:n}):this._getAmount({type:"MEME",method:"IN",vaultAddress:i,amount:n})}async calculateBuyAmountLocal(e){const{tokenName:t,amount:n,type:o,currentSupply:i}=e;if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Ui.NATIVE&&o!==Ui.EXACT)throw new G('Type must be either "native" or "exact"',"type","INVALID_TYPE");void 0!==i&&hn(i,"currentSupply");const r=!i;if(r&&!t)throw new G("Token name is required when currentSupply is not provided","tokenName","MISSING_TOKEN_NAME");t&&an(t);let a=i;if(r){a=(await this.fetchPoolDetailsForCalculation(t)).currentSupply}return o===Ui.EXACT?Mi.calculateBuyWithExact(n,a):Mi.calculateBuyWithNative(n,a)}async calculateSellAmountLocal(e){const{tokenName:t,amount:n,type:o,currentSupply:i,maxSupply:r,reverseBondingCurveMaxFeeFactor:a,reverseBondingCurveMinFeeFactor:s}=e;if(!n||"string"!=typeof n)throw new G("Amount is required and must be a string","amount","INVALID_AMOUNT");if(o!==Ui.EXACT&&o!==Ui.NATIVE)throw new G('Type must be either "exact" or "native"',"type","INVALID_TYPE");void 0!==i&&hn(i,"currentSupply");const c=!i||!r||void 0===a||void 0===s;if(c&&!t)throw new G("Token name is required when currentSupply, maxSupply, or fee factors are not provided","tokenName","MISSING_TOKEN_NAME");t&&an(t);let l=i,u=r,d=a,h=s;if(c&&t){const e=this.metadataCache.get(t);u=u??this.metadataCache.getMaxSupply(t),d=d??e?.reverseBondingCurveMaxFeeFactor,h=h??e?.reverseBondingCurveMinFeeFactor,l||(l=await this.fetchCurrentSupply(t));if(void 0===d||void 0===h){const e=await this.fetchPoolDetailsForCalculation(t);d=d??e.reverseBondingCurveMaxFeeFactor,h=h??e.reverseBondingCurveMinFeeFactor}}return o===Ui.EXACT?Mi.calculateSellWithExact(n,l,u,h,d):Mi.calculateSellWithNative(n,l,u,h,d)}async calculateBuyAmountForGraduation(e){const t="string"==typeof e?{tokenName:e}:e;if("object"==typeof e&&!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return $n(t,"tokenName")&&Un(t)&&Ln(t,"currentSupply")}(e))throw new Error("Invalid CalculateBuyAmountForGraduationOptions provided");const{tokenName:n,calculateAmountMode:o,currentSupply:i}=t;an(n);const r=await this.tokenResolver.resolveTokenToVault(n);if(!r)throw new G(K(n),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const a=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:r});if(1!==a.Status)throw oe(`Failed to fetch pool details: Status ${a.Status}`,a.Status);const s=a.Data,c=(await import("bignumber.js")).default,l=i??new c(s.maxSupply).minus(s.sellingTokenQuantity).toFixed(),u=s.sellingTokenQuantity;if("0"===u)throw new G(`Token ${n} is already graduated (no tokens remaining in pool)`,"tokenName","ALREADY_GRADUATED");const d={tokenName:n,amount:u,type:"exact",currentSupply:l,...void 0!==o&&{mode:o}};return{...await this.calculateBuyAmount(d),remainingTokens:u}}async launchToken(e){if(!this.bundleHttp)throw ie("Bundle backend client not configured. LaunchToken requires bundleHttp client.","bundleHttp");Di(e);const t=e.preBuyQuantity||"0";if(isNaN(Number(t))||Number(t)<0)throw new G("Pre-buy quantity must be a valid non-negative number string","preBuyQuantity","INVALID_PRE_BUY_QUANTITY");if(e.reverseBondingCurveConfiguration){const{minFeePortion:t,maxFeePortion:n}=e.reverseBondingCurveConfiguration,o=Number(t),i=Number(n);if(isNaN(o)||isNaN(i))throw new G("Reverse bonding curve fees must be valid numbers","reverseBondingCurveConfiguration","INVALID_BONDING_CURVE_CONFIG");if(o<.1)throw new G("Minimum fee must be >= 0.1","reverseBondingCurveConfiguration","INVALID_BONDING_CURVE_CONFIG");if(i>.5)throw new G("Maximum fee must be <= 0.5","reverseBondingCurveConfiguration","INVALID_BONDING_CURVE_CONFIG");if(i<o)throw new G("Maximum fee must be >= minimum fee","reverseBondingCurveConfiguration","INVALID_BONDING_CURVE_CONFIG")}let n="";if(e.tokenImage)if(e.tokenImage instanceof File||Buffer.isBuffer(e.tokenImage)){const t=await this.uploadImageByTokenName({tokenName:e.tokenName,options:{file:e.tokenImage,tokenName:e.tokenName}});if(!t)throw oe("Image upload failed: No URL returned");n=t}else"string"==typeof e.tokenImage&&(n=e.tokenImage);const o=`galaswap - operation - ${N()}-${Date.now()}-${this.http.getAddress()}`,i={tokenName:e.tokenName.trim(),tokenSymbol:e.tokenSymbol.trim().toUpperCase(),tokenDescription:e.tokenDescription.trim(),tokenImage:n.trim(),preBuyQuantity:t.toString(),tokenCategory:e.tokenCategory||"Unit",tokenCollection:e.tokenCollection||"Token",uniqueKey:o};e.websiteUrl?.trim()&&(i.websiteUrl=e.websiteUrl.trim()),e.telegramUrl?.trim()&&(i.telegramUrl=e.telegramUrl.trim()),e.twitterUrl?.trim()&&(i.twitterUrl=e.twitterUrl.trim()),i.reverseBondingCurveConfiguration={minFeePortion:e.reverseBondingCurveConfiguration?.minFeePortion?.toString()||"0.1",maxFeePortion:e.reverseBondingCurveConfiguration?.maxFeePortion?.toString()||"0.5"};const a=new Li(i),s=await this.http.signWithGalaChain("CreateSale",a,r.SIGN_TYPED_DATA),{signature:c,types:l,domain:u,prefix:d}=s,h={tokenName:a.tokenName,tokenSymbol:a.tokenSymbol,tokenDescription:a.tokenDescription,tokenImage:a.tokenImage,preBuyQuantity:a.preBuyQuantity,...a.websiteUrl&&{websiteUrl:a.websiteUrl},...a.telegramUrl&&{telegramUrl:a.telegramUrl},...a.twitterUrl&&{twitterUrl:a.twitterUrl},tokenCategory:a.tokenCategory,tokenCollection:a.tokenCollection,uniqueKey:a.uniqueKey,signature:c,types:l,domain:u,...d&&{prefix:d},...a.reverseBondingCurveConfiguration&&{reverseBondingCurveConfiguration:a.reverseBondingCurveConfiguration}},g=`${e.tokenName.trim()}$Unit$none$none`,p="GALA$Unit$none$none";let m;if(parseFloat(t)>0){const e=`$service$${g}$launchpad`;m=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${p}$${e}`,`$tokenBalance$${p}$${e}`]}else{const e=`$service$${g}$launchpad`;m=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`]}const f={signedDto:h,stringsInstructions:m,method:"CreateSale"},y=await this.bundleHttp.post("/bundle",f);if(y.error||!y.data)throw oe(y.message||"Token launch failed");return y.data}async fetchTokenDistribution(e){if(!e)throw te("tokenName","Token name");an(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");this.metadataCache.set(e,{vaultAddress:t});const n=encodeURIComponent(t),o=await this.http.get(`/holders/${n}`);if(!0===o.error||200!==o.status||!o.data)throw oe(o.message||"Failed to fetch token distribution",o.status);const i=o.data;if(!Array.isArray(i))throw oe("Invalid API response: expected array of holders",o.status);for(const e of i){if(!e.owner||"string"!=typeof e.owner)throw oe("Invalid holder data: missing or invalid owner field",o.status);if(!e.quantity||"string"!=typeof e.quantity)throw oe("Invalid holder data: missing or invalid quantity field",o.status);const t=parseFloat(e.quantity);if(isNaN(t)||!isFinite(t))throw oe(`Invalid holder quantity: "${e.quantity}"`,o.status)}const r=i.reduce((e,t)=>e.plus(t.quantity),new u(0));return{holders:i.map(e=>{const t=new u(e.quantity),n=r.isZero()?0:t.dividedBy(r).multipliedBy(100).toNumber();return{address:e.owner,balance:e.quantity,percentage:n}}),totalSupply:r.toFixed(),totalHolders:i.length,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw te("tokenName","Token name");an(e);const t=await this.http.get("/launchpad/get-badge/",{tokenName:e});if(t.error||!t.data)throw oe(t.message||"Failed to fetch token badges");return{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadgeByTokenName(e){const{tokenName:t,badgeType:n,badgeName:o}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const i=("volume"===n?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===o);return i?.isActive||!1}catch{return!1}}async calculateInitialBuyAmount(e){if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return"string"==typeof t.nativeTokenQuantity&&(void 0===t.vaultAddress||"string"==typeof t.vaultAddress)}(e))throw new G("Invalid pre-mint calculation data","data","INVALID_PRE_MINT_DATA");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not available. Please initialize SDK with galaChainBaseUrl.","galaChainHttp");try{const t={vaultAddress:"service|testToken",nativeTokenQuantity:e.nativeTokenQuantity,IsPreMint:!0},n=await this.galaChainHttp.post("/api/asset/launchpad-contract/CallMemeTokenOut",t);if(!Bi(n))throw oe("Malformed response data from GalaChain gateway");if(1!==n.Status)throw oe(`GalaChain calculation failed with status ${n.Status}`,n.Status);const{calculatedQuantity:o,extraFees:i}=n.Data;return{amount:o,reverseBondingCurveFee:i.reverseBondingCurve,transactionFee:i.transactionFees,gasFee:"1"}}catch(e){if(e instanceof Error){const t=new Error(`Pre-mint calculation failed: ${e.message}`);throw e.stack&&(t.stack=e.stack),t}throw new Error(`Pre-mint calculation failed: ${String(e)}`)}}async fetchPoolDetailsForCalculation(e){const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const n=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:t});if(1!==n.Status)throw oe(`Failed to fetch pool details: Status ${n.Status}`,n.Status);const o=n.Data,i=new(0,(await import("bignumber.js")).default)(o.maxSupply).minus(o.sellingTokenQuantity).toFixed(),r=o.sellingTokenQuantity,a=o.maxSupply;let s=.5,c=0;o.reverseBondingCurveConfiguration?(s=parseFloat(o.reverseBondingCurveConfiguration.maxFeePortion),c=parseFloat(o.reverseBondingCurveConfiguration.minFeePortion)):this.logger.debug(`Pool details missing reverseBondingCurveConfiguration for token ${e}, using defaults (min: 0.0, max: 0.5)`);const l=s-c;return this.metadataCache.set(e,{maxSupply:a,reverseBondingCurveMaxFeeFactor:s,reverseBondingCurveMinFeeFactor:c,reverseBondingCurveNetFeeFactor:l}),{currentSupply:i,remainingTokens:r,maxSupply:a,reverseBondingCurveMaxFeeFactor:s,reverseBondingCurveMinFeeFactor:c,reverseBondingCurveNetFeeFactor:l}}async fetchCurrentSupply(e){an(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new G(K(e),"tokenName","VAULT_NOT_FOUND");if(!this.galaChainHttp)throw ie("GalaChain HTTP client not configured");const n=await this.galaChainHttp.post("/api/asset/launchpad-contract/FetchSaleDetails",{vaultAddress:t});if(1!==n.Status)throw oe(`Failed to fetch pool details: Status ${n.Status}`,n.Status);const o=n.Data,i=new(0,(await import("bignumber.js")).default)(o.maxSupply).minus(o.sellingTokenQuantity).toFixed(),r=o.maxSupply;return this.metadataCache.set(e,{maxSupply:r}),i}getAddress(){return this.http.getAddress()}formatAddressForBackend(e){return dn(e)}validateTokenName(e){return an(e)}validatePagination(e){return sn(e)}async fetchTokenPrice(e){if(!this.dexApiHttp)throw ie("DEX API client not configured. Token price fetching requires dexApiHttp client.","dexApiHttp");if(!e||Array.isArray(e)&&0===e.length)throw te("symbols","At least one symbol");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.dexApiHttp.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),n=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&n.push({symbol:e.symbol,price:e.currentPrices.usd})}),n}catch(e){throw oe(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`,void 0,e instanceof Error?e:void 0)}}async fetchLaunchpadTokenSpotPrice(e){const t="string"==typeof e?{tokenName:e}:e;if("object"==typeof e&&!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return $n(t,"tokenName")&&Un(t)&&Ln(t,"currentSupply")}(e))throw new Error("Invalid FetchLaunchpadTokenSpotPriceOptions provided");const{tokenName:n,calculateAmountMode:o,currentSupply:i}=t;if(!n||"string"!=typeof n)throw te("tokenName","Token name (string)");try{const e={tokenName:n,amount:"1",type:"native",...void 0!==o&&{mode:o},...void 0!==i&&{currentSupply:i}},t=await this.calculateBuyAmount(e),r=(await this.fetchTokenPrice("GALA"))[0];if(!r)throw oe("GALA price not available");const a=Number(t.amount);if(a<=0)throw new G(`Invalid token amount calculation: ${a}`,"amount","INVALID_CALCULATION");const s=r.price/a;return{symbol:n.toUpperCase(),price:s}}catch(e){if(e instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${n}: ${e.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${n}: ${String(e)}`)}}warmCacheFromPoolData(e,t){this.metadataCache.warmFromPoolData(e,t)}getCacheStats(){return this.metadataCache.stats()}clearCache(e){this.metadataCache.clear(e)}}const qi={PROD:{launchpadBaseUrl:"https://lpad-backend-prod1.defi.gala.com",galaChainBaseUrl:"https://gateway-mainnet.galachain.com",bundleBaseUrl:"https://bundle-backend-prod1.defi.gala.com",webSocketUrl:"https://bundle-backend-prod1.defi.gala.com",dexApiBaseUrl:"https://dex-api-platform-dex-prod-gala.gala.com",dexBackendBaseUrl:"https://dex-backend-prod1.defi.gala.com",launchpadFrontendUrl:"https://lpad-frontend-prod1.defi.gala.com"},STAGE:{launchpadBaseUrl:"https://lpad-backend-dev1.defi.gala.com",galaChainBaseUrl:"https://galachain-gateway-chain-platform-stage-chain-platform-eks.stage.galachain.com",bundleBaseUrl:"https://bundle-backend-dev1.defi.gala.com",webSocketUrl:"https://bundle-backend-dev1.defi.gala.com",dexApiBaseUrl:"https://dex-api-platform-dex-stage-gala.gala.com",dexBackendBaseUrl:"https://dex-backend-dev1.defi.gala.com",launchpadFrontendUrl:"https://lpad-frontend-test1.defi.gala.com"}};function Ki(e){return qi[e]}class Gi extends Error{constructor(e,t){super(e),this.cause=t,this.name="WebSocketError"}}class Wi extends Gi{constructor(e,t){super(`WebSocket confirmation timeout for transaction ${e} after ${t}ms`),this.name="WebSocketTimeoutError"}}class ji extends Error{constructor(e,t,n){super(`Transaction ${e} failed with status: ${t}${n?` - ${n}`:""}`),this.transactionId=e,this.status=t,this.name="TransactionFailedError"}}function Hi(e,t){if(!e)throw new Gi(`Invalid WebSocket response received for transaction ${t}: response is null or undefined`);if("object"!=typeof e)throw new Gi(`Invalid WebSocket response received for transaction ${t}: expected object, got ${typeof e}`);if(!Object.prototype.hasOwnProperty.call(e,"status")&&!Object.prototype.hasOwnProperty.call(e,"Status"))throw new Gi(`Invalid WebSocket response received for transaction ${t}: missing status field`)}function zi(e,t,n,o){Hi(e,t);const i=e,r=i.data||{};if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(void 0!==t.inputQuantity&&"string"!=typeof t.inputQuantity||void 0!==t.outputQuantity&&"string"!=typeof t.outputQuantity||void 0!==t.totalFees&&"string"!=typeof t.totalFees||void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress)}(r))throw new Gi(`Invalid trade data received for transaction ${t}`);const a={transactionId:t,type:n,method:"native"===o.type?"native":"exact",inputAmount:r.inputQuantity||o.amount,outputAmount:r.outputQuantity||o.expectedAmount||"0",totalFees:r.totalFees||"0",tokenName:o.tokenName,vaultAddress:r.vaultAddress||"",timestamp:Date.now()};return void 0!==i.blockHash&&(a.blockHash=i.blockHash),void 0!==i.gasUsed&&(a.gasUsed=i.gasUsed),void 0!==o.slippageToleranceFactor&&(a.slippageTolerance=o.slippageToleranceFactor),a}const Vi="4.0.4-beta.0";class Qi{constructor(e){this.logger=e||new _({debug:!1,context:"LiquidityEventExtractor"})}walkPayloadForLiquidityEvents(e,t){const n=[],o=new WeakSet,i=(e,r=0)=>{if(r>50)this.logger.debug("Payload nesting exceeded maximum depth of 50");else if(e&&"string"!=typeof e&&"object"==typeof e){if(o.has(e))return;o.add(e);const a=this.extractLiquidityFromObject(e);a&&!t.has(a.transactionId)&&(n.push(a),t.add(a.transactionId));for(const t of Object.values(e))i(t,r+1)}};return i(e,0),n}extractLiquidityFromObject(e){const t=this.extractTransactionId(e);if(!t)return null;const n=e.Data,o=n&&"object"==typeof n&&!Array.isArray(n)?n:e,i=this.extractPositionId(o),r=this.extractPoolHash(o),a=this.extractAmounts(o),s=this.extractUserAddress(o),c=this.extractPoolFee(o);if(!(i&&r&&a&&s&&null!==c))return null;const l=this.extractPoolAlias(o),u=this.extractUserBalanceDelta(o),d=this.extractTimestamp(o),h=u?.token0Balance?.collection,g=u?.token1Balance?.collection,p={transactionId:t,positionId:i,poolHash:r,poolFee:c,amounts:a,userAddress:s};return void 0!==h&&(p.token0=h),void 0!==g&&(p.token1=g),void 0!==d&&(p.timestamp=d),void 0!==l&&(p.poolAlias=l),void 0!==u&&(p.userBalanceDelta=u),p}extractTransactionId(e){const t=["transactionId","txId","tx_id","hash","txHash","id"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPositionId(e){const t=["positionId","position_id","tokenId","nftId"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolHash(e){const t=["poolHash","pool_hash","poolId","pool"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolAlias(e){const t=["poolAlias","pool_alias"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}}extractAmounts(e){const t=e.amounts;if(Array.isArray(t)&&t.length>=2){const e=String(t[0]).trim(),n=String(t[1]).trim();if(e&&n)return[e,n]}const n=e.amount0||e.amount0Desired,o=e.amount1||e.amount1Desired;return void 0!==n&&void 0!==o?[String(n),String(o)]:null}extractUserAddress(e){const t=["userAddress","user","owner","from","sender","wallet","address"];for(const n of t){const t=e[n];if("string"==typeof t&&t.trim())return t}return null}extractPoolFee(e){const t=["poolFee","fee","feeTier","feeTierBps"];for(const n of t){const t=e[n];if("number"==typeof t)return this.normalizeFee(t);if("string"==typeof t){const e=Number(t);if(!Number.isNaN(e))return this.normalizeFee(e)}}return null}normalizeFee(e){return 1===e||1e4===e?1e4:.3===e||3e3===e?3e3:.05===e||500===e?500:Number.isInteger(e)?e:e<1?Math.round(1e4*e):e}extractTimestamp(e){const t=["timeStamp","timestamp","time","createdAt","date"];for(const n of t){const t=e[n];if("number"==typeof t)return t;if("string"==typeof t){const e=new Date(t).getTime();if(!Number.isNaN(e))return e}}}extractUserBalanceDelta(e){const t=e.userBalanceDelta;if(!t||"object"!=typeof t)return;const n=t,o=this.extractBalanceObject(n.token0Balance),i=this.extractBalanceObject(n.token1Balance);if(!o&&!i)return;const r={};return void 0!==o&&(r.token0Balance=o),void 0!==i&&(r.token1Balance=i),r}extractBalanceObject(e){if(!e||"object"!=typeof e)return;const t=e,n=t.collection,o=t.category,i=t.type,r=t.additionalKey,a=t.quantity,s=t.owner;return"string"==typeof n&&"string"==typeof o&&"string"==typeof i&&"string"==typeof r&&"string"==typeof a&&"string"==typeof s?{collection:n,category:o,type:i,additionalKey:r,quantity:a,owner:s}:void 0}}class Xi{constructor(t){this.wallet=t.wallet;let n=null,o="STAGE";t.env?(o=t.env,n=Ki(t.env)):t.baseUrl?.includes("prod")?(o="PROD",n=Ki("PROD")):(o="STAGE",n=Ki("STAGE")),this.environment=o,this.config={baseUrl:n.launchpadBaseUrl,galaChainBaseUrl:n.galaChainBaseUrl,bundleBaseUrl:n.bundleBaseUrl,webSocketUrl:n.webSocketUrl,dexApiBaseUrl:n.dexApiBaseUrl,dexBackendBaseUrl:n.dexBackendBaseUrl,launchpadFrontendUrl:n.launchpadFrontendUrl,timeout:3e4,debug:!1,...t},this.logger=new _({debug:this.config.debug??!1,context:"LaunchpadSDK"}),this.validateConfiguration(),this.slippageToleranceFactor=void 0===t.slippageToleranceFactor?Xi.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR:this.parseSlippageToleranceFactor(t.slippageToleranceFactor),this.maxAcceptableReverseBondingCurveFeeSlippageFactor=void 0===t.maxAcceptableReverseBondingCurveFeeSlippageFactor?Xi.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR:this.parseFeeSlippageFactor(t.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.calculateAmountMode=t.calculateAmountMode||Xi.DEFAULT_CALCULATE_AMOUNT_MODE,this.pricingConcurrency=t.pricingConcurrency||5,this.auth=new O({wallet:t.wallet,messagePrefix:"Create a GalaChain Wallet"}),this.http=new R(this.auth,this.config),this.galaChainHttp=new R(this.auth,{...this.config,baseUrl:this.config.galaChainBaseUrl}),this.bundleHttp=new R(this.auth,{...this.config,baseUrl:this.config.bundleBaseUrl}),this.dexApiHttp=new R(this.auth,{...this.config,baseUrl:this.config.dexApiBaseUrl}),this.dexBackendHttp=new R(this.auth,{...this.config,baseUrl:this.config.dexBackendBaseUrl}),this.galaChainPublicAxios=e.create({baseURL:this.config.galaChainBaseUrl,timeout:this.config.timeout||3e4,headers:{"Content-Type":"application/json",Accept:"application/json"}}),this.cache=new mi(t.debug||!1),this.launchpadService=new to(this.http),this.tokenResolverService=new ti(this.launchpadService.poolService),this.launchpadAPI=new Ri(this.http,this.tokenResolverService,this.logger,this.bundleHttp,this.galaChainHttp,this.dexApiHttp,this.calculateAmountMode),this.galaChainService=new Oo(this.galaChainHttp,t.wallet,this.tokenResolverService,t.debug||!1,this.galaChainPublicAxios),this.dexService=new _o(this.dexBackendHttp,this.cache,this.galaChainService,t.debug||!1),this.bundleService=new Xo(this.bundleHttp,this.tokenResolverService,this.config.debug||!1,t.wallet,t.wallet?this.getAddress():void 0,this.slippageToleranceFactor,this.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.websocketService=new ei({url:this.config.webSocketUrl},this.config.debug),this.priceHistoryService=new ii(this.dexBackendHttp,this.config.debug||!1,this.tokenResolverService),this.dexQuoteService=new wi(this.galaChainHttp,this.config.galaChainBaseUrl,t.debug||!1,t.dexQuoteNetworkTimeout||3e4),this.gswapService=new gi({privateKey:t.wallet?.privateKey,getWalletAddress:()=>this.wallet?this.getAddress():void 0,gatewayBaseUrl:this.config.galaChainBaseUrl,bundlerBaseUrl:this.config.bundleBaseUrl,galaChainBaseUrl:this.config.galaChainBaseUrl,dexBackendBaseUrl:this.config.dexBackendBaseUrl,dexBackendHttp:this.dexBackendHttp},this.websocketService,this.dexQuoteService),this.dexPoolService=new fi(this.dexBackendHttp,this.config.dexBackendBaseUrl,this.gswapService,this.pricingConcurrency,t.debug||!1)}createOverrideSdk(e){if(!e||"string"!=typeof e)throw ie("Invalid privateKey: must be a non-empty string","privateKey");if(!e.match(/^0x[a-fA-F0-9]{64}$/))throw ie('Invalid privateKey format: must be "0x" followed by 64 hexadecimal characters',"privateKey");const t=new n(e),o={...this.config,wallet:t};return new Xi(o)}getAddress(){return this.validateWallet(),this.auth.getAddress()}getEthereumAddress(){return this.validateWallet(),this.wallet.address}validateWallet(){if(!this.wallet)throw new G("Wallet is required for this operation",void 0,"WALLET_REQUIRED");return this.wallet}setWallet(e){if(!e||"object"!=typeof e||!("address"in e))throw new G("Invalid wallet: must be an ethers Wallet instance, received "+typeof e,"wallet","INVALID_WALLET");this.wallet=e,this.auth.setWallet(e)}getWallet(){return this.wallet}hasWallet(){return void 0!==this.wallet}getConfig(){const{wallet:e,...t}=this.config;return{...t,environment:this.environment,slippageToleranceFactor:this.slippageToleranceFactor,maxAcceptableReverseBondingCurveFeeSlippageFactor:this.maxAcceptableReverseBondingCurveFeeSlippageFactor,calculateAmountMode:this.calculateAmountMode}}getVersion(){return Vi}getUrlByTokenName(e){const t=this.config.launchpadFrontendUrl;if(!t)throw ie("launchpadFrontendUrl not configured in SDK","launchpadFrontendUrl");return`${t.replace(/\/$/,"")}/buy-sell/${e}`}async fetchPools(e){const t=await this.launchpadService.fetchPools(e||{});return await this.warmCacheFromPools(t.pools),t}async fetchAllPools(e){const t=await this.launchpadService.fetchAllPools(e);return await this.warmCacheFromPools(t.pools),t}async fetchDexPools(e={}){return this.dexPoolService.fetchDexPools(e)}async fetchAllDexPools(e={}){return this.dexPoolService.fetchAllDexPools(e)}async fetchCompositePoolData(e){return this.dexQuoteService.fetchCompositePoolData(e)}async calculateDexPoolQuoteExactAmountLocal(e){return this.dexQuoteService.calculateDexPoolQuoteExactAmountLocal(e)}async calculateDexPoolQuoteExactAmountExternal(e){return this.dexQuoteService.calculateDexPoolQuoteExactAmountExternal(e)}async calculateDexPoolQuoteExactAmount(e,t="local"){return this.dexQuoteService.calculateDexPoolQuoteExactAmount(e,t)}async fetchTokenDistribution(e){return this.launchpadService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.launchpadService.fetchTokenBadges(e)}async fetchTokenPrice(e){const{tokenName:t,tokenId:n,currentSupply:o,calculateAmountMode:i}=e;if(t&&!n){const e={tokenName:t};return o&&(e.currentSupply=o),i&&(e.calculateAmountMode=i),this.fetchLaunchpadTokenSpotPrice(e)}if(n&&!t)try{return await this.dexService.fetchTokenPrice({tokenId:n})}catch(e){const t=function(e){if(Wn(e)&&e.response)return e.response.status}(e);if(400===t||404===t){this.logger.debug(`DEX spot price not available (HTTP ${t}) for tokenId, attempting launchpad fallback`);try{const t=(await this.fetchTokenDetails(n)).name.trim().toLowerCase();if(!/^[a-z0-9]{3,20}$/.test(t))throw this.logger.error(`Token name extracted from GalaChain doesn't match launchpad format: "${t}"`),e;this.logger.debug(`Falling back to launchpad pricing using extracted token name: "${t}"`);const r={tokenName:t};return o&&(r.currentSupply=o),i&&(r.calculateAmountMode=i),this.fetchLaunchpadTokenSpotPrice(r)}catch(t){throw this.logger.error(`Launchpad fallback failed: ${Gn(t)}`),e}}throw e}if(!t&&!n)throw te("tokenName or tokenId","Either tokenName (for launchpad tokens) or tokenId (for DEX tokens) is required");throw new G("tokenName and tokenId are mutually exclusive - provide only one","params","INVALID_PARAMS")}async fetchGalaPrice(){return this.fetchTokenPrice({tokenId:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none"}})}async fetchLaunchpadTokenSpotPrice(e){const t="string"==typeof e?{tokenName:e}:e;return this.dexService.fetchLaunchpadTokenSpotPrice(t.tokenName,e=>this.launchpadAPI.calculateBuyAmount(e),e=>this.fetchPoolDetails(e))}async fetchTokenDetails(e){return this.dexService.fetchTokenDetails(e)}async fetchAllDexSeasons(){return this.dexService.fetchAllDexSeasons()}async fetchCurrentDexSeason(){return this.dexService.fetchCurrentDexSeason()}async fetchDexLeaderboardBySeasonId(e){return this.dexService.fetchDexLeaderboardBySeasonId(e)}async fetchCurrentDexLeaderboard(){return this.dexService.fetchCurrentDexLeaderboard()}async fetchDexAggregatedVolumeSummary(){return this.dexService.fetchDexAggregatedVolumeSummary()}async fetchLaunchTokenFee(){return this.galaChainService.fetchLaunchTokenFee()}async fetchTokenClassesWithSupply(e){return this.galaChainService.fetchTokenClassesWithSupply(e)}async fetchPoolDetails(e){const t=await this.resolveVaultAddress(e);if(!t)throw new Error(K(e));const n=(await this.galaChainService.fetchPoolDetails({vaultAddress:t})).Data,o=await this.launchpadAPI.fetchPoolDetailsForCalculation(e);return n.currentSupply=o.currentSupply,n.reverseBondingCurveMaxFeeFactor=o.reverseBondingCurveMaxFeeFactor,n.reverseBondingCurveMinFeeFactor=o.reverseBondingCurveMinFeeFactor,n.reverseBondingCurveNetFeeFactor=o.reverseBondingCurveNetFeeFactor,n.tokenName=e,n}async fetchPoolDetailsForCalculation(e){return this.launchpadAPI.fetchPoolDetailsForCalculation(e)}async isTokenGraduated(e){return(await this.fetchPoolDetails(e)).isGraduated}async fetchVolumeData(e){return this.launchpadService.fetchVolumeData(e)}async fetchTrades(e){return this.launchpadService.fetchTrades(e)}async fetchGalaBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n=t(e)||this.getAddress();return this.galaChainService.fetchGalaBalance({owner:n,collection:"GALA",category:"Unit",additionalKey:"none",type:"none",instance:"0"})}async fetchTokenBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n=t(e.address);if(e.tokenId){const{normalizeToTokenInstanceKey:t}=await Promise.resolve().then(function(){return Io}),o=t(e.tokenId),{collection:i,category:r,type:a,additionalKey:s}=o;return this.galaChainService.fetchTokenBalance({owner:n,collection:i,category:r,additionalKey:s,type:a,instance:"0"},e.withExpired??!1)}if(e.tokenName){const t=(await this.fetchTokensHeld({tokenName:e.tokenName,page:1,limit:1,...n&&{address:n}})).tokens[0];return t?{quantity:t.quantity,collection:t.collection||"Token",category:"Unit",tokenId:`${t.collection||"Token"}|Unit|${t.symbol}|none`,symbol:t.symbol,name:t.name}:null}throw te("tokenId or tokenName","Either tokenId or tokenName")}async fetchLockedBalance(e){const t=await this.fetchTokenBalance(e);if(!t)return null;const n="lockedHolds"in t||"lockedQuantity"in t;return{tokenId:t.tokenId,lockedQuantity:n?t.lockedQuantity??"0":"0",lockedHolds:n?t.lockedHolds??[]:[]}}async fetchAvailableBalance(e){const t=await this.fetchTokenBalance(e);if(!t)return null;const n="availableQuantity"in t;return{tokenId:t.tokenId,availableQuantity:n?t.availableQuantity??String(t.quantity):String(t.quantity),totalQuantity:String(t.quantity)}}async calculateBuyAmount(e){return this.launchpadAPI.calculateBuyAmount(e)}async calculateSellAmount(e){return this.launchpadAPI.calculateSellAmount(e)}async calculateBuyAmountLocal(e){return this.launchpadAPI.calculateBuyAmountLocal(e)}async calculateSellAmountLocal(e){return this.launchpadAPI.calculateSellAmountLocal(e)}async calculateBuyAmountExternal(e){return this.launchpadAPI.calculateBuyAmountExternal(e)}async calculateSellAmountExternal(e){return this.launchpadAPI.calculateSellAmountExternal(e)}async calculateBuyAmountForGraduation(e){return this.launchpadAPI.calculateBuyAmountForGraduation(e)}async graduateToken(e){const{tokenName:t,slippageToleranceFactor:n,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,privateKey:i,calculateAmountMode:r,currentSupply:a}=e;let s=t;void 0===r&&void 0===a||(s={tokenName:t,...void 0!==r&&{calculateAmountMode:r},...void 0!==a&&{currentSupply:a}});const c=await this.calculateBuyAmountForGraduation(s),l={tokenName:t,amount:c.remainingTokens,type:"exact",expectedAmount:c.amount,maxAcceptableReverseBondingCurveFee:c.reverseBondingCurveFee,slippageToleranceFactor:this.slippageToleranceFactor};return void 0!==n&&(l.slippageToleranceFactor=n),void 0!==o&&(l.maxAcceptableReverseBondingCurveFeeSlippageFactor=o),void 0!==i&&(l.privateKey=i),await this.buy(l)}async calculateInitialBuyAmount(e){const t={nativeTokenQuantity:e};return this.launchpadAPI.calculateInitialBuyAmount(t)}async buy(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.buy(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=(await this.bundleService.buyToken(e)).data,n=t?.transactionId;if(!n)throw re("No transaction ID returned from buy operation");return this.waitForConfirmation(n,t=>zi(t,n,"buy",e))}async sell(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.sell(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=(await this.bundleService.sellToken(e)).data,n=t?.transactionId;if(!n)throw re("No transaction ID returned from sell operation");return this.waitForConfirmation(n,t=>zi(t,n,"sell",e))}async getBundlerTransactionResult(e){return this.bundleService.getBundlerTransactionResult(e)}async launchToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.launchToken(o)}this.validateWallet(),await this.ensureWebSocketConnection();const t=await this.launchpadAPI.launchToken(e);return this.waitForConfirmation(t,n=>{Hi(n,t);const o=n?.data||{};if(!function(e){if(!e||"object"!=typeof e)return!1;const t=e;return!(void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress||void 0!==t.tokenStringKey&&"string"!=typeof t.tokenStringKey||void 0!==t.creatorAddress&&"string"!=typeof t.creatorAddress)}(o))throw new Gi(`Invalid launch data received for transaction ${t}`);const i={transactionId:t,vaultAddress:o.vaultAddress||"",tokenStringKey:o.tokenStringKey||"",tokenName:e.tokenName,tokenSymbol:e.tokenSymbol,creatorAddress:o.creatorAddress||this.getAddress(),timestamp:Date.now(),...n.blockHash&&{blockHash:n.blockHash},...n.gasUsed&&{gasUsed:n.gasUsed}};return"string"==typeof e.tokenImage&&(i.tokenImage=e.tokenImage),void 0!==e.preBuyQuantity&&(i.preBuyQuantity=e.preBuyQuantity),i.vaultAddress&&this.tokenResolverService.set(e.tokenName,i.vaultAddress),i})}async uploadTokenImage(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.uploadTokenImage(o)}return this.validateWallet(),this.launchpadService.uploadImageByTokenName(e)}async isTokenNameAvailable(e){return this.launchpadService.isTokenNameAvailable(e)}async isTokenSymbolAvailable(e){return this.launchpadService.isTokenSymbolAvailable(e)}async fetchProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n=t(e)||this.getAddress();return this.launchpadService.fetchProfile(n)}async updateProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n={...e,address:t(e.address)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.updateProfile(o)}return this.validateWallet(),this.launchpadService.updateProfile(n)}async uploadProfileImage(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n={...e,address:t(e.address)||this.getAddress()};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.uploadProfileImage(o)}return this.validateWallet(),this.launchpadService.uploadProfileImage(n)}async fetchTokensHeld(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n=t(e?.address)||this.getAddress(),o={page:e?.page||1,limit:e?.limit||10,address:n};return e?.tokenName&&(o.tokenName=e.tokenName),e?.search&&(o.search=e.search),this.launchpadService.fetchTokensHeld(o)}async fetchTokensCreated(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n={type:"DEFI",address:t(e?.address)||this.getAddress(),page:e?.page||1,limit:e?.limit||10};return e?.tokenName&&(n.tokenName=e.tokenName),e?.search&&(n.search=e.search),this.launchpadService.fetchTokenList(n)}async fetchPriceHistory(e){return this.priceHistoryService.fetchPriceHistory(e)}async fetchAllPriceHistory(e){return this.priceHistoryService.fetchAllPriceHistory(e)}async transferGala(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n={...e,recipientAddress:t(e.recipientAddress)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.transferGala(o)}return this.validateWallet(),this.galaChainService.transferGala(n)}async transferToken(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn}),n={...e,to:t(e.to)};if(n.privateKey){const e=this.createOverrideSdk(n.privateKey),{privateKey:t,...o}=n;return e.transferToken(o)}return this.validateWallet(),this.galaChainService.transferToken(n)}async resolveTokenClassKey(e){return this.galaChainService.resolveTokenClassKey(e)}async lockToken(e){if(e.lockAuthority){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return gn});e={...e,lockAuthority:t(e.lockAuthority)}}if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.lockToken(o)}return this.validateWallet(),this.galaChainService.lockToken(e)}async unlockToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:n,...o}=e;return t.unlockToken(o)}return this.validateWallet(),this.galaChainService.unlockToken(e)}async resolveVaultAddress(e){return this.tokenResolverService.resolveTokenToVault(e)}getCacheInfo(){return this.launchpadAPI.getCacheStats()}clearCache(e){this.launchpadAPI.clearCache(e)}validateConfiguration(){if(("number"!=typeof this.config.timeout||this.config.timeout<=0||this.config.timeout>3e5)&&(this.logger.warn(`Invalid timeout value: ${this.config.timeout}. Using default 30000ms.`),this.config.timeout=3e4),!this.config.baseUrl)throw ie("baseUrl is required in configuration","baseUrl");if(!this.config.webSocketUrl)throw ie("webSocketUrl is required in configuration","webSocketUrl");try{new URL(this.config.baseUrl)}catch{throw ie(`Invalid baseUrl format: ${this.config.baseUrl}`,"baseUrl")}try{new URL(this.config.webSocketUrl)}catch{throw ie(`Invalid webSocketUrl format: ${this.config.webSocketUrl}`,"webSocketUrl")}if(this.config.galaChainBaseUrl)try{new URL(this.config.galaChainBaseUrl)}catch{throw ie(`Invalid galaChainBaseUrl format: ${this.config.galaChainBaseUrl}`,"galaChainBaseUrl")}if(this.config.bundleBaseUrl)try{new URL(this.config.bundleBaseUrl)}catch{throw ie(`Invalid bundleBaseUrl format: ${this.config.bundleBaseUrl}`,"bundleBaseUrl")}if(this.config.launchpadFrontendUrl)try{new URL(this.config.launchpadFrontendUrl)}catch{throw ie(`Invalid launchpadFrontendUrl format: ${this.config.launchpadFrontendUrl}`,"launchpadFrontendUrl")}}parseSlippageToleranceFactor(e){const t=parseFloat(String(e));return isNaN(t)||t<0||t>1?(this.logger.warn(`Invalid slippage tolerance factor: ${e}, using default: ${Xi.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR}`),Xi.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR):t}parseFeeSlippageFactor(e){const t=parseFloat(String(e));return isNaN(t)||t<0||t>1?(this.logger.warn(`Invalid fee slippage factor: ${e}, using default: ${Xi.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR}`),Xi.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR):t}async ensureWebSocketConnection(){this.websocketService.isConnected()||(await this.websocketService.connect(),this.logger.debug("WebSocket connection established"))}async waitForConfirmation(e,t){this.logger.debug(`Waiting for confirmation of transaction: ${e}`);try{const n=await this.websocketService.waitForTransaction(e);if("completed"!==n.status)throw new ji(e,n.status,n.message);let o;try{o=t(n)}catch(t){if(t instanceof Gi)throw t;throw new Gi(`Failed to transform WebSocket response for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}return this.logger.debug(`Transaction confirmed: ${e}`,o),o}catch(t){if(this.logger.error(`Transaction confirmation failed: ${e}`,t),t instanceof ji||t instanceof Gi)throw t;throw new Gi(`WebSocket confirmation failed for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}}async warmCacheFromPools(e){if(!e||!Array.isArray(e))return;const{extractMetadataFromPoolData:t,isValidPoolForCaching:n}=await Promise.resolve().then(function(){return kr});e.forEach(e=>{if(!n(e))return;const o=t(e,this.logger);o&&this.launchpadAPI.warmCacheFromPoolData(e.tokenName,o)})}async getSwapQuoteExactInput(e,t,n){return this.gswapService.getSwapQuoteExactInput({fromToken:e,toToken:t,amount:n})}async getSwapQuoteExactOutput(e,t,n){return this.gswapService.getSwapQuoteExactOutput({fromToken:e,toToken:t,amount:n})}async executeSwap(e,t,n,o,i,r=.01){return this.validateWallet(),this.gswapService.executeSwap({fromToken:e,toToken:t,inputAmount:n,estimatedOutput:o,feeTier:i,slippageTolerance:r})}async getSwapUserAssets(e){return this.gswapService.getUserAssets(e)}async getAllSwapUserAssets(e){return this.gswapService.getAllUserAssets(e)}async getSwapPoolInfo(e,t){return this.gswapService.getPoolInfo(e,t)}async getSwapPoolPrice(e,t,n){return this.gswapService.getPositionCurrentPrice({token0:e,token1:t,feeTier:n})}async getSwapUserLiquidityPositions(e,t,n,o){let i,r;"string"==typeof n?(i=n,r=o):"object"==typeof n?r=n:o&&(r=o);return await this.gswapService.getUserLiquidityPositions(e,t,i,r)}async getAllSwapUserLiquidityPositions(e,t){const n=await this.gswapService.getAllSwapUserLiquidityPositions(e,t);if(!t?.withPrices){if(Array.isArray(n))return n;if(n&&"items"in n)return n.items}return n}async getSwapLiquidityPosition(e,t){return this.gswapService.getLiquidityPosition(e,t)}async getSwapLiquidityPositionById(e,t,n,o,i,r,a){return this.gswapService.getLiquidityPositionById(e,t,n,o,i,r,a)}async fetchSwapPositionDirect(e){return this.gswapService.fetchSwapPositionDirect(e)}async getSwapEstimateRemoveLiquidity(e){return this.gswapService.estimateRemoveLiquidity(e)}async addSwapLiquidityByPrice(e){return this.gswapService.addLiquidityByPrice(e)}async addSwapLiquidityByTicks(e){this.validateWallet();const t={token0:e.token0,token1:e.token1,fee:e.feeTier,tickLower:e.tickLower,tickUpper:e.tickUpper,amount0Desired:e.amount0Desired,amount1Desired:e.amount1Desired};return void 0!==e.amount0Min&&(t.amount0Min=e.amount0Min),void 0!==e.amount1Min&&(t.amount1Min=e.amount1Min),this.gswapService.addSwapLiquidityByTicks(t)}async removeSwapLiquidity(e){return this.validateWallet(),this.gswapService.removeLiquidity(e)}async collectSwapPositionFees(e){return this.validateWallet(),this.gswapService.collectPositionFees(e)}connectWebSocket(){this.websocketService.connect()}disconnectWebSocket(){this.websocketService.disconnect()}isWebSocketConnected(){return this.websocketService.isConnected()}subscribeToEvent(e,t){const n=this.websocketService.getSocket();return n?(n.on(e,t),()=>{n.off(e,t),this.logger.debug(`Unsubscribed from event: "${e}"`)}):(this.logger.warn(`⚠️ WebSocket not connected - subscribing to "${e}" without connection`),()=>{})}onDexPoolCreation(e,t){const n=1e3,o=Math.max(t?.intervalMs??3e4,n);t?.intervalMs&&t.intervalMs<n&&this.logger.warn(`Poll interval ${t.intervalMs}ms is below minimum 1000ms. Using minimum interval instead.`);const i=t?.minTVL,r=t?.tokens,a=new Map;let s=!0,c=null;let l=0;const u=async()=>{if(s){try{const t=await this.fetchDexPools({limit:20});l>0&&(this.logger.debug("Successfully recovered from polling errors"),l=0),t.pools.forEach(t=>{const n=(e=>`${e.token0}-${e.token1}-${e.fee}`)(t);if(!a.has(n)){if((e=>{if(a.set(e,!0),a.size>1e3){const e=a.keys().next().value;void 0!==e&&a.delete(e)}})(n),i){if((t.token0Tvl+t.token1Tvl)/2<i)return}if(r&&r.length>0){if(!(r.includes(t.token0)||r.includes(t.token1)))return}e(t)}})}catch(e){l++;const t=e instanceof Error?e.message:String(e);l>=5?this.logger.error(`Polling for new DEX pools failed ${l} consecutive times. Last error: ${t}. Continuing to retry...`):l>1?this.logger.warn(`Error polling for new DEX pools (attempt ${l}/5): ${t}`):this.logger.debug(`Error polling for new DEX pools: ${t}`)}if(s){const e=Math.min(Math.max(l-1,0),2),t=o*Math.pow(2,e);c=setTimeout(u,t)}}};return u(),()=>{s=!1,c&&clearTimeout(c),this.logger.debug("Stopped watching for DEX pool creation")}}onLaunchpadTokenCreation(e,t){const n=1e3,o=Math.max(t?.intervalMs??3e4,n);t?.intervalMs&&t.intervalMs<n&&this.logger.warn(`Poll interval ${t.intervalMs}ms is below minimum 1000ms. Using minimum interval instead.`);const i=t?.creatorAddress,r=new Map;let a=!0,s=null;let c=0;const l=async()=>{if(a){try{const t=await this.fetchPools({type:"recent",limit:20});c>0&&(this.logger.debug("Successfully recovered from polling errors"),c=0),t.pools.forEach(t=>{r.has(t.tokenName)||((e=>{if(r.set(e,!0),r.size>1e3){const e=r.keys().next().value;void 0!==e&&r.delete(e)}})(t.tokenName),i&&t.creatorAddress!==i||e(t))})}catch(e){c++;const t=e instanceof Error?e.message:String(e);c>=5?this.logger.error(`Polling for new launchpad tokens failed ${c} consecutive times. Last error: ${t}. Continuing to retry...`):c>1?this.logger.warn(`Error polling for new launchpad tokens (attempt ${c}/5): ${t}`):this.logger.debug(`Error polling for new launchpad tokens: ${t}`)}if(a){const e=Math.min(Math.max(c-1,0),2),t=o*Math.pow(2,e);s=setTimeout(l,t)}}};return l(),()=>{a=!1,s&&clearTimeout(s),this.logger.debug("Stopped watching for launchpad token creation")}}normalizeFee(e){if(null==e)return null;const t="number"==typeof e?e:Number.parseFloat(String(e).replace("%","").trim());return Number.isNaN(t)?null:1===t||1e4===t?1e4:.3===t||3e3===t?3e3:.05===t||500===t?500:!Number.isInteger(t)||500!==t&&3e3!==t&&1e4!==t?null:t}extractField(e,...t){if("object"!=typeof e||null===e)return"";const n=e;for(const e of t)if(n[e])return String(n[e]);return""}looksLikePoolPair(e){if("string"!=typeof e)return null;const t=e.trim();if(!t.includes("/"))return null;const n=t.split("/");return 3!==n.length?null:n[0]&&n[1]&&n[2]?t:null}buildPoolPairFromObject(e){if("object"!=typeof e||null===e)return null;const t=e,n=this.extractField(t,"token0ClassKey","token0Class","token0","token0Symbol")||"",o=this.extractField(t,"token1ClassKey","token1Class","token1","token1Symbol")||"",i=this.normalizeFee(t.feeTier??t.fee??t.feeTierBps??t.liquidityFeeBps??t.feeBps);return n&&o&&null!==i?`${n}/${o}/${i}`:null}parsePoolPairString(e){const t=e.split("/");if(3!==t.length)return null;const n=t[0].split("|")[0],o=t[1].split("|")[0],i=t[2];return n&&o&&i?{token0:n,token1:o,fee:i,poolPair:e}:null}serializeBalanceToken(e){if(!e||"object"!=typeof e)return"";const t=e;return[(t.collection??t.token??"")||"",(t.category??"")||"none",(t.type??"")||"none",(t.additionalKey??"none")||"none"].join("|")}buildPoolPairFromBalances(e){if("object"!=typeof e||null===e)return null;const t=e,n=t.userBalanceDelta??t.balanceDelta??t.delta;if(!n||"object"!=typeof n||null===n)return null;const o=n,i=o.token0Balance??o.token0??o.baseBalance??o.primaryBalance,r=o.token1Balance??o.token1??o.quoteBalance??o.secondaryBalance,a=this.serializeBalanceToken(i),s=this.serializeBalanceToken(r),c=this.normalizeFee(t.poolFee??t.feeTier??t.fee??t.feeTierBps??t.liquidityFeeBps);return a&&s&&null!==c?`${a}/${s}/${c}`:null}extractPoolDataFromPayload(e){if("string"==typeof e){const t=this.looksLikePoolPair(e);return t?this.parsePoolPairString(t):null}if("object"!=typeof e||null===e)return null;const t=e,n=this.looksLikePoolPair(t.poolPair);if(n)return this.parsePoolPairString(n);const o=this.buildPoolPairFromBalances(t);if(o)return this.parsePoolPairString(o);const i=this.buildPoolPairFromObject(t);if(i)return this.parsePoolPairString(i);if(t.pool&&"object"==typeof t.pool&&null!==t.pool){const e=this.extractPoolDataFromPayload(t.pool);if(e)return e}return null}matchesPoolFilter(e,t){if(t?.tokenFilter){if(!(e.token0===t.tokenFilter||e.token1===t.tokenFilter))return!1}if(t?.pairTokens){const[n,o]=t.pairTokens,i=e.token0===n||e.token1===n,r=e.token0===o||e.token1===o;if(!i||!r||n===o)return!1}if(void 0!==t?.feeTierFilter){if(this.normalizeFee(t.feeTierFilter)!==this.normalizeFee(e.fee))return!1}return!0}matchesCreatorFilter(e,t){return!t||e.creatorAddress===t}subscribeToTokenCreations(e,t){if(this.logger.debug("Subscribing to token creation broadcasts"+(t?.creatorFilter?` (filter: ${t.creatorFilter})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=(n,...o)=>{try{if(o.length>0&&"object"==typeof o[0]&&null!==o[0]){const n=o[0].data;if(n&&n.Data&&"object"==typeof n.Data){const o=n.Data;if("CreateSale"===o.functionName){const n={tokenName:o.tokenName||"",symbol:o.symbol||"",creatorAddress:o.creatorAddress||"",description:o.description||"",image:o.image||"",vaultAddress:o.vaultAddress||"",tokenStringKey:o.tokenStringKey||"",preBuyQuantity:o.initialBuyQuantity||"0",websiteUrl:o.websiteUrl||"",telegramUrl:o.telegramUrl||"",twitterUrl:o.twitterUrl||"",isFinalized:o.isFinalized||!1};this.matchesCreatorFilter(n,t?.creatorFilter)&&e(n)}}}}catch(e){this.logger.warn(`Error processing token creation broadcast: ${e instanceof Error?e.message:String(e)}`)}};let r=this.websocketService.getSocket();if(r)r.onAny(i),n=!0,this.logger.debug("Token creation broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const a=()=>{if(r=this.websocketService.getSocket(),!r&&e<Xi.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>a(),Xi.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS));if(!r&&e>=Xi.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${Xi.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS*Xi.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("Token creation broadcast subscription timeout:",e.message),void(t?.onError&&t.onError(e))}r&&(r.onAny(i),n=!0,this.logger.debug("Token creation broadcast listener registered"))};a()}return()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared token creation broadcast polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(i),n=!1,this.logger.debug("Stopped listening to token creation broadcasts"))}catch(e){this.logger.warn("Error removing token creation listener:",e)}}}walkPayloadForPools(e,t,n=new WeakSet){const o=[];if("string"==typeof e){const n=this.looksLikePoolPair(e);if(n){const e=this.parsePoolPairString(n);e&&!t.has(e.poolPair)&&(t.add(e.poolPair),o.push(e))}return o}if("object"!=typeof e||null===e)return o;if(n.has(e))return o;n.add(e);const i=this.extractPoolDataFromPayload(e);i&&!t.has(i.poolPair)&&(t.add(i.poolPair),o.push(i));for(const i of Object.values(e)){const e=this.walkPayloadForPools(i,t,n);o.push(...e)}return o}subscribeToDexPoolAdded(e,t){if(this.logger.debug("Subscribing to DEX pool creation broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=new Set,r=(n,...o)=>{try{for(const n of o){const o=this.walkPayloadForPools(n,i);for(const n of o)this.matchesPoolFilter(n,t)&&e(n)}}catch(e){this.logger.warn(`Error processing DEX pool broadcast: ${e instanceof Error?e.message:String(e)}`)}};let a=this.websocketService.getSocket();if(a)a.onAny(r),n=!0,this.logger.debug("DEX pool broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const i=()=>{if(a=this.websocketService.getSocket(),!a&&e<Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>i(),Xi.DEX_POOL_SOCKET_POLL_INTERVAL_MS));if(!a&&e>=Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS*Xi.DEX_POOL_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("DEX pool subscription timeout:",e.message),void(t?.onError&&t.onError(e))}a&&(a.onAny(r),n=!0,this.logger.debug("DEX pool broadcast listener registered"))};i()}return()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared DEX pool polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(r),n=!1,this.logger.debug("Stopped listening to DEX pool broadcasts"))}catch(e){this.logger.warn("Error removing DEX pool listener:",e)}}}subscribeToDexSwapExecuted(e,t){if(this.logger.debug("Subscribing to DEX swap execution broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=null,o=null,i=null,r=!1;const a=async e=>{const t=vi.parsePoolKey(e);if(!t)throw new Error(`Invalid pool key format: ${e}`);return await this.dexQuoteService.fetchCompositePoolData({token0:t.token0,token1:t.token1,fee:t.feeTier})},s=()=>{const c=this.websocketService.getSocket();if(!c)return this.logger.debug("WebSocket not yet ready for swap monitoring, polling..."),void(n=setTimeout(()=>s(),100));r=!0,o=new Pi(c,a,this.dexQuoteService,t||{},this.logger),i=o.subscribe(t||{},e),this.logger.debug("DEX swap monitoring subscription established")};return this.websocketService.getSocket()||(this.logger.debug("WebSocket not yet connected, initiating connection for swap monitoring..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed for swap monitoring: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)})),s(),()=>{try{n&&clearTimeout(n),i&&r&&i(),o&&o.shutdown().catch(e=>{this.logger.warn("Error shutting down swap monitor:",e)})}catch(e){this.logger.warn("Error cleaning up swap monitor:",e)}}}subscribeToDexLiquidityAdded(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityRemoved(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityChanged(e,t){return this.subscribeToDexLiquidityEvents(e,t)}subscribeToDexLiquidityEvents(e,t){if(this.logger.debug("Subscribing to DEX liquidity broadcasts"+(t?.tokenFilter?` (filter: ${t.tokenFilter})`:t?.pairTokens?` (pair: ${t.pairTokens.join("/")})`:"")),!this.websocketService)throw new Error("WebSocket service not initialized");let n=!1,o=null;const i=new Set,r=new Qi(this.logger),a=(n,...o)=>{try{for(const n of o){const o=r.walkPayloadForLiquidityEvents(n,i);for(const n of o)if(this.matchesLiquidityFilter(n,t))try{const t=e(n);t instanceof Promise&&t.catch(e=>{this.logger.warn(`Error in liquidity event callback: ${e instanceof Error?e.message:String(e)}`)})}catch(e){this.logger.warn(`Error in liquidity event callback: ${e instanceof Error?e.message:String(e)}`)}}}catch(e){this.logger.warn(`Error processing DEX liquidity broadcast: ${e instanceof Error?e.message:String(e)}`)}};let s=this.websocketService.getSocket();if(s)s.onAny(a),n=!0,this.logger.debug("DEX liquidity broadcast listener registered");else{this.logger.debug("WebSocket not yet connected, initiating connection for liquidity monitoring..."),this.websocketService.connect().catch(e=>{const n=new Error(`WebSocket connection failed: ${e instanceof Error?e.message:String(e)}`);this.logger.warn("Failed to establish WebSocket connection:",e),t?.onError&&t.onError(n)});let e=0;const i=()=>{if(s=this.websocketService.getSocket(),!s&&e<Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS)return e++,void(o=setTimeout(()=>i(),Xi.DEX_POOL_SOCKET_POLL_INTERVAL_MS));if(!s&&e>=Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS){const e=new Error(`WebSocket not available after ${Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS*Xi.DEX_POOL_SOCKET_POLL_INTERVAL_MS}ms`);return this.logger.warn("DEX liquidity subscription timeout:",e.message),void(t?.onError&&t.onError(e))}s&&(s.onAny(a),n=!0,this.logger.debug("DEX liquidity broadcast listener registered"))};i()}return async()=>{try{if(null!==o&&(clearTimeout(o),o=null,this.logger.debug("Cleared DEX liquidity polling timeout")),!n)return void this.logger.debug("Cleanup called before listener registration - no action needed");const e=this.websocketService.getSocket();e&&(e.offAny(a),n=!1,this.logger.debug("Stopped listening to DEX liquidity broadcasts"))}catch(e){this.logger.warn("Error removing DEX liquidity listener:",e)}}}matchesLiquidityFilter(e,t){if(!t)return!0;if(t.positionId&&e.positionId!==t.positionId)return!1;if(t.poolHash&&e.poolHash!==t.poolHash)return!1;if(void 0!==t.feeTierFilter){const n=this.normalizeFeeTier(t.feeTierFilter);if(e.poolFee!==n)return!1}if(t.userFilter&&e.userAddress!==t.userFilter)return!1;if(t.tokenFilter){const n=e.token0?.toLowerCase().includes(t.tokenFilter.toLowerCase()),o=e.token1?.toLowerCase().includes(t.tokenFilter.toLowerCase());if(!n&&!o)return!1}if(t.pairTokens){const[n,o]=t.pairTokens.map(e=>e.toLowerCase()),i=e.token0?.toLowerCase()||"",r=e.token1?.toLowerCase()||"",a=i.includes(n)&&r.includes(o),s=i.includes(o)&&r.includes(n);if(!a&&!s)return!1}if(t.minAmount){const n=parseFloat(t.minAmount),o=parseFloat(e.amounts[0])||0,i=parseFloat(e.amounts[1])||0;if(Math.abs(o)<n&&Math.abs(i)<n)return!1}return!0}normalizeFeeTier(e){if("number"==typeof e)return e>=100?e:Math.round(1e4*e);const t=e.replace("%","").trim(),n=parseFloat(t);return n>=100?n:Math.round(1e4*n)}async cleanup(){try{this.logger.debug("Starting cleanup..."),this.http.cleanup(),this.websocketService&&this.websocketService.disconnect(),this.logger.debug("Cleanup completed")}catch(e){this.logger.error("Error during cleanup:",e)}}static cleanupAll(e=!1){const t=new _({debug:e,context:"LaunchpadSDK"});t.debug("Starting global cleanup...");const{WebSocketService:n}=require("./services/WebSocketService");n.cleanupAll(e),t.debug("Global cleanup completed")}}Xi.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR=.15,Xi.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR=.01,Xi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY=pi.DEFAULT_LAUNCHPAD_TOKEN_MAX_SUPPLY,Xi.DEFAULT_CALCULATE_AMOUNT_MODE=_i.LOCAL,Xi.TOKEN_CREATION_SOCKET_WAIT_ATTEMPTS=30,Xi.TOKEN_CREATION_SOCKET_POLL_INTERVAL_MS=100,Xi.DEX_POOL_SOCKET_WAIT_ATTEMPTS=30,Xi.DEX_POOL_SOCKET_POLL_INTERVAL_MS=100;class Zi{static generateWallet(){try{const e=n.createRandom();if(!e.mnemonic?.phrase)throw new Error("Failed to generate wallet with mnemonic phrase");const t=this.toGalaAddress(e.address);return{privateKey:e.privateKey,address:e.address,galaAddress:t,mnemonic:e.mnemonic.phrase,wallet:new n(e.privateKey)}}catch(e){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const e=`test-wallet-${Date.now()}-${++this.testCounter}`,t="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64),o=new n(t),i=this.toGalaAddress(o.address);return{privateKey:o.privateKey,address:o.address,galaAddress:i,mnemonic:"test test test test test test test test test test test junk",wallet:o}}throw e}}static fromPrivateKey(e){const t=new n(e),o=this.toGalaAddress(t.address);return{privateKey:t.privateKey,address:t.address,galaAddress:o,mnemonic:"",wallet:t}}static fromMnemonic(e,t=0){try{const r=o.fromPhrase(e),a=i.fromMnemonic(r,`m/44'/60'/0'/0/${t}`),s=new n(a.privateKey),c=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:c,mnemonic:e,wallet:s}}catch(o){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const o=`test-mnemonic-index-${t}-${e}`,i="0x"+Buffer.from(o).toString("hex").padStart(64,"1").slice(0,64),r=new n(i),a=this.toGalaAddress(r.address);return{privateKey:r.privateKey,address:r.address,galaAddress:a,mnemonic:e,wallet:r}}throw o}}static toGalaAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new Error(`Invalid Ethereum address format: ${e}`);return`eth|${t}`}static toEthereumAddress(e){if(!e.startsWith("eth|"))throw new Error(`Invalid Gala address format: ${e}. Must start with 'eth|'`);const t=e.slice(4);if(!/^[a-fA-F0-9]{40}$/.test(t))throw new Error(`Invalid address in Gala format: ${e}`);return`0x${t}`}static isValidEthereumAddress(e){try{const t=e.replace(/^0x/i,"");return/^[a-fA-F0-9]{40}$/.test(t)}catch{return!1}}static isValidGalaAddress(e){try{if(!e.startsWith("eth|"))return!1;const t=e.slice(4);return/^[a-fA-F0-9]{40}$/.test(t)}catch{return!1}}static generateMultipleWallets(e=1){if(e<1||e>100)throw new Error("Count must be between 1 and 100");const t=[];if("undefined"!=typeof process&&"test"===process.env.NODE_ENV)for(let n=0;n<e;n++){const e=`test-multi-${n}-${Date.now()}-${++this.testCounter}`,o="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64);t.push(this.fromPrivateKey(o))}else for(let n=0;n<e;n++)t.push(this.generateWallet());return t}static getWalletSummary(e,t=!1){const n=["🔐 Wallet Information","═".repeat(50),`📍 Address: ${e.address}`,`🎮 Gala Address: ${e.galaAddress}`,`🌱 Mnemonic: ${e.mnemonic||"Not available"}`];return t?n.splice(3,0,`🔑 Private Key: ${e.privateKey}`):n.splice(3,0,"🔑 Private Key: [HIDDEN - use includeSensitive=true to show]"),n.push("═".repeat(50)),n.push("💾 IMPORTANT: Save your mnemonic phrase securely!"),n.push("This is your backup to recover the wallet."),n.join("\n")}}function Yi(e){if(void 0===e)return Zi.generateWallet();const t=e.trim();if(!t)throw new Error("Input cannot be empty string");if(function(e){const t=e.replace(/^0x/i,"");return/^[a-fA-F0-9]{64}$/.test(t)}(t))return Zi.fromPrivateKey(t);if(function(e){const t=e.split(/\s+/).filter(e=>e.length>0);if(12!==t.length&&24!==t.length)return!1;return t.every(e=>/^[a-zA-Z]+$/.test(e))}(t))return Zi.fromMnemonic(t);throw new Error(`Unable to detect input format. Expected:\n- Private key: 64 hexadecimal characters (with or without 0x prefix)\n- Mnemonic: 12 or 24 space-separated words\nReceived: "${t.slice(0,50)}${t.length>50?"...":""}"`)}function Ji(e){e||(e={});const{wallet:t,env:o,config:i={},...r}=e,a={...r,...i},{wallet:s,env:c,config:l,...u}=a;let d;if(t)if("string"==typeof t){d=Yi(t).wallet}else{if(!(t instanceof n))throw new Error("Invalid wallet input. Expected string (private key or mnemonic) or Wallet instance.");d=t}else{d=Yi().wallet}const h={wallet:d,...o&&{env:o},debug:!1,timeout:3e4,...u};return new Xi(h)}function er(){const e=F.join(process.cwd(),"..","..",".env");x.existsSync(e)&&C.config({path:e});const t=F.join(process.cwd(),".env");x.existsSync(t)&&C.config({path:t})}function tr(e,t){const n=process.env[e];if(!n){throw new Error(t?`${e} not set in root or local .env (${t})`:`${e} not set in root or local .env`)}return n}function nr(e,t){return process.env[e]??t}Zi.testCounter=0;class or{static async quickSetup(e={}){const t=e.environment||this.detectEnvironment(),n=this.setupWallet(e.privateKey),o={wallet:n.wallet,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t),...e.config||{}},i=new Xi(o),r={sdk:i,wallet:n,config:o};if(!1!==e.autoValidate){const e=await this.validateSetup(i,n);return{...r,validation:e}}return r}static async readOnlySetup(e={}){const t=e.environment||this.detectEnvironment(),n={wallet:void 0,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t),...e.config||{}};return{sdk:new Xi(n),config:n}}static async validateSetup(e,t){const n=[],o=[],i={canTrade:!1,canCreateTokens:!1,hasBalance:!1,connectionHealthy:!1};try{const t=await e.fetchGalaBalance(e.getAddress());if(i.connectionHealthy=!0,t&&t.quantity){const e=parseFloat(t.quantity);i.hasBalance=e>0,i.canTrade=e>=.1,i.canCreateTokens=e>=100,0===e?o.push("Wallet has zero GALA balance - cannot perform transactions"):e<.1?o.push("GALA balance too low for trading (minimum 0.1 GALA)"):e<100&&o.push("GALA balance too low for token creation (minimum 100 GALA)")}else n.push("Failed to fetch GALA balance: No balance returned")}catch(e){n.push(`Balance check error: ${e instanceof Error?e.message:String(e)}`)}try{const t=await e.fetchPools({type:"recent",page:1,limit:1});t.pools&&0!==t.pools.length||o.push("Pool listing not accessible - some features may be limited")}catch(e){o.push(`Pool access test failed: ${e instanceof Error?e.message:String(e)}`)}return{ready:0===n.length&&i.connectionHealthy,sdk:e,wallet:t||Zi.generateWallet(),issues:n,warnings:o,capabilities:i}}static getRecommendedConfig(e,t="general"){const n={environment:e,autoValidate:!0};switch(e){case"production":Object.assign(n,{debug:!1,timeout:3e4});break;case"development":Object.assign(n,{debug:!0,timeout:45e3});break;case"testing":Object.assign(n,{debug:!0,timeout:6e4})}switch(t){case"trading":n.timeout=1.5*(n.timeout||3e4);break;case"creation":n.timeout=2*(n.timeout||3e4);break;case"monitoring":n.timeout=.5*(n.timeout||3e4)}return n}static async multiWalletSetup(e,t="development"){const n={};for(const[o,i]of Object.entries(e)){const{sdk:e}=await this.quickSetup({environment:t,privateKey:i,agentId:`multi-wallet-${o}`,autoValidate:!1});n[o]=e}return n}static detectEnvironment(){const e=process.env.NODE_ENV?.toLowerCase();return"development"===e?"development":"test"===e||"testing"===e?"testing":"production"}static setupWallet(e){if(!e){const e=process.env.PRIVATE_KEY;return e?Zi.fromPrivateKey(e):Zi.generateWallet()}return"generate"===e?Zi.generateWallet():Zi.fromPrivateKey(e)}static getDefaultBaseUrl(e){return"production"===e?"https://lpad-backend-prod1.defi.gala.com":"https://lpad-backend-dev1.defi.gala.com"}static getDefaultTimeout(e){switch(e){case"production":default:return 3e4;case"development":return 45e3;case"testing":return 6e4}}static getEnvironmentDefaults(e){const t={};if("production"===e)t.bundleBaseUrl="https://bundle-backend-prod1.defi.gala.com",t.galaChainBaseUrl="https://galachain-gateway-chain-platform-prod-chain-platform-eks.prod.galachain.com";else t.bundleBaseUrl="https://bundle-backend-dev1.defi.gala.com",t.galaChainBaseUrl="https://galachain-gateway-chain-platform-stage-chain-platform-eks.stage.galachain.com";return t}}function ir(e){return e.filter(e=>new u(e.liquidity).isGreaterThan(0))}function rr(e,t){const n=new u(t);return e.filter(e=>new u(e.liquidity).isGreaterThanOrEqualTo(n))}function ar(e,t="desc"){return[...e].sort((e,n)=>{const o=new u(e.liquidity),i=new u(n.liquidity);return"desc"===t?i.minus(o).toNumber():o.minus(i).toNumber()})}function sr(e,t,n){const o=t.toUpperCase(),i=n.toUpperCase();return e.filter(e=>{const t=e.token0.toUpperCase(),n=e.token1.toUpperCase();return t===o&&n===i||t===i&&n===o})}function cr(e,t){const n=t.toUpperCase();return e.filter(e=>e.token0.toUpperCase()===n||e.token1.toUpperCase()===n)}function lr(e,t){return e.filter(e=>e.feeTier===t)}function ur(e){const t=new Map;return e.forEach(e=>{const n=`${e.token0.toUpperCase()}/${e.token1.toUpperCase()}`;t.has(n)||t.set(n,[]),t.get(n).push(e)}),t}function dr(e){const t=new Map;return e.forEach(e=>{t.has(e.feeTier)||t.set(e.feeTier,[]),t.get(e.feeTier).push(e)}),t}function hr(e,t,n,o){const i=t.toUpperCase(),r=n.toUpperCase();return e.filter(e=>{const t=e.token0.toUpperCase(),n=e.token1.toUpperCase();return(t===i&&n===r||t===r&&n===i)&&e.feeTier===o})}function gr(e){const t=new Map;return e.forEach(e=>{const n=`${e.token0.toUpperCase()}|${e.token1.toUpperCase()}|${e.feeTier}`;t.has(n)||t.set(n,[]),t.get(n).push(e)}),t}const pr={PAGE:"page",LIMIT:"limit",TOKEN_NAME:"tokenName",VAULT_ADDRESS:"vaultAddress",USER_ADDRESS:"userAddress",TRADE_TYPE:"tradeType",POOL_TYPE:"type",SEARCH:"search",SORT_ORDER:"sortOrder",START_DATE:"startDate",END_DATE:"endDate"};class mr{static fastValidation(e,t,n,o=mr.DEFAULT_CONFIG){const i=Date.now();let r=!0;try{const a=new u(e.sqrtPrice),s=new u(t.sqrtPrice);n.zeroForOne?s.gte(a)&&(this.logger.error("Fast validation failed: price did not decrease for zeroForOne swap",{originalSqrtPrice:a.toString(),updatedSqrtPrice:s.toString(),zeroForOne:n.zeroForOne}),r=!1):s.lte(a)&&(this.logger.error("Fast validation failed: price did not increase for oneForZero swap",{originalSqrtPrice:a.toString(),updatedSqrtPrice:s.toString(),zeroForOne:n.zeroForOne}),r=!1);const c=new u(e.liquidity),l=new u(t.liquidity);if(!c.isZero()){const e=l.minus(c).abs().div(c);e.gt(o.maxLiquidityChangePct)&&this.logger.warn("Fast validation warning: large liquidity change detected (could be legitimate)",{originalLiquidity:c.toString(),updatedLiquidity:l.toString(),changePct:e.times(100).toFixed(2)})}const d=new u(e.feeGrowthGlobal0),h=new u(t.feeGrowthGlobal0),g=new u(e.feeGrowthGlobal1),p=new u(t.feeGrowthGlobal1);n.zeroForOne?p.lt(g)&&(this.logger.error("Fast validation failed: feeGrowthGlobal1 decreased for zeroForOne",{originalFeeGrowth1:g.toString(),updatedFeeGrowth1:p.toString()}),r=!1):h.lt(d)&&(this.logger.error("Fast validation failed: feeGrowthGlobal0 decreased for oneForZero",{originalFeeGrowth0:d.toString(),updatedFeeGrowth0:h.toString()}),r=!1);const m=new u(e.protocolFeesToken0),f=new u(t.protocolFeesToken0),y=new u(e.protocolFeesToken1),w=new u(t.protocolFeesToken1);f.lt(m)&&(this.logger.error("Fast validation failed: protocolFeesToken0 decreased",{originalProtocolFees0:m.toString(),updatedProtocolFees0:f.toString()}),r=!1),w.lt(y)&&(this.logger.error("Fast validation failed: protocolFeesToken1 decreased",{originalProtocolFees1:y.toString(),updatedProtocolFees1:w.toString()}),r=!1);const k=Date.now()-i;return this.logger.debug("Fast validation completed",{passed:r,elapsedMs:k}),r}catch(e){return this.logger.error("Fast validation exception",e),!1}}static moderateValidation(e,t,n=mr.DEFAULT_CONFIG){const o=Date.now(),i=[];let r=0;try{if(t.actualSqrtPrice){const o=new u(e.sqrtPrice),a=new u(t.actualSqrtPrice),s=this.calculateDriftPercentage(o,a);r=s,s>100*n.maxDriftThreshold&&i.push(`Price drift detected: ${s.toFixed(4)}% (threshold: ${(100*n.maxDriftThreshold).toFixed(4)}%)`),this.logger.debug("Price drift comparison",{calculatedSqrtPrice:o.toString(),actualSqrtPrice:a.toString(),driftPct:s.toFixed(4)})}const a=new u(e.sqrtPrice),s=new u(2).pow(96),c=a.dividedBy(s),l=A(c),d=e.tick??0,h=Math.abs(l-d);h>n.maxTickDrift&&i.push(`Tick/price mismatch: tick=${d}, calculated=${l}, drift=${h}`);const g=new u(e.feeGrowthGlobal0),p=new u(e.feeGrowthGlobal1),m=new u(e.liquidity);try{uo(g,p,m)}catch(e){i.push(e.message)}const f=0===i.length,y=!f||r>100*n.maxDriftThreshold,w=Date.now()-o;return this.logger.debug("Moderate validation completed",{isValid:f,shouldRefetch:y,driftPercentage:r,errorCount:i.length,elapsedMs:w}),this.buildValidationResult(f,r,y,i)}catch(e){this.logger.error("Moderate validation exception",e);const t=e instanceof Error?e.message:String(e);return this.buildValidationResult(!1,0,!0,[`Exception during validation: ${t}`])}}static async fullValidation(e,t,n){const o=Date.now(),i=[];let r=0;try{this.logger.debug("Starting full validation with fresh pool data fetch",{poolKey:e});const a=await n(),s=new u(t.pool.sqrtPrice),c=new u(a.pool.sqrtPrice),l=this.calculateDriftPercentage(s,c);r=Math.max(r,l),l>100*this.DEFAULT_CONFIG.maxPriceDriftPct&&i.push(`Price drift: ${l.toFixed(4)}% (cached: ${s.toString()}, fresh: ${c.toString()})`);const d=new u(t.pool.liquidity),h=new u(a.pool.liquidity),g=this.calculateDriftPercentage(d,h);r=Math.max(r,g),g>100*this.DEFAULT_CONFIG.maxLiquidityDriftPct&&i.push(`Liquidity drift: ${g.toFixed(4)}% (cached: ${d.toString()}, fresh: ${h.toString()})`);const p=Object.keys(t.tickDataMap).length,m=Object.keys(a.tickDataMap).length;if(m>0){const e=Math.abs(m-p)/m;e>this.DEFAULT_CONFIG.maxTickCountDriftPct&&i.push(`Tick data incomplete: cached has ${p} ticks, fresh has ${m} ticks (${(100*e).toFixed(2)}% difference)`)}const f=0===i.length,y=!f,w=Date.now()-o;return this.logger.debug("Full validation completed",{poolKey:e,isValid:f,shouldRefetch:y,maxDriftPercentage:r,priceDrift:l,liquidityDrift:g,cachedTickCount:p,freshTickCount:m,errorCount:i.length,elapsedMs:w}),this.buildValidationResult(f,r,y,i)}catch(t){this.logger.error("Full validation exception",{poolKey:e,error:t});const n=t instanceof Error?t.message:String(t);return this.buildValidationResult(!1,0,!0,[`Exception during full validation: ${n}`])}}static calculateDriftPercentage(e,t){if(t.isZero())return this.logger.warn("Cannot calculate drift: actual value is zero"),1/0;return t.minus(e).abs().div(t).times(100).toNumber()}static buildValidationResult(e,t,n,o=[]){let i;if(n&&o.length>0){const e=o.join(" ").toLowerCase();i=e.includes("drift")?"drift":e.includes("tick")&&e.includes("mismatch")?"tick-mismatch":e.includes("tick")?"missing-tick-data":"manual"}const r={isValid:e,driftPercentage:t,shouldRefetch:n,validationErrors:o};return void 0!==i&&(r.refetchReason=i),r}}mr.logger=new _({debug:!1,context:"PoolStateValidator"}),mr.DEFAULT_CONFIG={maxDriftThreshold:.001,maxLiquidityChangePct:.5,maxTickDrift:1,maxTickCountDriftPct:.1,maxPriceDriftPct:.001,maxLiquidityDriftPct:.01};class fr{static calculatePoolStateHash(e){const t=`${e.sqrtPrice.toString()}|${e.liquidity.toString()}|${e.tick||0}`;return $.createHash("sha256").update(t).digest("hex").substring(0,16)}constructor(e,t){this.logger=new _({debug:t?.debug??!1,context:"PoolStateManager"});const n={maxIterations:t?.maxIterations??100,enableBigNumberCache:t?.enableBigNumberCache??!0,roundingMode:t?.roundingMode??u.ROUND_DOWN,debug:t?.debug??!1,maxSwapsSinceRefetch:t?.maxSwapsSinceRefetch??50,maxCumulativeDrift:t?.maxCumulativeDrift??5,strictValidation:t?.strictValidation??!1,enablePerformanceWarnings:t?.enablePerformanceWarnings??!0,performanceWarningThreshold:t?.performanceWarningThreshold??100};this.config={...n,...t?.onRefetchNeeded?{onRefetchNeeded:t.onRefetchNeeded}:{}},this.validationConfig=mr.DEFAULT_CONFIG,this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata={lastFullRefetch:Date.now(),swapsSinceRefetch:0,cumulativeDrift:0,processedTransactions:[],stateHash:fr.calculatePoolStateHash(this.pool)};if(!mr.fastValidation(this.pool,this.pool,{transactionId:"init",timestamp:Date.now(),amountSpecified:"0",zeroForOne:!1,exactInput:!0},this.validationConfig)&&this.config.strictValidation)throw new Error("Initial pool state validation failed");this.logger.info("PoolStateManager initialized",{pool:{sqrtPrice:this.pool.sqrtPrice.toString(),liquidity:this.pool.liquidity.toString(),tick:this.pool.tick},config:this.config})}async applySwapDelta(e){const t=Date.now();if(this.metadata.processedTransactions.includes(e.transactionId))throw this.logger.warn("Duplicate swap transaction",{transactionId:e.transactionId}),new Error(`Duplicate transaction ID: ${e.transactionId}`);try{const t={pool:this.pool,tickDataMap:this.tickDataMap},n={maxIterations:this.config.maxIterations,enableBigNumberCache:this.config.enableBigNumberCache,roundingMode:this.config.roundingMode,debugLogging:this.config.debug},o=Ai.calculateSwapDelta(t,e,n);this.lastSwapMetrics={calculationTimeMs:o.metadata.calculationTimeMs,swapSteps:o.metadata.swapSteps,timestamp:Date.now()},this.config.enablePerformanceWarnings&&o.metadata.calculationTimeMs>this.config.performanceWarningThreshold&&this.logger.warn("Slow swap calculation",{calculationTimeMs:o.metadata.calculationTimeMs,swapSteps:o.metadata.swapSteps,threshold:this.config.performanceWarningThreshold});if(!mr.fastValidation(t.pool,o.updatedPool,e,this.validationConfig)&&this.config.strictValidation)throw new Error("Swap validation failed");if(e.actualAmount0&&e.actualAmount1&&e.actualSqrtPrice){const t=new u(e.actualAmount0),n=new u(e.actualAmount1),i=o.amount0.minus(t).abs(),r=o.amount1.minus(n).abs(),a=t.isZero()?0:i.div(t.abs()).times(100).toNumber(),s=n.isZero()?0:r.div(n.abs()).times(100).toNumber(),c=Math.max(a,s);c>1&&(this.logger.warn("Drift detected in swap delta",{driftPercentage:c.toFixed(2),swapId:e.transactionId}),this.metadata.cumulativeDrift+=c)}if((this.metadata.swapsSinceRefetch>this.config.maxSwapsSinceRefetch||this.metadata.cumulativeDrift>this.config.maxCumulativeDrift)&&(this.logger.info("Triggering full refetch due to drift accumulation",{swapsSinceRefetch:this.metadata.swapsSinceRefetch,cumulativeDrift:this.metadata.cumulativeDrift.toFixed(2)}),this.config.onRefetchNeeded)){const e=await this.config.onRefetchNeeded(this.pool,this.tickDataMap);this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata.lastFullRefetch=Date.now(),this.metadata.swapsSinceRefetch=0,this.metadata.cumulativeDrift=0,this.metadata.stateHash=fr.calculatePoolStateHash(this.pool),this.logger.info("Full refetch completed",{sqrtPrice:this.pool.sqrtPrice.toString()})}return this.pool=o.updatedPool,Object.assign(this.tickDataMap,o.updatedTicks),this.metadata.swapsSinceRefetch++,this.metadata.stateHash=fr.calculatePoolStateHash(this.pool),this.metadata.processedTransactions.push(e.transactionId),this.metadata.processedTransactions.length>1e3&&(this.metadata.processedTransactions=this.metadata.processedTransactions.slice(-1e3)),this.logger.debug("Swap delta applied",{transactionId:e.transactionId,amount0:o.amount0.toString(),amount1:o.amount1.toString(),sqrtPriceNew:this.pool.sqrtPrice.toString()}),o}catch(n){const o=n instanceof Error?n.message:String(n);if(this.logger.error("Failed to apply swap delta",{transactionId:e.transactionId,error:o}),this.config.strictValidation)throw n;return{updatedPool:this.pool,updatedTicks:{},amount0:new u(0),amount1:new u(0),feeAmount0:new u(0),feeAmount1:new u(0),ticksCrossed:[],metadata:{calculationTimeMs:Date.now()-t,swapSteps:0,priceHitLimit:!1}}}}forceFullRefetch(e){this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata.lastFullRefetch=Date.now(),this.metadata.swapsSinceRefetch=0,this.metadata.cumulativeDrift=0,this.logger.info("Full refetch forced",{sqrtPrice:this.pool.sqrtPrice.toString()})}getPoolState(){return this.pool}getTickDataMap(){return{...this.tickDataMap}}getMetadata(){return{...this.metadata}}getLastSwapMetrics(){if(this.lastSwapMetrics)return{...this.lastSwapMetrics}}isRefetchRecommended(){return this.metadata.swapsSinceRefetch>this.config.maxSwapsSinceRefetch||this.metadata.cumulativeDrift>this.config.maxCumulativeDrift}reset(e){this.pool=e.pool,this.tickDataMap=e.tickDataMap,this.metadata={lastFullRefetch:Date.now(),swapsSinceRefetch:0,cumulativeDrift:0,processedTransactions:[],stateHash:fr.calculatePoolStateHash(this.pool)},this.logger.info("PoolStateManager reset",{sqrtPrice:this.pool.sqrtPrice.toString()})}}function yr(e,t){return new fr(e,t)}class wr extends Cn{constructor(e=!1){super(e),this.cache={},this.cacheExpiry=36e5}async resolveTokenMetadata(e){const t=this.getCacheKey(e),n=this.cache[t];if(n&&!this.isCacheExpired(n.timestamp))return this.logger.debug(`Using cached metadata for token: ${t}`),n.data;const o=this.extractMetadata(e);return this.cache[t]={data:o,timestamp:Date.now()},o}async getTokenSymbol(e){return(await this.resolveTokenMetadata(e)).symbol}async getTokenDecimals(e){return(await this.resolveTokenMetadata(e)).decimals}clearCache(e){e?(delete this.cache[e],this.logger.debug(`Cleared cache for token: ${e}`)):(this.cache={},this.logger.debug("Cleared all token metadata cache"))}getCacheStats(){const e=Object.keys(this.cache);return{size:e.length,entries:e}}getCacheKey(e){if("string"==typeof e)return e.toLowerCase();return`${e.type||e.symbol||"unknown"}|${e.additionalKey||"none"}`.toLowerCase()}extractMetadata(e){let t,n,o="Token",i="Unit",r="none";if("string"==typeof e)if(e.includes("|")){const a=e.split("|");"Token"===a[0]&&a[2]?(o=a[0],i=a[1]||"Unit",t=a[2],r=a[3]||"none",n=t):(t=a[0],n=t,i=a[1]||"Unit",r=a[3]||"none")}else t=e,n=e;else t=e.type||"unknown",o=e.collection||"Token",i=e.category||"Unit",r=e.additionalKey||"none",n=e.symbol||("Token"===o?t:o)||"unknown";const a=this.getDecimalsForToken(n);return{symbol:n.toUpperCase(),decimals:a,collection:o,category:i,type:t.toUpperCase(),additionalKey:r,verified:!1}}getDecimalsForToken(e){return{GALA:8,GUSDC:6,USDC:6,USDT:6,WETH:18,DAI:18}[e.toUpperCase()]??18}isCacheExpired(e){return Date.now()-e>this.cacheExpiry}setCacheExpiry(e){this.cacheExpiry=e,this.logger.debug(`Set token metadata cache expiry to ${e}ms`)}}"undefined"!=typeof process&&process.env&&(process.env.CORE_CHAINCODE_LOGGING_LEVEL=process.env.CORE_CHAINCODE_LOGGING_LEVEL||"ERROR");var kr=Object.freeze({__proto__:null,extractMetadataFromPoolData:function(e,t){const n={};if(e.vaultAddress&&(n.vaultAddress=e.vaultAddress),void 0!==e.reverseBondingCurveMinFeePortion){const o=parseFloat(e.reverseBondingCurveMinFeePortion);isNaN(o)?t&&t.debug(`Skipping invalid reverseBondingCurveMinFeePortion for ${e.tokenName}: "${e.reverseBondingCurveMinFeePortion}"`):n.reverseBondingCurveMinFeeFactor=o}if(void 0!==e.reverseBondingCurveMaxFeePortion){const o=parseFloat(e.reverseBondingCurveMaxFeePortion);isNaN(o)?t&&t.debug(`Skipping invalid reverseBondingCurveMaxFeePortion for ${e.tokenName}: "${e.reverseBondingCurveMaxFeePortion}"`):n.reverseBondingCurveMaxFeeFactor=o}return void 0!==n.reverseBondingCurveMaxFeeFactor&&void 0!==n.reverseBondingCurveMinFeeFactor&&(n.reverseBondingCurveNetFeeFactor=n.reverseBondingCurveMaxFeeFactor-n.reverseBondingCurveMinFeeFactor),Object.keys(n).length>0?n:null},isValidPoolForCaching:function(e){if(null===e||"object"!=typeof e)return!1;const t=e;return"tokenName"in t&&"string"==typeof t.tokenName&&t.tokenName.length>0}});export{or as AgentConfig,_i as CALCULATION_MODES,yi as COMPOSITE_POOL_FETCH_CONCURRENCY,j as ConfigurationError,J as DexPoolNotFoundError,Y as DexQuoteError,Zn as FileValidationError,Mo as GALA_DECIMALS,qo as GALA_TOKEN_CLASS_KEY,X as GSwapAssetError,Q as GSwapPoolError,z as GSwapQuoteError,V as GSwapSwapError,Xe as IMAGE_EXTENSIONS,Ro as LAUNCHPAD_TOKEN_DECIMALS,Xi as LaunchpadSDK,Do as LockError,Eo as LockErrorType,Nn as MAX_CONCURRENT_POOL_FETCHES,W as NetworkError,Dn as PAGINATION_DEFAULTS,Fn as POOL_FETCH_CONFIG,Oi as POOL_TYPES,fr as PoolStateManager,pr as QUERY_FIELD_NAMES,Yo as SDKTransactionStatus,Vi as SDK_VERSION,Ui as TRADING_TYPES,wr as TokenMetadataService,H as TransactionError,ji as TransactionFailedError,G as ValidationError,Gi as WebSocketError,Wi as WebSocketTimeoutError,ye as addressFormatSchema,ze as amountMethodSchema,He as amountTypeSchema,Ye as browserFileSchema,Je as bufferFileSchema,bt as buyTokensDataSchema,Nt as calculatePreMintDataSchema,je as checkPoolOptionsSchema,Ji as createLaunchpadSDK,Ie as createLimitSchema,ut as createPaginatedResultSchema,yr as createPoolStateManager,vt as createTradeDataSchema,Yi as createWallet,we as ethereumAddressSchema,gt as fetchGalaBalanceOptionsSchema,Dt as fetchPoolDetailsDataSchema,yt as fetchTokenBalanceOptionsSchema,Fe as fileSizeSchema,Ze as fileUploadSchema,xe as filenameSchema,lr as filterByFeeTier,ir as filterByLiquidity,rr as filterByMinLiquidity,hr as filterByPoolKey,cr as filterByToken,sr as filterByTokenPair,ke as flexibleAddressSchema,et as flexibleFileSchema,Go as formatGalaForDTO,Wo as formatLaunchpadTokenForDTO,fe as fullNameSchema,Pt as getAmountOptionsSchema,nr as getEnv,tr as getEnvOrThrow,St as getTradeOptionsSchema,Qe as graduateTokenOptionsSchema,Ve as graphDataOptionsSchema,dr as groupByFeeTier,gr as groupByPoolKey,ur as groupByTokenPair,tt as imageExtensionSchema,nt as imageFilenameSchema,Ce as imageMimeTypeSchema,Ge as imageUploadOptionsSchema,Po as isLockTokenData,No as isUnlockTokenData,$e as isoDateStringSchema,Ke as launchTokenDataSchema,er as loadEnvWithFallback,Te as nonNegativeDecimalStringSchema,Ae as optionalUrlSchema,Ee as pageNumberSchema,lt as paginationResultMetaSchema,We as poolFetchTypeSchema,at as poolPaginationSchema,be as positiveDecimalStringSchema,Be as privateKeySchema,uo as requireNonNegative,lo as requirePositive,ho as requirePositiveWithContext,qe as reverseBondingCurveConfigSchema,Ft as reverseBondingCurveConfigurationSchema,me as searchQuerySchema,Tt as sellTokensDataSchema,ar as sortByLiquidity,Pe as standardLimitSchema,ot as standardPaginationSchema,Le as timestampSchema,Me as tokenCategorySchema,Re as tokenCollectionSchema,ge as tokenDescriptionSchema,ft as tokenHoldSchema,ht as tokenListOptionsSchema,de as tokenNameSchema,he as tokenSymbolSchema,_e as tokenUrlsSchema,It as tradeCalculationMethodSchema,Et as tradeCalculationTypeSchema,De as tradeLimitSchema,At as tradeListParamsSchema,rt as tradePaginationSchema,ct as tradePaginationWithFiltersSchema,kt as tradeTypeBackendSchema,wt as tradeTypeSchema,Ue as transactionIdSchema,Oe as uniqueKeySchema,pt as updateProfileDataSchema,mt as uploadProfileImageOptionsSchema,Se as urlSchema,Ne as userLimitSchema,it as userPaginationSchema,pe as userTokenNameSchema,dt as userTokenTypeSchema,st as userTokensPaginationSchema,Bt as validateAddress,Ot as validateAmountString,Zt as validateBuyTokensData,nn as validateCalculatePreMintData,Wt as validateCheckPoolOptions,Xt as validateCreateTradeData,Ht as validateFetchGalaBalanceOptions,on as validateFetchPoolDetailsData,Qt as validateFetchTokenBalanceOptions,_t as validateFullName,tn as validateGetAmountOptions,Jt as validateGetTradeOptions,Gt as validateImageUploadOptions,qt as validateLaunchTokenData,Mt as validateSearchQuery,Yt as validateSellTokensData,Lt as validateTokenDescription,jt as validateTokenListOptions,Ct as validateTokenName,$t as validateTokenSymbol,Kt as validateTokenUrls,en as validateTradeListParams,zt as validateUpdateProfileData,Vt as validateUploadProfileImageOptions,Rt as validateUserTokenName,Ut as validateVaultAddress,ve as vaultAddressSchema};