@gala-chain/launchpad-sdk 3.7.0 โ†’ 3.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("ethers"),require("axios"),require("@gala-chain/connect"),require("zod"),require("@gala-chain/api"),require("bignumber.js"),require("uuid"),require("socket.io-client")):"function"==typeof define&&define.amd?define(["exports","ethers","axios","@gala-chain/connect","zod","@gala-chain/api","bignumber.js","uuid","socket.io-client"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).GalaLaunchpadSDK={},e.ethers,e.axios,e.GalaChainConnect,e.z,e.GalaChainAPI,e.BigNumber,e.uuid,e.io)}(this,function(e,t,a,n,r,s,o,i,c){"use strict";var l;!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 d extends Error{constructor(e,t,a){super(t),this.type=e,this.originalError=a,this.name="AuthError"}}class u{constructor(e){if(this.wallet=e.wallet,this.messagePrefix=e.messagePrefix||"Create a GalaChain Wallet",""===e.messagePrefix)throw new d(l.SIGNATURE_FAILED,"Message prefix cannot be empty");this.validateWallet()}async generateSignature(){try{const e=Date.now(),t=`${this.messagePrefix} ${e}`,a=await this.wallet.signMessage(t);return{message:t,signature:a,address:this.formatAddress(this.wallet.address),timestamp:e}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate signature for authentication",e instanceof Error?e:new Error(String(e)))}}getAddress(){return this.formatAddress(this.wallet.address)}getEthereumAddress(){return this.wallet.address}formatAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new d(l.INVALID_ADDRESS,`Invalid Ethereum address format: ${e}`);return`eth|${t}`}async signMessage(e){try{return{message:e,signature:await this.wallet.signMessage(e),address:this.wallet.address,timestamp:Date.now()}}catch(e){const t=e instanceof Error?e.message:String(e);throw new d(l.SIGNATURE_FAILED,t,e instanceof Error?e:new Error(String(e)))}}async generateAuthHeaders(e,t){try{const a=Date.now(),n=`${this.messagePrefix} ${t.toUpperCase()} ${e} ${a}`,r=await this.wallet.signMessage(n);return{"x-signature":r,"x-address":this.formatAddress(this.wallet.address),"x-message":n,"x-timestamp":a.toString()}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate authentication headers",e instanceof Error?e:new Error(String(e)))}}async signTypedData(e,t,a){try{return await this.wallet.signTypedData(e,t,a)}catch(e){throw new d(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 d(l.SIGNATURE_FAILED,"Custom message must be a non-empty string");try{const t=await this.wallet.signMessage(e);return{message:e,signature:t,address:this.formatAddress(this.wallet.address),timestamp:Date.now()}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate custom message signature",e instanceof Error?e:new Error(String(e)))}}validateWallet(){if(!this.wallet)throw new d(l.WALLET_NOT_CONNECTED,"Wallet is required for authentication");if(!this.wallet.address)throw new d(l.WALLET_NOT_CONNECTED,"Wallet address is not available");if(!this.wallet.privateKey&&!this.wallet.signMessage)throw new d(l.WALLET_NOT_CONNECTED,"Wallet must have a private key for signing messages")}}function h(e){const t={};for(const[a,n]of Object.entries(e))null!=n&&("string"==typeof n?t[a]=n:"number"==typeof n||"boolean"==typeof n?t[a]=n.toString():Array.isArray(n)?t[a]=n.join(","):t[a]="object"==typeof n?JSON.stringify(n):String(n));return t}class m{constructor(e,t={}){this.auth=e,this.debug=t.debug??!1,this.axios=a.create({baseURL:t.baseUrl||"https://lpad-backend-dev1.defi.gala.com",timeout:t.timeout||3e4,headers:{Accept:"application/json",...t.headers}}),this.setupInterceptors()}async request(e){try{const t={method:e.method,url:e.url,data:e.data,...e.params&&{params:h(e.params)},...e.headers&&{headers:e.headers},...e.timeout&&{timeout:e.timeout}};if(e.data instanceof FormData&&(t.headers&&t.headers["Content-Type"]&&delete t.headers["Content-Type"],this.debug),this.debug){e.data instanceof FormData||e.data}const a=await this.axios.request(t);return this.debug,a.data}catch(e){throw this.debug,e}}async get(e,t,a){return this.request({method:"GET",url:e,...t&&{params:t},...a&&{headers:a}})}async post(e,t,a){return this.request({method:"POST",url:e,data:t,...a&&{headers:a}})}async put(e,t,a){return this.request({method:"PUT",url:e,data:t,...a&&{headers:a}})}async delete(e,t,a){return this.request({method:"DELETE",url:e,...t&&{params:t},...a&&{headers:a}})}async patch(e,t,a){return this.request({method:"PATCH",url:e,data:t,...a&&{headers:a}})}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.auth.getEthereumAddress()}async signMessage(e){return(await this.auth.signMessage(e)).signature}async signTypedData(e,t,a){return await this.auth.signTypedData(e,t,a)}async signCustomMessage(e){try{const t=await this.auth.generateCustomSignature(e);return this.debug,{signature:t.signature,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}}catch(e){throw this.debug,new Error(`Failed to generate custom signature for message: ${e instanceof Error?e.message:"Unknown error"}`)}}async signWithGalaChain(e,t,a=n.SigningType.SIGN_TYPED_DATA){const r=this.auth.wallet.privateKey;if(!r)throw new Error("Wallet private key not available for @gala-chain signing");const s=new n.SigningClient(r);return await s.sign(e,t,a)}setupInterceptors(){this.axios.interceptors.request.use(async e=>{try{const t=await this.auth.generateSignature();return e.headers||(e.headers={}),e.headers.Sign=t.signature,e.data instanceof FormData||(e.headers["Content-Type"]="application/json"),this.debug,e}catch(e){throw this.debug,e}},e=>Promise.reject(e)),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.debug}else e.request,this.debug;return Promise.reject(e)})}}class g extends Error{constructor(e,t,a){super(e),this.field=t,this.code=a,this.name="ValidationError"}}function p(e,t,a){const{MIN_PAGE:n,MAX_PAGE:r,MIN_LIMIT:s,MAX_LIMIT:o}=a.PAGINATION;if("number"!=typeof e||e<n||e>r)throw new g(`Page must be a number between ${n} and ${r}`,"page","INVALID_PAGE");if("number"!=typeof t||t<s||t>o)throw new g(`Limit must be a number between ${s} and ${o}`,"limit","INVALID_LIMIT")}const f={ETH_ADDRESS:/^0x[0-9a-fA-F]{40}$/,BACKEND_ADDRESS:/^eth\|[0-9a-fA-F]{40}$/};function y(e){return"string"==typeof e&&e.trim().length>0}function w(e){return!(!e||"string"!=typeof e)&&(f.ETH_ADDRESS.test(e)||f.BACKEND_ADDRESS.test(e))}function k(e){return e.startsWith("0x")?`eth|${e.slice(2)}`:e}const A=r.z.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"),v=r.z.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"),T=r.z.string().min(1,"Token description is required").max(500,"Token description must be at most 500 characters"),E=r.z.string().min(1,"Token name must be at least 1 character").max(50,"Token name must be at most 50 characters"),b=r.z.string().min(1,"Search query must be at least 1 character").max(100,"Search query must be at most 100 characters"),N=r.z.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"),S=r.z.string().regex(f.BACKEND_ADDRESS,"Address must be in format: eth|[40-hex-chars]"),I=r.z.string().regex(f.ETH_ADDRESS,"Invalid Ethereum address format"),D=r.z.string().refine(e=>f.BACKEND_ADDRESS.test(e)||f.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),C=r.z.string().refine(e=>f.BACKEND_ADDRESS.test(e)||/^service\|Token\$Unit\$[A-Z0-9]+\$eth:[0-9a-fA-F]{40}\$launchpad$/.test(e),"Invalid vault address format"),F=r.z.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>0,"Amount must be greater than zero"),$=r.z.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>=0,"Amount must be zero or greater"),P=r.z.string().regex(/^(?!0+(\.0+)?$)\d+(\.\d+)?$/,"Amount must be a positive, non-zero number"),_=r.z.string().url("Must be a valid URL").regex(/^https?:\/\//,"URL must start with http:// or https://"),O=r.z.string().optional().refine(e=>!e||/^https?:\/\/.+\..+/.test(e),"Must be a valid URL if provided"),U=r.z.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 L(e=100){return r.z.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 x=L(100),R=L(20),B=L(20),M=r.z.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"),K=r.z.string().max(255,"Filename must be at most 255 characters"),G=r.z.enum(["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"]),z=r.z.string().min(1,"Comment message is required").max(500,"Comment must be at most 500 characters"),q=r.z.string().datetime("Must be a valid ISO 8601 date string"),V=r.z.number().int("Timestamp must be an integer").min(0,"Timestamp must be non-negative"),W=r.z.string().regex(/^0x[a-fA-F0-9]{64}$/,"Private key must be format: 0x + 64 hex characters"),j=r.z.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"),H=r.z.string().regex(/^galaconnect-operation-[a-z0-9-]+$/,"Unique key must be format: galaconnect-operation-{unique-id}"),Q=r.z.object({websiteUrl:O,telegramUrl:O,twitterUrl:O}).refine(e=>e.websiteUrl||e.telegramUrl||e.twitterUrl,"At least one social URL (website, telegram, or twitter) is required"),X=r.z.string().min(1,"Token category must not be empty").default("Unit"),Y=r.z.string().min(1,"Token collection must not be empty").default("Token"),J=r.z.object({minFeePortion:F,maxFeePortion:F}),Z=r.z.object({tokenName:A,tokenSymbol:v,tokenDescription:T,tokenImage:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer),r.z.string().url("Token image must be a valid URL")]).optional(),preBuyQuantity:$.default("0"),websiteUrl:O,telegramUrl:O,twitterUrl:O,tokenCategory:X,tokenCollection:Y,reverseBondingCurveConfiguration:J.optional(),privateKey:W.optional()}),ee=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),tokenName:A}),te=r.z.enum(["recent","popular"]),ae=r.z.object({tokenName:A.optional(),symbol:v.optional()}).refine(e=>e.tokenName||e.symbol,"At least one of tokenName or symbol is required"),ne=r.z.enum(["NATIVE","MEME"]),re=r.z.enum(["IN","OUT"]),se=r.z.object({from:r.z.number().int("From timestamp must be an integer").min(173e6,"From timestamp must be at least 173000000"),to:r.z.number().int("To timestamp must be an integer").min(173e6,"To timestamp must be at least 173000000"),resolution:r.z.number().int("Resolution must be an integer").min(1,"Resolution must be at least 1"),tokenName:A}),oe=[".png",".jpg",".jpeg",".gif",".webp",".svg"],ie=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),name:K,size:M,type:G}),ce=r.z.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"),le=r.z.instanceof(Buffer).refine(e=>e.length>=1&&e.length<=10485760,"Buffer size must be between 1 byte and 10MB"),de=r.z.union([ce,le]),ue=r.z.enum([".png",".jpg",".jpeg",".gif",".webp",".svg"]),he=K.refine(e=>{const t=e.slice(e.lastIndexOf(".")).toLowerCase();return oe.includes(t)},`Filename must end with one of: ${oe.join(", ")}`),me=r.z.object({page:U,limit:x}),ge=r.z.object({page:U,limit:R}),pe=r.z.object({page:U,limit:B}),fe=r.z.object({page:U,limit:L(50)}),ye=me.extend({type:r.z.enum(["recent","popular"]).optional(),tokenName:r.z.string().min(1).max(50).optional(),search:r.z.string().min(1).max(100).optional()}),we=ge.extend({tokenName:r.z.string().min(1).max(50).optional(),search:r.z.string().min(1).max(100).optional()}),ke=pe.extend({tradeType:r.z.enum(["BUY","SELL"]).optional(),tokenName:r.z.string().min(1).max(50).optional(),userAddress:r.z.string().regex(/^(0x[a-fA-F0-9]{40}|eth\|[a-fA-F0-9]{40})$/).optional(),startDate:r.z.string().datetime().optional(),endDate:r.z.string().datetime().optional(),sortOrder:r.z.enum(["ASC","DESC"]).default("DESC")}),Ae=r.z.object({page:r.z.number().int().min(1),limit:r.z.number().int().min(1),total:r.z.number().int().min(0),totalPages:r.z.number().int().min(0),hasNext:r.z.boolean(),hasPrevious:r.z.boolean()});const ve=r.z.enum(["all","DEFI","ASSET"]),Te=ge.extend({type:ve.optional(),address:D.optional(),search:b.optional(),tokenName:E.optional()}),Ee=r.z.object({walletAddress:D,amount:P}),be=r.z.object({address:D.optional(),refresh:r.z.boolean().optional()}),Ne=r.z.object({profileImage:r.z.string(),fullName:N,address:D,privateKey:W.optional()}),Se=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),address:D.optional(),privateKey:W.optional()}),Ie=r.z.object({address:D,tokenId:r.z.union([r.z.string(),r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string()}),r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string(),instance:r.z.string()})]).optional(),tokenClassKey:r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string()}).optional(),tokenName:E.optional()}).refine(e=>void 0!==e.tokenId||void 0!==e.tokenName,"At least one token identifier (tokenId or tokenName) is required"),De=r.z.enum(["buy","sell"]),Ce=r.z.enum(["BUY","SELL"]),Fe=r.z.object({tradeType:De,tokenAmount:F,vaultAddress:C,userAddress:D,slippageTolerance:F.optional(),deadline:r.z.number().int().positive().optional()}),$e=r.z.object({tokenSymbol:v,nativeTokenQuantity:F,expectedToken:F,maxAcceptableReverseBondingCurveFee:$.default("0").optional()}),Pe=r.z.object({tokenSymbol:v,tokenQuantity:F,expectedNativeToken:F,maxAcceptableReverseBondingCurveFee:$.default("0").optional()}),_e=pe.extend({tokenName:E.optional()}),Oe=r.z.object({page:r.z.number().int().min(1).max(1e3).default(1).optional(),limit:r.z.number().int().min(1).max(20).default(10).optional()}),Ue=r.z.enum(["NATIVE","MEME"]),Le=r.z.enum(["IN","OUT"]),xe=r.z.object({type:Ue,method:Le,vaultAddress:C,amount:F}),Re=r.z.object({nativeTokenQuantity:F}),Be=r.z.object({vaultAddress:C}),Me=r.z.object({minFeePortion:F,maxFeePortion:F});function Ke(e){return t=>{const a=e.safeParse(t);return{success:a.success,data:a.success?a.data:void 0,errors:a.success?void 0:a.error.errors.map(e=>e.message)}}}const Ge=Ke(A),ze=Ke(v),qe=Ke(T),Ve=Ke(D),We=Ke(C),je=Ke(F),He=Ke(P),Qe=Ke(N),Xe=Ke(b),Ye=Ke(E),Je=Ke(Z),Ze=Ke(Q),et=Ke(ee),tt=Ke(ae),at=Ke(Te),nt=Ke(Ee),rt=Ke(be),st=Ke(Ne),ot=Ke(Se),it=Ke(Ie),ct=Ke(Fe),lt=Ke($e),dt=Ke(Pe),ut=Ke(_e),ht=Ke(Oe),mt=Ke(xe),gt=Ke(Re),pt=Ke(Be);function ft(e,t){throw new g(e.join("; "),t,"VALIDATION_ERROR")}function yt(e){const t=Ge(e);!t.success&&t.errors&&ft(t.errors,"tokenName")}function wt(e){const t=ye.safeParse(e);t.success||ft(t.error.errors.map(e=>e.message),"pagination")}function kt(e){const t=tt(e);!t.success&&t.errors&&ft(t.errors,"options")}function At(e){const t=mt(e);!t.success&&t.errors&&ft(t.errors,"options")}function vt(e){const t=se.safeParse(e);t.success||ft(t.error.errors.map(e=>e.message),"options")}function Tt(e){const t=D.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}var Et=Object.freeze({__proto__:null,normalizeAddressInput:function(e){if(!e)return;const t=D.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:Tt,validateCheckPoolOptions:kt,validateGetAmountOptions:At,validateGetGraphOptions:vt,validatePagination:wt,validateTokenName:yt});function bt(e){if(!e)return[];return(Array.isArray(e)?e:e.token??[]).map(e=>({...e,createdAt:new Date(e.createdAt),updatedAt:new Date(e.updatedAt)}))}function Nt(e,t){const a=Number(e.page)||t.page,n=Number(e.limit)||t.limit,r=Number(e.total)||Number(e.data?.count)||0;return{page:a,limit:n,total:r,totalPages:Math.ceil(r/n)}}function St(e,t){return{hasNext:e<t,hasPrevious:e>1}}function It(e,t,a=!1){const n=!0===e.error||200!==e.status,r=a&&!e.data;if(n||r)throw new Error(e.message||t)}const Dt="/launchpad/upload-image",Ct="/launchpad/fetch-pool",Ft="/launchpad/check-pool",$t="/launchpad/get-graph-data",Pt="/holders",_t="/launchpad/get-badge/",Ot="/trade/",Ut="/token/commment",Lt="/token/commment",xt="/user/profile",Rt="/user/profile",Bt="/user/token-list",Mt="/user/token-hold",Kt="/user/transfer-faucets";function Gt(e){return e&&"object"==typeof e&&"string"==typeof e.tokenName&&(void 0===e.from||"number"==typeof e.from)&&(void 0===e.to||"number"==typeof e.to)&&(void 0===e.resolution||"number"==typeof e.resolution)}class zt{constructor(e){this.http=e}async fetchPools(e={}){let t;wt({page:e.page??1,limit:e.limit??10}),e.tokenName&&yt(e.tokenName),"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const a={page:e.page||1,limit:e.limit||10};e.search&&(a.search=e.search),e.tokenName&&(a.tokenName=e.tokenName),t&&(a.type=t);const n={page:a.page.toString(),limit:a.limit.toString()};void 0!==a.type&&(n.type=a.type),void 0!==a.tokenName&&(n.tokenName=a.tokenName),void 0!==a.search&&(n.search=a.search);const r=h(n),s=await this.http.get(Ct,r);It(s,"Failed to fetch pools",!0);return{pools:function(e){if(!e)return[];let t=[];return e.tokens?t=Array.isArray(e.tokens)?e.tokens.map(e=>({...e,createdAt:new Date(e.created_at??e.createdAt)})):[{...e.tokens,createdAt:new Date(e.tokens.created_at??e.tokens.createdAt)}]:e.pools&&Array.isArray(e.pools)&&(t=e.pools.map(e=>({...e,createdAt:new Date(e.created_at??e.createdAt)}))),t}(s.data),page:s.data.page,limit:s.data.limit,total:s.data.total,totalPages:s.data.totalPages,hasNext:s.data.page<s.data.totalPages,hasPrevious:s.data.page>1}}async checkPool(e){kt(e),e.tokenName&&yt(e.tokenName);const t=h(e),a=await this.http.get(Ft,t);It(a,"Failed to check pool");const n=a.data;return e.symbol?n?.isSymbolExist??!1:e.tokenName?n?.isNameExist??!1:n?.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(!Gt(e))throw new g("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:a,to:n,resolution:r}=e;if(yt(t),!a||!n||!r)throw new g("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const s={tokenName:t,from:a,to:n,resolution:r};vt(s);const o=h(s),i=await this.http.get($t,o);return It(i,"Failed to fetch graph data",!0),{dataPoints:i.data}}async fetchTokenDistribution(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.resolveTokenNameToVault(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);const a=encodeURIComponent(t),n=await this.http.get(`${Pt}/${a}`);return It(n,"Failed to fetch token distribution",!0),{holders:n.data.holders||[],totalSupply:n.data.totalSupply||"0",totalHolders:n.data.totalHolders||0,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.http.get(_t,{tokenName:e});return It(t,"Failed to fetch token badges",!0),{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadge(e){const{tokenName:t,badgeType:a,badgeName:n}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const r=("volume"===a?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===n);return r?.isActive||!1}catch{return!1}}async resolveTokenNameToVault(e){try{const t=await this.fetchPools({tokenName:e});return t.pools&&Array.isArray(t.pools)&&t.pools.length>0?t.pools[0].vaultAddress||null:t.pools&&t.pools.tokens&&t.pools.tokens.vaultAddress||null}catch{return null}}}function qt(e,t={}){const{stringifyFields:a=[],optionalFields:n=[],fieldMappings:r={}}=t,s={};for(const[t,o]of Object.entries(e)){const e=t;if(n.includes(e)&&void 0===o)continue;if(n.includes(e)&&"string"==typeof o&&0===o.trim().length)continue;const i=r[e],c=i?String(i):t;a.includes(e)?s[c]=String(o):s[c]=o}return h(s)}const Vt={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20}};class Wt{constructor(e){this.http=e}async fetchTrades(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||void 0!==t.tradeType&&"buy"!==t.tradeType&&"sell"!==t.tradeType||void 0!==t.userAddress&&"string"!=typeof t.userAddress||void 0!==t.page&&"number"!=typeof t.page||void 0!==t.limit&&"number"!=typeof t.limit)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");var t;const{tokenName:a,tradeType:n,userAddress:r,page:s=1,limit:o=10,startDate:i,endDate:c,sortOrder:l}=e;if(!y(a))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");p(s,o,Vt);const d=function(e,t,a){return qt({tokenName:e,page:t,limit:a},{stringifyFields:["page","limit"]})}(a,s,o),u=await this.http.get(Ot,d),h=(m=u.data)?(Array.isArray(m)?m:m.trades?m.trades:[]).map(e=>({...e,createdAt:new Date(e.createdAt),updatedAt:new Date(e.updatedAt)})):[];var m;const f=Nt(u,{page:s,limit:o}),w=St(f.page,f.totalPages);return{trades:h,...f,...w}}}function jt(e,t){if(!e)return t||new Date;if(e instanceof Date)return isNaN(e.getTime())?t||new Date:e;try{const a=new Date(e);return isNaN(a.getTime())?t||new Date:a}catch(e){return t||new Date}}function Ht(e,t){return new g(`Token "${e}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND")}const Qt={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:50},CONTENT:{MIN_LENGTH:1,MAX_LENGTH:500}};class Xt{constructor(e,t){this.http=e,this.poolService=t}async fetchComments(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||void 0!==t.page&&"number"!=typeof t.page||void 0!==t.limit&&"number"!=typeof t.limit)throw new g("Invalid options provided. Expected { tokenName: string, page?: number, limit?: number }","options","INVALID_OPTIONS");var t;const{tokenName:a,page:n=1,limit:r=10}=e;if(!y(a))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");p(n,r,Qt);const s=await this.poolService.resolveTokenNameToVault(a);if(!s)throw Ht(a);const o=qt({vaultAddress:s,page:n,limit:r},{stringifyFields:["page","limit"]}),i=await this.http.get(Ut,o);It(i,"Failed to fetch comments");return{comments:i.data.comments.map(e=>({...e,createdAt:jt(e.createdAt)})),total:i.data.count}}async postComment(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||"string"!=typeof t.content)throw new g("Invalid options provided. Expected { tokenName: string, content: string }","options","INVALID_OPTIONS");var t;const{tokenName:a,content:n}=e;if(!y(a))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>=Qt.CONTENT.MIN_LENGTH&&t.length<=Qt.CONTENT.MAX_LENGTH}(n))throw new g(`Comment content must be between ${Qt.CONTENT.MIN_LENGTH} and ${Qt.CONTENT.MAX_LENGTH} characters`,"content","INVALID_CONTENT");const r=await this.poolService.resolveTokenNameToVault(a);if(!r)throw Ht(a);const s={userAddress:this.http.getAddress(),vaultAddress:r,content:n};It(await this.http.post(Lt,s),"Failed to create comment")}}const Yt={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 Jt(e){return!(!e||"string"!=typeof e)&&Yt.USER_ADDRESS.PATTERN.test(e)}const Zt="Update profile";class ea{constructor(e){this.http=e}async fetchProfile(e){const t=e??this.http.getAddress();if(!Jt(t))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");const a={userAddress:t};return await this.http.get(xt,a)}async updateProfile(e){this.validateUpdateProfileData(e);let t=e.profileImage;if(!t||""===t.trim())try{const a=await this.fetchProfile(e.address);t=a.data?.profileImage||""}catch{t=""}const a={profileImage:t,fullName:e.fullName,userAddress:e.address},n=await this.http.signCustomMessage(Zt),r={address:n.address,message:Zt,publickey:n.ethereumAddress,sign:n.signature};It(await this.http.put(Rt,a,r),"Profile update failed")}async uploadProfileImage(e){this.validateUploadProfileImageOptions(e);const t=e.address??this.http.getAddress();try{const a=new FormData;if("undefined"!=typeof File&&e.file instanceof File)a.append("image",e.file);else{if(!Buffer.isBuffer(e.file))throw new g("Invalid file type","file","INVALID_FILE_TYPE");{const n=`profile-image-${t}.png`,r=new Blob([e.file],{type:"image/png"});a.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`${Dt}?tokenName=${encodeURIComponent(t)}`,data:a,headers:{}});return It(n,"Image upload failed"),"string"==typeof n.data?n.data:""}catch(e){if(e instanceof g)throw e;throw new g(`Profile image upload failed: ${e instanceof Error?e.message:"Unknown error"}`,"file","UPLOAD_FAILED")}}async fetchTokenList(e){this.validateGetTokenListOptions(e);const t=qt({page:e.page,limit:e.limit,type:"all"!==e.type&&e.type?e.type:"DEFI",address:e.address,search:e.search,tokenName:e.tokenName},{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),a=await this.http.get(Bt,t);It(a,"Failed to fetch token list",!0);const n=bt(a.data),r=Nt(a,{page:e.page||1,limit:e.limit||10}),s=St(r.page,r.totalPages);return{tokens:n,...r,...s}}async fetchTokensHeld(e){this.validateGetTokenListOptions(e);const t=qt({page:e.page,limit:e.limit,address:e.address,search:e.search,tokenName:e.tokenName},{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),a=await this.http.get(Mt,t);It(a,"Failed to fetch tokens held",!0);const n=bt(a.data),r=Nt(a,{page:e.page||1,limit:e.limit||10}),s=St(r.page,r.totalPages);return{tokens:n,...r,...s}}async fetchTokensCreated(e={}){const{page:t=1,limit:a=10,search:n,tokenName:r}=e,s={type:"DEFI",address:this.http.getAddress(),page:t,limit:a};return void 0!==n&&(s.search=n),void 0!==r&&(s.tokenName=r),this.fetchTokenList(s)}validateGetTokenListOptions(e){if(p(e.page,e.limit,Yt),void 0!==e.address&&!Jt(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>=Yt.SEARCH.MIN_LENGTH&&t.length<=Yt.SEARCH.MAX_LENGTH))throw new g(`Search query must be between ${Yt.SEARCH.MIN_LENGTH} and ${Yt.SEARCH.MAX_LENGTH} characters`,"search","INVALID_SEARCH");var t,a;if(void 0!==e.tokenName&&e.tokenName.trim().length>0&&!((a=e.tokenName)&&"string"==typeof a&&a.length>=Yt.TOKEN_NAME.MIN_LENGTH&&a.length<=Yt.TOKEN_NAME.MAX_LENGTH))throw new g(`Token name must be between ${Yt.TOKEN_NAME.MIN_LENGTH} and ${Yt.TOKEN_NAME.MAX_LENGTH} characters`,"tokenName","INVALID_TOKEN_NAME")}validateUpdateProfileData(e){if(!Jt(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>=Yt.PROFILE.FULL_NAME.MIN_LENGTH&&t.length<=Yt.PROFILE.FULL_NAME.MAX_LENGTH&&Yt.PROFILE.FULL_NAME.ALPHABETS_ONLY_PATTERN.test(t)))throw new g(`Full name must be between ${Yt.PROFILE.FULL_NAME.MIN_LENGTH} and ${Yt.PROFILE.FULL_NAME.MAX_LENGTH} characters`,"fullName","INVALID_FULL_NAME");var t}validateUploadProfileImageOptions(e){if(e.address&&!Jt(e.address))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS")}}class ta extends Error{constructor(e,t,a){super(e),this.filename=t,this.mimeType=a,this.name="FileValidationError"}}function aa(e,t,a){if(!e)throw new ta("File is required",t,a);if("undefined"!=typeof File&&e instanceof File){const t=ce.safeParse(e);if(!t.success){const a=t.error.errors.map(e=>e.message).join("; ");throw new ta(a,e.name,e.type)}return}if(Buffer.isBuffer(e)){if(!t)throw new ta("Filename is required when uploading Buffer objects",t,a);const n=le.safeParse(e);if(!n.success){const e=n.error.errors.map(e=>e.message).join("; ");throw new ta(e,t,a)}if(t.length>255)throw new ta(`Filename length ${t.length} exceeds maximum allowed length of 255 characters`,t,a);const r=["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"];if(!r.includes(a))throw new ta(`Invalid file type "${a}" is not allowed. Allowed types: ${r.join(", ")}`,t,a);const s=function(e){if(!e)return"";const t=e.lastIndexOf(".");if(-1===t)return"";return e.substring(t).toLowerCase()}(t),o=[".png",".jpg",".jpeg",".gif",".webp",".svg"];if(!o.includes(s))throw new ta(`File extension "${s}" is not allowed. Allowed extensions: ${o.join(", ")}`,t,a);const i=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"}}(s);if(i!==a&&"application/octet-stream"!==i)throw new ta(`File extension "${s}" does not match MIME type "${a}"`,t,a);return}throw new ta("File must be a File object (browser) or Buffer (Node.js)",t,a)}class na{constructor(e){this.http=e}async uploadImageByTokenName(e){const{tokenName:t,options:a}=e;yt(t);const n=`${t}.png`;aa(a.file,n,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&a.file instanceof File)e.append("image",a.file);else{if(!Buffer.isBuffer(a.file))throw new Error("Unsupported file type");{const n=`${a.tokenName??t}.png`,r=new Blob([a.file],{type:"image/png"});e.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`${Dt}?tokenName=${encodeURIComponent(a.tokenName??t)}`,data:e,headers:{}});return It(n,"Image upload failed"),"string"==typeof n.data?n.data:""}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw new Error("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.");throw e}}}class ra{constructor(e){this.http=e}async transferFaucets(e){this.validateTransferFaucetsData(e);const t={userAddress:e.walletAddress,amount:e.amount};It(await this.http.post(Kt,t),"Faucet transfer failed")}validateTransferFaucetsData(e){if(!Jt(e.walletAddress))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(!(t=e.amount)||"string"!=typeof t||!Yt.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 sa{constructor(e){this.http=e,this.poolService=new zt(e),this.tradeService=new Wt(e),this.commentService=new Xt(e,this.poolService),this.userService=new ea(e),this.imageService=new na(e),this.faucetService=new ra(e)}async uploadImageByTokenName(e){return this.imageService.uploadImageByTokenName(e)}async fetchPools(e={}){return this.poolService.fetchPools(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 yt(e)}}class oa extends n.ChainCallDTO{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,a,n,r){let s;if("string"==typeof n){const e=n.split("$");if(4!==e.length)throw new Error(`Invalid token class key format: ${n}. Expected format: collection$category$type$additionalKey`);s={collection:e[0],category:e[1],type:e[2],additionalKey:e[3],instance:"0"}}else s={collection:n.collection,category:n.category,type:n.type,additionalKey:n.additionalKey,instance:"0"};return new oa({from:e,to:t,quantity:a,tokenInstance:s,uniqueKey:r||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,a,n){return new oa({from:e,to:t,quantity:a,tokenInstance:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"},uniqueKey:n||`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 ia{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},a={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,a,e),domain:t,types:a,signerPublicKey:this.wallet.signingKey.publicKey}}static toGalaChainAddress(e){const a=e.replace("0x","");return`eth|${t.ethers.getAddress(`0x${a}`).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){const t=e.split("$");if(4!==t.length)throw new Error(`Invalid token class key format: ${e}. Expected format: collection$category$type$additionalKey`);return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3],instance:"0"}}}function ca(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")}var la,da=Object.freeze({__proto__:null,normalizeToTokenInstanceKey:ca});!function(e){e.DEBUG="DEBUG",e.INFO="INFO",e.WARN="WARN",e.ERROR="ERROR"}(la||(la={}));class ua{constructor(e){this.levelPriority={[la.DEBUG]:0,[la.INFO]:1,[la.WARN]:2,[la.ERROR]:3},this.debugEnabled=e.debug,this.context=e.context||"SDK",this.minLevel=e.minLevel||(e.debug?la.DEBUG:la.INFO)}debug(e,t){this.log(la.DEBUG,e,t)}info(e,t){this.log(la.INFO,e,t)}warn(e,t){this.log(la.WARN,e,t)}error(e,t){this.log(la.ERROR,e,t)}log(e,t,a){if(this.levelPriority[e]<this.levelPriority[this.minLevel])return;if(e===la.DEBUG&&!this.debugEnabled)return;const n=`[${(new Date).toISOString()}] [${this.context}] [${e}]`,r=this.getConsoleMethod(e);void 0!==a?a instanceof Error?r(`${n} ${t}`,a.message,a.stack):r(`${n} ${t}`,a):r(`${n} ${t}`)}getConsoleMethod(e){switch(e){case la.DEBUG:return console.debug;case la.INFO:return console.info;case la.WARN:return console.warn;case la.ERROR:return console.error;default:return console.log}}child(e){return new ua({debug:this.debugEnabled,context:`${this.context}:${e}`,minLevel:this.minLevel})}isDebugEnabled(){return this.debugEnabled&&this.levelPriority[la.DEBUG]>=this.levelPriority[this.minLevel]}}const ha={MAX_UNIQUE_KEY_LENGTH:64,UNIQUE_KEY_PATTERN:/^(galaswap-operation-|galaconnect-operation-)/,TOKEN_NAME_PATTERN:/^[a-zA-Z0-9]+$/};var ma;!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"}(ma||(ma={}));class ga extends Error{constructor(e,t,a){super(e),this.type=t,this.details=a,this.name="TransferError"}}class pa{constructor(e,t,a,n=!1){this.http=e,this.wallet=t,this.tokenResolver=a,this.signatureHelper=new ia(t),this.logger=new ua({debug:n,context:"GalaChainService"})}async fetchPoolDetails(e){this.validateFetchPoolDetailsData(e);const t=await this.http.post("/api/asset/launchpad-contract/FetchSaleDetails",e);if(1!==t.Status)throw new Error(`Failed to fetch pool details: Status ${t.Status}`);return t}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(1!==t.Status||!t.Data||0===t.Data.length)return null;const a=t.Data.find(t=>t.collection===e.collection&&t.category===e.category&&t.additionalKey===e.additionalKey&&t.type===e.type);if(!a||"0"===a.quantity)return null;const n=`${a.collection}|${a.category}|${a.additionalKey}|${a.type}`;return{quantity:a.quantity,collection:a.collection,category:a.category,tokenId:n}}catch(e){throw new Error(`Failed to fetch token balance from GalaChain: ${e.message||"Unknown error"}`)}}async transferGala(e){this.validateTransferGalaData(e);try{const t=k(e.recipientAddress),a=k(this.wallet.address),n=oa.forGALA(a,t,e.amount,e.uniqueKey),r=await this.signatureHelper.signTransferToken(n.toSigningPayload()),s=new oa({...n.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full GALA Transfer Request Payload:",JSON.stringify(s,null,2));const o=await this.http.post("/api/asset/token-contract/TransferToken",s);if(this.logger.debug("[DEBUG] Transfer response:",JSON.stringify(o,null,2)),o&&"object"==typeof o){if("Status"in o&&1===o.Status&&"Data"in o){const e=o;return Array.isArray(e.Data)&&e.Data.length>0?"gala-transfer-successful":"transfer-successful-no-id"}if("transactionId"in o&&o.transactionId)return o.transactionId}throw new ga("Transfer succeeded but transaction ID could not be extracted",ma.NETWORK_ERROR)}catch(t){throw this.handleTransferError(t,"GALA transfer failed",e)}}async transferToken(e){this.validateTransferTokenData(e);try{const t=k(e.to),a=k(this.wallet.address);let n;if(e.tokenId)n=ca(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId),this.logger.debug("[DEBUG] Normalized Token Instance:",JSON.stringify(n,null,2));else{if(!e.tokenName)throw new ga("Must provide either tokenId or tokenName for token identification",ma.TOKEN_NOT_FOUND);n=await this.resolveTokenInstance(e.tokenName)}const r=new oa({from:a,to:t,quantity:e.amount,tokenInstance:n,uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),s=await this.signatureHelper.signTransferToken(r.toSigningPayload()),o=new oa({...r.toSigningPayload(),signedPayload:s});this.logger.debug("[DEBUG] Full Transfer Request Payload:",JSON.stringify(o,null,2));const i=await this.http.post("/api/asset/token-contract/TransferToken",o);if(this.logger.debug("[DEBUG] Token transfer response:",JSON.stringify(i,null,2)),i&&"object"==typeof i){if("Status"in i&&1===i.Status&&"Data"in i){const e=i;return Array.isArray(e.Data)&&e.Data.length>0?"token-transfer-successful":"transfer-successful-no-id"}if("transactionId"in i&&i.transactionId)return i.transactionId}throw new ga("Transfer succeeded but transaction ID could not be extracted",ma.NETWORK_ERROR)}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 ga)throw t;throw new ga(`Failed to resolve token class key for '${e}': ${t instanceof Error?t.message:String(t)}`,ma.TOKEN_NOT_FOUND,{tokenName:e})}}validateTransferGalaData(e){if(!((t=e)&&"object"==typeof t&&"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)throw new g("Invalid GALA transfer data: missing required fields");var t;if(!w(e.recipientAddress))throw new ga("Invalid recipient address format",ma.INVALID_RECIPIENT,{recipientAddress:e.recipientAddress});if(parseFloat(e.amount)<=0)throw new ga("Transfer amount must be positive",ma.INVALID_AMOUNT,{amount:e.amount});if(e.uniqueKey){if(e.uniqueKey.length>ha.MAX_UNIQUE_KEY_LENGTH)throw new g(`Unique key too long. Maximum length: ${ha.MAX_UNIQUE_KEY_LENGTH}`);if(!ha.UNIQUE_KEY_PATTERN.test(e.uniqueKey))throw new ga('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',ma.INVALID_AMOUNT,{uniqueKey:e.uniqueKey})}}validateTransferTokenData(e){if(!((t=e)&&"object"==typeof t&&"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)throw new g("Invalid token transfer data: missing required fields");var t;if(!w(e.to))throw new ga("Invalid recipient address format",ma.INVALID_RECIPIENT,{recipientAddress:e.to});if(!e.tokenId&&!e.tokenName)throw new ga("Must provide either tokenId or tokenName for token identification",ma.TOKEN_NOT_FOUND);if(e.tokenName&&!ha.TOKEN_NAME_PATTERN.test(e.tokenName))throw new ga("Invalid token name format",ma.TOKEN_NOT_FOUND,{tokenName:e.tokenName});if(parseFloat(e.amount)<=0)throw new ga("Transfer amount must be positive",ma.INVALID_AMOUNT,{amount:e.amount});if(e.uniqueKey){if(e.uniqueKey.length>ha.MAX_UNIQUE_KEY_LENGTH)throw new g(`Unique key too long. Maximum length: ${ha.MAX_UNIQUE_KEY_LENGTH}`);if(!ha.UNIQUE_KEY_PATTERN.test(e.uniqueKey))throw new ga('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',ma.INVALID_AMOUNT,{uniqueKey:e.uniqueKey})}}async resolveTokenInstance(e){try{const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new ga(`Token '${e}' not found or not available for transfer`,ma.TOKEN_NOT_FOUND,{tokenName:e});const a=this.resolveTokenInstanceFromVaultAddress(t);return this.logger.debug(`[DEBUG] Token resolution for '${e}':\n Vault Address: ${t}\n Token Instance: ${JSON.stringify(a,null,2)}`),a}catch(t){if(t instanceof ga)throw t;throw new ga(`Failed to resolve token '${e}': ${t instanceof Error?t.message:String(t)}`,ma.TOKEN_NOT_FOUND,{tokenName:e})}}resolveTokenInstanceFromVaultAddress(e){const[t,a]=e.split("|");if(!a)throw new ga(`Invalid vault address format: missing token components. Address: ${e}`,ma.TOKEN_NOT_FOUND);const n=a.split("$");if(n.length<4)throw new ga(`Invalid vault address format: insufficient token components. Expected 4+, got ${n.length}. Address: ${e}`,ma.TOKEN_NOT_FOUND);return{collection:n[0],category:n[1],type:n[2],additionalKey:n[3],instance:"0"}}handleTransferError(e,t,a){if(e instanceof ga)return e;if(e instanceof g)return new ga(e.message,ma.INVALID_AMOUNT);if(400===e.response?.status)return new ga(e.response.data?.message||"Invalid transfer request",ma.INVALID_AMOUNT);if(403===e.response?.status)return new ga("Insufficient balance for transfer",ma.INSUFFICIENT_BALANCE);if(404===e.response?.status){const e={};return"tokenName"in a&&(e.tokenName=a.tokenName),new ga("Token not found",ma.TOKEN_NOT_FOUND,e)}return"ECONNABORTED"===e.code||"ETIMEDOUT"===e.code?new ga("Transfer request timed out",ma.NETWORK_ERROR):new ga(e.message||t,ma.NETWORK_ERROR)}}class fa{constructor(e){this.http=e}async fetchTokenSpotPrice(e){if(!e||Array.isArray(e)&&0===e.length)throw new Error("At least one symbol must be provided");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.http.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),a=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&a.push({symbol:e.symbol,price:e.currentPrices.usd})}),a}catch(e){throw new Error(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`)}}async fetchLaunchpadTokenSpotPrice(e,t){if(!e||"string"!=typeof e)throw new Error("Token name is required and must be a string");try{const a=await t({tokenName:e,amount:"1",type:"native"}),n=(await this.fetchTokenSpotPrice("GALA"))[0];if(!n)throw new Error("GALA price not available");const r=Number(a.amount)/1e18;if(r<=0)throw new Error(`Invalid token amount calculation: ${r}`);const s=n.price/r;return{symbol:e.toUpperCase(),price:s}}catch(t){if(t instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${String(t)}`)}}}function ya(e,t=18){const a=parseFloat(e);if(0===a)return"0";return a.toFixed(t).replace(/\.?0+$/,"")}function wa(e){return ya(e,8)}function ka(e){return ya(e,18)}class Aa extends n.ChainCallDTO{constructor(e,t,a="0",n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=wa(t),this.expectedToken=ka(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class va extends n.ChainCallDTO{constructor(e,t,a,n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=ka(t),this.expectedNativeToken=wa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class Ta extends n.ChainCallDTO{constructor(e,t,a="0",n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=ka(t),this.expectedNativeToken=wa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class Ea extends n.ChainCallDTO{constructor(e,t,a,n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=wa(t),this.expectedToken=ka(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}const ba={BuyNativeDto:Aa,BuyExactDto:va,SellExactDto:Ta,SellNativeDto:Ea};var Na,Sa,Ia;!function(e){e[e.METAMASK=0]="METAMASK",e[e.TRUST_WALLET=1]="TRUST_WALLET",e[e.GALA_WALLET=2]="GALA_WALLET"}(Na||(Na={}));class Da{constructor(e,t=!1){this.walletProvider=e,this.debug=t}async signDTO(e,t,a,r=Na.METAMASK){try{this.debug;const s=this.generateEIP712Types(t,e),o=n.calculatePersonalSignPrefix(e),i={...e,prefix:o};let c,l,d;switch(r){case Na.GALA_WALLET:({signature:c,domain:l}=await this.signWithGalaWallet(s,i,t,a));break;case Na.TRUST_WALLET:c=await this.signWithTrustWallet(i),l={name:"ethereum",chainId:1};break;case Na.METAMASK:default:({signature:c,domain:l}=await this.signWithMetaMask(s,i))}return d=r===Na.TRUST_WALLET?{...i,signature:c}:{...e,signature:c,types:s,domain:l},this.debug,d}catch(e){throw this.debug,new Error(`Failed to sign DTO: ${e.message}`)}}async signWithMetaMask(e,t){try{let a,n;if(this.walletProvider.signTypedData&&!this.walletProvider.getNetwork)a={name:"ethereum",chainId:1},n=await this.walletProvider.signTypedData(a,e,t);else{if(!this.walletProvider.getNetwork||!this.walletProvider.signTypedData)throw new Error("Wallet provider does not support typed data signing");{const r=await this.walletProvider.getNetwork();a={name:r.name,chainId:Number(r.chainId)},n=await this.walletProvider.signTypedData(a,e,t)}}return{signature:n,domain:a}}catch(e){throw new Error(`MetaMask/ethers signing failed: ${e.message}`)}}async signWithTrustWallet(e){try{const a=(t=e,JSON.stringify(t));let n;if(!this.walletProvider.signMessage)throw new Error("Wallet provider does not support signMessage");return n=await this.walletProvider.signMessage(a),n}catch(e){throw new Error(`TrustWallet signing failed: ${e.message}`)}var t}async signWithGalaWallet(e,t,a,n){try{const r={name:"ethereum",chainId:1};if("undefined"==typeof window)return this.debug,await this.signWithMetaMask(e,t);const s={domain:r,types:e,message:t,Primary_type:a},o=window;if(!o?.gala)throw new Error("GalaWallet not found in window object");await o.gala.setAddress(n);return{signature:await o.gala.request({method:"eth_signTypedData",params:[JSON.stringify(s),n]}),domain:r}}catch(e){throw new Error(`GalaWallet signing failed: ${e.message}`)}}generateEIP712Types(e,t){const a={};a[e]=[];const n=(e,t,r,s=!1)=>{if(void 0!==t){if(Array.isArray(t)){const o=n(e,t[0],r,!0);return s||a[r].push({name:e,type:(o??e)+"[]"}),o?o+"[]":void 0}if("object"==typeof t&&null!==t){if(a[e])throw new Error(`Type name collision not supported: ${e}`);return a[e]=[],Object.entries(t).forEach(([t,a])=>{n(t,a,e)}),s||a[r].push({name:e,type:e}),e}{let n;switch(typeof t){case"string":n="string";break;case"number":n="uint256";break;case"boolean":n="bool";break;default:throw new Error(`Unsupported type for fieldName ${e}: ${typeof t}, value: ${t}`)}return s||a[r].push({name:e,type:n}),n}}};return Object.entries(t).forEach(([t,a])=>{n(t,a,e)}),this.debug,a}detectWalletPreference(){if("undefined"==typeof window)return Na.METAMASK;const e=window;return e?.gala?Na.GALA_WALLET:e?.trustWallet?.isTrust?Na.TRUST_WALLET:Na.METAMASK}}class Ca{constructor(e=!1){this.debug=e}generateStringsInstructions(e){try{this.debug;const t=this.extractTokenSymbolFromVault(e),a=this.createTokenInstance(t),n=this.createGalaInstance(),r=`$service$${a.toStringKey()}$launchpad`,s=`$tokenBalance$${a.toStringKey()}$${e}`,o=`$tokenBalance$${a.toStringKey()}$${e}`,i=`$tokenBalance$${n.toStringKey()}$${e}`,c=[r,s,o,i,`$tokenBalance$${n.toStringKey()}$${e}`];return this.debug,c}catch(e){throw this.debug,new Error(`Failed to generate stringsInstructions: ${e.message}`)}}createTokenInstance(e){const t=new s.TokenClassKey;return t.collection=e.toLowerCase(),t.category="Unit",t.type="none",t.additionalKey="none",this.debug,t}createGalaInstance(){const e=new s.TokenClassKey;return e.collection="GALA",e.category="Unit",e.type="none",e.additionalKey="none",this.debug,e}extractTokenSymbolFromVault(e){if(!e||"string"!=typeof e)throw new Error("Vault address is required and must be a string");const t=e.split("$");if(t.length<3)throw new Error(`Invalid vault address format: ${e}`);const a=t[2];if(!a||0===a.trim().length)throw new Error(`Empty token symbol in vault address: ${e}`);return this.debug,a}validateVaultAddress(e){if(!e||"string"!=typeof e)throw new Error("Vault address must be a non-empty string");if(!e.startsWith("service|Token$Unit$"))throw new Error('Vault address must start with "service|Token$Unit$"');if(!e.endsWith("$launchpad"))throw new Error('Vault address must end with "$launchpad"');const t=e.split("$");if(t.length<5)throw new Error('Vault address must have at least 5 parts separated by "$"');const a=t[2];if(!a||!/^[A-Za-z]{1,10}$/.test(a))throw new Error("Token symbol must be 1-10 letters (case insensitive)");return this.debug,!0}generateTokenClassKeyString(e,t,a,n){return`${e}$${t}$${a}$${n}`}parseTokenClassKeyString(e){const t=e.split("$");if(4!==t.length)throw new Error(`Invalid TokenClassKey string format: ${e}`);return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3]}}}function Fa(e,t,a){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 n=new o(e);if(n.isNaN())throw new Error(`Invalid expected amount: ${e}. Must be a valid number`);if(0===t)return e;const r=n.multipliedBy(t);let s;switch(a){case"buy-native":case"sell-exact":s=n.minus(r);break;case"buy-exact":case"sell-native":s=n.plus(r);break;default:throw new Error(`Unknown operation type: ${a}`)}return s.isLessThan(0)&&(s=new o(0)),s.toFixed()}class $a{constructor(e,t,a=!1,n,r,s=.05,o=.01){this.httpClient=e,this.tokenResolver=t,this.walletProvider=n,this.userAddress=r,this.defaultSlippageToleranceFactor=s,this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor=o,this.bundleEndpoint="/bundle",this.logger=new ua({debug:a,context:"BundleService"}),n&&r&&(this.signatureService=new Da(n,a),this.tokenKeyService=new Ca(a))}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 a=await this.httpClient.post(this.bundleEndpoint,t);return this.logger.debug("๐Ÿ“ฅ Bundle API response:",{success:a.success,hasData:!!a.data,error:a.error}),this.handleBundleResponse(a)}catch(e){return this.logger.error("โŒ Bundle transaction submission failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}validateBundleData(e){if(!e)throw new Error("Bundle data is required");if(!e.signedDto)throw new Error("Signed DTO is required");if(!e.method||"string"!=typeof e.method)throw new Error("Method name is required");if(!Array.isArray(e.stringsInstructions))throw new Error("stringsInstructions must be an array");if(0===e.stringsInstructions.length)throw new Error("stringsInstructions cannot be empty");const t=["BuyWithNative","BuyExactToken","SellExactToken","SellWithNative"];if(!t.includes(e.method))throw new Error(`Unsupported method: ${e.method}. Supported: ${t.join(", ")}`);e.stringsInstructions.forEach((e,t)=>{if("string"!=typeof e||0===e.length)throw new Error(`stringsInstructions[${t}] must be a non-empty string`);if(!e.startsWith("$"))throw new Error(`stringsInstructions[${t}] must start with '$': ${e}`)}),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){return"string"==typeof e?e:e?.response?.data?.error?e.response.data.error:e?.response?.data?.message?e.response.data.message:e?.message?e.message:"Unknown bundle transaction error"}async getBundlerTransactionResult(e){try{if(!e||"string"!=typeof e)throw new Error("Transaction ID is required");this.logger.debug("๐Ÿ” Checking bundler transaction result:",e);const t=await this.httpClient.get(`${this.bundleEndpoint}?id=${e}`);return this.logger.debug("๐Ÿ“Š Bundler transaction result:",t),{success:!0,data:t}}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 new Error("Transaction ID is required");this.logger.debug("๐Ÿšซ Cancelling transaction:",e);const t=await this.httpClient.delete(`${this.bundleEndpoint}/${e}`);return this.logger.debug("๐Ÿ—‘๏ธ Transaction cancellation response:",t),{success:!0,data:t}}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.httpClient.get(`${this.bundleEndpoint}/health`);return this.logger.debug("๐Ÿ’š Bundle service health:",e),{success:!0,data:e}}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:a,type:n,expectedAmount:r,maxAcceptableReverseBondingCurveFee:s,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,slippageToleranceFactor:i}=e,c=i??this.defaultSlippageToleranceFactor,l=o??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let d=s||"0";s&&(d=Fa(s,l,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:s,slippageFactor:l,adjustedMaxFee:d}));const u=await this.resolveTokenNameToVault(t);if(!u)throw Ht(t);if("native"===n){if(!r)throw new g("expectedAmount is required for native buy operations. Use getBuyTokenAmount() first to calculate expected tokens.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"buy-native");this.logger.debug("BuyNative slippage applied:",{originalExpectedTokens:r,slippageFactor:c,adjustedMinTokens:e});const t=new ba.BuyNativeDto(u,a,e,{maxAcceptableReverseBondingCurveFee:d});return await this.executeBundleTransaction(t,"BuyWithNative",u)}{if(!r)throw new g("expectedAmount is required for exact buy operations. Use getBuyTokenAmount() first to calculate expected GALA cost.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"buy-exact");this.logger.debug("BuyExact slippage applied:",{originalExpectedGalaCost:r,slippageFactor:c,adjustedMaxGalaCost:e});const t=new ba.BuyExactDto(u,a,e,{maxAcceptableReverseBondingCurveFee:d});return await this.executeBundleTransaction(t,"BuyExactToken",u)}}async sellToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:a,type:n,expectedAmount:r,maxAcceptableReverseBondingCurveFee:s,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,slippageToleranceFactor:i}=e,c=i??this.defaultSlippageToleranceFactor,l=o??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let d=s||"0";s&&(d=Fa(s,l,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:s,slippageFactor:l,adjustedMaxFee:d}));const u=await this.resolveTokenNameToVault(t);if(!u)throw Ht(t);if("exact"===n){if(!r)throw new g("expectedAmount is required for exact sell operations. Use getSellTokenAmount() first to calculate expected GALA.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"sell-exact");this.logger.debug("SellExact slippage applied:",{originalExpectedGala:r,slippageFactor:c,adjustedMinGala:e});const t=new ba.SellExactDto(u,a,e,{maxAcceptableReverseBondingCurveFee:s||"0"});return await this.executeBundleTransaction(t,"SellExactToken",u)}{if(!r)throw new g("expectedAmount is required for native sell operations. Use getSellTokenAmount() first to calculate tokens to sell.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"sell-native");this.logger.debug("SellNative slippage applied:",{originalExpectedTokensToSell:r,slippageFactor:c,adjustedMaxTokensToSell:e});const t=new ba.SellNativeDto(u,a,e,{maxAcceptableReverseBondingCurveFee:s||"0"});return await this.executeBundleTransaction(t,"SellWithNative",u)}}ensureTradingServicesAvailable(){if(!this.signatureService||!this.tokenKeyService)throw new Error("Trading services not available. BundleService requires walletProvider and userAddress for trading operations.");if(!this.userAddress)throw new Error("User address is required for trading operations.")}async executeBundleTransaction(e,t,a){this.ensureTradingServicesAvailable();try{e.uniqueKey=`galaswap - operation - ${i.v4()}-${Date.now()}-${this.userAddress}`;const n=await this.signatureService.signDTO(e,t,this.userAddress),r=this.tokenKeyService.generateStringsInstructions(a),s={stringsInstructions:r,method:t,signedDto:n};this.logger.debug("๐Ÿ“ฆ Bundle transaction data:",{method:t,stringsInstructions:r,dtoKeys:Object.keys(n)});const o=await this.submitTransaction(s);if(o.success&&o.data)return this.logger.debug("โœ… Bundle transaction submitted:",o.data),{success:!0,data:{transactionId:o.data,message:"Transaction submitted successfully. Monitor WebSocket for completion."}};throw new Error(o.error||"Bundle transaction failed")}catch(e){throw this.logger.error("โŒ Bundle transaction error:",e),e}}async resolveTokenNameToVault(e){return await this.tokenResolver.resolveTokenToVault(e)}}!function(e){e.PROCESSED="PROCESSED",e.COMPLETED="COMPLETED",e.SUCCESS="SUCCESS",e.FAILED="FAILED",e.ERROR="ERROR",e.PROCESSING="PROCESSING",e.PENDING="PENDING"}(Sa||(Sa={})),e.SDKTransactionStatus=void 0,(Ia=e.SDKTransactionStatus||(e.SDKTransactionStatus={})).PENDING="pending",Ia.PROCESSING="processing",Ia.COMPLETED="completed",Ia.FAILED="failed",Ia.TIMEOUT="timeout";const Pa={[Sa.PROCESSED]:e.SDKTransactionStatus.COMPLETED,[Sa.COMPLETED]:e.SDKTransactionStatus.COMPLETED,[Sa.SUCCESS]:e.SDKTransactionStatus.COMPLETED,[Sa.FAILED]:e.SDKTransactionStatus.FAILED,[Sa.ERROR]:e.SDKTransactionStatus.FAILED,[Sa.PROCESSING]:e.SDKTransactionStatus.PROCESSING,[Sa.PENDING]:e.SDKTransactionStatus.PENDING};class _a{constructor(e,t=!1){this.socket=null,this.listeners=new Map,this.timeouts=new Map,this.reconnectCount=0,this.config={reconnectAttempts:5,reconnectDelay:2e3,timeout:3e5,...e},this.debug=t,this.logger=new ua({debug:t,context:"WebSocketService"}),this.isSocketIOAvailable=this.checkSocketIOAvailability()}checkSocketIOAvailability(){try{return"function"==typeof c.io||(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=c.io(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.debug&&this.socket.onAny((e,...t)=>{this.logger.debug(`๐Ÿ“ก [WebSocket Event] "${e}":`,JSON.stringify(t,null,2))})}catch(e){t(e)}})}async monitorTransaction(t,a){this.listeners.set(t,a),this.logger.debug(`๐Ÿ“ก Starting to monitor transaction: ${t}`),this.logger.debug(`๐Ÿ“ก WebSocket connected: ${!!this.socket&&this.socket.connected}`);const n=setTimeout(()=>{if(this.listeners.has(t)){const n={transactionId:t,status:e.SDKTransactionStatus.TIMEOUT,message:"Transaction monitoring timeout - no response after 60 seconds",timestamp:Date.now()};this.logger.debug(`๐Ÿ“ก Transaction timeout for ${t}`),a(n),this.listeners.delete(t),this.timeouts.delete(t),this.socket?.off(t)}},6e4);if(this.timeouts.set(t,n),this.socket&&this.socket.connected)this.socket.off(t),this.logger.debug(`๐Ÿ“ก Listening for transaction updates: ${t}`),this.logger.debug(`๐Ÿ“ก WebSocket connection ID: ${this.socket.id}`),this.logger.debug(`๐Ÿ“ก WebSocket URL: ${this.config.url}`),this.socket.on(t,n=>{this.logger.debug(`๐Ÿ“ก Socket.IO transaction update for ${t}:`,JSON.stringify(n,null,2));const r=n?.status||n?.Status||n?.data?.status||n?.data?.Status;let s=n?.message||n?.Message||n?.data?.message||n?.data?.Message||n?.error||n?.data?.error;s&&"string"==typeof s||(s=r===Sa.FAILED||r===Sa.ERROR?"Transaction failed - check transaction details":r===Sa.COMPLETED||r===Sa.PROCESSED||r===Sa.SUCCESS?"Transaction completed successfully":r?`Transaction status: ${r}`:"Unknown transaction status");const o={transactionId:t,status:this.mapSocketStatus(r),message:s,timestamp:Date.now(),blockHash:n?.blockHash||n?.data?.blockHash,gasUsed:n?.gasUsed||n?.data?.gasUsed};if(this.logger.debug(`๐Ÿ“ก Mapped status for ${t}: ${r} -> ${o.status}`),this.logger.debug(`๐Ÿ“ก Final message: "${s}"`),a(o),o.status===e.SDKTransactionStatus.COMPLETED||o.status===e.SDKTransactionStatus.FAILED){this.listeners.delete(t);const e=this.timeouts.get(t);e&&(clearTimeout(e),this.timeouts.delete(t)),this.socket?.off(t),this.logger.debug(`๐Ÿ“ก Cleaned up listener for ${t} (${o.status})`)}});else{const n={transactionId:t,status:e.SDKTransactionStatus.FAILED,message:"WebSocket not connected - cannot monitor transaction",timestamp:Date.now()};a(n),this.listeners.delete(t),this.timeouts.delete(t)}}async waitForTransaction(t){return new Promise((a,n)=>{this.monitorTransaction(t,t=>{t.status===e.SDKTransactionStatus.COMPLETED?a(t):t.status!==e.SDKTransactionStatus.FAILED&&t.status!==e.SDKTransactionStatus.TIMEOUT||n(new Error(`Transaction ${t.status}: ${t.message}`))})})}mapSocketStatus(t){const a=t?.toUpperCase();return Pa[a]||e.SDKTransactionStatus.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.socket.disconnect(),this.socket=null,this.logger.debug("๐Ÿ”Œ Socket.IO disconnected"))}isConnected(){return this.socket?.connected||!1}}class Oa{constructor(e){this.poolService=e,this.cache=new Map}async resolveTokenToVault(e){if(!y(e))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");const t=e.trim().toLowerCase(),a=this.get(t);if(a)return a;try{const a=await this.poolService.resolveTokenNameToVault(e);return a&&this.set(t,a),a}catch{return null}}async resolveTokenClassKey(e){const t=await this.resolveTokenToVault(e);if(!t)throw new Error(`Token '${e}' not found. Cannot resolve to TokenClassKey.`);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:a}of e)this.set(t,a)}parseVaultAddressToTokenClassKey(e){const t=e.split("|");if(2!==t.length)throw new Error(`Invalid vault address format: ${e}. Expected format: service|Token$Unit$...$launchpad`);const a=t[1].split("$");if(a.length<4)throw new Error(`Invalid vault address format: ${e}. Expected at least 4 parts after service|`);return{collection:a[0],category:a[1],type:a[2],additionalKey:a[3]}}}function Ua(e){const t=function(e){const t=Je(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 La="/api/asset/launchpad-contract/CallNativeTokenIn",xa="/api/asset/launchpad-contract/CallNativeTokenOut",Ra="/api/asset/launchpad-contract/CallMemeTokenIn",Ba="/api/asset/launchpad-contract/CallMemeTokenOut";class Ma extends n.ChainCallDTO{constructor(e){super(),this.tokenName=e.tokenName,this.tokenSymbol=e.tokenSymbol,this.tokenDescription=e.tokenDescription,this.tokenImage=e.tokenImage,this.preBuyQuantity=e.preBuyQuantity,this.websiteUrl=e.websiteUrl,this.telegramUrl=e.telegramUrl,this.twitterUrl=e.twitterUrl,this.tokenCategory=e.tokenCategory,this.tokenCollection=e.tokenCollection,this.uniqueKey=e.uniqueKey,e.reverseBondingCurveConfiguration&&(this.reverseBondingCurveConfiguration=e.reverseBondingCurveConfiguration)}}function Ka(e){return e&&"object"==typeof e&&"number"==typeof e.Status&&e.Data&&"object"==typeof e.Data&&"string"==typeof e.Data.calculatedQuantity&&e.Data.extraFees&&"object"==typeof e.Data.extraFees&&"string"==typeof e.Data.extraFees.reverseBondingCurve&&"string"==typeof e.Data.extraFees.transactionFees}const Ga={NATIVE:"native",EXACT:"exact"};class za{constructor(e,t,a,n,r){this.http=e,this.tokenResolver=t,this.bundleHttp=a,this.galaChainHttp=n,this.dexApiHttp=r}async uploadImageByTokenName(e){const{tokenName:t,options:a}=e;yt(t);const n=`${t}.png`;aa(a.file,n,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&a.file instanceof File)e.append("image",a.file);else{if(!Buffer.isBuffer(a.file))throw new Error("Unsupported file type");{const n=`${a.tokenName||t}.png`,r=new Blob([a.file],{type:"image/png"});e.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`/launchpad/upload-image?tokenName=${encodeURIComponent(a.tokenName||t)}`,data:e,headers:{}});if(!0===n.error||200!==n.status||!n.data?.imageUrl)throw new Error(n.message||"Image upload failed - no URL returned");return n.data.imageUrl}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw new Error("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.");throw e}}async fetchPoolsFromAPI(e){wt(e),e.tokenName&&yt(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 a=h(t),n=await this.http.get("/launchpad/fetch-pool",a);if(!0===n.error||200!==n.status||!n.data)throw new Error(n.message||"Failed to fetch pools");let r=[];return n.data.tokens?r=Array.isArray(n.data.tokens)?n.data.tokens.map(e=>({...e,createdAt:new Date(e.created_at||e.createdAt)})):[{...n.data.tokens,createdAt:new Date(n.data.tokens.created_at||n.data.tokens.createdAt)}]:n.data.pools&&Array.isArray(n.data.pools)&&(r=n.data.pools.map(e=>({...e,createdAt:new Date(e.created_at||e.createdAt)}))),{pools:r,page:n.data.page,limit:n.data.limit,total:n.data.total,totalPages:n.data.totalPages,hasNext:n.data.page<n.data.totalPages,hasPrevious:n.data.page>1}}async _getAmount(e){if(At(e),!this.galaChainHttp)throw new Error("GalaChain client not configured. Direct GalaChain calls require galaChainHttp client.");const{endpoint:t,body:a}=((e,t,a,n)=>{if("NATIVE"===e&&"IN"===t)return{endpoint:La,body:{vaultAddress:a,tokenQuantity:n,IsPreMint:!1}};if("NATIVE"===e&&"OUT"===t)return{endpoint:xa,body:{vaultAddress:a,tokenQuantity:n,IsPreMint:!1}};if("MEME"===e&&"IN"===t)return{endpoint:Ra,body:{vaultAddress:a,nativeTokenQuantity:n,IsPreMint:!1}};if("MEME"===e&&"OUT"===t)return{endpoint:Ba,body:{vaultAddress:a,nativeTokenQuantity:n,IsPreMint:!1}};throw new Error(`Unsupported operation: ${e}-${t}`)})(e.type,e.method,e.vaultAddress,e.amount);try{const e=await this.galaChainHttp.post(t,a);if(!Ka(e))throw new Error("Malformed response data from GalaChain gateway");if(1!==e.Status)throw new Error(`GalaChain calculation failed with status ${e.Status}`);const{calculatedQuantity:n,extraFees:r}=e.Data;return{amount:n,reverseBondingCurveFee:r.reverseBondingCurve,transactionFee:r.transactionFees,gasFee:"1"}}catch(e){throw e}}async checkPool(e){kt(e),e.tokenName&&yt(e.tokenName);const t=h(e),a=await this.http.get("/launchpad/check-pool",t);if(!0===a.error||200!==a.status)throw new Error(a.message||"Failed to check pool");const n=a.data;return e.symbol?n?.isSymbolExist??!1:e.tokenName?n?.isNameExist??!1:n?.exists??!1}async fetchVolumeData(e){if(!Gt(e))throw new g("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:a,to:n,resolution:r}=e;if(yt(t),!a||!n||!r)throw new g("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const s={tokenName:t,from:a,to:n,resolution:r};vt(s);const o=h(s),i=await this.http.get("/launchpad/get-graph-data",o);if(!0===i.error||200!==i.status||!i.data)throw new Error(i.message||"Failed to fetch graph data");return{dataPoints:i.data}}async fetchPools(e={}){let t;"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const a={page:e.page||1,limit:e.limit||10};return e.search&&(a.search=e.search),e.tokenName&&(a.tokenName=e.tokenName),t&&(a.type=t),this.fetchPoolsFromAPI(a)}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:a,type:n}=e;if(!t||"string"!=typeof t)throw new g("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!a||"string"!=typeof a)throw new g("Amount is required and must be a string","amount","INVALID_AMOUNT");const r=await this.tokenResolver.resolveTokenToVault(t);if(!r)throw new g(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");if(n!==Ga.NATIVE&&n!==Ga.EXACT)throw new g('Type must be either "native" or "exact"',"type","INVALID_TYPE");return n===Ga.EXACT?this._getAmount({type:"NATIVE",method:"IN",vaultAddress:r,amount:a}):this._getAmount({type:"MEME",method:"OUT",vaultAddress:r,amount:a})}async calculateSellAmount(e){const{tokenName:t,amount:a,type:n}=e,r=await this.tokenResolver.resolveTokenToVault(t);if(!r)throw new g(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return n===Ga.EXACT?this._getAmount({type:"NATIVE",method:"OUT",vaultAddress:r,amount:a}):this._getAmount({type:"MEME",method:"IN",vaultAddress:r,amount:a})}async launchToken(e){if(!this.bundleHttp)throw new Error("Bundle backend client not configured. LaunchToken requires bundleHttp client.");Ua(e);const t=e.preBuyQuantity||"0";if(isNaN(Number(t))||Number(t)<0)throw new Error("Pre-buy quantity must be a valid non-negative number string");if(e.reverseBondingCurveConfiguration){const{minFeePortion:t,maxFeePortion:a}=e.reverseBondingCurveConfiguration,n=Number(t),r=Number(a);if(isNaN(n)||isNaN(r)||n<=0||r<=0||n>=r)throw new Error("Reverse bonding curve configuration must have valid min/max fee portions with min < max")}let a="";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 new Error("Image upload failed: No URL returned");a=t}else"string"==typeof e.tokenImage&&(a=e.tokenImage);const r=`galaswap - operation - ${i.v4()}-${Date.now()}-${this.http.getAddress()}`,s={tokenName:e.tokenName.trim(),tokenSymbol:e.tokenSymbol.trim().toUpperCase(),tokenDescription:e.tokenDescription.trim(),tokenImage:a.trim(),preBuyQuantity:t.toString(),websiteUrl:e.websiteUrl||"",telegramUrl:e.telegramUrl||"",twitterUrl:e.twitterUrl||"",tokenCategory:e.tokenCategory||"Unit",tokenCollection:e.tokenCollection||"Token",uniqueKey:r};e.reverseBondingCurveConfiguration&&(s.reverseBondingCurveConfiguration={minFeePortion:e.reverseBondingCurveConfiguration.minFeePortion.toString(),maxFeePortion:e.reverseBondingCurveConfiguration.maxFeePortion.toString()});const o=new Ma(s),c=await this.http.signWithGalaChain("CreateSale",o,n.SigningType.SIGN_TYPED_DATA),{signature:l,types:d,domain:u,prefix:h}=c,m={tokenName:o.tokenName,tokenSymbol:o.tokenSymbol,tokenDescription:o.tokenDescription,tokenImage:o.tokenImage,preBuyQuantity:o.preBuyQuantity,websiteUrl:o.websiteUrl,telegramUrl:o.telegramUrl,twitterUrl:o.twitterUrl,tokenCategory:o.tokenCategory,tokenCollection:o.tokenCollection,uniqueKey:o.uniqueKey,signature:l,types:d,domain:u,...h&&{prefix:h},...o.reverseBondingCurveConfiguration&&{reverseBondingCurveConfiguration:o.reverseBondingCurveConfiguration}},g=`${e.tokenName.trim()}$Unit$none$none`,p="GALA$Unit$none$none";let f;if(parseFloat(t)>0){const e=`$service$${g}$launchpad`;f=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${p}$${e}`,`$tokenBalance$${p}$${e}`]}else{const e=`$service$${g}$launchpad`;f=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`]}const y={signedDto:m,stringsInstructions:f,method:"CreateSale"},w=await this.bundleHttp.post("/bundle",y);if(w.error||!w.data)throw new Error(w.message||"Token launch failed");return w.data}async fetchTokenDistribution(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);const a=encodeURIComponent(t),n=await this.http.get(`/holders/${a}`);if(!0===n.error||200!==n.status||!n.data)throw new Error(n.message||"Failed to fetch token distribution");return{holders:n.data.holders||[],totalSupply:n.data.totalSupply||"0",totalHolders:n.data.totalHolders||0,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.http.get("/launchpad/get-badge/",{tokenName:e});if(t.error||!t.data)throw new Error(t.message||"Failed to fetch token badges");return{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadgeByTokenName(e){const{tokenName:t,badgeType:a,badgeName:n}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const r=("volume"===a?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===n);return r?.isActive||!1}catch{return!1}}async calculateInitialBuyAmount(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.nativeTokenQuantity||void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress)throw new Error("Invalid pre-mint calculation data");var t;if(!this.galaChainHttp)throw new Error("GalaChain HTTP client not available. Please initialize SDK with galaChainBaseUrl.");try{const t={vaultAddress:"service|testToken",nativeTokenQuantity:e.nativeTokenQuantity,IsPreMint:!0},a=await this.galaChainHttp.post("/api/asset/launchpad-contract/CallMemeTokenOut",t);if(!Ka(a))throw new Error("Malformed response data from GalaChain gateway");if(1!==a.Status)throw new Error(`GalaChain calculation failed with status ${a.Status}`);const{calculatedQuantity:n,extraFees:r}=a.Data;return{amount:n,reverseBondingCurveFee:r.reverseBondingCurve,transactionFee:r.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)}`)}}getAddress(){return this.http.getAddress()}formatAddressForBackend(e){return Tt(e)}validateTokenName(e){return yt(e)}validatePagination(e){return wt(e)}async fetchTokenSpotPrice(e){if(!this.dexApiHttp)throw new Error("DEX API client not configured. Token price fetching requires dexApiHttp client.");if(!e||Array.isArray(e)&&0===e.length)throw new Error("At least one symbol must be provided");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.dexApiHttp.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),a=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&a.push({symbol:e.symbol,price:e.currentPrices.usd})}),a}catch(e){throw new Error(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`)}}async fetchLaunchpadTokenSpotPrice(e){if(!e||"string"!=typeof e)throw new Error("Token name is required and must be a string");try{const t=await this.calculateBuyAmount({tokenName:e,amount:"1",type:"native"}),a=(await this.fetchTokenSpotPrice("GALA"))[0];if(!a)throw new Error("GALA price not available");const n=Number(t.amount)/1e18;if(n<=0)throw new Error(`Invalid token amount calculation: ${n}`);const r=a.price/n;return{symbol:e.toUpperCase(),price:r}}catch(t){if(t instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${String(t)}`)}}}const qa={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",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",launchpadFrontendUrl:"https://lpad-frontend-dev1.defi.gala.com"}};function Va(e){return qa[e]}class Wa extends Error{constructor(e,t){super(e),this.cause=t,this.name="WebSocketError"}}class ja extends Error{constructor(e,t,a){super(`Transaction ${e} failed with status: ${t}${a?` - ${a}`:""}`),this.transactionId=e,this.status=t,this.name="TransactionFailedError"}}function Ha(e){return e&&"object"==typeof e}function Qa(e,t){if(!e)throw new Wa(`Invalid WebSocket response received for transaction ${t}: response is null or undefined`);if("object"!=typeof e)throw new Wa(`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 Wa(`Invalid WebSocket response received for transaction ${t}: missing status field`)}class Xa{constructor(e){let t=null;t=e.env?Va(e.env):e.baseUrl?.includes("prod")?Va("PROD"):Va("STAGE"),this.config={baseUrl:t.launchpadBaseUrl,galaChainBaseUrl:t.galaChainBaseUrl,bundleBaseUrl:t.bundleBaseUrl,webSocketUrl:t.webSocketUrl,dexApiBaseUrl:t.dexApiBaseUrl,launchpadFrontendUrl:t.launchpadFrontendUrl,timeout:3e4,debug:!1,...e},this.validateConfiguration(),this.slippageToleranceFactor=void 0===e.slippageToleranceFactor?Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR:this.parseSlippageToleranceFactor(e.slippageToleranceFactor),this.maxAcceptableReverseBondingCurveFeeSlippageFactor=void 0===e.maxAcceptableReverseBondingCurveFeeSlippageFactor?Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR:this.parseFeeSlippageFactor(e.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.auth=new u({wallet:e.wallet,messagePrefix:"Create a GalaChain Wallet"}),this.http=new m(this.auth,this.config),this.galaChainHttp=new m(this.auth,{...this.config,baseUrl:this.config.galaChainBaseUrl}),this.bundleHttp=new m(this.auth,{...this.config,baseUrl:this.config.bundleBaseUrl}),this.dexApiHttp=new m(this.auth,{...this.config,baseUrl:this.config.dexApiBaseUrl}),this.launchpadService=new sa(this.http),this.tokenResolverService=new Oa(this.launchpadService.poolService),this.launchpadAPI=new za(this.http,this.tokenResolverService,this.bundleHttp,this.galaChainHttp,this.dexApiHttp),this.galaChainService=new pa(this.galaChainHttp,e.wallet,this.tokenResolverService,e.debug||!1),this.dexService=new fa(this.dexApiHttp),this.bundleService=new $a(this.bundleHttp,this.tokenResolverService,this.config.debug||!1,e.wallet,this.getAddress(),this.slippageToleranceFactor,this.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.websocketService=new _a({url:this.config.webSocketUrl},this.config.debug)}createOverrideSdk(e){if(!e||"string"!=typeof e)throw new Error("Invalid privateKey: must be a non-empty string");if(!e.match(/^0x[a-fA-F0-9]{64}$/))throw new Error('Invalid privateKey format: must be "0x" followed by 64 hexadecimal characters');const a=new t.Wallet(e),n={...this.config,wallet:a};return new Xa(n)}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.config.wallet.address}getConfig(){const{wallet:e,...t}=this.config;return{...t,slippageToleranceFactor:this.slippageToleranceFactor,maxAcceptableReverseBondingCurveFeeSlippageFactor:this.maxAcceptableReverseBondingCurveFeeSlippageFactor}}getUrlByTokenName(e){const t=this.config.launchpadFrontendUrl;if(!t)throw new Error("launchpadFrontendUrl not configured in SDK");return`${t.replace(/\/$/,"")}/buy-sell/${e}`}async fetchPools(e){return this.launchpadService.fetchPools(e||{})}async fetchTokenDistribution(e){return this.launchpadService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.launchpadService.fetchTokenBadges(e)}async fetchTokenSpotPrice(e){return this.dexService.fetchTokenSpotPrice(e)}async fetchGalaSpotPrice(){const e=await this.dexService.fetchTokenSpotPrice("GALA");if(0===e.length)throw new Error("Failed to fetch GALA price - no price data returned");const t=e.find(e=>"GALA"===e.symbol);if(!t)throw new Error("GALA price not found in response");return t}async fetchLaunchpadTokenSpotPrice(e){return this.launchpadAPI.fetchLaunchpadTokenSpotPrice(e)}async fetchPoolDetails(e){const t=await this.resolveVaultAddress(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);return(await this.galaChainService.fetchPoolDetails({vaultAddress:t})).Data}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 Et}),a=t(e)||this.getAddress();return this.galaChainService.fetchGalaBalance({owner:a,collection:"GALA",category:"Unit",additionalKey:"none",type:"none",instance:"0"})}async fetchTokenBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a=t(e.address);if(e.tokenId){const{normalizeToTokenInstanceKey:t}=await Promise.resolve().then(function(){return da}),n=t(e.tokenId),{collection:r,category:s,type:o,additionalKey:i}=n;return this.galaChainService.fetchTokenBalance({owner:a,collection:r,category:s,additionalKey:i,type:o,instance:"0"})}if(e.tokenName){const t=(await this.fetchTokensHeld({tokenName:e.tokenName,page:1,limit:1,...a&&{address:a}})).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 new Error("Must provide 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 calculateInitialBuyAmount(e){const t={nativeTokenQuantity:e};return this.launchpadAPI.calculateInitialBuyAmount(t)}async buy(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.buy(n)}await this.ensureWebSocketConnection();const t=await this.bundleService.buyToken(e),a=t.data?.transactionId;if(!a)throw new Error("No transaction ID returned from buy operation");return this.waitForConfirmation(a,t=>{Qa(t,a);const n=t?.data||{};if(!Ha(n))throw new Wa(`Invalid trade data received for transaction ${a}`);const r={transactionId:a,type:"buy",method:"native"===e.type?"native":"exact",inputAmount:n.inputQuantity||e.amount,outputAmount:n.outputQuantity||e.expectedAmount||"0",totalFees:n.totalFees||"0",tokenName:e.tokenName,vaultAddress:n.vaultAddress||"",blockHash:t.blockHash,gasUsed:t.gasUsed,timestamp:Date.now()};return void 0!==e.slippageToleranceFactor&&(r.slippageTolerance=e.slippageToleranceFactor),r})}async sell(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.sell(n)}await this.ensureWebSocketConnection();const t=await this.bundleService.sellToken(e),a=t.data?.transactionId;if(!a)throw new Error("No transaction ID returned from sell operation");return this.waitForConfirmation(a,t=>{Qa(t,a);const n=t?.data||{};if(!Ha(n))throw new Wa(`Invalid trade data received for transaction ${a}`);const r={transactionId:a,type:"sell",method:"native"===e.type?"native":"exact",inputAmount:n.inputQuantity||e.amount,outputAmount:n.outputQuantity||e.expectedAmount||"0",totalFees:n.totalFees||"0",tokenName:e.tokenName,vaultAddress:n.vaultAddress||"",blockHash:t.blockHash,gasUsed:t.gasUsed,timestamp:Date.now()};return void 0!==e.slippageToleranceFactor&&(r.slippageTolerance=e.slippageToleranceFactor),r})}async getBundlerTransactionResult(e){return this.bundleService.getBundlerTransactionResult(e)}async postComment(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.postComment(n)}return this.launchpadService.postComment(e)}async launchToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.launchToken(n)}await this.ensureWebSocketConnection();const t=await this.launchpadAPI.launchToken(e);return this.waitForConfirmation(t,a=>{Qa(a,t);const n=a?.data||{};if(!function(e){return e&&"object"==typeof e}(n))throw new Wa(`Invalid launch data received for transaction ${t}`);const r={transactionId:t,vaultAddress:n.vaultAddress||"",tokenStringKey:n.tokenStringKey||"",tokenName:e.tokenName,tokenSymbol:e.tokenSymbol,creatorAddress:n.creatorAddress||this.getAddress(),blockHash:a.blockHash,gasUsed:a.gasUsed,timestamp:Date.now()};return"string"==typeof e.tokenImage&&(r.tokenImage=e.tokenImage),void 0!==e.preBuyQuantity&&(r.preBuyQuantity=e.preBuyQuantity),r.vaultAddress&&this.tokenResolverService.set(e.tokenName,r.vaultAddress),r})}async uploadTokenImage(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.uploadTokenImage(n)}return 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 Et}),a=t(e)||this.getAddress();return this.launchpadService.fetchProfile(a)}async updateProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,address:t(e.address)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.updateProfile(n)}return this.launchpadService.updateProfile(a)}async uploadProfileImage(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,address:t(e.address)||this.getAddress()};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.uploadProfileImage(n)}return this.launchpadService.uploadProfileImage(a)}async retrieveGalaFromFaucet(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={walletAddress:t(e)||this.getAddress(),amount:"5"};return this.launchpadService.transferFaucets(a)}async fetchTokensHeld(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a=t(e?.address)||this.getAddress(),n={page:e?.page||1,limit:e?.limit||10,address:a};return e?.tokenName&&(n.tokenName=e.tokenName),e?.search&&(n.search=e.search),this.launchpadService.fetchTokensHeld(n)}async fetchTokensCreated(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={type:"DEFI",address:t(e?.address)||this.getAddress(),page:e?.page||1,limit:e?.limit||10};return e?.tokenName&&(a.tokenName=e.tokenName),e?.search&&(a.search=e.search),this.launchpadService.fetchTokenList(a)}async transferGala(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,recipientAddress:t(e.recipientAddress)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.transferGala(n)}return this.galaChainService.transferGala(a)}async transferToken(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,to:t(e.to)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.transferToken(n)}return this.galaChainService.transferToken(a)}async resolveTokenClassKey(e){return this.galaChainService.resolveTokenClassKey(e)}async resolveVaultAddress(e){return this.tokenResolverService.resolveTokenToVault(e)}validateConfiguration(){if(("number"!=typeof this.config.timeout||this.config.timeout<=0||this.config.timeout>3e5)&&(this.config.debug,this.config.timeout=3e4),!this.config.baseUrl)throw new Error("baseUrl is required in configuration");if(!this.config.webSocketUrl)throw new Error("webSocketUrl is required in configuration");try{new URL(this.config.baseUrl)}catch{throw new Error(`Invalid baseUrl format: ${this.config.baseUrl}`)}try{new URL(this.config.webSocketUrl)}catch{throw new Error(`Invalid webSocketUrl format: ${this.config.webSocketUrl}`)}if(this.config.galaChainBaseUrl)try{new URL(this.config.galaChainBaseUrl)}catch{throw new Error(`Invalid galaChainBaseUrl format: ${this.config.galaChainBaseUrl}`)}if(this.config.bundleBaseUrl)try{new URL(this.config.bundleBaseUrl)}catch{throw new Error(`Invalid bundleBaseUrl format: ${this.config.bundleBaseUrl}`)}if(this.config.launchpadFrontendUrl)try{new URL(this.config.launchpadFrontendUrl)}catch{throw new Error(`Invalid launchpadFrontendUrl format: ${this.config.launchpadFrontendUrl}`)}}parseSlippageToleranceFactor(e){const t=parseFloat(e);return isNaN(t)||t<0||t>1?(this.config.debug,Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR):t}parseFeeSlippageFactor(e){const t=parseFloat(e);return isNaN(t)||t<0||t>1?(this.config.debug,Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR):t}async ensureWebSocketConnection(){this.websocketService.isConnected()||(await this.websocketService.connect(),this.config.debug)}async waitForConfirmation(e,t){this.config.debug;try{const a=await this.websocketService.waitForTransaction(e);if("completed"!==a.status)throw new ja(e,a.status,a.message);let n;try{n=t(a)}catch(t){if(t instanceof Wa)throw t;throw new Wa(`Failed to transform WebSocket response for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}return this.config.debug,n}catch(t){if(this.config.debug,t instanceof ja||t instanceof Wa)throw t;throw new Wa(`WebSocket confirmation failed for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}}async cleanup(){try{this.config.debug,this.websocketService&&this.websocketService.disconnect(),this.config.debug}catch(e){this.config.debug}}static cleanupAll(e=!1){const{WebSocketService:t}=require("./services/WebSocketService");t.cleanupAll(e)}}Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR=.15,Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR=.01;class Ya{static generateWallet(){try{const e=t.Wallet.createRandom();if(!e.mnemonic?.phrase)throw new Error("Failed to generate wallet with mnemonic phrase");const a=this.toGalaAddress(e.address);return{privateKey:e.privateKey,address:e.address,galaAddress:a,mnemonic:e.mnemonic.phrase,wallet:new t.Wallet(e.privateKey)}}catch(e){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const e=`test-wallet-${Date.now()}-${++this.testCounter}`,a="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64),n=new t.Wallet(a),r=this.toGalaAddress(n.address);return{privateKey:n.privateKey,address:n.address,galaAddress:r,mnemonic:"test test test test test test test test test test test junk",wallet:n}}throw e}}static fromPrivateKey(e){const a=new t.Wallet(e),n=this.toGalaAddress(a.address);return{privateKey:a.privateKey,address:a.address,galaAddress:n,mnemonic:"",wallet:a}}static fromMnemonic(e,a=0){try{const n=t.Mnemonic.fromPhrase(e),r=t.HDNodeWallet.fromMnemonic(n,`m/44'/60'/0'/0/${a}`),s=new t.Wallet(r.privateKey),o=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:o,mnemonic:e,wallet:s}}catch(n){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const n=`test-mnemonic-index-${a}-${e}`,r="0x"+Buffer.from(n).toString("hex").padStart(64,"1").slice(0,64),s=new t.Wallet(r),o=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:o,mnemonic:e,wallet:s}}throw n}}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 a=0;a<e;a++){const e=`test-multi-${a}-${Date.now()}-${++this.testCounter}`,n="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64);t.push(this.fromPrivateKey(n))}else for(let a=0;a<e;a++)t.push(this.generateWallet());return t}static getWalletSummary(e,t=!1){const a=["๐Ÿ” Wallet Information","โ•".repeat(50),`๐Ÿ“ Address: ${e.address}`,`๐ŸŽฎ Gala Address: ${e.galaAddress}`,`๐ŸŒฑ Mnemonic: ${e.mnemonic||"Not available"}`];return t?a.splice(3,0,`๐Ÿ”‘ Private Key: ${e.privateKey}`):a.splice(3,0,"๐Ÿ”‘ Private Key: [HIDDEN - use includeSensitive=true to show]"),a.push("โ•".repeat(50)),a.push("๐Ÿ’พ IMPORTANT: Save your mnemonic phrase securely!"),a.push("This is your backup to recover the wallet."),a.join("\n")}}function Ja(e){if(void 0===e)return Ya.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 Ya.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 Ya.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?"...":""}"`)}Ya.testCounter=0;e.AgentConfig=class{static async quickSetup(e={}){const t=e.environment||this.detectEnvironment(),a=this.setupWallet(e.privateKey),n={wallet:a.wallet,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t)},r=new Xa(n),s={sdk:r,wallet:a,config:n};if(!1!==e.autoValidate){const e=await this.validateSetup(r,a);return{...s,validation:e}}return s}static async validateSetup(e,t){const a=[],n=[],r={canTrade:!1,canCreateTokens:!1,hasBalance:!1,connectionHealthy:!1};try{const t=await e.fetchGalaBalance(e.getAddress());if(r.connectionHealthy=!0,t&&t.quantity){const e=parseFloat(t.quantity);r.hasBalance=e>0,r.canTrade=e>=.1,r.canCreateTokens=e>=100,0===e?n.push("Wallet has zero GALA balance - cannot perform transactions"):e<.1?n.push("GALA balance too low for trading (minimum 0.1 GALA)"):e<100&&n.push("GALA balance too low for token creation (minimum 100 GALA)")}else a.push("Failed to fetch GALA balance: No balance returned")}catch(e){a.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||n.push("Pool listing not accessible - some features may be limited")}catch(e){n.push(`Pool access test failed: ${e instanceof Error?e.message:String(e)}`)}return{ready:0===a.length&&r.connectionHealthy,sdk:e,wallet:t||Ya.generateWallet(),issues:a,warnings:n,capabilities:r}}static getRecommendedConfig(e,t="general"){const a={environment:e,autoValidate:!0};switch(e){case"production":Object.assign(a,{debug:!1,timeout:3e4});break;case"development":Object.assign(a,{debug:!0,timeout:45e3});break;case"testing":Object.assign(a,{debug:!0,timeout:6e4})}switch(t){case"trading":a.timeout=1.5*(a.timeout||3e4);break;case"creation":a.timeout=2*(a.timeout||3e4);break;case"monitoring":a.timeout=.5*(a.timeout||3e4)}return a}static async multiWalletSetup(e,t="development"){const a={};for(const[n,r]of Object.entries(e)){const{sdk:e}=await this.quickSetup({environment:t,privateKey:r,agentId:`multi-wallet-${n}`,autoValidate:!1});a[n]=e}return a}static detectEnvironment(){const e=process.env.NODE_ENV?.toLowerCase();return"production"===e?"production":"test"===e||"testing"===e?"testing":"development"}static setupWallet(e){if(!e){const e=process.env.PRIVATE_KEY;return e?Ya.fromPrivateKey(e):Ya.generateWallet()}return"generate"===e?Ya.generateWallet():Ya.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}},e.FileValidationError=ta,e.GALA_DECIMALS=8,e.GALA_TOKEN_CLASS_KEY={collection:"GALA",category:"Unit",type:"none",additionalKey:"none"},e.IMAGE_EXTENSIONS=oe,e.LAUNCHPAD_TOKEN_DECIMALS=18,e.LaunchpadSDK=Xa,e.POOL_TYPES={RECENT:"recent",POPULAR:"popular"},e.SDK_VERSION="3.7.0",e.TRADING_TYPES=Ga,e.TransactionFailedError=ja,e.ValidationError=g,e.WebSocketError=Wa,e.WebSocketTimeoutError=class extends Wa{constructor(e,t){super(`WebSocket confirmation timeout for transaction ${e} after ${t}ms`),this.name="WebSocketTimeoutError"}},e.addressFormatSchema=S,e.amountMethodSchema=re,e.amountTypeSchema=ne,e.browserFileSchema=ce,e.bufferFileSchema=le,e.buyTokensDataSchema=$e,e.calculatePreMintDataSchema=Re,e.checkPoolOptionsSchema=ae,e.commentMessageSchema=z,e.commentPaginationSchema=fe,e.createLaunchpadSDK=function(e){const{wallet:a,env:n,config:r={}}=e||{};let s;if(a)if("string"==typeof a){s=Ja(a).wallet}else{if(!(a instanceof t.Wallet))throw new Error("Invalid wallet input. Expected string (private key or mnemonic) or Wallet instance.");s=a}else{s=Ja().wallet}const o={wallet:s,...n&&{env:n},debug:!1,timeout:3e4,...r};return new Xa(o)},e.createLimitSchema=L,e.createPaginatedResultSchema=function(e){return r.z.object({data:r.z.array(e),page:r.z.number().int().min(1),limit:r.z.number().int().min(1),total:r.z.number().int().min(0),totalPages:r.z.number().int().min(0),hasNext:r.z.boolean(),hasPrevious:r.z.boolean()})},e.createTradeDataSchema=Fe,e.createWallet=Ja,e.ethereumAddressSchema=I,e.faucetAmountSchema=P,e.fetchGalaBalanceOptionsSchema=be,e.fetchPoolDetailsDataSchema=Be,e.fetchTokenBalanceOptionsSchema=Ie,e.fileSizeSchema=M,e.fileUploadSchema=ie,e.filenameSchema=K,e.flexibleAddressSchema=D,e.flexibleFileSchema=de,e.formatGalaForDTO=wa,e.formatLaunchpadTokenForDTO=ka,e.fullNameSchema=N,e.getAmountOptionsSchema=xe,e.getTradeOptionsSchema=_e,e.graphDataOptionsSchema=se,e.imageExtensionSchema=ue,e.imageFilenameSchema=he,e.imageMimeTypeSchema=G,e.imageUploadOptionsSchema=ee,e.isoDateStringSchema=q,e.launchTokenDataSchema=Z,e.nonNegativeDecimalStringSchema=$,e.optionalUrlSchema=O,e.pageNumberSchema=U,e.paginationResultMetaSchema=Ae,e.poolFetchTypeSchema=te,e.poolPaginationSchema=ye,e.positiveDecimalStringSchema=F,e.privateKeySchema=W,e.reverseBondingCurveConfigSchema=J,e.reverseBondingCurveConfigurationSchema=Me,e.searchQuerySchema=b,e.sellTokensDataSchema=Pe,e.standardLimitSchema=x,e.standardPaginationSchema=me,e.timestampSchema=V,e.tokenCategorySchema=X,e.tokenCollectionSchema=Y,e.tokenDescriptionSchema=T,e.tokenListOptionsSchema=Te,e.tokenNameSchema=A,e.tokenSymbolSchema=v,e.tokenUrlsSchema=Q,e.tradeCalculationMethodSchema=Le,e.tradeCalculationTypeSchema=Ue,e.tradeLimitSchema=B,e.tradeListParamsSchema=Oe,e.tradePaginationSchema=pe,e.tradePaginationWithFiltersSchema=ke,e.tradeTypeBackendSchema=Ce,e.tradeTypeSchema=De,e.transactionIdSchema=j,e.transferFaucetsDataSchema=Ee,e.uniqueKeySchema=H,e.updateProfileDataSchema=Ne,e.uploadProfileImageOptionsSchema=Se,e.urlSchema=_,e.userLimitSchema=R,e.userPaginationSchema=ge,e.userTokenNameSchema=E,e.userTokenTypeSchema=ve,e.userTokensPaginationSchema=we,e.validateAddress=Ve,e.validateAmountString=je,e.validateBuyTokensData=lt,e.validateCalculatePreMintData=gt,e.validateCheckPoolOptions=tt,e.validateCreateTradeData=ct,e.validateFaucetAmount=He,e.validateFetchGalaBalanceOptions=rt,e.validateFetchPoolDetailsData=pt,e.validateFetchTokenBalanceOptions=it,e.validateFullName=Qe,e.validateGetAmountOptions=mt,e.validateGetTradeOptions=ut,e.validateImageUploadOptions=et,e.validateLaunchTokenData=Je,e.validateSearchQuery=Xe,e.validateSellTokensData=dt,e.validateTokenDescription=qe,e.validateTokenListOptions=at,e.validateTokenName=Ge,e.validateTokenSymbol=ze,e.validateTokenUrls=Ze,e.validateTradeListParams=ht,e.validateTransferFaucetsData=nt,e.validateUpdateProfileData=st,e.validateUploadProfileImageOptions=ot,e.validateUserTokenName=Ye,e.validateVaultAddress=We,e.vaultAddressSchema=C});
1
+ !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports,require("ethers"),require("axios"),require("@gala-chain/connect"),require("zod"),require("@gala-chain/api"),require("bignumber.js"),require("uuid"),require("socket.io-client")):"function"==typeof define&&define.amd?define(["exports","ethers","axios","@gala-chain/connect","zod","@gala-chain/api","bignumber.js","uuid","socket.io-client"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).GalaLaunchpadSDK={},e.ethers,e.axios,e.GalaChainConnect,e.z,e.GalaChainAPI,e.BigNumber,e.uuid,e.io)}(this,function(e,t,a,n,r,s,o,i,c){"use strict";var l;!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 d extends Error{constructor(e,t,a){super(t),this.type=e,this.originalError=a,this.name="AuthError"}}class u{constructor(e){if(this.wallet=e.wallet,this.messagePrefix=e.messagePrefix||"Create a GalaChain Wallet",""===e.messagePrefix)throw new d(l.SIGNATURE_FAILED,"Message prefix cannot be empty");this.validateWallet()}async generateSignature(){try{const e=Date.now(),t=`${this.messagePrefix} ${e}`,a=await this.wallet.signMessage(t);return{message:t,signature:a,address:this.formatAddress(this.wallet.address),timestamp:e}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate signature for authentication",e instanceof Error?e:new Error(String(e)))}}getAddress(){return this.formatAddress(this.wallet.address)}getEthereumAddress(){return this.wallet.address}formatAddress(e){const t=e.replace(/^0x/i,"");if(!/^[a-fA-F0-9]{40}$/.test(t))throw new d(l.INVALID_ADDRESS,`Invalid Ethereum address format: ${e}`);return`eth|${t}`}async signMessage(e){try{return{message:e,signature:await this.wallet.signMessage(e),address:this.wallet.address,timestamp:Date.now()}}catch(e){const t=e instanceof Error?e.message:String(e);throw new d(l.SIGNATURE_FAILED,t,e instanceof Error?e:new Error(String(e)))}}async generateAuthHeaders(e,t){try{const a=Date.now(),n=`${this.messagePrefix} ${t.toUpperCase()} ${e} ${a}`,r=await this.wallet.signMessage(n);return{"x-signature":r,"x-address":this.formatAddress(this.wallet.address),"x-message":n,"x-timestamp":a.toString()}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate authentication headers",e instanceof Error?e:new Error(String(e)))}}async signTypedData(e,t,a){try{return await this.wallet.signTypedData(e,t,a)}catch(e){throw new d(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 d(l.SIGNATURE_FAILED,"Custom message must be a non-empty string");try{const t=await this.wallet.signMessage(e);return{message:e,signature:t,address:this.formatAddress(this.wallet.address),timestamp:Date.now()}}catch(e){throw new d(l.SIGNATURE_FAILED,"Failed to generate custom message signature",e instanceof Error?e:new Error(String(e)))}}validateWallet(){if(!this.wallet)throw new d(l.WALLET_NOT_CONNECTED,"Wallet is required for authentication");if(!this.wallet.address)throw new d(l.WALLET_NOT_CONNECTED,"Wallet address is not available");if(!this.wallet.privateKey&&!this.wallet.signMessage)throw new d(l.WALLET_NOT_CONNECTED,"Wallet must have a private key for signing messages")}}function h(e){const t={};for(const[a,n]of Object.entries(e))null!=n&&("string"==typeof n?t[a]=n:"number"==typeof n||"boolean"==typeof n?t[a]=n.toString():Array.isArray(n)?t[a]=n.join(","):t[a]="object"==typeof n?JSON.stringify(n):String(n));return t}class m{constructor(e,t={}){this.auth=e,this.debug=t.debug??!1,this.axios=a.create({baseURL:t.baseUrl||"https://lpad-backend-dev1.defi.gala.com",timeout:t.timeout||3e4,headers:{Accept:"application/json",...t.headers}}),this.setupInterceptors()}async request(e){try{const t={method:e.method,url:e.url,data:e.data,...e.params&&{params:h(e.params)},...e.headers&&{headers:e.headers},...e.timeout&&{timeout:e.timeout}};if(e.data instanceof FormData&&(t.headers&&t.headers["Content-Type"]&&delete t.headers["Content-Type"],this.debug),this.debug){e.data instanceof FormData||e.data}const a=await this.axios.request(t);return this.debug,a.data}catch(e){throw this.debug,e}}async get(e,t,a){return this.request({method:"GET",url:e,...t&&{params:t},...a&&{headers:a}})}async post(e,t,a){return this.request({method:"POST",url:e,data:t,...a&&{headers:a}})}async put(e,t,a){return this.request({method:"PUT",url:e,data:t,...a&&{headers:a}})}async delete(e,t,a){return this.request({method:"DELETE",url:e,...t&&{params:t},...a&&{headers:a}})}async patch(e,t,a){return this.request({method:"PATCH",url:e,data:t,...a&&{headers:a}})}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.auth.getEthereumAddress()}async signMessage(e){return(await this.auth.signMessage(e)).signature}async signTypedData(e,t,a){return await this.auth.signTypedData(e,t,a)}async signCustomMessage(e){try{const t=await this.auth.generateCustomSignature(e);return this.debug,{signature:t.signature,address:t.address,ethereumAddress:this.auth.getEthereumAddress()}}catch(e){throw this.debug,new Error(`Failed to generate custom signature for message: ${e instanceof Error?e.message:"Unknown error"}`)}}async signWithGalaChain(e,t,a=n.SigningType.SIGN_TYPED_DATA){const r=this.auth.wallet.privateKey;if(!r)throw new Error("Wallet private key not available for @gala-chain signing");const s=new n.SigningClient(r);return await s.sign(e,t,a)}setupInterceptors(){this.requestInterceptorId=this.axios.interceptors.request.use(async e=>{try{const t=await this.auth.generateSignature();return e.headers||(e.headers={}),e.headers.Sign=t.signature,e.data instanceof FormData||(e.headers["Content-Type"]="application/json"),this.debug,e}catch(e){throw this.debug,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.debug}else e.request,this.debug;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.debug}}class g extends Error{constructor(e,t,a){super(e),this.field=t,this.code=a,this.name="ValidationError"}}function p(e,t,a){const{MIN_PAGE:n,MAX_PAGE:r,MIN_LIMIT:s,MAX_LIMIT:o}=a.PAGINATION;if("number"!=typeof e||e<n||e>r)throw new g(`Page must be a number between ${n} and ${r}`,"page","INVALID_PAGE");if("number"!=typeof t||t<s||t>o)throw new g(`Limit must be a number between ${s} and ${o}`,"limit","INVALID_LIMIT")}const f={ETH_ADDRESS:/^0x[0-9a-fA-F]{40}$/,BACKEND_ADDRESS:/^eth\|[0-9a-fA-F]{40}$/};function y(e){return"string"==typeof e&&e.trim().length>0}function w(e){return!(!e||"string"!=typeof e)&&(f.ETH_ADDRESS.test(e)||f.BACKEND_ADDRESS.test(e))}function A(e){return e.startsWith("0x")?`eth|${e.slice(2)}`:e}const k=r.z.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"),v=r.z.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"),T=r.z.string().min(1,"Token description is required").max(500,"Token description must be at most 500 characters"),E=r.z.string().min(1,"Token name must be at least 1 character").max(50,"Token name must be at most 50 characters"),b=r.z.string().min(1,"Search query must be at least 1 character").max(100,"Search query must be at most 100 characters"),N=r.z.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"),S=r.z.string().regex(f.BACKEND_ADDRESS,"Address must be in format: eth|[40-hex-chars]"),I=r.z.string().regex(f.ETH_ADDRESS,"Invalid Ethereum address format"),D=r.z.string().refine(e=>f.BACKEND_ADDRESS.test(e)||f.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),C=r.z.string().refine(e=>f.BACKEND_ADDRESS.test(e)||/^service\|Token\$Unit\$[A-Z0-9]+\$eth:[0-9a-fA-F]{40}\$launchpad$/.test(e),"Invalid vault address format"),F=r.z.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>0,"Amount must be greater than zero"),$=r.z.string().regex(/^\d+(\.\d+)?$/,"Must be a valid decimal number").refine(e=>parseFloat(e)>=0,"Amount must be zero or greater"),P=r.z.string().regex(/^(?!0+(\.0+)?$)\d+(\.\d+)?$/,"Amount must be a positive, non-zero number"),_=r.z.string().url("Must be a valid URL").regex(/^https?:\/\//,"URL must start with http:// or https://"),O=r.z.string().optional().refine(e=>!e||/^https?:\/\/.+\..+/.test(e),"Must be a valid URL if provided"),U=r.z.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 L(e=100){return r.z.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 x=L(100),R=L(20),B=L(20),M=r.z.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"),K=r.z.string().max(255,"Filename must be at most 255 characters"),G=r.z.enum(["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"]),z=r.z.string().min(1,"Comment message is required").max(500,"Comment must be at most 500 characters"),q=r.z.string().datetime("Must be a valid ISO 8601 date string"),V=r.z.number().int("Timestamp must be an integer").min(0,"Timestamp must be non-negative"),W=r.z.string().regex(/^0x[a-fA-F0-9]{64}$/,"Private key must be format: 0x + 64 hex characters"),j=r.z.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"),H=r.z.string().regex(/^galaconnect-operation-[a-z0-9-]+$/,"Unique key must be format: galaconnect-operation-{unique-id}"),Q=r.z.object({websiteUrl:O,telegramUrl:O,twitterUrl:O}).refine(e=>e.websiteUrl||e.telegramUrl||e.twitterUrl,"At least one social URL (website, telegram, or twitter) is required"),X=r.z.string().min(1,"Token category must not be empty").default("Unit"),Y=r.z.string().min(1,"Token collection must not be empty").default("Token"),J=r.z.object({minFeePortion:F,maxFeePortion:F}),Z=r.z.object({tokenName:k,tokenSymbol:v,tokenDescription:T,tokenImage:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer),r.z.string().url("Token image must be a valid URL")]).optional(),preBuyQuantity:$.default("0"),websiteUrl:O,telegramUrl:O,twitterUrl:O,tokenCategory:X,tokenCollection:Y,reverseBondingCurveConfiguration:J.optional(),privateKey:W.optional()}),ee=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),tokenName:k}),te=r.z.enum(["recent","popular"]),ae=r.z.object({tokenName:k.optional(),symbol:v.optional()}).refine(e=>e.tokenName||e.symbol,"At least one of tokenName or symbol is required"),ne=r.z.enum(["NATIVE","MEME"]),re=r.z.enum(["IN","OUT"]),se=r.z.object({from:r.z.number().int("From timestamp must be an integer").min(173e6,"From timestamp must be at least 173000000"),to:r.z.number().int("To timestamp must be an integer").min(173e6,"To timestamp must be at least 173000000"),resolution:r.z.number().int("Resolution must be an integer").min(1,"Resolution must be at least 1"),tokenName:k}),oe=[".png",".jpg",".jpeg",".gif",".webp",".svg"],ie=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),name:K,size:M,type:G}),ce=r.z.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"),le=r.z.instanceof(Buffer).refine(e=>e.length>=1&&e.length<=10485760,"Buffer size must be between 1 byte and 10MB"),de=r.z.union([ce,le]),ue=r.z.enum([".png",".jpg",".jpeg",".gif",".webp",".svg"]),he=K.refine(e=>{const t=e.slice(e.lastIndexOf(".")).toLowerCase();return oe.includes(t)},`Filename must end with one of: ${oe.join(", ")}`),me=r.z.object({page:U,limit:x}),ge=r.z.object({page:U,limit:R}),pe=r.z.object({page:U,limit:B}),fe=r.z.object({page:U,limit:L(50)}),ye=me.extend({type:r.z.enum(["recent","popular"]).optional(),tokenName:r.z.string().min(1).max(50).optional(),search:r.z.string().min(1).max(100).optional()}),we=ge.extend({tokenName:r.z.string().min(1).max(50).optional(),search:r.z.string().min(1).max(100).optional()}),Ae=pe.extend({tradeType:r.z.enum(["BUY","SELL"]).optional(),tokenName:r.z.string().min(1).max(50).optional(),userAddress:r.z.string().regex(/^(0x[a-fA-F0-9]{40}|eth\|[a-fA-F0-9]{40})$/).optional(),startDate:r.z.string().datetime().optional(),endDate:r.z.string().datetime().optional(),sortOrder:r.z.enum(["ASC","DESC"]).default("DESC")}),ke=r.z.object({page:r.z.number().int().min(1),limit:r.z.number().int().min(1),total:r.z.number().int().min(0),totalPages:r.z.number().int().min(0),hasNext:r.z.boolean(),hasPrevious:r.z.boolean()});const ve=r.z.enum(["all","DEFI","ASSET"]),Te=ge.extend({type:ve.optional(),address:D.optional(),search:b.optional(),tokenName:E.optional()}),Ee=r.z.object({walletAddress:D,amount:P}),be=r.z.object({address:D.optional(),refresh:r.z.boolean().optional()}),Ne=r.z.object({profileImage:r.z.string(),fullName:N,address:D,privateKey:W.optional()}),Se=r.z.object({file:r.z.union([r.z.instanceof(File),r.z.instanceof(Buffer)]),address:D.optional(),privateKey:W.optional()}),Ie=r.z.object({address:D,tokenId:r.z.union([r.z.string(),r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string()}),r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string(),instance:r.z.string()})]).optional(),tokenClassKey:r.z.object({collection:r.z.string(),category:r.z.string(),type:r.z.string(),additionalKey:r.z.string()}).optional(),tokenName:E.optional()}).refine(e=>void 0!==e.tokenId||void 0!==e.tokenName,"At least one token identifier (tokenId or tokenName) is required"),De=r.z.enum(["buy","sell"]),Ce=r.z.enum(["BUY","SELL"]),Fe=r.z.object({tradeType:De,tokenAmount:F,vaultAddress:C,userAddress:D,slippageTolerance:F.optional(),deadline:r.z.number().int().positive().optional()}),$e=r.z.object({tokenSymbol:v,nativeTokenQuantity:F,expectedToken:F,maxAcceptableReverseBondingCurveFee:$.default("0").optional()}),Pe=r.z.object({tokenSymbol:v,tokenQuantity:F,expectedNativeToken:F,maxAcceptableReverseBondingCurveFee:$.default("0").optional()}),_e=pe.extend({tokenName:E.optional()}),Oe=r.z.object({page:r.z.number().int().min(1).max(1e3).default(1).optional(),limit:r.z.number().int().min(1).max(20).default(10).optional()}),Ue=r.z.enum(["NATIVE","MEME"]),Le=r.z.enum(["IN","OUT"]),xe=r.z.object({type:Ue,method:Le,vaultAddress:C,amount:F}),Re=r.z.object({nativeTokenQuantity:F}),Be=r.z.object({vaultAddress:C}),Me=r.z.object({minFeePortion:F,maxFeePortion:F});function Ke(e){return t=>{const a=e.safeParse(t);return{success:a.success,data:a.success?a.data:void 0,errors:a.success?void 0:a.error.errors.map(e=>e.message)}}}const Ge=Ke(k),ze=Ke(v),qe=Ke(T),Ve=Ke(D),We=Ke(C),je=Ke(F),He=Ke(P),Qe=Ke(N),Xe=Ke(b),Ye=Ke(E),Je=Ke(Z),Ze=Ke(Q),et=Ke(ee),tt=Ke(ae),at=Ke(Te),nt=Ke(Ee),rt=Ke(be),st=Ke(Ne),ot=Ke(Se),it=Ke(Ie),ct=Ke(Fe),lt=Ke($e),dt=Ke(Pe),ut=Ke(_e),ht=Ke(Oe),mt=Ke(xe),gt=Ke(Re),pt=Ke(Be);function ft(e,t){throw new g(e.join("; "),t,"VALIDATION_ERROR")}function yt(e){const t=Ge(e);!t.success&&t.errors&&ft(t.errors,"tokenName")}function wt(e){const t=ye.safeParse(e);t.success||ft(t.error.errors.map(e=>e.message),"pagination")}function At(e){const t=tt(e);!t.success&&t.errors&&ft(t.errors,"options")}function kt(e){const t=mt(e);!t.success&&t.errors&&ft(t.errors,"options")}function vt(e){const t=se.safeParse(e);t.success||ft(t.error.errors.map(e=>e.message),"options")}function Tt(e){const t=D.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}var Et=Object.freeze({__proto__:null,normalizeAddressInput:function(e){if(!e)return;const t=D.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:Tt,validateCheckPoolOptions:At,validateGetAmountOptions:kt,validateGetGraphOptions:vt,validatePagination:wt,validateTokenName:yt});function bt(e){if(!e)return[];return(Array.isArray(e)?e:e.token??[]).map(e=>({...e,createdAt:new Date(e.createdAt),updatedAt:new Date(e.updatedAt)}))}function Nt(e,t){const a=Number(e.page)||t.page,n=Number(e.limit)||t.limit,r=Number(e.total)||Number(e.data?.count)||0;return{page:a,limit:n,total:r,totalPages:Math.ceil(r/n)}}function St(e,t){return{hasNext:e<t,hasPrevious:e>1}}function It(e,t,a=!1){const n=!0===e.error||200!==e.status,r=a&&!e.data;if(n||r)throw new Error(e.message||t)}const Dt="/launchpad/upload-image",Ct="/launchpad/fetch-pool",Ft="/launchpad/check-pool",$t="/launchpad/get-graph-data",Pt="/holders",_t="/launchpad/get-badge/",Ot="/trade/",Ut="/token/commment",Lt="/token/commment",xt="/user/profile",Rt="/user/profile",Bt="/user/token-list",Mt="/user/token-hold",Kt="/user/transfer-faucets";function Gt(e){return e&&"object"==typeof e&&"string"==typeof e.tokenName&&(void 0===e.from||"number"==typeof e.from)&&(void 0===e.to||"number"==typeof e.to)&&(void 0===e.resolution||"number"==typeof e.resolution)}class zt{constructor(e){this.http=e}async fetchPools(e={}){let t;wt({page:e.page??1,limit:e.limit??10}),e.tokenName&&yt(e.tokenName),"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const a={page:e.page||1,limit:e.limit||10};e.search&&(a.search=e.search),e.tokenName&&(a.tokenName=e.tokenName),t&&(a.type=t);const n={page:a.page.toString(),limit:a.limit.toString()};void 0!==a.type&&(n.type=a.type),void 0!==a.tokenName&&(n.tokenName=a.tokenName),void 0!==a.search&&(n.search=a.search);const r=h(n),s=await this.http.get(Ct,r);It(s,"Failed to fetch pools",!0);return{pools:function(e){if(!e)return[];let t=[];return e.tokens?t=Array.isArray(e.tokens)?e.tokens.map(e=>({...e,createdAt:new Date(e.created_at??e.createdAt)})):[{...e.tokens,createdAt:new Date(e.tokens.created_at??e.tokens.createdAt)}]:e.pools&&Array.isArray(e.pools)&&(t=e.pools.map(e=>({...e,createdAt:new Date(e.created_at??e.createdAt)}))),t}(s.data),page:s.data.page,limit:s.data.limit,total:s.data.total,totalPages:s.data.totalPages,hasNext:s.data.page<s.data.totalPages,hasPrevious:s.data.page>1}}async checkPool(e){At(e),e.tokenName&&yt(e.tokenName);const t=h(e),a=await this.http.get(Ft,t);It(a,"Failed to check pool");const n=a.data;return e.symbol?n?.isSymbolExist??!1:e.tokenName?n?.isNameExist??!1:n?.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(!Gt(e))throw new g("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:a,to:n,resolution:r}=e;if(yt(t),!a||!n||!r)throw new g("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const s={tokenName:t,from:a,to:n,resolution:r};vt(s);const o=h(s),i=await this.http.get($t,o);return It(i,"Failed to fetch graph data",!0),{dataPoints:i.data}}async fetchTokenDistribution(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.resolveTokenNameToVault(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);const a=encodeURIComponent(t),n=await this.http.get(`${Pt}/${a}`);return It(n,"Failed to fetch token distribution",!0),{holders:n.data.holders||[],totalSupply:n.data.totalSupply||"0",totalHolders:n.data.totalHolders||0,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.http.get(_t,{tokenName:e});return It(t,"Failed to fetch token badges",!0),{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadge(e){const{tokenName:t,badgeType:a,badgeName:n}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const r=("volume"===a?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===n);return r?.isActive||!1}catch{return!1}}async resolveTokenNameToVault(e){try{const t=await this.fetchPools({tokenName:e});return t.pools&&Array.isArray(t.pools)&&t.pools.length>0?t.pools[0].vaultAddress||null:t.pools&&t.pools.tokens&&t.pools.tokens.vaultAddress||null}catch{return null}}}function qt(e,t={}){const{stringifyFields:a=[],optionalFields:n=[],fieldMappings:r={}}=t,s={};for(const[t,o]of Object.entries(e)){const e=t;if(n.includes(e)&&void 0===o)continue;if(n.includes(e)&&"string"==typeof o&&0===o.trim().length)continue;const i=r[e],c=i?String(i):t;a.includes(e)?s[c]=String(o):s[c]=o}return h(s)}const Vt={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:20}};class Wt{constructor(e){this.http=e}async fetchTrades(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||void 0!==t.tradeType&&"buy"!==t.tradeType&&"sell"!==t.tradeType||void 0!==t.userAddress&&"string"!=typeof t.userAddress||void 0!==t.page&&"number"!=typeof t.page||void 0!==t.limit&&"number"!=typeof t.limit)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");var t;const{tokenName:a,tradeType:n,userAddress:r,page:s=1,limit:o=10,startDate:i,endDate:c,sortOrder:l}=e;if(!y(a))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");p(s,o,Vt);const d=function(e,t,a){return qt({tokenName:e,page:t,limit:a},{stringifyFields:["page","limit"]})}(a,s,o),u=await this.http.get(Ot,d),h=(m=u.data)?(Array.isArray(m)?m:m.trades?m.trades:[]).map(e=>({...e,createdAt:new Date(e.createdAt),updatedAt:new Date(e.updatedAt)})):[];var m;const f=Nt(u,{page:s,limit:o}),w=St(f.page,f.totalPages);return{trades:h,...f,...w}}}function jt(e,t){if(!e)return t||new Date;if(e instanceof Date)return isNaN(e.getTime())?t||new Date:e;try{const a=new Date(e);return isNaN(a.getTime())?t||new Date:a}catch(e){return t||new Date}}function Ht(e,t){return new g(`Token "${e}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND")}const Qt={PAGINATION:{MIN_PAGE:1,MAX_PAGE:1e3,MIN_LIMIT:1,MAX_LIMIT:50},CONTENT:{MIN_LENGTH:1,MAX_LENGTH:500}};class Xt{constructor(e,t){this.http=e,this.poolService=t}async fetchComments(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||void 0!==t.page&&"number"!=typeof t.page||void 0!==t.limit&&"number"!=typeof t.limit)throw new g("Invalid options provided. Expected { tokenName: string, page?: number, limit?: number }","options","INVALID_OPTIONS");var t;const{tokenName:a,page:n=1,limit:r=10}=e;if(!y(a))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");p(n,r,Qt);const s=await this.poolService.resolveTokenNameToVault(a);if(!s)throw Ht(a);const o=qt({vaultAddress:s,page:n,limit:r},{stringifyFields:["page","limit"]}),i=await this.http.get(Ut,o);It(i,"Failed to fetch comments");return{comments:i.data.comments.map(e=>({...e,createdAt:jt(e.createdAt)})),total:i.data.count}}async postComment(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.tokenName||"string"!=typeof t.content)throw new g("Invalid options provided. Expected { tokenName: string, content: string }","options","INVALID_OPTIONS");var t;const{tokenName:a,content:n}=e;if(!y(a))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>=Qt.CONTENT.MIN_LENGTH&&t.length<=Qt.CONTENT.MAX_LENGTH}(n))throw new g(`Comment content must be between ${Qt.CONTENT.MIN_LENGTH} and ${Qt.CONTENT.MAX_LENGTH} characters`,"content","INVALID_CONTENT");const r=await this.poolService.resolveTokenNameToVault(a);if(!r)throw Ht(a);const s={userAddress:this.http.getAddress(),vaultAddress:r,content:n};It(await this.http.post(Lt,s),"Failed to create comment")}}const Yt={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 Jt(e){return!(!e||"string"!=typeof e)&&Yt.USER_ADDRESS.PATTERN.test(e)}const Zt="Update profile";class ea{constructor(e){this.http=e}async fetchProfile(e){const t=e??this.http.getAddress();if(!Jt(t))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");const a={userAddress:t};return await this.http.get(xt,a)}async updateProfile(e){this.validateUpdateProfileData(e);let t=e.profileImage;if(!t||""===t.trim())try{const a=await this.fetchProfile(e.address);t=a.data?.profileImage||""}catch{t=""}const a={profileImage:t,fullName:e.fullName,userAddress:e.address},n=await this.http.signCustomMessage(Zt),r={address:n.address,message:Zt,publickey:n.ethereumAddress,sign:n.signature};It(await this.http.put(Rt,a,r),"Profile update failed")}async uploadProfileImage(e){this.validateUploadProfileImageOptions(e);const t=e.address??this.http.getAddress();try{const a=new FormData;if("undefined"!=typeof File&&e.file instanceof File)a.append("image",e.file);else{if(!Buffer.isBuffer(e.file))throw new g("Invalid file type","file","INVALID_FILE_TYPE");{const n=`profile-image-${t}.png`,r=new Blob([e.file],{type:"image/png"});a.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`${Dt}?tokenName=${encodeURIComponent(t)}`,data:a,headers:{}});return It(n,"Image upload failed"),"string"==typeof n.data?n.data:""}catch(e){if(e instanceof g)throw e;throw new g(`Profile image upload failed: ${e instanceof Error?e.message:"Unknown error"}`,"file","UPLOAD_FAILED")}}async fetchTokenList(e){this.validateGetTokenListOptions(e);const t=qt({page:e.page,limit:e.limit,type:"all"!==e.type&&e.type?e.type:"DEFI",address:e.address,search:e.search,tokenName:e.tokenName},{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),a=await this.http.get(Bt,t);It(a,"Failed to fetch token list",!0);const n=bt(a.data),r=Nt(a,{page:e.page||1,limit:e.limit||10}),s=St(r.page,r.totalPages);return{tokens:n,...r,...s}}async fetchTokensHeld(e){this.validateGetTokenListOptions(e);const t=qt({page:e.page,limit:e.limit,address:e.address,search:e.search,tokenName:e.tokenName},{stringifyFields:["page","limit"],optionalFields:["address","search","tokenName"]}),a=await this.http.get(Mt,t);It(a,"Failed to fetch tokens held",!0);const n=bt(a.data),r=Nt(a,{page:e.page||1,limit:e.limit||10}),s=St(r.page,r.totalPages);return{tokens:n,...r,...s}}async fetchTokensCreated(e={}){const{page:t=1,limit:a=10,search:n,tokenName:r}=e,s={type:"DEFI",address:this.http.getAddress(),page:t,limit:a};return void 0!==n&&(s.search=n),void 0!==r&&(s.tokenName=r),this.fetchTokenList(s)}validateGetTokenListOptions(e){if(p(e.page,e.limit,Yt),void 0!==e.address&&!Jt(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>=Yt.SEARCH.MIN_LENGTH&&t.length<=Yt.SEARCH.MAX_LENGTH))throw new g(`Search query must be between ${Yt.SEARCH.MIN_LENGTH} and ${Yt.SEARCH.MAX_LENGTH} characters`,"search","INVALID_SEARCH");var t,a;if(void 0!==e.tokenName&&e.tokenName.trim().length>0&&!((a=e.tokenName)&&"string"==typeof a&&a.length>=Yt.TOKEN_NAME.MIN_LENGTH&&a.length<=Yt.TOKEN_NAME.MAX_LENGTH))throw new g(`Token name must be between ${Yt.TOKEN_NAME.MIN_LENGTH} and ${Yt.TOKEN_NAME.MAX_LENGTH} characters`,"tokenName","INVALID_TOKEN_NAME")}validateUpdateProfileData(e){if(!Jt(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>=Yt.PROFILE.FULL_NAME.MIN_LENGTH&&t.length<=Yt.PROFILE.FULL_NAME.MAX_LENGTH&&Yt.PROFILE.FULL_NAME.ALPHABETS_ONLY_PATTERN.test(t)))throw new g(`Full name must be between ${Yt.PROFILE.FULL_NAME.MIN_LENGTH} and ${Yt.PROFILE.FULL_NAME.MAX_LENGTH} characters`,"fullName","INVALID_FULL_NAME");var t}validateUploadProfileImageOptions(e){if(e.address&&!Jt(e.address))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS")}}class ta extends Error{constructor(e,t,a){super(e),this.filename=t,this.mimeType=a,this.name="FileValidationError"}}function aa(e,t,a){if(!e)throw new ta("File is required",t,a);if("undefined"!=typeof File&&e instanceof File){const t=ce.safeParse(e);if(!t.success){const a=t.error.errors.map(e=>e.message).join("; ");throw new ta(a,e.name,e.type)}return}if(Buffer.isBuffer(e)){if(!t)throw new ta("Filename is required when uploading Buffer objects",t,a);const n=le.safeParse(e);if(!n.success){const e=n.error.errors.map(e=>e.message).join("; ");throw new ta(e,t,a)}if(t.length>255)throw new ta(`Filename length ${t.length} exceeds maximum allowed length of 255 characters`,t,a);const r=["image/png","image/jpg","image/jpeg","image/gif","image/webp","image/svg+xml"];if(!r.includes(a))throw new ta(`Invalid file type "${a}" is not allowed. Allowed types: ${r.join(", ")}`,t,a);const s=function(e){if(!e)return"";const t=e.lastIndexOf(".");if(-1===t)return"";return e.substring(t).toLowerCase()}(t),o=[".png",".jpg",".jpeg",".gif",".webp",".svg"];if(!o.includes(s))throw new ta(`File extension "${s}" is not allowed. Allowed extensions: ${o.join(", ")}`,t,a);const i=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"}}(s);if(i!==a&&"application/octet-stream"!==i)throw new ta(`File extension "${s}" does not match MIME type "${a}"`,t,a);return}throw new ta("File must be a File object (browser) or Buffer (Node.js)",t,a)}class na{constructor(e){this.http=e}async uploadImageByTokenName(e){const{tokenName:t,options:a}=e;yt(t);const n=`${t}.png`;aa(a.file,n,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&a.file instanceof File)e.append("image",a.file);else{if(!Buffer.isBuffer(a.file))throw new Error("Unsupported file type");{const n=`${a.tokenName??t}.png`,r=new Blob([a.file],{type:"image/png"});e.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`${Dt}?tokenName=${encodeURIComponent(a.tokenName??t)}`,data:e,headers:{}});return It(n,"Image upload failed"),"string"==typeof n.data?n.data:""}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw new Error("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.");throw e}}}class ra{constructor(e){this.http=e}async transferFaucets(e){this.validateTransferFaucetsData(e);const t={userAddress:e.walletAddress,amount:e.amount};It(await this.http.post(Kt,t),"Faucet transfer failed")}validateTransferFaucetsData(e){if(!Jt(e.walletAddress))throw new g("Address must be in format: eth|[40-hex-chars]","address","INVALID_ADDRESS");if(!(t=e.amount)||"string"!=typeof t||!Yt.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 sa{constructor(e){this.http=e,this.poolService=new zt(e),this.tradeService=new Wt(e),this.commentService=new Xt(e,this.poolService),this.userService=new ea(e),this.imageService=new na(e),this.faucetService=new ra(e)}async uploadImageByTokenName(e){return this.imageService.uploadImageByTokenName(e)}async fetchPools(e={}){return this.poolService.fetchPools(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 yt(e)}}class oa extends n.ChainCallDTO{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,a,n,r){let s;if("string"==typeof n){const e=n.split("$");if(4!==e.length)throw new Error(`Invalid token class key format: ${n}. Expected format: collection$category$type$additionalKey`);s={collection:e[0],category:e[1],type:e[2],additionalKey:e[3],instance:"0"}}else s={collection:n.collection,category:n.category,type:n.type,additionalKey:n.additionalKey,instance:"0"};return new oa({from:e,to:t,quantity:a,tokenInstance:s,uniqueKey:r||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`})}static forGALA(e,t,a,n){return new oa({from:e,to:t,quantity:a,tokenInstance:{collection:"GALA",category:"Unit",type:"none",additionalKey:"none",instance:"0"},uniqueKey:n||`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 ia{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},a={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,a,e),domain:t,types:a,signerPublicKey:this.wallet.signingKey.publicKey}}static toGalaChainAddress(e){const a=e.replace("0x","");return`eth|${t.ethers.getAddress(`0x${a}`).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){const t=e.split("$");if(4!==t.length)throw new Error(`Invalid token class key format: ${e}. Expected format: collection$category$type$additionalKey`);return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3],instance:"0"}}}function ca(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")}var la,da=Object.freeze({__proto__:null,normalizeToTokenInstanceKey:ca});!function(e){e.DEBUG="DEBUG",e.INFO="INFO",e.WARN="WARN",e.ERROR="ERROR"}(la||(la={}));class ua{constructor(e){this.levelPriority={[la.DEBUG]:0,[la.INFO]:1,[la.WARN]:2,[la.ERROR]:3},this.debugEnabled=e.debug,this.context=e.context||"SDK",this.minLevel=e.minLevel||(e.debug?la.DEBUG:la.INFO)}debug(e,t){this.log(la.DEBUG,e,t)}info(e,t){this.log(la.INFO,e,t)}warn(e,t){this.log(la.WARN,e,t)}error(e,t){this.log(la.ERROR,e,t)}log(e,t,a){if(this.levelPriority[e]<this.levelPriority[this.minLevel])return;if(e===la.DEBUG&&!this.debugEnabled)return;const n=`[${(new Date).toISOString()}] [${this.context}] [${e}]`,r=this.getConsoleMethod(e);void 0!==a?a instanceof Error?r(`${n} ${t}`,a.message,a.stack):r(`${n} ${t}`,a):r(`${n} ${t}`)}getConsoleMethod(e){switch(e){case la.DEBUG:return console.debug;case la.INFO:return console.info;case la.WARN:return console.warn;case la.ERROR:return console.error;default:return console.log}}child(e){return new ua({debug:this.debugEnabled,context:`${this.context}:${e}`,minLevel:this.minLevel})}isDebugEnabled(){return this.debugEnabled&&this.levelPriority[la.DEBUG]>=this.levelPriority[this.minLevel]}}const ha={MAX_UNIQUE_KEY_LENGTH:64,UNIQUE_KEY_PATTERN:/^(galaswap-operation-|galaconnect-operation-)/,TOKEN_NAME_PATTERN:/^[a-zA-Z0-9]+$/};var ma;!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"}(ma||(ma={}));class ga extends Error{constructor(e,t,a){super(e),this.type=t,this.details=a,this.name="TransferError"}}class pa{constructor(e,t,a,n=!1){this.http=e,this.wallet=t,this.tokenResolver=a,this.signatureHelper=new ia(t),this.logger=new ua({debug:n,context:"GalaChainService"})}async fetchPoolDetails(e){this.validateFetchPoolDetailsData(e);const t=await this.http.post("/api/asset/launchpad-contract/FetchSaleDetails",e);if(1!==t.Status)throw new Error(`Failed to fetch pool details: Status ${t.Status}`);return t}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(1!==t.Status||!t.Data||0===t.Data.length)return null;const a=t.Data.find(t=>t.collection===e.collection&&t.category===e.category&&t.additionalKey===e.additionalKey&&t.type===e.type);if(!a||"0"===a.quantity)return null;const n=`${a.collection}|${a.category}|${a.additionalKey}|${a.type}`;return{quantity:a.quantity,collection:a.collection,category:a.category,tokenId:n}}catch(e){throw new Error(`Failed to fetch token balance from GalaChain: ${e.message||"Unknown error"}`)}}async transferGala(e){this.validateTransferGalaData(e);try{const t=A(e.recipientAddress),a=A(this.wallet.address),n=oa.forGALA(a,t,e.amount,e.uniqueKey),r=await this.signatureHelper.signTransferToken(n.toSigningPayload()),s=new oa({...n.toSigningPayload(),signedPayload:r});this.logger.debug("[DEBUG] Full GALA Transfer Request Payload:",JSON.stringify(s,null,2));const o=await this.http.post("/api/asset/token-contract/TransferToken",s);if(this.logger.debug("[DEBUG] Transfer response:",JSON.stringify(o,null,2)),o&&"object"==typeof o){if("Status"in o&&1===o.Status&&"Data"in o){const e=o;return Array.isArray(e.Data)&&e.Data.length>0?"gala-transfer-successful":"transfer-successful-no-id"}if("transactionId"in o&&o.transactionId)return o.transactionId}throw new ga("Transfer succeeded but transaction ID could not be extracted",ma.NETWORK_ERROR)}catch(t){throw this.handleTransferError(t,"GALA transfer failed",e)}}async transferToken(e){this.validateTransferTokenData(e);try{const t=A(e.to),a=A(this.wallet.address);let n;if(e.tokenId)n=ca(e.tokenId),this.logger.debug("[DEBUG] Using provided tokenId:",e.tokenId),this.logger.debug("[DEBUG] Normalized Token Instance:",JSON.stringify(n,null,2));else{if(!e.tokenName)throw new ga("Must provide either tokenId or tokenName for token identification",ma.TOKEN_NOT_FOUND);n=await this.resolveTokenInstance(e.tokenName)}const r=new oa({from:a,to:t,quantity:e.amount,tokenInstance:n,uniqueKey:e.uniqueKey||`galaconnect-operation-${Date.now()}_${Math.random().toString(36).substring(2,8)}`}),s=await this.signatureHelper.signTransferToken(r.toSigningPayload()),o=new oa({...r.toSigningPayload(),signedPayload:s});this.logger.debug("[DEBUG] Full Transfer Request Payload:",JSON.stringify(o,null,2));const i=await this.http.post("/api/asset/token-contract/TransferToken",o);if(this.logger.debug("[DEBUG] Token transfer response:",JSON.stringify(i,null,2)),i&&"object"==typeof i){if("Status"in i&&1===i.Status&&"Data"in i){const e=i;return Array.isArray(e.Data)&&e.Data.length>0?"token-transfer-successful":"transfer-successful-no-id"}if("transactionId"in i&&i.transactionId)return i.transactionId}throw new ga("Transfer succeeded but transaction ID could not be extracted",ma.NETWORK_ERROR)}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 ga)throw t;throw new ga(`Failed to resolve token class key for '${e}': ${t instanceof Error?t.message:String(t)}`,ma.TOKEN_NOT_FOUND,{tokenName:e})}}validateTransferGalaData(e){if(!((t=e)&&"object"==typeof t&&"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)throw new g("Invalid GALA transfer data: missing required fields");var t;if(!w(e.recipientAddress))throw new ga("Invalid recipient address format",ma.INVALID_RECIPIENT,{recipientAddress:e.recipientAddress});if(parseFloat(e.amount)<=0)throw new ga("Transfer amount must be positive",ma.INVALID_AMOUNT,{amount:e.amount});if(e.uniqueKey){if(e.uniqueKey.length>ha.MAX_UNIQUE_KEY_LENGTH)throw new g(`Unique key too long. Maximum length: ${ha.MAX_UNIQUE_KEY_LENGTH}`);if(!ha.UNIQUE_KEY_PATTERN.test(e.uniqueKey))throw new ga('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',ma.INVALID_AMOUNT,{uniqueKey:e.uniqueKey})}}validateTransferTokenData(e){if(!((t=e)&&"object"==typeof t&&"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)throw new g("Invalid token transfer data: missing required fields");var t;if(!w(e.to))throw new ga("Invalid recipient address format",ma.INVALID_RECIPIENT,{recipientAddress:e.to});if(!e.tokenId&&!e.tokenName)throw new ga("Must provide either tokenId or tokenName for token identification",ma.TOKEN_NOT_FOUND);if(e.tokenName&&!ha.TOKEN_NAME_PATTERN.test(e.tokenName))throw new ga("Invalid token name format",ma.TOKEN_NOT_FOUND,{tokenName:e.tokenName});if(parseFloat(e.amount)<=0)throw new ga("Transfer amount must be positive",ma.INVALID_AMOUNT,{amount:e.amount});if(e.uniqueKey){if(e.uniqueKey.length>ha.MAX_UNIQUE_KEY_LENGTH)throw new g(`Unique key too long. Maximum length: ${ha.MAX_UNIQUE_KEY_LENGTH}`);if(!ha.UNIQUE_KEY_PATTERN.test(e.uniqueKey))throw new ga('Invalid unique key format. Must start with "galaswap-operation-" or "galaconnect-operation-"',ma.INVALID_AMOUNT,{uniqueKey:e.uniqueKey})}}async resolveTokenInstance(e){try{const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new ga(`Token '${e}' not found or not available for transfer`,ma.TOKEN_NOT_FOUND,{tokenName:e});const a=this.resolveTokenInstanceFromVaultAddress(t);return this.logger.debug(`[DEBUG] Token resolution for '${e}':\n Vault Address: ${t}\n Token Instance: ${JSON.stringify(a,null,2)}`),a}catch(t){if(t instanceof ga)throw t;throw new ga(`Failed to resolve token '${e}': ${t instanceof Error?t.message:String(t)}`,ma.TOKEN_NOT_FOUND,{tokenName:e})}}resolveTokenInstanceFromVaultAddress(e){const[t,a]=e.split("|");if(!a)throw new ga(`Invalid vault address format: missing token components. Address: ${e}`,ma.TOKEN_NOT_FOUND);const n=a.split("$");if(n.length<4)throw new ga(`Invalid vault address format: insufficient token components. Expected 4+, got ${n.length}. Address: ${e}`,ma.TOKEN_NOT_FOUND);return{collection:n[0],category:n[1],type:n[2],additionalKey:n[3],instance:"0"}}handleTransferError(e,t,a){if(e instanceof ga)return e;if(e instanceof g)return new ga(e.message,ma.INVALID_AMOUNT);if(400===e.response?.status)return new ga(e.response.data?.message||"Invalid transfer request",ma.INVALID_AMOUNT);if(403===e.response?.status)return new ga("Insufficient balance for transfer",ma.INSUFFICIENT_BALANCE);if(404===e.response?.status){const e={};return"tokenName"in a&&(e.tokenName=a.tokenName),new ga("Token not found",ma.TOKEN_NOT_FOUND,e)}return"ECONNABORTED"===e.code||"ETIMEDOUT"===e.code?new ga("Transfer request timed out",ma.NETWORK_ERROR):new ga(e.message||t,ma.NETWORK_ERROR)}}class fa{constructor(e){this.http=e}async fetchTokenSpotPrice(e){if(!e||Array.isArray(e)&&0===e.length)throw new Error("At least one symbol must be provided");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.http.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),a=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&a.push({symbol:e.symbol,price:e.currentPrices.usd})}),a}catch(e){throw new Error(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`)}}async fetchLaunchpadTokenSpotPrice(e,t){if(!e||"string"!=typeof e)throw new Error("Token name is required and must be a string");try{const a=await t({tokenName:e,amount:"1",type:"native"}),n=(await this.fetchTokenSpotPrice("GALA"))[0];if(!n)throw new Error("GALA price not available");const r=Number(a.amount)/1e18;if(r<=0)throw new Error(`Invalid token amount calculation: ${r}`);const s=n.price/r;return{symbol:e.toUpperCase(),price:s}}catch(t){if(t instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${String(t)}`)}}}function ya(e,t=18){const a=parseFloat(e);if(0===a)return"0";return a.toFixed(t).replace(/\.?0+$/,"")}function wa(e){return ya(e,8)}function Aa(e){return ya(e,18)}class ka extends n.ChainCallDTO{constructor(e,t,a="0",n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=wa(t),this.expectedToken=Aa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class va extends n.ChainCallDTO{constructor(e,t,a,n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Aa(t),this.expectedNativeToken=wa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class Ta extends n.ChainCallDTO{constructor(e,t,a="0",n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.tokenQuantity=Aa(t),this.expectedNativeToken=wa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}class Ea extends n.ChainCallDTO{constructor(e,t,a,n={maxAcceptableReverseBondingCurveFee:"0"}){super(),this.vaultAddress=e,this.nativeTokenQuantity=wa(t),this.expectedToken=Aa(a),this.extraFees={maxAcceptableReverseBondingCurveFee:wa(n.maxAcceptableReverseBondingCurveFee)}}}const ba={BuyNativeDto:ka,BuyExactDto:va,SellExactDto:Ta,SellNativeDto:Ea};var Na,Sa,Ia;!function(e){e[e.METAMASK=0]="METAMASK",e[e.TRUST_WALLET=1]="TRUST_WALLET",e[e.GALA_WALLET=2]="GALA_WALLET"}(Na||(Na={}));class Da{constructor(e,t=!1){this.walletProvider=e,this.debug=t}async signDTO(e,t,a,r=Na.METAMASK){try{this.debug;const s=this.generateEIP712Types(t,e),o=n.calculatePersonalSignPrefix(e),i={...e,prefix:o};let c,l,d;switch(r){case Na.GALA_WALLET:({signature:c,domain:l}=await this.signWithGalaWallet(s,i,t,a));break;case Na.TRUST_WALLET:c=await this.signWithTrustWallet(i),l={name:"ethereum",chainId:1};break;case Na.METAMASK:default:({signature:c,domain:l}=await this.signWithMetaMask(s,i))}return d=r===Na.TRUST_WALLET?{...i,signature:c}:{...e,signature:c,types:s,domain:l},this.debug,d}catch(e){throw this.debug,new Error(`Failed to sign DTO: ${e.message}`)}}async signWithMetaMask(e,t){try{let a,n;if(this.walletProvider.signTypedData&&!this.walletProvider.getNetwork)a={name:"ethereum",chainId:1},n=await this.walletProvider.signTypedData(a,e,t);else{if(!this.walletProvider.getNetwork||!this.walletProvider.signTypedData)throw new Error("Wallet provider does not support typed data signing");{const r=await this.walletProvider.getNetwork();a={name:r.name,chainId:Number(r.chainId)},n=await this.walletProvider.signTypedData(a,e,t)}}return{signature:n,domain:a}}catch(e){throw new Error(`MetaMask/ethers signing failed: ${e.message}`)}}async signWithTrustWallet(e){try{const a=(t=e,JSON.stringify(t));let n;if(!this.walletProvider.signMessage)throw new Error("Wallet provider does not support signMessage");return n=await this.walletProvider.signMessage(a),n}catch(e){throw new Error(`TrustWallet signing failed: ${e.message}`)}var t}async signWithGalaWallet(e,t,a,n){try{const r={name:"ethereum",chainId:1};if("undefined"==typeof window)return this.debug,await this.signWithMetaMask(e,t);const s={domain:r,types:e,message:t,Primary_type:a},o=window;if(!o?.gala)throw new Error("GalaWallet not found in window object");await o.gala.setAddress(n);return{signature:await o.gala.request({method:"eth_signTypedData",params:[JSON.stringify(s),n]}),domain:r}}catch(e){throw new Error(`GalaWallet signing failed: ${e.message}`)}}generateEIP712Types(e,t){const a={};a[e]=[];const n=(e,t,r,s=!1)=>{if(void 0!==t){if(Array.isArray(t)){const o=n(e,t[0],r,!0);return s||a[r].push({name:e,type:(o??e)+"[]"}),o?o+"[]":void 0}if("object"==typeof t&&null!==t){if(a[e])throw new Error(`Type name collision not supported: ${e}`);return a[e]=[],Object.entries(t).forEach(([t,a])=>{n(t,a,e)}),s||a[r].push({name:e,type:e}),e}{let n;switch(typeof t){case"string":n="string";break;case"number":n="uint256";break;case"boolean":n="bool";break;default:throw new Error(`Unsupported type for fieldName ${e}: ${typeof t}, value: ${t}`)}return s||a[r].push({name:e,type:n}),n}}};return Object.entries(t).forEach(([t,a])=>{n(t,a,e)}),this.debug,a}detectWalletPreference(){if("undefined"==typeof window)return Na.METAMASK;const e=window;return e?.gala?Na.GALA_WALLET:e?.trustWallet?.isTrust?Na.TRUST_WALLET:Na.METAMASK}}class Ca{constructor(e=!1){this.debug=e}generateStringsInstructions(e){try{this.debug;const t=this.extractTokenSymbolFromVault(e),a=this.createTokenInstance(t),n=this.createGalaInstance(),r=`$service$${a.toStringKey()}$launchpad`,s=`$tokenBalance$${a.toStringKey()}$${e}`,o=`$tokenBalance$${a.toStringKey()}$${e}`,i=`$tokenBalance$${n.toStringKey()}$${e}`,c=[r,s,o,i,`$tokenBalance$${n.toStringKey()}$${e}`];return this.debug,c}catch(e){throw this.debug,new Error(`Failed to generate stringsInstructions: ${e.message}`)}}createTokenInstance(e){const t=new s.TokenClassKey;return t.collection=e.toLowerCase(),t.category="Unit",t.type="none",t.additionalKey="none",this.debug,t}createGalaInstance(){const e=new s.TokenClassKey;return e.collection="GALA",e.category="Unit",e.type="none",e.additionalKey="none",this.debug,e}extractTokenSymbolFromVault(e){if(!e||"string"!=typeof e)throw new Error("Vault address is required and must be a string");const t=e.split("$");if(t.length<3)throw new Error(`Invalid vault address format: ${e}`);const a=t[2];if(!a||0===a.trim().length)throw new Error(`Empty token symbol in vault address: ${e}`);return this.debug,a}validateVaultAddress(e){if(!e||"string"!=typeof e)throw new Error("Vault address must be a non-empty string");if(!e.startsWith("service|Token$Unit$"))throw new Error('Vault address must start with "service|Token$Unit$"');if(!e.endsWith("$launchpad"))throw new Error('Vault address must end with "$launchpad"');const t=e.split("$");if(t.length<5)throw new Error('Vault address must have at least 5 parts separated by "$"');const a=t[2];if(!a||!/^[A-Za-z]{1,10}$/.test(a))throw new Error("Token symbol must be 1-10 letters (case insensitive)");return this.debug,!0}generateTokenClassKeyString(e,t,a,n){return`${e}$${t}$${a}$${n}`}parseTokenClassKeyString(e){const t=e.split("$");if(4!==t.length)throw new Error(`Invalid TokenClassKey string format: ${e}`);return{collection:t[0],category:t[1],type:t[2],additionalKey:t[3]}}}function Fa(e,t,a){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 n=new o(e);if(n.isNaN())throw new Error(`Invalid expected amount: ${e}. Must be a valid number`);if(0===t)return e;const r=n.multipliedBy(t);let s;switch(a){case"buy-native":case"sell-exact":s=n.minus(r);break;case"buy-exact":case"sell-native":s=n.plus(r);break;default:throw new Error(`Unknown operation type: ${a}`)}return s.isLessThan(0)&&(s=new o(0)),s.toFixed()}class $a{constructor(e,t,a=!1,n,r,s=.05,o=.01){this.httpClient=e,this.tokenResolver=t,this.walletProvider=n,this.userAddress=r,this.defaultSlippageToleranceFactor=s,this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor=o,this.bundleEndpoint="/bundle",this.logger=new ua({debug:a,context:"BundleService"}),n&&r&&(this.signatureService=new Da(n,a),this.tokenKeyService=new Ca(a))}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 a=await this.httpClient.post(this.bundleEndpoint,t);return this.logger.debug("๐Ÿ“ฅ Bundle API response:",{success:a.success,hasData:!!a.data,error:a.error}),this.handleBundleResponse(a)}catch(e){return this.logger.error("โŒ Bundle transaction submission failed:",e),{success:!1,error:this.formatErrorMessage(e)}}}validateBundleData(e){if(!e)throw new Error("Bundle data is required");if(!e.signedDto)throw new Error("Signed DTO is required");if(!e.method||"string"!=typeof e.method)throw new Error("Method name is required");if(!Array.isArray(e.stringsInstructions))throw new Error("stringsInstructions must be an array");if(0===e.stringsInstructions.length)throw new Error("stringsInstructions cannot be empty");const t=["BuyWithNative","BuyExactToken","SellExactToken","SellWithNative"];if(!t.includes(e.method))throw new Error(`Unsupported method: ${e.method}. Supported: ${t.join(", ")}`);e.stringsInstructions.forEach((e,t)=>{if("string"!=typeof e||0===e.length)throw new Error(`stringsInstructions[${t}] must be a non-empty string`);if(!e.startsWith("$"))throw new Error(`stringsInstructions[${t}] must start with '$': ${e}`)}),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){return"string"==typeof e?e:e?.response?.data?.error?e.response.data.error:e?.response?.data?.message?e.response.data.message:e?.message?e.message:"Unknown bundle transaction error"}async getBundlerTransactionResult(e){try{if(!e||"string"!=typeof e)throw new Error("Transaction ID is required");this.logger.debug("๐Ÿ” Checking bundler transaction result:",e);const t=await this.httpClient.get(`${this.bundleEndpoint}?id=${e}`);return this.logger.debug("๐Ÿ“Š Bundler transaction result:",t),{success:!0,data:t}}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 new Error("Transaction ID is required");this.logger.debug("๐Ÿšซ Cancelling transaction:",e);const t=await this.httpClient.delete(`${this.bundleEndpoint}/${e}`);return this.logger.debug("๐Ÿ—‘๏ธ Transaction cancellation response:",t),{success:!0,data:t}}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.httpClient.get(`${this.bundleEndpoint}/health`);return this.logger.debug("๐Ÿ’š Bundle service health:",e),{success:!0,data:e}}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:a,type:n,expectedAmount:r,maxAcceptableReverseBondingCurveFee:s,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,slippageToleranceFactor:i}=e,c=i??this.defaultSlippageToleranceFactor,l=o??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let d=s||"0";s&&(d=Fa(s,l,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:s,slippageFactor:l,adjustedMaxFee:d}));const u=await this.resolveTokenNameToVault(t);if(!u)throw Ht(t);if("native"===n){if(!r)throw new g("expectedAmount is required for native buy operations. Use getBuyTokenAmount() first to calculate expected tokens.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"buy-native");this.logger.debug("BuyNative slippage applied:",{originalExpectedTokens:r,slippageFactor:c,adjustedMinTokens:e});const t=new ba.BuyNativeDto(u,a,e,{maxAcceptableReverseBondingCurveFee:d});return await this.executeBundleTransaction(t,"BuyWithNative",u)}{if(!r)throw new g("expectedAmount is required for exact buy operations. Use getBuyTokenAmount() first to calculate expected GALA cost.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"buy-exact");this.logger.debug("BuyExact slippage applied:",{originalExpectedGalaCost:r,slippageFactor:c,adjustedMaxGalaCost:e});const t=new ba.BuyExactDto(u,a,e,{maxAcceptableReverseBondingCurveFee:d});return await this.executeBundleTransaction(t,"BuyExactToken",u)}}async sellToken(e){this.ensureTradingServicesAvailable();const{tokenName:t,amount:a,type:n,expectedAmount:r,maxAcceptableReverseBondingCurveFee:s,maxAcceptableReverseBondingCurveFeeSlippageFactor:o,slippageToleranceFactor:i}=e,c=i??this.defaultSlippageToleranceFactor,l=o??this.defaultMaxAcceptableReverseBondingCurveFeeSlippageFactor;let d=s||"0";s&&(d=Fa(s,l,"buy-exact"),this.logger.debug("Reverse bonding curve fee slippage applied:",{baseFee:s,slippageFactor:l,adjustedMaxFee:d}));const u=await this.resolveTokenNameToVault(t);if(!u)throw Ht(t);if("exact"===n){if(!r)throw new g("expectedAmount is required for exact sell operations. Use getSellTokenAmount() first to calculate expected GALA.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"sell-exact");this.logger.debug("SellExact slippage applied:",{originalExpectedGala:r,slippageFactor:c,adjustedMinGala:e});const t=new ba.SellExactDto(u,a,e,{maxAcceptableReverseBondingCurveFee:s||"0"});return await this.executeBundleTransaction(t,"SellExactToken",u)}{if(!r)throw new g("expectedAmount is required for native sell operations. Use getSellTokenAmount() first to calculate tokens to sell.","expectedAmount","EXPECTED_AMOUNT_REQUIRED");const e=Fa(r,c,"sell-native");this.logger.debug("SellNative slippage applied:",{originalExpectedTokensToSell:r,slippageFactor:c,adjustedMaxTokensToSell:e});const t=new ba.SellNativeDto(u,a,e,{maxAcceptableReverseBondingCurveFee:s||"0"});return await this.executeBundleTransaction(t,"SellWithNative",u)}}ensureTradingServicesAvailable(){if(!this.signatureService||!this.tokenKeyService)throw new Error("Trading services not available. BundleService requires walletProvider and userAddress for trading operations.");if(!this.userAddress)throw new Error("User address is required for trading operations.")}async executeBundleTransaction(e,t,a){this.ensureTradingServicesAvailable();try{e.uniqueKey=`galaswap - operation - ${i.v4()}-${Date.now()}-${this.userAddress}`;const n=await this.signatureService.signDTO(e,t,this.userAddress),r=this.tokenKeyService.generateStringsInstructions(a),s={stringsInstructions:r,method:t,signedDto:n};this.logger.debug("๐Ÿ“ฆ Bundle transaction data:",{method:t,stringsInstructions:r,dtoKeys:Object.keys(n)});const o=await this.submitTransaction(s);if(o.success&&o.data)return this.logger.debug("โœ… Bundle transaction submitted:",o.data),{success:!0,data:{transactionId:o.data,message:"Transaction submitted successfully. Monitor WebSocket for completion."}};throw new Error(o.error||"Bundle transaction failed")}catch(e){throw this.logger.error("โŒ Bundle transaction error:",e),e}}async resolveTokenNameToVault(e){return await this.tokenResolver.resolveTokenToVault(e)}}!function(e){e.PROCESSED="PROCESSED",e.COMPLETED="COMPLETED",e.SUCCESS="SUCCESS",e.FAILED="FAILED",e.ERROR="ERROR",e.PROCESSING="PROCESSING",e.PENDING="PENDING"}(Sa||(Sa={})),e.SDKTransactionStatus=void 0,(Ia=e.SDKTransactionStatus||(e.SDKTransactionStatus={})).PENDING="pending",Ia.PROCESSING="processing",Ia.COMPLETED="completed",Ia.FAILED="failed",Ia.TIMEOUT="timeout";const Pa={[Sa.PROCESSED]:e.SDKTransactionStatus.COMPLETED,[Sa.COMPLETED]:e.SDKTransactionStatus.COMPLETED,[Sa.SUCCESS]:e.SDKTransactionStatus.COMPLETED,[Sa.FAILED]:e.SDKTransactionStatus.FAILED,[Sa.ERROR]:e.SDKTransactionStatus.FAILED,[Sa.PROCESSING]:e.SDKTransactionStatus.PROCESSING,[Sa.PENDING]:e.SDKTransactionStatus.PENDING};class _a{constructor(e,t=!1){this.socket=null,this.listeners=new Map,this.timeouts=new Map,this.reconnectCount=0,this.hasOnAnyListener=!1,this.config={reconnectAttempts:5,reconnectDelay:2e3,timeout:3e5,...e},this.debug=t,this.logger=new ua({debug:t,context:"WebSocketService"}),this.isSocketIOAvailable=this.checkSocketIOAvailability()}checkSocketIOAvailability(){try{return"function"==typeof c.io||(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=c.io(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.debug&&(this.socket.onAny((e,...t)=>{this.logger.debug(`๐Ÿ“ก [WebSocket Event] "${e}":`,JSON.stringify(t,null,2))}),this.hasOnAnyListener=!0)}catch(e){t(e)}})}async monitorTransaction(t,a){this.listeners.set(t,a),this.logger.debug(`๐Ÿ“ก Starting to monitor transaction: ${t}`),this.logger.debug(`๐Ÿ“ก WebSocket connected: ${!!this.socket&&this.socket.connected}`);const n=setTimeout(()=>{if(this.listeners.has(t)){const n={transactionId:t,status:e.SDKTransactionStatus.TIMEOUT,message:"Transaction monitoring timeout - no response after 60 seconds",timestamp:Date.now()};this.logger.debug(`๐Ÿ“ก Transaction timeout for ${t}`),a(n),this.listeners.delete(t),this.timeouts.delete(t),this.socket?.off(t)}},6e4);if(this.timeouts.set(t,n),this.socket&&this.socket.connected)this.socket.off(t),this.logger.debug(`๐Ÿ“ก Listening for transaction updates: ${t}`),this.logger.debug(`๐Ÿ“ก WebSocket connection ID: ${this.socket.id}`),this.logger.debug(`๐Ÿ“ก WebSocket URL: ${this.config.url}`),this.socket.on(t,n=>{this.logger.debug(`๐Ÿ“ก Socket.IO transaction update for ${t}:`,JSON.stringify(n,null,2));const r=n?.status||n?.Status||n?.data?.status||n?.data?.Status;let s=n?.message||n?.Message||n?.data?.message||n?.data?.Message||n?.error||n?.data?.error;s&&"string"==typeof s||(s=r===Sa.FAILED||r===Sa.ERROR?"Transaction failed - check transaction details":r===Sa.COMPLETED||r===Sa.PROCESSED||r===Sa.SUCCESS?"Transaction completed successfully":r?`Transaction status: ${r}`:"Unknown transaction status");const o={transactionId:t,status:this.mapSocketStatus(r),message:s,timestamp:Date.now(),blockHash:n?.blockHash||n?.data?.blockHash,gasUsed:n?.gasUsed||n?.data?.gasUsed};if(this.logger.debug(`๐Ÿ“ก Mapped status for ${t}: ${r} -> ${o.status}`),this.logger.debug(`๐Ÿ“ก Final message: "${s}"`),a(o),o.status===e.SDKTransactionStatus.COMPLETED||o.status===e.SDKTransactionStatus.FAILED){this.listeners.delete(t);const e=this.timeouts.get(t);e&&(clearTimeout(e),this.timeouts.delete(t)),this.socket?.off(t),this.logger.debug(`๐Ÿ“ก Cleaned up listener for ${t} (${o.status})`)}});else{const n={transactionId:t,status:e.SDKTransactionStatus.FAILED,message:"WebSocket not connected - cannot monitor transaction",timestamp:Date.now()};a(n),this.listeners.delete(t),this.timeouts.delete(t)}}async waitForTransaction(t){return new Promise((a,n)=>{this.monitorTransaction(t,t=>{t.status===e.SDKTransactionStatus.COMPLETED?a(t):t.status!==e.SDKTransactionStatus.FAILED&&t.status!==e.SDKTransactionStatus.TIMEOUT||n(new Error(`Transaction ${t.status}: ${t.message}`))})})}mapSocketStatus(t){const a=t?.toUpperCase();return Pa[a]||e.SDKTransactionStatus.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.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}}class Oa{constructor(e){this.poolService=e,this.cache=new Map}async resolveTokenToVault(e){if(!y(e))throw new g("Token name is required and must be a non-empty string","tokenName","INVALID_TOKEN_NAME");const t=e.trim().toLowerCase(),a=this.get(t);if(a)return a;try{const a=await this.poolService.resolveTokenNameToVault(e);return a&&this.set(t,a),a}catch{return null}}async resolveTokenClassKey(e){const t=await this.resolveTokenToVault(e);if(!t)throw new Error(`Token '${e}' not found. Cannot resolve to TokenClassKey.`);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:a}of e)this.set(t,a)}parseVaultAddressToTokenClassKey(e){const t=e.split("|");if(2!==t.length)throw new Error(`Invalid vault address format: ${e}. Expected format: service|Token$Unit$...$launchpad`);const a=t[1].split("$");if(a.length<4)throw new Error(`Invalid vault address format: ${e}. Expected at least 4 parts after service|`);return{collection:a[0],category:a[1],type:a[2],additionalKey:a[3]}}}function Ua(e){const t=function(e){const t=Je(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 La="/api/asset/launchpad-contract/CallNativeTokenIn",xa="/api/asset/launchpad-contract/CallNativeTokenOut",Ra="/api/asset/launchpad-contract/CallMemeTokenIn",Ba="/api/asset/launchpad-contract/CallMemeTokenOut";class Ma extends n.ChainCallDTO{constructor(e){super(),this.tokenName=e.tokenName,this.tokenSymbol=e.tokenSymbol,this.tokenDescription=e.tokenDescription,this.tokenImage=e.tokenImage,this.preBuyQuantity=e.preBuyQuantity,this.websiteUrl=e.websiteUrl,this.telegramUrl=e.telegramUrl,this.twitterUrl=e.twitterUrl,this.tokenCategory=e.tokenCategory,this.tokenCollection=e.tokenCollection,this.uniqueKey=e.uniqueKey,e.reverseBondingCurveConfiguration&&(this.reverseBondingCurveConfiguration=e.reverseBondingCurveConfiguration)}}function Ka(e){return e&&"object"==typeof e&&"number"==typeof e.Status&&e.Data&&"object"==typeof e.Data&&"string"==typeof e.Data.calculatedQuantity&&e.Data.extraFees&&"object"==typeof e.Data.extraFees&&"string"==typeof e.Data.extraFees.reverseBondingCurve&&"string"==typeof e.Data.extraFees.transactionFees}const Ga={NATIVE:"native",EXACT:"exact"};class za{constructor(e,t,a,n,r){this.http=e,this.tokenResolver=t,this.bundleHttp=a,this.galaChainHttp=n,this.dexApiHttp=r}async uploadImageByTokenName(e){const{tokenName:t,options:a}=e;yt(t);const n=`${t}.png`;aa(a.file,n,"image/png");try{const e=new FormData;if("undefined"!=typeof File&&a.file instanceof File)e.append("image",a.file);else{if(!Buffer.isBuffer(a.file))throw new Error("Unsupported file type");{const n=`${a.tokenName||t}.png`,r=new Blob([a.file],{type:"image/png"});e.append("image",r,n)}}const n=await this.http.request({method:"POST",url:`/launchpad/upload-image?tokenName=${encodeURIComponent(a.tokenName||t)}`,data:e,headers:{}});if(!0===n.error||200!==n.status||!n.data?.imageUrl)throw new Error(n.message||"Image upload failed - no URL returned");return n.data.imageUrl}catch(e){if(e instanceof Error&&e.message.includes("FormData"))throw new Error("File upload failed: FormData not supported in this environment. Ensure you have proper polyfills for Node.js environments.");throw e}}async fetchPoolsFromAPI(e){wt(e),e.tokenName&&yt(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 a=h(t),n=await this.http.get("/launchpad/fetch-pool",a);if(!0===n.error||200!==n.status||!n.data)throw new Error(n.message||"Failed to fetch pools");let r=[];return n.data.tokens?r=Array.isArray(n.data.tokens)?n.data.tokens.map(e=>({...e,createdAt:new Date(e.created_at||e.createdAt)})):[{...n.data.tokens,createdAt:new Date(n.data.tokens.created_at||n.data.tokens.createdAt)}]:n.data.pools&&Array.isArray(n.data.pools)&&(r=n.data.pools.map(e=>({...e,createdAt:new Date(e.created_at||e.createdAt)}))),{pools:r,page:n.data.page,limit:n.data.limit,total:n.data.total,totalPages:n.data.totalPages,hasNext:n.data.page<n.data.totalPages,hasPrevious:n.data.page>1}}async _getAmount(e){if(kt(e),!this.galaChainHttp)throw new Error("GalaChain client not configured. Direct GalaChain calls require galaChainHttp client.");const{endpoint:t,body:a}=((e,t,a,n)=>{if("NATIVE"===e&&"IN"===t)return{endpoint:La,body:{vaultAddress:a,tokenQuantity:n,IsPreMint:!1}};if("NATIVE"===e&&"OUT"===t)return{endpoint:xa,body:{vaultAddress:a,tokenQuantity:n,IsPreMint:!1}};if("MEME"===e&&"IN"===t)return{endpoint:Ra,body:{vaultAddress:a,nativeTokenQuantity:n,IsPreMint:!1}};if("MEME"===e&&"OUT"===t)return{endpoint:Ba,body:{vaultAddress:a,nativeTokenQuantity:n,IsPreMint:!1}};throw new Error(`Unsupported operation: ${e}-${t}`)})(e.type,e.method,e.vaultAddress,e.amount);try{const e=await this.galaChainHttp.post(t,a);if(!Ka(e))throw new Error("Malformed response data from GalaChain gateway");if(1!==e.Status)throw new Error(`GalaChain calculation failed with status ${e.Status}`);const{calculatedQuantity:n,extraFees:r}=e.Data;return{amount:n,reverseBondingCurveFee:r.reverseBondingCurve,transactionFee:r.transactionFees,gasFee:"1"}}catch(e){throw e}}async checkPool(e){At(e),e.tokenName&&yt(e.tokenName);const t=h(e),a=await this.http.get("/launchpad/check-pool",t);if(!0===a.error||200!==a.status)throw new Error(a.message||"Failed to check pool");const n=a.data;return e.symbol?n?.isSymbolExist??!1:e.tokenName?n?.isNameExist??!1:n?.exists??!1}async fetchVolumeData(e){if(!Gt(e))throw new g("Invalid options provided. Expected { tokenName: string, from?: number, to?: number, resolution?: number }","options","INVALID_OPTIONS");const{tokenName:t,from:a,to:n,resolution:r}=e;if(yt(t),!a||!n||!r)throw new g("Graph options (from, to, resolution) are required","options","MISSING_GRAPH_OPTIONS");const s={tokenName:t,from:a,to:n,resolution:r};vt(s);const o=h(s),i=await this.http.get("/launchpad/get-graph-data",o);if(!0===i.error||200!==i.status||!i.data)throw new Error(i.message||"Failed to fetch graph data");return{dataPoints:i.data}}async fetchPools(e={}){let t;"recent"===e.type?t="RECENT":"popular"===e.type&&(t="POPULAR");const a={page:e.page||1,limit:e.limit||10};return e.search&&(a.search=e.search),e.tokenName&&(a.tokenName=e.tokenName),t&&(a.type=t),this.fetchPoolsFromAPI(a)}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:a,type:n}=e;if(!t||"string"!=typeof t)throw new g("Token name is required and must be a string","tokenName","INVALID_TOKEN_NAME");if(!a||"string"!=typeof a)throw new g("Amount is required and must be a string","amount","INVALID_AMOUNT");const r=await this.tokenResolver.resolveTokenToVault(t);if(!r)throw new g(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");if(n!==Ga.NATIVE&&n!==Ga.EXACT)throw new g('Type must be either "native" or "exact"',"type","INVALID_TYPE");return n===Ga.EXACT?this._getAmount({type:"NATIVE",method:"IN",vaultAddress:r,amount:a}):this._getAmount({type:"MEME",method:"OUT",vaultAddress:r,amount:a})}async calculateSellAmount(e){const{tokenName:t,amount:a,type:n}=e,r=await this.tokenResolver.resolveTokenToVault(t);if(!r)throw new g(`Token "${t}" not found. Please verify the token name is correct.`,"tokenName","TOKEN_NOT_FOUND");return n===Ga.EXACT?this._getAmount({type:"NATIVE",method:"OUT",vaultAddress:r,amount:a}):this._getAmount({type:"MEME",method:"IN",vaultAddress:r,amount:a})}async launchToken(e){if(!this.bundleHttp)throw new Error("Bundle backend client not configured. LaunchToken requires bundleHttp client.");Ua(e);const t=e.preBuyQuantity||"0";if(isNaN(Number(t))||Number(t)<0)throw new Error("Pre-buy quantity must be a valid non-negative number string");if(e.reverseBondingCurveConfiguration){const{minFeePortion:t,maxFeePortion:a}=e.reverseBondingCurveConfiguration,n=Number(t),r=Number(a);if(isNaN(n)||isNaN(r)||n<=0||r<=0||n>=r)throw new Error("Reverse bonding curve configuration must have valid min/max fee portions with min < max")}let a="";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 new Error("Image upload failed: No URL returned");a=t}else"string"==typeof e.tokenImage&&(a=e.tokenImage);const r=`galaswap - operation - ${i.v4()}-${Date.now()}-${this.http.getAddress()}`,s={tokenName:e.tokenName.trim(),tokenSymbol:e.tokenSymbol.trim().toUpperCase(),tokenDescription:e.tokenDescription.trim(),tokenImage:a.trim(),preBuyQuantity:t.toString(),websiteUrl:e.websiteUrl||"",telegramUrl:e.telegramUrl||"",twitterUrl:e.twitterUrl||"",tokenCategory:e.tokenCategory||"Unit",tokenCollection:e.tokenCollection||"Token",uniqueKey:r};e.reverseBondingCurveConfiguration&&(s.reverseBondingCurveConfiguration={minFeePortion:e.reverseBondingCurveConfiguration.minFeePortion.toString(),maxFeePortion:e.reverseBondingCurveConfiguration.maxFeePortion.toString()});const o=new Ma(s),c=await this.http.signWithGalaChain("CreateSale",o,n.SigningType.SIGN_TYPED_DATA),{signature:l,types:d,domain:u,prefix:h}=c,m={tokenName:o.tokenName,tokenSymbol:o.tokenSymbol,tokenDescription:o.tokenDescription,tokenImage:o.tokenImage,preBuyQuantity:o.preBuyQuantity,websiteUrl:o.websiteUrl,telegramUrl:o.telegramUrl,twitterUrl:o.twitterUrl,tokenCategory:o.tokenCategory,tokenCollection:o.tokenCollection,uniqueKey:o.uniqueKey,signature:l,types:d,domain:u,...h&&{prefix:h},...o.reverseBondingCurveConfiguration&&{reverseBondingCurveConfiguration:o.reverseBondingCurveConfiguration}},g=`${e.tokenName.trim()}$Unit$none$none`,p="GALA$Unit$none$none";let f;if(parseFloat(t)>0){const e=`$service$${g}$launchpad`;f=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${g}$${e}`,`$tokenBalance$${p}$${e}`,`$tokenBalance$${p}$${e}`]}else{const e=`$service$${g}$launchpad`;f=[e,`$token$${g}$${e}`,`$tokenBalance$${g}$${e}`]}const y={signedDto:m,stringsInstructions:f,method:"CreateSale"},w=await this.bundleHttp.post("/bundle",y);if(w.error||!w.data)throw new Error(w.message||"Token launch failed");return w.data}async fetchTokenDistribution(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.tokenResolver.resolveTokenToVault(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);const a=encodeURIComponent(t),n=await this.http.get(`/holders/${a}`);if(!0===n.error||200!==n.status||!n.data)throw new Error(n.message||"Failed to fetch token distribution");return{holders:n.data.holders||[],totalSupply:n.data.totalSupply||"0",totalHolders:n.data.totalHolders||0,lastUpdated:new Date}}async fetchTokenBadges(e){if(!e)throw new Error("Token name is required");yt(e);const t=await this.http.get("/launchpad/get-badge/",{tokenName:e});if(t.error||!t.data)throw new Error(t.message||"Failed to fetch token badges");return{volumeBadges:t.data.volumeBadge||[],engagementBadges:t.data.engagementBadge||[]}}async hasTokenBadgeByTokenName(e){const{tokenName:t,badgeType:a,badgeName:n}=e;try{const e=await this.fetchTokenBadges(t);if(!e)return!1;const r=("volume"===a?e.volumeBadges:e.engagementBadges).find(e=>e.badgeName===n);return r?.isActive||!1}catch{return!1}}async calculateInitialBuyAmount(e){if(!(t=e)||"object"!=typeof t||"string"!=typeof t.nativeTokenQuantity||void 0!==t.vaultAddress&&"string"!=typeof t.vaultAddress)throw new Error("Invalid pre-mint calculation data");var t;if(!this.galaChainHttp)throw new Error("GalaChain HTTP client not available. Please initialize SDK with galaChainBaseUrl.");try{const t={vaultAddress:"service|testToken",nativeTokenQuantity:e.nativeTokenQuantity,IsPreMint:!0},a=await this.galaChainHttp.post("/api/asset/launchpad-contract/CallMemeTokenOut",t);if(!Ka(a))throw new Error("Malformed response data from GalaChain gateway");if(1!==a.Status)throw new Error(`GalaChain calculation failed with status ${a.Status}`);const{calculatedQuantity:n,extraFees:r}=a.Data;return{amount:n,reverseBondingCurveFee:r.reverseBondingCurve,transactionFee:r.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)}`)}}getAddress(){return this.http.getAddress()}formatAddressForBackend(e){return Tt(e)}validateTokenName(e){return yt(e)}validatePagination(e){return wt(e)}async fetchTokenSpotPrice(e){if(!this.dexApiHttp)throw new Error("DEX API client not configured. Token price fetching requires dexApiHttp client.");if(!e||Array.isArray(e)&&0===e.length)throw new Error("At least one symbol must be provided");const t=Array.isArray(e)?e.join(","):e;try{const e=await this.dexApiHttp.request({method:"GET",url:"/v1/tokens",params:{symbols:t}}),a=[];return e.tokens&&Array.isArray(e.tokens)&&e.tokens.forEach(e=>{e.currentPrices&&e.symbol&&a.push({symbol:e.symbol,price:e.currentPrices.usd})}),a}catch(e){throw new Error(`Failed to fetch token prices: ${e instanceof Error?e.message:e}`)}}async fetchLaunchpadTokenSpotPrice(e){if(!e||"string"!=typeof e)throw new Error("Token name is required and must be a string");try{const t=await this.calculateBuyAmount({tokenName:e,amount:"1",type:"native"}),a=(await this.fetchTokenSpotPrice("GALA"))[0];if(!a)throw new Error("GALA price not available");const n=Number(t.amount)/1e18;if(n<=0)throw new Error(`Invalid token amount calculation: ${n}`);const r=a.price/n;return{symbol:e.toUpperCase(),price:r}}catch(t){if(t instanceof Error)throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${t.message}`);throw new Error(`Failed to calculate launchpad token spot price for ${e}: ${String(t)}`)}}}const qa={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",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",launchpadFrontendUrl:"https://lpad-frontend-dev1.defi.gala.com"}};function Va(e){return qa[e]}class Wa extends Error{constructor(e,t){super(e),this.cause=t,this.name="WebSocketError"}}class ja extends Error{constructor(e,t,a){super(`Transaction ${e} failed with status: ${t}${a?` - ${a}`:""}`),this.transactionId=e,this.status=t,this.name="TransactionFailedError"}}function Ha(e){return e&&"object"==typeof e}function Qa(e,t){if(!e)throw new Wa(`Invalid WebSocket response received for transaction ${t}: response is null or undefined`);if("object"!=typeof e)throw new Wa(`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 Wa(`Invalid WebSocket response received for transaction ${t}: missing status field`)}class Xa{constructor(e){let t=null;t=e.env?Va(e.env):e.baseUrl?.includes("prod")?Va("PROD"):Va("STAGE"),this.config={baseUrl:t.launchpadBaseUrl,galaChainBaseUrl:t.galaChainBaseUrl,bundleBaseUrl:t.bundleBaseUrl,webSocketUrl:t.webSocketUrl,dexApiBaseUrl:t.dexApiBaseUrl,launchpadFrontendUrl:t.launchpadFrontendUrl,timeout:3e4,debug:!1,...e},this.validateConfiguration(),this.slippageToleranceFactor=void 0===e.slippageToleranceFactor?Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR:this.parseSlippageToleranceFactor(e.slippageToleranceFactor),this.maxAcceptableReverseBondingCurveFeeSlippageFactor=void 0===e.maxAcceptableReverseBondingCurveFeeSlippageFactor?Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR:this.parseFeeSlippageFactor(e.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.auth=new u({wallet:e.wallet,messagePrefix:"Create a GalaChain Wallet"}),this.http=new m(this.auth,this.config),this.galaChainHttp=new m(this.auth,{...this.config,baseUrl:this.config.galaChainBaseUrl}),this.bundleHttp=new m(this.auth,{...this.config,baseUrl:this.config.bundleBaseUrl}),this.dexApiHttp=new m(this.auth,{...this.config,baseUrl:this.config.dexApiBaseUrl}),this.launchpadService=new sa(this.http),this.tokenResolverService=new Oa(this.launchpadService.poolService),this.launchpadAPI=new za(this.http,this.tokenResolverService,this.bundleHttp,this.galaChainHttp,this.dexApiHttp),this.galaChainService=new pa(this.galaChainHttp,e.wallet,this.tokenResolverService,e.debug||!1),this.dexService=new fa(this.dexApiHttp),this.bundleService=new $a(this.bundleHttp,this.tokenResolverService,this.config.debug||!1,e.wallet,this.getAddress(),this.slippageToleranceFactor,this.maxAcceptableReverseBondingCurveFeeSlippageFactor),this.websocketService=new _a({url:this.config.webSocketUrl},this.config.debug)}createOverrideSdk(e){if(!e||"string"!=typeof e)throw new Error("Invalid privateKey: must be a non-empty string");if(!e.match(/^0x[a-fA-F0-9]{64}$/))throw new Error('Invalid privateKey format: must be "0x" followed by 64 hexadecimal characters');const a=new t.Wallet(e),n={...this.config,wallet:a};return new Xa(n)}getAddress(){return this.auth.getAddress()}getEthereumAddress(){return this.config.wallet.address}getConfig(){const{wallet:e,...t}=this.config;return{...t,slippageToleranceFactor:this.slippageToleranceFactor,maxAcceptableReverseBondingCurveFeeSlippageFactor:this.maxAcceptableReverseBondingCurveFeeSlippageFactor}}getUrlByTokenName(e){const t=this.config.launchpadFrontendUrl;if(!t)throw new Error("launchpadFrontendUrl not configured in SDK");return`${t.replace(/\/$/,"")}/buy-sell/${e}`}async fetchPools(e){return this.launchpadService.fetchPools(e||{})}async fetchTokenDistribution(e){return this.launchpadService.fetchTokenDistribution(e)}async fetchTokenBadges(e){return this.launchpadService.fetchTokenBadges(e)}async fetchTokenSpotPrice(e){return this.dexService.fetchTokenSpotPrice(e)}async fetchGalaSpotPrice(){const e=await this.dexService.fetchTokenSpotPrice("GALA");if(0===e.length)throw new Error("Failed to fetch GALA price - no price data returned");const t=e.find(e=>"GALA"===e.symbol);if(!t)throw new Error("GALA price not found in response");return t}async fetchLaunchpadTokenSpotPrice(e){return this.launchpadAPI.fetchLaunchpadTokenSpotPrice(e)}async fetchPoolDetails(e){const t=await this.resolveVaultAddress(e);if(!t)throw new Error(`Could not find vault address for token: ${e}`);return(await this.galaChainService.fetchPoolDetails({vaultAddress:t})).Data}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 Et}),a=t(e)||this.getAddress();return this.galaChainService.fetchGalaBalance({owner:a,collection:"GALA",category:"Unit",additionalKey:"none",type:"none",instance:"0"})}async fetchTokenBalance(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a=t(e.address);if(e.tokenId){const{normalizeToTokenInstanceKey:t}=await Promise.resolve().then(function(){return da}),n=t(e.tokenId),{collection:r,category:s,type:o,additionalKey:i}=n;return this.galaChainService.fetchTokenBalance({owner:a,collection:r,category:s,additionalKey:i,type:o,instance:"0"})}if(e.tokenName){const t=(await this.fetchTokensHeld({tokenName:e.tokenName,page:1,limit:1,...a&&{address:a}})).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 new Error("Must provide 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 calculateInitialBuyAmount(e){const t={nativeTokenQuantity:e};return this.launchpadAPI.calculateInitialBuyAmount(t)}async buy(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.buy(n)}await this.ensureWebSocketConnection();const t=await this.bundleService.buyToken(e),a=t.data?.transactionId;if(!a)throw new Error("No transaction ID returned from buy operation");return this.waitForConfirmation(a,t=>{Qa(t,a);const n=t?.data||{};if(!Ha(n))throw new Wa(`Invalid trade data received for transaction ${a}`);const r={transactionId:a,type:"buy",method:"native"===e.type?"native":"exact",inputAmount:n.inputQuantity||e.amount,outputAmount:n.outputQuantity||e.expectedAmount||"0",totalFees:n.totalFees||"0",tokenName:e.tokenName,vaultAddress:n.vaultAddress||"",blockHash:t.blockHash,gasUsed:t.gasUsed,timestamp:Date.now()};return void 0!==e.slippageToleranceFactor&&(r.slippageTolerance=e.slippageToleranceFactor),r})}async sell(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.sell(n)}await this.ensureWebSocketConnection();const t=await this.bundleService.sellToken(e),a=t.data?.transactionId;if(!a)throw new Error("No transaction ID returned from sell operation");return this.waitForConfirmation(a,t=>{Qa(t,a);const n=t?.data||{};if(!Ha(n))throw new Wa(`Invalid trade data received for transaction ${a}`);const r={transactionId:a,type:"sell",method:"native"===e.type?"native":"exact",inputAmount:n.inputQuantity||e.amount,outputAmount:n.outputQuantity||e.expectedAmount||"0",totalFees:n.totalFees||"0",tokenName:e.tokenName,vaultAddress:n.vaultAddress||"",blockHash:t.blockHash,gasUsed:t.gasUsed,timestamp:Date.now()};return void 0!==e.slippageToleranceFactor&&(r.slippageTolerance=e.slippageToleranceFactor),r})}async getBundlerTransactionResult(e){return this.bundleService.getBundlerTransactionResult(e)}async postComment(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.postComment(n)}return this.launchpadService.postComment(e)}async launchToken(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.launchToken(n)}await this.ensureWebSocketConnection();const t=await this.launchpadAPI.launchToken(e);return this.waitForConfirmation(t,a=>{Qa(a,t);const n=a?.data||{};if(!function(e){return e&&"object"==typeof e}(n))throw new Wa(`Invalid launch data received for transaction ${t}`);const r={transactionId:t,vaultAddress:n.vaultAddress||"",tokenStringKey:n.tokenStringKey||"",tokenName:e.tokenName,tokenSymbol:e.tokenSymbol,creatorAddress:n.creatorAddress||this.getAddress(),blockHash:a.blockHash,gasUsed:a.gasUsed,timestamp:Date.now()};return"string"==typeof e.tokenImage&&(r.tokenImage=e.tokenImage),void 0!==e.preBuyQuantity&&(r.preBuyQuantity=e.preBuyQuantity),r.vaultAddress&&this.tokenResolverService.set(e.tokenName,r.vaultAddress),r})}async uploadTokenImage(e){if(e.privateKey){const t=this.createOverrideSdk(e.privateKey),{privateKey:a,...n}=e;return t.uploadTokenImage(n)}return 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 Et}),a=t(e)||this.getAddress();return this.launchpadService.fetchProfile(a)}async updateProfile(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,address:t(e.address)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.updateProfile(n)}return this.launchpadService.updateProfile(a)}async uploadProfileImage(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,address:t(e.address)||this.getAddress()};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.uploadProfileImage(n)}return this.launchpadService.uploadProfileImage(a)}async retrieveGalaFromFaucet(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={walletAddress:t(e)||this.getAddress(),amount:"5"};return this.launchpadService.transferFaucets(a)}async fetchTokensHeld(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a=t(e?.address)||this.getAddress(),n={page:e?.page||1,limit:e?.limit||10,address:a};return e?.tokenName&&(n.tokenName=e.tokenName),e?.search&&(n.search=e.search),this.launchpadService.fetchTokensHeld(n)}async fetchTokensCreated(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={type:"DEFI",address:t(e?.address)||this.getAddress(),page:e?.page||1,limit:e?.limit||10};return e?.tokenName&&(a.tokenName=e.tokenName),e?.search&&(a.search=e.search),this.launchpadService.fetchTokenList(a)}async transferGala(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,recipientAddress:t(e.recipientAddress)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.transferGala(n)}return this.galaChainService.transferGala(a)}async transferToken(e){const{normalizeAddressInput:t}=await Promise.resolve().then(function(){return Et}),a={...e,to:t(e.to)};if(a.privateKey){const e=this.createOverrideSdk(a.privateKey),{privateKey:t,...n}=a;return e.transferToken(n)}return this.galaChainService.transferToken(a)}async resolveTokenClassKey(e){return this.galaChainService.resolveTokenClassKey(e)}async resolveVaultAddress(e){return this.tokenResolverService.resolveTokenToVault(e)}validateConfiguration(){if(("number"!=typeof this.config.timeout||this.config.timeout<=0||this.config.timeout>3e5)&&(this.config.debug,this.config.timeout=3e4),!this.config.baseUrl)throw new Error("baseUrl is required in configuration");if(!this.config.webSocketUrl)throw new Error("webSocketUrl is required in configuration");try{new URL(this.config.baseUrl)}catch{throw new Error(`Invalid baseUrl format: ${this.config.baseUrl}`)}try{new URL(this.config.webSocketUrl)}catch{throw new Error(`Invalid webSocketUrl format: ${this.config.webSocketUrl}`)}if(this.config.galaChainBaseUrl)try{new URL(this.config.galaChainBaseUrl)}catch{throw new Error(`Invalid galaChainBaseUrl format: ${this.config.galaChainBaseUrl}`)}if(this.config.bundleBaseUrl)try{new URL(this.config.bundleBaseUrl)}catch{throw new Error(`Invalid bundleBaseUrl format: ${this.config.bundleBaseUrl}`)}if(this.config.launchpadFrontendUrl)try{new URL(this.config.launchpadFrontendUrl)}catch{throw new Error(`Invalid launchpadFrontendUrl format: ${this.config.launchpadFrontendUrl}`)}}parseSlippageToleranceFactor(e){const t=parseFloat(e);return isNaN(t)||t<0||t>1?(this.config.debug,Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR):t}parseFeeSlippageFactor(e){const t=parseFloat(e);return isNaN(t)||t<0||t>1?(this.config.debug,Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR):t}async ensureWebSocketConnection(){this.websocketService.isConnected()||(await this.websocketService.connect(),this.config.debug)}async waitForConfirmation(e,t){this.config.debug;try{const a=await this.websocketService.waitForTransaction(e);if("completed"!==a.status)throw new ja(e,a.status,a.message);let n;try{n=t(a)}catch(t){if(t instanceof Wa)throw t;throw new Wa(`Failed to transform WebSocket response for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}return this.config.debug,n}catch(t){if(this.config.debug,t instanceof ja||t instanceof Wa)throw t;throw new Wa(`WebSocket confirmation failed for transaction ${e}`,t instanceof Error?t:new Error(String(t)))}}async cleanup(){try{this.config.debug,this.http.cleanup(),this.websocketService&&this.websocketService.disconnect(),this.config.debug}catch(e){this.config.debug}}static cleanupAll(e=!1){const{WebSocketService:t}=require("./services/WebSocketService");t.cleanupAll(e)}}Xa.DEFAULT_SLIPPAGE_TOLERANCE_FACTOR=.15,Xa.DEFAULT_MAX_ACCEPTABLE_REVERSE_BONDING_CURVE_FEE_SLIPPAGE_FACTOR=.01;class Ya{static generateWallet(){try{const e=t.Wallet.createRandom();if(!e.mnemonic?.phrase)throw new Error("Failed to generate wallet with mnemonic phrase");const a=this.toGalaAddress(e.address);return{privateKey:e.privateKey,address:e.address,galaAddress:a,mnemonic:e.mnemonic.phrase,wallet:new t.Wallet(e.privateKey)}}catch(e){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const e=`test-wallet-${Date.now()}-${++this.testCounter}`,a="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64),n=new t.Wallet(a),r=this.toGalaAddress(n.address);return{privateKey:n.privateKey,address:n.address,galaAddress:r,mnemonic:"test test test test test test test test test test test junk",wallet:n}}throw e}}static fromPrivateKey(e){const a=new t.Wallet(e),n=this.toGalaAddress(a.address);return{privateKey:a.privateKey,address:a.address,galaAddress:n,mnemonic:"",wallet:a}}static fromMnemonic(e,a=0){try{const n=t.Mnemonic.fromPhrase(e),r=t.HDNodeWallet.fromMnemonic(n,`m/44'/60'/0'/0/${a}`),s=new t.Wallet(r.privateKey),o=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:o,mnemonic:e,wallet:s}}catch(n){if("undefined"!=typeof process&&"test"===process.env.NODE_ENV){const n=`test-mnemonic-index-${a}-${e}`,r="0x"+Buffer.from(n).toString("hex").padStart(64,"1").slice(0,64),s=new t.Wallet(r),o=this.toGalaAddress(s.address);return{privateKey:s.privateKey,address:s.address,galaAddress:o,mnemonic:e,wallet:s}}throw n}}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 a=0;a<e;a++){const e=`test-multi-${a}-${Date.now()}-${++this.testCounter}`,n="0x"+Buffer.from(e).toString("hex").padStart(64,"1").slice(0,64);t.push(this.fromPrivateKey(n))}else for(let a=0;a<e;a++)t.push(this.generateWallet());return t}static getWalletSummary(e,t=!1){const a=["๐Ÿ” Wallet Information","โ•".repeat(50),`๐Ÿ“ Address: ${e.address}`,`๐ŸŽฎ Gala Address: ${e.galaAddress}`,`๐ŸŒฑ Mnemonic: ${e.mnemonic||"Not available"}`];return t?a.splice(3,0,`๐Ÿ”‘ Private Key: ${e.privateKey}`):a.splice(3,0,"๐Ÿ”‘ Private Key: [HIDDEN - use includeSensitive=true to show]"),a.push("โ•".repeat(50)),a.push("๐Ÿ’พ IMPORTANT: Save your mnemonic phrase securely!"),a.push("This is your backup to recover the wallet."),a.join("\n")}}function Ja(e){if(void 0===e)return Ya.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 Ya.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 Ya.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?"...":""}"`)}Ya.testCounter=0;e.AgentConfig=class{static async quickSetup(e={}){const t=e.environment||this.detectEnvironment(),a=this.setupWallet(e.privateKey),n={wallet:a.wallet,baseUrl:e.baseUrl||this.getDefaultBaseUrl(t),timeout:e.timeout||this.getDefaultTimeout(t),debug:e.debug??"production"!==t,...this.getEnvironmentDefaults(t)},r=new Xa(n),s={sdk:r,wallet:a,config:n};if(!1!==e.autoValidate){const e=await this.validateSetup(r,a);return{...s,validation:e}}return s}static async validateSetup(e,t){const a=[],n=[],r={canTrade:!1,canCreateTokens:!1,hasBalance:!1,connectionHealthy:!1};try{const t=await e.fetchGalaBalance(e.getAddress());if(r.connectionHealthy=!0,t&&t.quantity){const e=parseFloat(t.quantity);r.hasBalance=e>0,r.canTrade=e>=.1,r.canCreateTokens=e>=100,0===e?n.push("Wallet has zero GALA balance - cannot perform transactions"):e<.1?n.push("GALA balance too low for trading (minimum 0.1 GALA)"):e<100&&n.push("GALA balance too low for token creation (minimum 100 GALA)")}else a.push("Failed to fetch GALA balance: No balance returned")}catch(e){a.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||n.push("Pool listing not accessible - some features may be limited")}catch(e){n.push(`Pool access test failed: ${e instanceof Error?e.message:String(e)}`)}return{ready:0===a.length&&r.connectionHealthy,sdk:e,wallet:t||Ya.generateWallet(),issues:a,warnings:n,capabilities:r}}static getRecommendedConfig(e,t="general"){const a={environment:e,autoValidate:!0};switch(e){case"production":Object.assign(a,{debug:!1,timeout:3e4});break;case"development":Object.assign(a,{debug:!0,timeout:45e3});break;case"testing":Object.assign(a,{debug:!0,timeout:6e4})}switch(t){case"trading":a.timeout=1.5*(a.timeout||3e4);break;case"creation":a.timeout=2*(a.timeout||3e4);break;case"monitoring":a.timeout=.5*(a.timeout||3e4)}return a}static async multiWalletSetup(e,t="development"){const a={};for(const[n,r]of Object.entries(e)){const{sdk:e}=await this.quickSetup({environment:t,privateKey:r,agentId:`multi-wallet-${n}`,autoValidate:!1});a[n]=e}return a}static detectEnvironment(){const e=process.env.NODE_ENV?.toLowerCase();return"production"===e?"production":"test"===e||"testing"===e?"testing":"development"}static setupWallet(e){if(!e){const e=process.env.PRIVATE_KEY;return e?Ya.fromPrivateKey(e):Ya.generateWallet()}return"generate"===e?Ya.generateWallet():Ya.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}},e.FileValidationError=ta,e.GALA_DECIMALS=8,e.GALA_TOKEN_CLASS_KEY={collection:"GALA",category:"Unit",type:"none",additionalKey:"none"},e.IMAGE_EXTENSIONS=oe,e.LAUNCHPAD_TOKEN_DECIMALS=18,e.LaunchpadSDK=Xa,e.POOL_TYPES={RECENT:"recent",POPULAR:"popular"},e.SDK_VERSION="3.7.0",e.TRADING_TYPES=Ga,e.TransactionFailedError=ja,e.ValidationError=g,e.WebSocketError=Wa,e.WebSocketTimeoutError=class extends Wa{constructor(e,t){super(`WebSocket confirmation timeout for transaction ${e} after ${t}ms`),this.name="WebSocketTimeoutError"}},e.addressFormatSchema=S,e.amountMethodSchema=re,e.amountTypeSchema=ne,e.browserFileSchema=ce,e.bufferFileSchema=le,e.buyTokensDataSchema=$e,e.calculatePreMintDataSchema=Re,e.checkPoolOptionsSchema=ae,e.commentMessageSchema=z,e.commentPaginationSchema=fe,e.createLaunchpadSDK=function(e){const{wallet:a,env:n,config:r={}}=e||{};let s;if(a)if("string"==typeof a){s=Ja(a).wallet}else{if(!(a instanceof t.Wallet))throw new Error("Invalid wallet input. Expected string (private key or mnemonic) or Wallet instance.");s=a}else{s=Ja().wallet}const o={wallet:s,...n&&{env:n},debug:!1,timeout:3e4,...r};return new Xa(o)},e.createLimitSchema=L,e.createPaginatedResultSchema=function(e){return r.z.object({data:r.z.array(e),page:r.z.number().int().min(1),limit:r.z.number().int().min(1),total:r.z.number().int().min(0),totalPages:r.z.number().int().min(0),hasNext:r.z.boolean(),hasPrevious:r.z.boolean()})},e.createTradeDataSchema=Fe,e.createWallet=Ja,e.ethereumAddressSchema=I,e.faucetAmountSchema=P,e.fetchGalaBalanceOptionsSchema=be,e.fetchPoolDetailsDataSchema=Be,e.fetchTokenBalanceOptionsSchema=Ie,e.fileSizeSchema=M,e.fileUploadSchema=ie,e.filenameSchema=K,e.flexibleAddressSchema=D,e.flexibleFileSchema=de,e.formatGalaForDTO=wa,e.formatLaunchpadTokenForDTO=Aa,e.fullNameSchema=N,e.getAmountOptionsSchema=xe,e.getTradeOptionsSchema=_e,e.graphDataOptionsSchema=se,e.imageExtensionSchema=ue,e.imageFilenameSchema=he,e.imageMimeTypeSchema=G,e.imageUploadOptionsSchema=ee,e.isoDateStringSchema=q,e.launchTokenDataSchema=Z,e.nonNegativeDecimalStringSchema=$,e.optionalUrlSchema=O,e.pageNumberSchema=U,e.paginationResultMetaSchema=ke,e.poolFetchTypeSchema=te,e.poolPaginationSchema=ye,e.positiveDecimalStringSchema=F,e.privateKeySchema=W,e.reverseBondingCurveConfigSchema=J,e.reverseBondingCurveConfigurationSchema=Me,e.searchQuerySchema=b,e.sellTokensDataSchema=Pe,e.standardLimitSchema=x,e.standardPaginationSchema=me,e.timestampSchema=V,e.tokenCategorySchema=X,e.tokenCollectionSchema=Y,e.tokenDescriptionSchema=T,e.tokenListOptionsSchema=Te,e.tokenNameSchema=k,e.tokenSymbolSchema=v,e.tokenUrlsSchema=Q,e.tradeCalculationMethodSchema=Le,e.tradeCalculationTypeSchema=Ue,e.tradeLimitSchema=B,e.tradeListParamsSchema=Oe,e.tradePaginationSchema=pe,e.tradePaginationWithFiltersSchema=Ae,e.tradeTypeBackendSchema=Ce,e.tradeTypeSchema=De,e.transactionIdSchema=j,e.transferFaucetsDataSchema=Ee,e.uniqueKeySchema=H,e.updateProfileDataSchema=Ne,e.uploadProfileImageOptionsSchema=Se,e.urlSchema=_,e.userLimitSchema=R,e.userPaginationSchema=ge,e.userTokenNameSchema=E,e.userTokenTypeSchema=ve,e.userTokensPaginationSchema=we,e.validateAddress=Ve,e.validateAmountString=je,e.validateBuyTokensData=lt,e.validateCalculatePreMintData=gt,e.validateCheckPoolOptions=tt,e.validateCreateTradeData=ct,e.validateFaucetAmount=He,e.validateFetchGalaBalanceOptions=rt,e.validateFetchPoolDetailsData=pt,e.validateFetchTokenBalanceOptions=it,e.validateFullName=Qe,e.validateGetAmountOptions=mt,e.validateGetTradeOptions=ut,e.validateImageUploadOptions=et,e.validateLaunchTokenData=Je,e.validateSearchQuery=Xe,e.validateSellTokensData=dt,e.validateTokenDescription=qe,e.validateTokenListOptions=at,e.validateTokenName=Ge,e.validateTokenSymbol=ze,e.validateTokenUrls=Ze,e.validateTradeListParams=ht,e.validateTransferFaucetsData=nt,e.validateUpdateProfileData=st,e.validateUploadProfileImageOptions=ot,e.validateUserTokenName=Ye,e.validateVaultAddress=We,e.vaultAddressSchema=C});
@@ -31,6 +31,7 @@ export declare class WebSocketService {
31
31
  private debug;
32
32
  private isSocketIOAvailable;
33
33
  private readonly logger;
34
+ private hasOnAnyListener;
34
35
  constructor(config: WebSocketConfig, debug?: boolean);
35
36
  /**
36
37
  * Check if Socket.IO is available in the current environment
@@ -1 +1 @@
1
- {"version":3,"file":"WebSocketService.d.ts","sourceRoot":"","sources":["../../src/services/WebSocketService.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH,OAAO,EAEL,oBAAoB,EAErB,MAAM,0BAA0B,CAAC;AAElC,MAAM,WAAW,iBAAiB;IAChC,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,oBAAoB,CAAC;IAC7B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,eAAe;IAC9B,GAAG,EAAE,MAAM,CAAC;IACZ,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,MAAM,CAAkB;IAChC,OAAO,CAAC,SAAS,CACL;IACZ,OAAO,CAAC,QAAQ,CAA0C;IAC1D,OAAO,CAAC,cAAc,CAAK;IAC3B,OAAO,CAAC,KAAK,CAAU;IACvB,OAAO,CAAC,mBAAmB,CAAU;IACrC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;gBAEpB,MAAM,EAAE,eAAe,EAAE,KAAK,GAAE,OAAe;IAc3D;;;OAGG;IACH,OAAO,CAAC,yBAAyB;IAiBjC;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAkE9B;;OAEG;IACG,kBAAkB,CACtB,aAAa,EAAE,MAAM,EACrB,QAAQ,EAAE,CAAC,MAAM,EAAE,iBAAiB,KAAK,IAAI,GAC5C,OAAO,CAAC,IAAI,CAAC;IA8HhB;;OAEG;IACG,kBAAkB,CAAC,aAAa,EAAE,MAAM,GAAG,OAAO,CAAC,iBAAiB,CAAC;IAY3E;;OAEG;IACH,OAAO,CAAC,eAAe;IAKvB;;OAEG;YACW,eAAe;IAmB7B;;OAEG;IACH,UAAU,IAAI,IAAI;IAelB;;OAEG;IACH,WAAW,IAAI,OAAO;CAGvB"}
1
+ {"version":3,"file":"WebSocketService.d.ts","sourceRoot":"","sources":["../../src/services/WebSocketService.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAIH,OAAO,EAEL,oBAAoB,EAErB,MAAM,0BAA0B,CAAC;AAElC,MAAM,WAAW,iBAAiB;IAChC,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,oBAAoB,CAAC;IAC7B,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,eAAe;IAC9B,GAAG,EAAE,MAAM,CAAC;IACZ,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,MAAM,CAAkB;IAChC,OAAO,CAAC,SAAS,CACL;IACZ,OAAO,CAAC,QAAQ,CAA0C;IAC1D,OAAO,CAAC,cAAc,CAAK;IAC3B,OAAO,CAAC,KAAK,CAAU;IACvB,OAAO,CAAC,mBAAmB,CAAU;IACrC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAS;IAChC,OAAO,CAAC,gBAAgB,CAAS;gBAErB,MAAM,EAAE,eAAe,EAAE,KAAK,GAAE,OAAe;IAc3D;;;OAGG;IACH,OAAO,CAAC,yBAAyB;IAiBjC;;OAEG;IACG,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC;IAmE9B;;OAEG;IACG,kBAAkB,CACtB,aAAa,EAAE,MAAM,EACrB,QAAQ,EAAE,CAAC,MAAM,EAAE,iBAAiB,KAAK,IAAI,GAC5C,OAAO,CAAC,IAAI,CAAC;IA8HhB;;OAEG;IACG,kBAAkB,CAAC,aAAa,EAAE,MAAM,GAAG,OAAO,CAAC,iBAAiB,CAAC;IAY3E;;OAEG;IACH,OAAO,CAAC,eAAe;IAKvB;;OAEG;YACW,eAAe;IAmB7B;;OAEG;IACH,UAAU,IAAI,IAAI;IA4BlB;;OAEG;IACH,WAAW,IAAI,OAAO;CAGvB"}