@hfunlabs/hypurr-connect 0.1.9 → 0.1.10
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/README.md +80 -64
- package/dist/index.d.ts +21 -15
- package/dist/index.js +333 -257
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
- package/src/GrpcExchangeTransport.ts +16 -10
- package/src/HypurrConnectProvider.tsx +341 -129
- package/src/LoginModal.tsx +15 -74
- package/src/grpc.ts +2 -2
- package/src/index.ts +0 -2
- package/src/types.ts +27 -5
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/HypurrConnectProvider.tsx","../src/agent.ts","../src/grpc.ts","../src/GrpcExchangeTransport.ts","../src/LoginModal.tsx","../src/icons/MetaMaskColorIcon.tsx","../src/icons/TelegramColorIcon.tsx","../src/TelegramLoginWidget.tsx","../src/types.ts"],"sourcesContent":["import {\n ExchangeClient,\n HttpTransport,\n type IRequestTransport,\n} from \"@hfunlabs/hyperliquid\";\nimport { PrivateKeySigner, signUserSignedAction } from \"@hfunlabs/hyperliquid/signing\";\nimport type { TelegramUserResponse } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service\";\nimport type {\n TelegramUser as HypurrTelegramUser,\n TelegramChatWalletPack,\n} from \"hypurr-grpc/ts/hypurr/user\";\nimport type { HyperliquidWallet } from \"hypurr-grpc/ts/hypurr/wallet\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n type ReactNode,\n} from \"react\";\nimport {\n AGENT_NAME,\n clearAgent as clearStoredAgent,\n fetchActiveAgent,\n generateAgentKey,\n isAgentValid,\n isDeadAgentError,\n loadAgent,\n saveAgent,\n} from \"./agent\";\nimport { createStaticClient, createTelegramClient } from \"./grpc\";\nimport { GrpcExchangeTransport } from \"./GrpcExchangeTransport\";\nimport type {\n AuthMethod,\n EoaSigner,\n HypurrConnectConfig,\n HypurrConnectState,\n HypurrUser,\n SignTypedDataFn,\n StoredAgent,\n TelegramLoginData,\n} from \"./types\";\n\n/** @internal context value — extends the public type with fields used only by library internals */\ninterface InternalConnectState extends HypurrConnectState {\n loginTelegram: (data: TelegramLoginData) => void;\n botUsername: string;\n useWidget: boolean;\n}\n\nconst TELEGRAM_STORAGE_KEY = \"hypurr-connect-tg-user\";\n\nfunction toAuthDataMap(data: TelegramLoginData): Record<string, string> {\n const map: Record<string, string> = {\n id: String(data.id),\n first_name: data.first_name,\n auth_date: String(data.auth_date),\n hash: data.hash,\n };\n if (data.last_name) map.last_name = data.last_name;\n if (data.username) map.username = data.username;\n if (data.photo_url) map.photo_url = data.photo_url;\n return map;\n}\n\nfunction isInvalidTelegramAuthError(err: unknown): boolean {\n const msg =\n err instanceof Error\n ? err.message\n : typeof err === \"object\" && err !== null && \"message\" in err\n ? String((err as { message: unknown }).message)\n : String(err);\n return /invalid telegram auth data/i.test(msg);\n}\n\nconst HypurrConnectContext = createContext<InternalConnectState | null>(null);\n\nexport function useHypurrConnect(): HypurrConnectState {\n const ctx = useContext(HypurrConnectContext);\n if (!ctx)\n throw new Error(\n \"useHypurrConnect must be used within <HypurrConnectProvider>\",\n );\n return ctx;\n}\n\n/** @internal — gives library components access to fields not on the public API */\nexport function useHypurrConnectInternal(): InternalConnectState {\n const ctx = useContext(HypurrConnectContext);\n if (!ctx)\n throw new Error(\n \"useHypurrConnectInternal must be used within <HypurrConnectProvider>\",\n );\n return ctx;\n}\n\nexport function HypurrConnectProvider({\n config,\n children,\n}: {\n config: HypurrConnectConfig;\n children: ReactNode;\n}) {\n const tgClient = useMemo(() => createTelegramClient(config), [config]);\n const staticClient = useMemo(() => createStaticClient(config), [config]);\n\n // ── Telegram auth state ──────────────────────────────────────\n const [tgLoginData, setTgLoginData] = useState<TelegramLoginData | null>(\n () => {\n try {\n const stored = localStorage.getItem(TELEGRAM_STORAGE_KEY);\n return stored ? JSON.parse(stored) : null;\n } catch {\n return null;\n }\n },\n );\n const [tgUser, setTgUser] = useState<HypurrTelegramUser | null>(null);\n const [tgLoading, setTgLoading] = useState(false);\n const [tgError, setTgError] = useState<string | null>(null);\n\n const authDataMap = useMemo(\n () => (tgLoginData ? toAuthDataMap(tgLoginData) : {}),\n [tgLoginData],\n );\n\n const [tgUserTick, setTgUserTick] = useState(0);\n\n // Auto-disconnect when the server rejects telegram auth data.\n const onInvalidAuthRef = useRef<(() => void) | null>(null);\n onInvalidAuthRef.current = () => {\n console.warn(\"[HypurrConnect] Invalid telegram auth data — disconnecting.\");\n setTgLoginData(null);\n setTgUser(null);\n setTgError(null);\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n };\n\n useEffect(() => {\n if (!tgLoginData) return;\n let cancelled = false;\n setTgLoading(true);\n setTgError(null);\n\n (async () => {\n try {\n const authData = toAuthDataMap(tgLoginData);\n const [{ response: userResp }, { response: walletsResp }] =\n await Promise.all([\n tgClient.telegramUser({ authData }),\n tgClient.telegramUserWallets({ authData }),\n ]);\n if (cancelled) return;\n const user = (userResp as TelegramUserResponse).user ?? null;\n if (user) {\n // TelegramUser.wallets lacks twap/scale sessions; replace with\n // the full wallets from TelegramUserWallets which populates them.\n user.wallets = walletsResp.wallets;\n }\n setTgUser(user);\n } catch (err) {\n if (cancelled) return;\n if (isInvalidTelegramAuthError(err)) {\n onInvalidAuthRef.current?.();\n return;\n }\n console.error(\"[HypurrConnect] gRPC TelegramUser failed:\", err);\n setTgError(err instanceof Error ? err.message : String(err));\n } finally {\n if (!cancelled) setTgLoading(false);\n }\n })();\n\n return () => {\n cancelled = true;\n };\n }, [tgLoginData, tgClient, tgUserTick]);\n\n // ── EOA auth state ───────────────────────────────────────────\n const [eoaAddress, setEoaAddress] = useState<`0x${string}` | null>(null);\n const [agent, setAgent] = useState<StoredAgent | null>(null);\n const [eoaLoading, setEoaLoading] = useState(false);\n const [eoaError, setEoaError] = useState<string | null>(null);\n const eoaSignerRef = useRef<EoaSigner | null>(null);\n\n // ── Derived auth ─────────────────────────────────────────────\n const authMethod: AuthMethod = tgLoginData\n ? \"telegram\"\n : eoaAddress\n ? \"eoa\"\n : null;\n\n // ── Multi-wallet state (Telegram) ─────────────────────────────\n const [wallets, setWallets] = useState<HyperliquidWallet[]>([]);\n const [selectedWalletId, setSelectedWalletId] = useState<number>(0);\n const [packs, setPacks] = useState<TelegramChatWalletPack[]>([]);\n\n const refreshWallets = useCallback(() => setTgUserTick((t) => t + 1), []);\n\n useEffect(() => {\n if (authMethod !== \"telegram\" || !tgUser) {\n setWallets([]);\n setSelectedWalletId(0);\n setPacks([]);\n return;\n }\n\n const userWallets = tgUser.wallets ?? [];\n setWallets(userWallets);\n setPacks(tgUser.packs ?? []);\n\n const defaultId = tgUser.walletId || userWallets[0]?.id || 0;\n setSelectedWalletId((prev) => {\n if (prev && userWallets.some((w) => w.id === prev)) return prev;\n return defaultId;\n });\n }, [authMethod, tgUser]);\n\n const selectedWallet = useMemo(\n () => wallets.find((w) => w.id === selectedWalletId) ?? wallets[0] ?? null,\n [wallets, selectedWalletId],\n );\n\n const selectWallet = useCallback(\n (walletId: number) => {\n if (wallets.some((w) => w.id === walletId)) {\n setSelectedWalletId(walletId);\n }\n },\n [wallets],\n );\n\n // ── Session poller (TWAP + Scale for selected wallet) ──────\n const pollInterval = config.sessionPollInterval ?? 5_000;\n\n useEffect(() => {\n if (authMethod !== \"telegram\" || !selectedWalletId || !pollInterval) return;\n\n let cancelled = false;\n\n const poll = async () => {\n try {\n const [{ response: twapResp }, { response: scaleResp }] =\n await Promise.all([\n tgClient.hyperliquidWalletTwapSessions({\n authData: authDataMap,\n walletId: selectedWalletId,\n }),\n tgClient.hyperliquidWalletScaleSessions({\n authData: authDataMap,\n walletId: selectedWalletId,\n }),\n ]);\n if (cancelled) return;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: twapResp.sessions,\n scaleSessions: scaleResp.sessions,\n }\n : w,\n ),\n );\n } catch (err) {\n if (isInvalidTelegramAuthError(err)) {\n onInvalidAuthRef.current?.();\n return;\n }\n // Silently ignore poll errors — next tick will retry.\n }\n };\n\n // Fire immediately on mount / wallet switch, then on interval.\n poll();\n const id = setInterval(poll, pollInterval);\n return () => {\n cancelled = true;\n clearInterval(id);\n };\n }, [authMethod, selectedWalletId, pollInterval, tgClient, authDataMap]);\n\n const user = useMemo<HypurrUser | null>(() => {\n if (tgLoginData && authMethod === \"telegram\" && selectedWallet) {\n return {\n address: selectedWallet.ethereumAddress,\n walletId: selectedWallet.id,\n displayName: tgLoginData.username\n ? `@${tgLoginData.username}`\n : tgLoginData.first_name,\n photoUrl: tgLoginData.photo_url,\n authMethod: \"telegram\",\n telegramId: String(tgLoginData.id),\n hfunScore: tgUser?.reputation?.hfunScore,\n reputationScore: tgUser?.reputation?.reputationScore,\n };\n }\n if (eoaAddress && authMethod === \"eoa\") {\n return {\n address: eoaAddress,\n walletId: 0,\n displayName: `${eoaAddress.slice(0, 6)}...${eoaAddress.slice(-4)}`,\n authMethod: \"eoa\",\n };\n }\n return null;\n }, [tgLoginData, selectedWallet, eoaAddress, authMethod, tgUser]);\n\n // ── Exchange client ──────────────────────────────────────────\n // Telegram: GrpcExchangeTransport → HyperliquidCoreAction (server signs)\n // EOA: dual wallet — agent key for L1 actions, master signer for user-signed\n // actions (transfers, withdrawals, etc.). The dual wallet inspects the\n // EIP-712 domain name to decide which key signs each request.\n // When a signer is available but no agent exists yet, the dual wallet\n // auto-provisions an agent on the first L1 action (triggers one extra\n // wallet popup for the approveAgent user-signed action).\n\n const onDeadAgentRef = useRef<((address: `0x${string}`) => void) | null>(\n null,\n );\n onDeadAgentRef.current = (addr: `0x${string}`) => {\n clearStoredAgent(addr);\n setAgent(null);\n setEoaError(\"Agent expired or was deregistered. Please reconnect.\");\n };\n\n // Mutable slot for the agent signer — the dual wallet reads this so it can\n // pick up a newly provisioned agent without waiting for a React re-render.\n const agentSignerRef = useRef<PrivateKeySigner | null>(\n agent ? new PrivateKeySigner(agent.privateKey) : null,\n );\n useEffect(() => {\n agentSignerRef.current = agent\n ? new PrivateKeySigner(agent.privateKey)\n : null;\n }, [agent]);\n\n // Lock to prevent concurrent auto-provisioning attempts\n const provisioningRef = useRef<Promise<PrivateKeySigner> | null>(null);\n\n const agentReady =\n authMethod === \"telegram\" || (authMethod === \"eoa\" && !!agent);\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const exchange = useMemo<ExchangeClient<any> | null>(() => {\n if (authMethod === \"telegram\" && user?.address) {\n const transport = new GrpcExchangeTransport({\n isTestnet: config.isTestnet ?? false,\n telegramClient: tgClient,\n authDataMap,\n walletId: user.walletId,\n onAuthError: () => onInvalidAuthRef.current?.(),\n });\n return new ExchangeClient({\n transport,\n externalSigning: true,\n userAddress: user.address as `0x${string}`,\n });\n }\n\n if (authMethod === \"eoa\" && eoaAddress) {\n const hasSigner = !!eoaSignerRef.current;\n\n if (!agent && !hasSigner) {\n const noAgentTransport: IRequestTransport = {\n isTestnet: config.isTestnet ?? false,\n request(): Promise<never> {\n throw new Error(\n \"[HypurrConnect] No agent key approved and no wallet signer available. \" +\n \"Either call approveAgent(signTypedDataAsync) or pass a signer to \" +\n \"connectEoa(address, { signTypedData, chainId }).\",\n );\n },\n };\n return new ExchangeClient({\n transport: noAgentTransport,\n externalSigning: true,\n userAddress: eoaAddress,\n });\n }\n\n const isTestnet = config.isTestnet ?? false;\n const inner = new HttpTransport({ isTestnet });\n const deadAgentAddr = eoaAddress;\n const guardedTransport: IRequestTransport = {\n isTestnet: inner.isTestnet,\n async request<T>(\n endpoint: \"info\" | \"exchange\" | \"explorer\",\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n try {\n return await inner.request<T>(endpoint, payload, signal);\n } catch (err) {\n if (endpoint === \"exchange\" && isDeadAgentError(err)) {\n onDeadAgentRef.current?.(deadAgentAddr);\n }\n throw err;\n }\n },\n };\n\n const signerRef = eoaSignerRef;\n const agentRef = agentSignerRef;\n const provRef = provisioningRef;\n const ownerAddress = eoaAddress;\n\n /**\n * Auto-provision an agent key when one doesn't exist yet.\n *\n * Bypasses the SDK's `executeUserSignedAction` (and its per-address\n * semaphore) to avoid deadlocking when called from inside\n * `dualWallet.signTypedData`, which is already inside the SDK's\n * `executeL1Action` lock for the same address.\n */\n const ensureAgent = async (): Promise<PrivateKeySigner> => {\n const existing = agentRef.current;\n if (existing) return existing;\n\n if (provRef.current) return provRef.current;\n\n const signer = signerRef.current;\n if (!signer) {\n throw new Error(\n \"[HypurrConnect] No wallet signer available to auto-provision agent. \" +\n \"Pass a signer to connectEoa(address, { signTypedData, chainId }).\",\n );\n }\n\n provRef.current = (async () => {\n try {\n const { privateKey, address: agentAddress } =\n await generateAgentKey();\n\n const chainIdHex = `0x${signer.chainId.toString(16)}` as `0x${string}`;\n const nonce = Date.now();\n const action = {\n type: \"approveAgent\" as const,\n signatureChainId: chainIdHex,\n hyperliquidChain: (isTestnet ? \"Testnet\" : \"Mainnet\") as\n | \"Testnet\"\n | \"Mainnet\",\n agentAddress: agentAddress.toLowerCase() as `0x${string}`,\n agentName: AGENT_NAME,\n nonce,\n };\n\n const approveAgentTypes = {\n \"HyperliquidTransaction:ApproveAgent\": [\n { name: \"hyperliquidChain\", type: \"string\" },\n { name: \"agentAddress\", type: \"address\" },\n { name: \"agentName\", type: \"string\" },\n { name: \"nonce\", type: \"uint64\" },\n ],\n };\n\n const wallet = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n signTypedData(params: any) {\n return signer.signTypedData(params);\n },\n getAddresses: async () => [ownerAddress] as `0x${string}`[],\n getChainId: async () => signer.chainId,\n };\n\n const signature = await signUserSignedAction({\n wallet,\n action,\n types: approveAgentTypes,\n });\n\n const apiUrl = isTestnet\n ? \"https://api-ui.hyperliquid-testnet.xyz/exchange\"\n : \"https://api.hyperliquid.xyz/exchange\";\n\n const res = await fetch(apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ action, signature, nonce }),\n });\n\n const body = await res.json();\n if (body?.status === \"err\") {\n throw new Error(\n `approveAgent API error: ${body.response ?? JSON.stringify(body)}`,\n );\n }\n\n const remote = await fetchActiveAgent(ownerAddress, isTestnet);\n const validUntil =\n remote?.validUntil ?? Date.now() + 7 * 24 * 60 * 60 * 1000;\n\n const stored: StoredAgent = {\n privateKey,\n address: agentAddress,\n approvedAt: Date.now(),\n validUntil,\n };\n saveAgent(ownerAddress, stored);\n\n const newSigner = new PrivateKeySigner(privateKey);\n agentRef.current = newSigner;\n setAgent(stored);\n\n return newSigner;\n } finally {\n provRef.current = null;\n }\n })();\n\n return provRef.current;\n };\n\n // Dual wallet: routes signing based on the EIP-712 domain.\n // \"Exchange\" domain → L1 action → agent key signs (auto-provisions if needed).\n // \"HyperliquidSignTransaction\" domain → user-signed → master wallet (popup).\n const dualWallet = {\n address: ownerAddress,\n async signTypedData(params: {\n domain: {\n name?: string;\n version?: string;\n chainId?: number;\n verifyingContract?: `0x${string}`;\n salt?: `0x${string}`;\n };\n types: Record<string, { name: string; type: string }[]>;\n primaryType: string;\n message: Record<string, unknown>;\n }): Promise<`0x${string}`> {\n if (params.domain.name === \"HyperliquidSignTransaction\") {\n const signer = signerRef.current;\n if (!signer) {\n throw new Error(\n \"[HypurrConnect] No wallet signer available for user-signed actions. \" +\n \"Pass a signer to connectEoa(address, { signTypedData, chainId }).\",\n );\n }\n return signer.signTypedData(\n params as Parameters<typeof signer.signTypedData>[0],\n );\n }\n\n const agentSigner = await ensureAgent();\n return agentSigner.signTypedData(params);\n },\n };\n\n return new ExchangeClient({\n transport: guardedTransport,\n wallet: dualWallet,\n signatureChainId: () => {\n const id = signerRef.current?.chainId ?? 42161;\n return `0x${id.toString(16)}` as `0x${string}`;\n },\n });\n }\n\n return null;\n }, [\n authMethod,\n user,\n agent,\n eoaAddress,\n config.isTestnet,\n tgClient,\n authDataMap,\n ]);\n\n const handleClearAgent = useCallback(() => {\n if (eoaAddress) {\n clearStoredAgent(eoaAddress);\n setAgent(null);\n }\n }, [eoaAddress]);\n\n // ── Wallet management (Telegram only) ───────────────────────\n const createWallet = useCallback(\n async (name: string): Promise<HyperliquidWallet> => {\n const { response } = await tgClient.hyperliquidWalletCreate({\n authData: authDataMap,\n name,\n });\n refreshWallets();\n if (!response.wallet)\n throw new Error(\"Wallet creation returned no wallet\");\n return response.wallet;\n },\n [tgClient, authDataMap, refreshWallets],\n );\n\n const deleteWallet = useCallback(\n async (walletId: number): Promise<void> => {\n await tgClient.hyperliquidWalletDelete({\n authData: authDataMap,\n walletId,\n });\n if (walletId === selectedWalletId) {\n const remaining = wallets.filter((w) => w.id !== walletId);\n setSelectedWalletId(remaining[0]?.id ?? 0);\n }\n refreshWallets();\n },\n [tgClient, authDataMap, selectedWalletId, wallets, refreshWallets],\n );\n\n const createWalletPack = useCallback(\n async (name: string): Promise<number> => {\n const { response } = await tgClient.telegramChatWalletPackCreate({\n authData: authDataMap,\n name,\n });\n refreshWallets();\n return response.packId;\n },\n [tgClient, authDataMap, refreshWallets],\n );\n\n const addPackLabel = useCallback(\n async (params: {\n walletAddress: string;\n walletLabel: string;\n packId: number;\n }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelAdd({\n authData: authDataMap,\n ...params,\n });\n refreshWallets();\n },\n [tgClient, authDataMap, refreshWallets],\n );\n\n const modifyPackLabel = useCallback(\n async (params: {\n walletLabelOld: string;\n walletLabelNew: string;\n packId: number;\n }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelModify({\n authData: authDataMap,\n ...params,\n });\n refreshWallets();\n },\n [tgClient, authDataMap, refreshWallets],\n );\n\n const removePackLabel = useCallback(\n async (params: { walletLabel: string; packId: number }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelRemove({\n authData: authDataMap,\n ...params,\n });\n refreshWallets();\n },\n [tgClient, authDataMap, refreshWallets],\n );\n\n // ── TWAP session management (Telegram only) ─────────────────\n const createTwap = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidTwapCreate>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidTwapCreate({\n authData: authDataMap,\n walletId: selectedWalletId,\n ...params,\n });\n if (!response.session)\n throw new Error(\"TWAP creation returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? { ...w, twapSessions: [...w.twapSessions, session] }\n : w,\n ),\n );\n return session;\n },\n [tgClient, authDataMap, selectedWalletId],\n );\n\n const modifyTwap = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidTwapModify>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidTwapModify({\n authData: authDataMap,\n walletId: selectedWalletId,\n ...params,\n });\n if (!response.session)\n throw new Error(\"TWAP modify returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: w.twapSessions.map((s) =>\n s.id === session.id ? session : s,\n ),\n }\n : w,\n ),\n );\n return session;\n },\n [tgClient, authDataMap, selectedWalletId],\n );\n\n const cancelTwap = useCallback(\n async (sessionId: number) => {\n await tgClient.hyperliquidTwapCancel({\n authData: authDataMap,\n walletId: selectedWalletId,\n twapSessionId: sessionId,\n });\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: w.twapSessions.filter((s) => s.id !== sessionId),\n }\n : w,\n ),\n );\n },\n [tgClient, authDataMap, selectedWalletId],\n );\n\n // ── Scale session management (Telegram only) ───────────────\n const createScale = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidScaleCreate>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidScaleCreate({\n authData: authDataMap,\n walletId: selectedWalletId,\n ...params,\n });\n if (!response.session)\n throw new Error(\"Scale creation returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? { ...w, scaleSessions: [...w.scaleSessions, session] }\n : w,\n ),\n );\n return session;\n },\n [tgClient, authDataMap, selectedWalletId],\n );\n\n const cancelScale = useCallback(\n async (sessionId: number) => {\n await tgClient.hyperliquidScaleCancel({\n authData: authDataMap,\n walletId: selectedWalletId,\n scaleSessionId: sessionId,\n });\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n scaleSessions: w.scaleSessions.filter(\n (s) => s.id !== sessionId,\n ),\n }\n : w,\n ),\n );\n },\n [tgClient, authDataMap, selectedWalletId],\n );\n\n // ── Login modal state ────────────────────────────────────────\n const [loginModalOpen, setLoginModalOpen] = useState(false);\n const openLoginModal = useCallback(() => setLoginModalOpen(true), []);\n const closeLoginModal = useCallback(() => setLoginModalOpen(false), []);\n\n // ── Auth actions ─────────────────────────────────────────────\n const loginTelegram = useCallback((data: TelegramLoginData) => {\n setTgLoginData(data);\n localStorage.setItem(TELEGRAM_STORAGE_KEY, JSON.stringify(data));\n setEoaAddress(null);\n setAgent(null);\n setEoaError(null);\n }, []);\n\n const connectEoa = useCallback(\n (address: `0x${string}`, signer?: EoaSigner) => {\n eoaSignerRef.current = signer ?? null;\n setEoaAddress(address);\n setTgLoginData(null);\n setTgUser(null);\n setTgError(null);\n setEoaError(null);\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n\n const existing = loadAgent(address);\n if (existing && existing.validUntil > Date.now()) {\n setAgent(existing);\n } else {\n if (existing) clearStoredAgent(address);\n setAgent(null);\n }\n },\n [],\n );\n\n const approveAgentFn = useCallback(\n async (signTypedDataAsync: SignTypedDataFn, chainId: number) => {\n if (!eoaAddress) {\n throw new Error(\n \"[HypurrConnect] Cannot approve agent: no EOA wallet connected. Call connectEoa(address) first.\",\n );\n }\n\n eoaSignerRef.current = { signTypedData: signTypedDataAsync, chainId };\n\n setEoaLoading(true);\n setEoaError(null);\n try {\n const existing = loadAgent(eoaAddress);\n if (existing) {\n const isTestnet = config.isTestnet ?? false;\n const valid = await isAgentValid(existing, eoaAddress, isTestnet);\n if (valid) {\n setAgent(existing);\n return;\n }\n clearStoredAgent(eoaAddress);\n }\n\n const { privateKey, address: agentAddress } = await generateAgentKey();\n const isTestnet = config.isTestnet ?? false;\n\n const chainIdHex = `0x${chainId.toString(16)}` as `0x${string}`;\n const nonce = Date.now();\n const action = {\n type: \"approveAgent\" as const,\n signatureChainId: chainIdHex,\n hyperliquidChain: (isTestnet ? \"Testnet\" : \"Mainnet\") as\n | \"Testnet\"\n | \"Mainnet\",\n agentAddress: agentAddress.toLowerCase() as `0x${string}`,\n agentName: AGENT_NAME,\n nonce,\n };\n\n const approveAgentTypes = {\n \"HyperliquidTransaction:ApproveAgent\": [\n { name: \"hyperliquidChain\", type: \"string\" },\n { name: \"agentAddress\", type: \"address\" },\n { name: \"agentName\", type: \"string\" },\n { name: \"nonce\", type: \"uint64\" },\n ],\n };\n\n const wallet = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n signTypedData(params: any) {\n return signTypedDataAsync(params);\n },\n getAddresses: async () => [eoaAddress] as `0x${string}`[],\n getChainId: async () => chainId,\n };\n\n const signature = await signUserSignedAction({\n wallet,\n action,\n types: approveAgentTypes,\n });\n\n const apiUrl = isTestnet\n ? \"https://api.hyperliquid-testnet.xyz/exchange\"\n : \"https://api.hyperliquid.xyz/exchange\";\n\n const res = await fetch(apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ action, signature, nonce }),\n });\n\n const body = await res.json();\n if (body?.status === \"err\") {\n throw new Error(\n `approveAgent API error: ${body.response ?? JSON.stringify(body)}`,\n );\n }\n\n const remote = await fetchActiveAgent(eoaAddress, isTestnet);\n const validUntil =\n remote?.validUntil ?? Date.now() + 7 * 24 * 60 * 60 * 1000;\n\n const stored: StoredAgent = {\n privateKey,\n address: agentAddress,\n approvedAt: Date.now(),\n validUntil,\n };\n saveAgent(eoaAddress, stored);\n setAgent(stored);\n } catch (err) {\n console.error(\"[HypurrConnect] EOA agent approval failed:\", err);\n setEoaError(err instanceof Error ? err.message : String(err));\n setAgent(null);\n } finally {\n setEoaLoading(false);\n }\n },\n [eoaAddress, config.isTestnet],\n );\n\n const logout = useCallback(() => {\n setTgLoginData(null);\n setTgUser(null);\n setTgError(null);\n setEoaAddress(null);\n setAgent(null);\n setEoaError(null);\n eoaSignerRef.current = null;\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n }, []);\n\n // ── Context value ────────────────────────────────────────────\n const value = useMemo<InternalConnectState>(\n () => ({\n user,\n isLoggedIn: !!user,\n isLoading: tgLoading || eoaLoading,\n error: tgError ?? eoaError,\n authMethod,\n exchange,\n\n wallets,\n selectedWalletId,\n selectWallet,\n\n createWallet,\n deleteWallet,\n refreshWallets,\n\n packs,\n createWalletPack,\n addPackLabel,\n modifyPackLabel,\n removePackLabel,\n\n createTwap,\n modifyTwap,\n cancelTwap,\n\n createScale,\n cancelScale,\n\n loginModalOpen,\n openLoginModal,\n closeLoginModal,\n\n loginTelegram,\n connectEoa,\n approveAgent: approveAgentFn,\n logout,\n\n agent,\n agentReady,\n clearAgent: handleClearAgent,\n\n botId: config.telegram?.botId ?? \"\",\n botUsername: config.telegram?.botUsername ?? \"\",\n useWidget: config.telegram?.useWidget ?? false,\n\n authDataMap,\n telegramClient: tgClient,\n staticClient,\n }),\n [\n user,\n tgLoading,\n eoaLoading,\n tgError,\n eoaError,\n authMethod,\n exchange,\n wallets,\n selectedWalletId,\n selectWallet,\n createWallet,\n deleteWallet,\n refreshWallets,\n packs,\n createWalletPack,\n addPackLabel,\n modifyPackLabel,\n removePackLabel,\n createTwap,\n modifyTwap,\n cancelTwap,\n createScale,\n cancelScale,\n loginModalOpen,\n openLoginModal,\n closeLoginModal,\n loginTelegram,\n connectEoa,\n approveAgentFn,\n logout,\n agent,\n agentReady,\n handleClearAgent,\n config.telegram?.botId,\n config.telegram?.botUsername,\n config.telegram?.useWidget,\n authDataMap,\n tgClient,\n staticClient,\n ],\n );\n\n return (\n <HypurrConnectContext.Provider value={value}>\n {children}\n </HypurrConnectContext.Provider>\n );\n}\n","import type { StoredAgent } from \"./types\";\n\nexport const AGENT_NAME = \"hypurr-connect\";\n\nconst AGENT_STORAGE_PREFIX = \"hypurr-connect-agent\";\n\nfunction storageKey(masterAddress: string): string {\n return `${AGENT_STORAGE_PREFIX}:${masterAddress.toLowerCase()}`;\n}\n\nexport function loadAgent(masterAddress: string): StoredAgent | null {\n try {\n const raw = localStorage.getItem(storageKey(masterAddress));\n return raw ? JSON.parse(raw) : null;\n } catch {\n return null;\n }\n}\n\nexport function saveAgent(masterAddress: string, agent: StoredAgent): void {\n localStorage.setItem(storageKey(masterAddress), JSON.stringify(agent));\n}\n\nexport function clearAgent(masterAddress: string): void {\n localStorage.removeItem(storageKey(masterAddress));\n}\n\n/**\n * Generate a random 32-byte private key and derive its address using the\n * SDK's PrivateKeySigner (no viem dependency needed).\n */\nexport async function generateAgentKey(): Promise<{\n privateKey: `0x${string}`;\n address: `0x${string}`;\n}> {\n const bytes = crypto.getRandomValues(new Uint8Array(32));\n const hex = Array.from(bytes)\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n const privateKey = `0x${hex}` as `0x${string}`;\n\n const { PrivateKeySigner } = await import(\"@hfunlabs/hyperliquid/signing\");\n const signer = new PrivateKeySigner(privateKey);\n return { privateKey, address: signer.address };\n}\n\ninterface ExtraAgent {\n address: string;\n name: string;\n validUntil: number;\n}\n\n/**\n * Query the Hyperliquid info API for the named agents registered to a user.\n * Returns the matching entry for AGENT_NAME if it exists and is still valid.\n */\nexport async function fetchActiveAgent(\n userAddress: string,\n isTestnet: boolean,\n): Promise<ExtraAgent | null> {\n const url = isTestnet\n ? \"https://api.hyperliquid-testnet.xyz/info\"\n : \"https://api.hyperliquid.xyz/info\";\n\n const res = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ type: \"extraAgents\", user: userAddress }),\n });\n\n if (!res.ok) return null;\n\n const agents: unknown = await res.json();\n if (!Array.isArray(agents)) return null;\n\n const nowMs = Date.now();\n const match = (agents as ExtraAgent[]).find(\n (a) => a.name === AGENT_NAME && a.validUntil * 1000 > nowMs,\n );\n if (!match) return null;\n return { ...match, validUntil: match.validUntil * 1000 };\n}\n\n/**\n * Checks whether a stored agent is still valid: the address must appear in the\n * on-chain `extraAgents` list and not be expired.\n */\nexport async function isAgentValid(\n stored: StoredAgent,\n userAddress: string,\n isTestnet: boolean,\n): Promise<boolean> {\n if (stored.validUntil <= Date.now()) return false;\n\n const remote = await fetchActiveAgent(userAddress, isTestnet);\n if (!remote) return false;\n\n return (\n remote.address.toLowerCase() === stored.address.toLowerCase() &&\n remote.validUntil > Date.now()\n );\n}\n\nconst DEAD_AGENT_PATTERNS = [\n /agent address .+ is not valid/i,\n /unknown signer/i,\n /not authorized/i,\n /not an agent/i,\n];\n\n/**\n * Returns true if the error indicates the agent has been pruned, expired,\n * or is otherwise no longer registered on-chain.\n */\nexport function isDeadAgentError(err: unknown): boolean {\n const msg =\n err instanceof Error\n ? err.message\n : typeof err === \"object\" && err !== null && \"message\" in err\n ? String((err as { message: unknown }).message)\n : String(err);\n return DEAD_AGENT_PATTERNS.some((p) => p.test(msg));\n}\n","import { GrpcWebFetchTransport } from \"@protobuf-ts/grpcweb-transport\";\nimport { StaticClient } from \"hypurr-grpc/ts/hypurr/static/static_service.client\";\nimport { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\nimport type { HypurrConnectConfig } from \"./types\";\n\nconst GRPC_URL = \"https://grpc.hypurr.fun\";\n\nfunction createTransport(config: HypurrConnectConfig) {\n return new GrpcWebFetchTransport({\n baseUrl: GRPC_URL,\n timeout: config.grpcTimeout ?? 15_000,\n });\n}\n\nexport function createTelegramClient(\n config: HypurrConnectConfig,\n): TelegramClient {\n return new TelegramClient(createTransport(config));\n}\n\nexport function createStaticClient(config: HypurrConnectConfig): StaticClient {\n return new StaticClient(createTransport(config));\n}\n","import type { IRequestTransport } from \"@hfunlabs/hyperliquid\";\nimport type { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\n\nexport interface GrpcExchangeTransportConfig {\n isTestnet?: boolean;\n telegramClient: TelegramClient;\n authDataMap: Record<string, string>;\n walletId: number;\n onAuthError?: () => void;\n}\n\ninterface ExchangePayload {\n action: { type: string; [key: string]: unknown };\n nonce: number;\n signature?: { r: string; s: string; v: number };\n vaultAddress?: string | null;\n}\n\n/**\n * Routes exchange requests through the Hypurr gRPC backend (HyperliquidCoreAction)\n * for server-side signing. The backend handles signature generation.\n *\n * The SDK-generated nonce is forwarded; the action payload is JSON-encoded as bytes.\n * Info/explorer requests are proxied directly to the Hyperliquid HTTP API.\n */\nexport class GrpcExchangeTransport implements IRequestTransport {\n isTestnet: boolean;\n private telegramClient: TelegramClient;\n private authDataMap: Record<string, string>;\n private walletId: number;\n private infoUrl: string;\n private onAuthError?: () => void;\n\n constructor(config: GrpcExchangeTransportConfig) {\n this.isTestnet = config.isTestnet ?? false;\n this.telegramClient = config.telegramClient;\n this.authDataMap = config.authDataMap;\n this.walletId = config.walletId;\n this.onAuthError = config.onAuthError;\n this.infoUrl = this.isTestnet\n ? \"https://api.hyperliquid-testnet.xyz\"\n : \"https://api.hyperliquid.xyz\";\n }\n\n async request<T>(\n endpoint: \"info\" | \"exchange\" | \"explorer\",\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n if (endpoint === \"exchange\") {\n return this.exchangeViaGrpc<T>(payload as ExchangePayload, signal);\n }\n return this.directRequest<T>(endpoint, payload, signal);\n }\n\n private async exchangeViaGrpc<T>(\n payload: ExchangePayload,\n signal?: AbortSignal,\n ): Promise<T> {\n if (signal?.aborted) {\n throw new DOMException(\"Request aborted\", \"AbortError\");\n }\n\n const actionBytes = new TextEncoder().encode(\n JSON.stringify(payload.action),\n );\n\n console.debug(\"[GrpcExchangeTransport] sending action:\", payload.action);\n\n let response;\n try {\n ({ response } = await this.telegramClient.hyperliquidCoreAction({\n authData: this.authDataMap,\n walletId: this.walletId,\n action: actionBytes,\n nonce: payload.nonce || Date.now(),\n }));\n } catch (err) {\n if (\n this.onAuthError &&\n err instanceof Error &&\n /invalid telegram auth data/i.test(err.message)\n ) {\n this.onAuthError();\n }\n throw err;\n }\n\n if (signal?.aborted) {\n throw new DOMException(\"Request aborted\", \"AbortError\");\n }\n\n const { status, result, error } = response;\n\n console.debug(\n \"[GrpcExchangeTransport] gRPC status:\",\n status,\n \"error:\",\n error,\n );\n\n if (error) {\n throw new Error(`GrpcExchangeTransport: ${error}`);\n }\n\n // The backend's `result` contains the raw Hyperliquid API JSON response.\n // The SDK's assertSuccessResponse expects the same shape as the HTTP API,\n // e.g. { status: \"ok\", response: { type: \"order\", data: { statuses: [...] } } }\n if (result && result.length > 0) {\n const parsed = JSON.parse(new TextDecoder().decode(result));\n console.debug(\"[GrpcExchangeTransport] parsed result:\", parsed);\n return parsed as T;\n }\n\n // Fallback: return a minimal success shape if no result bytes\n return { status: \"ok\", response: status } as T;\n }\n\n private async directRequest<T>(\n endpoint: string,\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n const res = await fetch(`${this.infoUrl}/${endpoint}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(payload),\n signal,\n });\n\n if (!res.ok) {\n throw new Error(`HTTP ${res.status}: ${await res.text()}`);\n }\n\n return (await res.json()) as T;\n }\n}\n","import {\n AnimatePresence,\n motion,\n useAnimationControls,\n type PanInfo,\n} from \"framer-motion\";\nimport {\n useCallback,\n useEffect,\n useSyncExternalStore,\n type CSSProperties,\n type ReactNode,\n} from \"react\";\nimport { useHypurrConnectInternal } from \"./HypurrConnectProvider\";\nimport { MetaMaskColorIcon } from \"./icons/MetaMaskColorIcon\";\nimport { TelegramColorIcon } from \"./icons/TelegramColorIcon\";\nimport { TelegramLoginWidget } from \"./TelegramLoginWidget\";\nimport type { TelegramLoginData } from \"./types\";\n\nexport interface LoginModalProps {\n onConnectWallet: () => void;\n walletIcon?: ReactNode;\n}\n\nconst MOBILE_BREAKPOINT = 640;\n\nconst btnStyle: CSSProperties = {\n display: \"flex\",\n height: 53,\n width: \"100%\",\n alignItems: \"center\",\n gap: 12,\n overflow: \"hidden\",\n borderRadius: 6,\n background: \"rgba(255,255,255,0.05)\",\n padding: \"0 24px\",\n fontSize: 14,\n fontWeight: 600,\n letterSpacing: \"-0.01em\",\n color: \"#fff\",\n cursor: \"pointer\",\n border: \"none\",\n transition: \"background 150ms\",\n};\n\nconst btnHoverBg = { background: \"rgba(255,255,255,0.1)\" };\n\nconst backdropStyle: CSSProperties = {\n position: \"fixed\",\n inset: 0,\n zIndex: 100,\n background: \"rgba(0,0,0,0.6)\",\n backdropFilter: \"blur(2px)\",\n WebkitBackdropFilter: \"blur(2px)\",\n};\n\nconst modalWrapperStyle: CSSProperties = {\n position: \"fixed\",\n inset: 0,\n zIndex: 101,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n padding: 16,\n};\n\nconst modalBoxStyle: CSSProperties = {\n display: \"flex\",\n width: 400,\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 16,\n overflow: \"hidden\",\n borderRadius: 12,\n border: \"1px solid rgba(255,255,255,0.1)\",\n background: \"#282828\",\n padding: 24,\n};\n\nconst headingStyle: CSSProperties = {\n fontSize: 16,\n fontWeight: 700,\n letterSpacing: \"-0.025em\",\n color: \"#fff\",\n margin: 0,\n};\n\nconst dividerStyle: CSSProperties = {\n height: 1,\n width: \"100%\",\n background: \"rgba(255,255,255,0.05)\",\n};\n\nconst iconSize: CSSProperties = { width: 20, height: 20 };\n\nconst mobileQuery =\n typeof window !== \"undefined\"\n ? window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT}px)`)\n : null;\n\nfunction subscribeMobile(cb: () => void) {\n mobileQuery?.addEventListener(\"change\", cb);\n return () => mobileQuery?.removeEventListener(\"change\", cb);\n}\n\nfunction getSnapshotMobile() {\n return mobileQuery?.matches ?? false;\n}\n\nfunction useIsMobile() {\n return useSyncExternalStore(subscribeMobile, getSnapshotMobile, () => false);\n}\n\nfunction HoverButton({\n onClick,\n children,\n}: {\n onClick: () => void;\n children: ReactNode;\n}) {\n return (\n <motion.button\n type=\"button\"\n onClick={onClick}\n style={btnStyle}\n whileHover={btnHoverBg}\n >\n {children}\n </motion.button>\n );\n}\n\nexport function LoginModal({ onConnectWallet, walletIcon }: LoginModalProps) {\n const {\n loginTelegram,\n loginModalOpen,\n closeLoginModal,\n botId,\n botUsername,\n useWidget,\n } = useHypurrConnectInternal();\n\n const handleTelegramAuth = useCallback(\n (user: TelegramLoginData) => {\n loginTelegram(user);\n closeLoginModal();\n },\n [loginTelegram, closeLoginModal],\n );\n\n useEffect(() => {\n if (!loginModalOpen) return;\n function onMessage(e: MessageEvent) {\n if (e.origin !== \"https://oauth.telegram.org\") return;\n try {\n const data = typeof e.data === \"string\" ? JSON.parse(e.data) : e.data;\n if (data?.event === \"auth_result\" && data.result) {\n const r = data.result;\n handleTelegramAuth({\n id: r.id,\n first_name: r.first_name ?? \"\",\n last_name: r.last_name ?? undefined,\n username: r.username ?? undefined,\n photo_url: r.photo_url ?? undefined,\n auth_date: r.auth_date,\n hash: r.hash,\n });\n }\n } catch {\n /* ignore non-JSON */\n }\n }\n window.addEventListener(\"message\", onMessage);\n return () => window.removeEventListener(\"message\", onMessage);\n }, [loginModalOpen, handleTelegramAuth]);\n\n const openTelegramOAuth = useCallback(() => {\n const origin = encodeURIComponent(window.location.origin);\n const url = `https://oauth.telegram.org/auth?bot_id=${botId}&origin=${origin}&request_access=write`;\n const w = 550;\n const h = 470;\n const left = window.screenX + (window.outerWidth - w) / 2;\n const top = window.screenY + (window.outerHeight - h) / 2;\n window.open(\n url,\n \"telegram_auth\",\n `width=${w},height=${h},left=${left},top=${top}`,\n );\n }, [botId]);\n\n const isMobile = useIsMobile();\n\n const modalContent = (\n <>\n <div\n style={{\n display: \"flex\",\n width: \"100%\",\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 8,\n overflow: \"hidden\",\n }}\n >\n {useWidget && botUsername ? (\n <TelegramLoginWidget\n botUsername={botUsername}\n onAuth={handleTelegramAuth}\n />\n ) : (\n <HoverButton onClick={openTelegramOAuth}>\n <TelegramColorIcon style={iconSize} />\n Telegram\n </HoverButton>\n )}\n </div>\n\n <div style={dividerStyle} />\n\n <HoverButton\n onClick={() => {\n closeLoginModal();\n onConnectWallet();\n }}\n >\n {walletIcon ?? <MetaMaskColorIcon style={iconSize} />}\n Wallet\n </HoverButton>\n </>\n );\n\n return (\n <AnimatePresence>\n {loginModalOpen &&\n (isMobile ? (\n <MobileDrawer key=\"drawer\" onClose={closeLoginModal}>\n {modalContent}\n </MobileDrawer>\n ) : (\n <>\n <motion.div\n key=\"backdrop\"\n style={backdropStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.15 }}\n onClick={closeLoginModal}\n />\n <motion.div\n key=\"modal-wrapper\"\n style={modalWrapperStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.05 }}\n onClick={closeLoginModal}\n >\n <motion.div\n style={modalBoxStyle}\n initial={{ opacity: 0, scale: 0.95, y: 10 }}\n animate={{ opacity: 1, scale: 1, y: 0 }}\n exit={{ opacity: 0, scale: 0.95, y: 10 }}\n transition={{ duration: 0.2, ease: \"easeOut\" }}\n onClick={(e) => e.stopPropagation()}\n >\n <p style={headingStyle}>Connect</p>\n {modalContent}\n </motion.div>\n </motion.div>\n </>\n ))}\n </AnimatePresence>\n );\n}\n\nconst drawerSheetStyle: CSSProperties = {\n position: \"fixed\",\n left: 0,\n right: 0,\n bottom: 0,\n zIndex: 101,\n display: \"flex\",\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 16,\n borderTopLeftRadius: 12,\n borderTopRightRadius: 12,\n borderLeft: \"1px solid rgba(255,255,255,0.1)\",\n borderRight: \"1px solid rgba(255,255,255,0.1)\",\n borderTop: \"1px solid rgba(255,255,255,0.1)\",\n background: \"#282828\",\n padding: \"12px 24px max(24px, env(safe-area-inset-bottom))\",\n};\n\nconst drawerBgStyle: CSSProperties = {\n position: \"absolute\",\n left: 0,\n right: 0,\n top: 0,\n bottom: \"-100vh\",\n zIndex: -1,\n background: \"#282828\",\n borderTopLeftRadius: 12,\n borderTopRightRadius: 12,\n};\n\nconst grabHandleAreaStyle: CSSProperties = {\n width: \"100%\",\n cursor: \"grab\",\n paddingBottom: 4,\n};\n\nconst grabHandleStyle: CSSProperties = {\n margin: \"0 auto\",\n height: 4,\n width: 100,\n borderRadius: 9999,\n background: \"rgba(255,255,255,0.05)\",\n};\n\nfunction MobileDrawer({\n children,\n onClose,\n}: {\n children: ReactNode;\n onClose: () => void;\n}) {\n const controls = useAnimationControls();\n\n const handleDragEnd = useCallback(\n (_: unknown, info: PanInfo) => {\n if (info.offset.y > 100 || info.velocity.y > 500) {\n onClose();\n } else {\n controls.start({ y: 0 });\n }\n },\n [onClose, controls],\n );\n\n return (\n <>\n <motion.div\n key=\"drawer-backdrop\"\n style={backdropStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.15 }}\n onClick={onClose}\n />\n\n <motion.div\n key=\"drawer-sheet\"\n style={drawerSheetStyle}\n initial={{ y: \"100%\" }}\n animate={{ y: 0 }}\n exit={{ y: \"100%\" }}\n transition={{ type: \"tween\", duration: 0.3, ease: [0.32, 0.72, 0, 1] }}\n drag=\"y\"\n dragConstraints={{ top: 0, bottom: 0 }}\n dragElastic={{ top: 0, bottom: 0.4 }}\n onDragEnd={handleDragEnd}\n >\n <div style={drawerBgStyle} />\n\n <div style={grabHandleAreaStyle}>\n <div style={grabHandleStyle} />\n </div>\n\n <p style={headingStyle}>Connect</p>\n\n {children}\n </motion.div>\n </>\n );\n}\n","import type { CSSProperties } from \"react\";\n\nexport function MetaMaskColorIcon({ style }: { style?: CSSProperties }) {\n return (\n <svg\n width=\"24\"\n height=\"24\"\n viewBox=\"0 0 24 24\"\n style={style}\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <g clipPath=\"url(#clip0_2567_1088)\">\n <path\n d=\"M19.8188 19.418L15.9421 18.2871L13.0186 19.9994L10.9788 19.9985L8.05356 18.2871L4.17862 19.418L3 15.5193L4.17875 11.1924L3 7.5341L4.17875 3L10.2336 6.54437H13.7639L19.8188 3L20.9976 7.5341L19.8188 11.1924L20.9976 15.5193L19.8188 19.418Z\"\n fill=\"#FF5C16\"\n />\n <path\n d=\"M4.17969 3L10.2347 6.54685L9.99394 8.98101L4.17969 3ZM8.05476 15.5209L10.7189 17.5093L8.05476 18.2869V15.5209ZM10.5059 12.2335L9.99394 8.98275L6.71642 11.1934L6.71464 11.1925V11.1941L6.72479 13.4695L8.05387 12.2336L10.5059 12.2335ZM19.819 3L13.7641 6.54685L14.004 8.98101L19.819 3ZM15.9441 15.5209L13.2798 17.5093L15.9441 18.2869V15.5209ZM17.2833 11.194V11.1924L17.2825 11.1932L14.0049 8.98275L13.4929 12.2335H15.944L17.2739 13.4693L17.2833 11.194Z\"\n fill=\"#FF5C16\"\n />\n <path\n d=\"M8.05369 18.2867L4.17875 19.4177L3 15.5207H8.05369V18.2867ZM10.5049 12.2324L11.245 16.9321L10.2191 14.319L6.72296 13.4691L8.0528 12.2325L10.5049 12.2324ZM15.9438 18.2867L19.8188 19.4177L20.9976 15.5206H15.9438C15.9438 15.5207 15.9438 18.2867 15.9438 18.2867ZM13.4927 12.2324L12.7526 16.9321L13.7783 14.319L17.2748 13.4691L15.944 12.2325L13.4927 12.2324Z\"\n fill=\"#E34807\"\n />\n <path\n d=\"M3 15.5194L4.17875 11.1924H6.71358L6.72283 13.4686L10.2194 14.3185L11.2451 16.9315L10.7178 17.5069L8.05369 15.5185H3V15.5194ZM20.9976 15.5194L19.8188 11.1924H17.2839L17.2746 13.4686L13.7783 14.3185L12.7524 16.9315L13.2796 17.5069L15.9439 15.5185H20.9976V15.5194ZM13.7639 6.54443H10.2336L9.99389 8.97859L11.2453 16.9289H12.7526L14.0047 8.97859L13.7639 6.54443Z\"\n fill=\"#FF8D5D\"\n />\n <path\n d=\"M4.17875 3L3 7.5341L4.17875 11.1924H6.71358L9.99287 8.98114L4.17875 3ZM9.77231 13.1766H8.62399L7.9988 13.7771L10.2202 14.3166L9.77231 13.1757V13.1766ZM19.8188 3L20.9976 7.5341L19.8188 11.1924H17.2839L14.0047 8.98114L19.8188 3ZM14.2269 13.1766H15.3769L16.0021 13.7778L13.7782 14.3184L14.2269 13.1757V13.1766ZM13.0178 18.4484L13.2798 17.5086L12.7524 16.9332H11.244L10.7168 17.5086L10.9787 18.4484\"\n fill=\"#661800\"\n />\n <path\n d=\"M13.0173 18.4482V20.0001H10.9785V18.4482H13.0173Z\"\n fill=\"#C0C4CD\"\n />\n <path\n d=\"M8.05469 18.2854L10.9807 19.9994V18.4475L10.7187 17.5078L8.05469 18.2854ZM15.944 18.2854L13.0179 19.9994V18.4475L13.2799 17.5078L15.944 18.2854Z\"\n fill=\"#E7EBF6\"\n />\n </g>\n <defs>\n <clipPath id=\"clip0_2567_1088\">\n <rect\n width=\"18\"\n height=\"17\"\n fill=\"white\"\n transform=\"translate(3 3)\"\n />\n </clipPath>\n </defs>\n </svg>\n );\n}\n","import type { CSSProperties } from \"react\";\n\nexport function TelegramColorIcon({ style }: { style?: CSSProperties }) {\n return (\n <svg\n width=\"24\"\n height=\"24\"\n viewBox=\"0 0 24 24\"\n style={style}\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n d=\"M12 21C16.9706 21 21 16.9706 21 12C21 7.02944 16.9706 3 12 3C7.02944 3 3 7.02944 3 12C3 16.9706 7.02944 21 12 21Z\"\n fill=\"url(#paint0_linear_2571_1084)\"\n />\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M7.07426 11.905C9.69794 10.7619 11.4475 10.0083 12.3229 9.64417C14.8222 8.60458 15.3416 8.424 15.6801 8.41803C15.7546 8.41672 15.921 8.43517 16.0289 8.52267C16.1199 8.59655 16.145 8.69635 16.1569 8.7664C16.1689 8.83645 16.1839 8.99602 16.172 9.1207C16.0366 10.5438 15.4505 13.9973 15.1523 15.5912C15.0262 16.2657 14.7778 16.4918 14.5373 16.514C14.0146 16.562 13.6178 16.1686 13.1115 15.8367C12.3194 15.3175 11.8719 14.9943 11.103 14.4876C10.2145 13.902 10.7905 13.5802 11.2969 13.0542C11.4294 12.9166 13.7322 10.822 13.7768 10.632C13.7824 10.6082 13.7875 10.5196 13.7349 10.4729C13.6823 10.4261 13.6046 10.4421 13.5486 10.4548C13.4691 10.4728 12.2037 11.3092 9.75232 12.964C9.39313 13.2106 9.06779 13.3308 8.7763 13.3245C8.45496 13.3176 7.83681 13.1428 7.37729 12.9934C6.81366 12.8102 6.3657 12.7134 6.40471 12.4022C6.42503 12.2401 6.64821 12.0744 7.07426 11.905Z\"\n fill=\"white\"\n />\n <defs>\n <linearGradient\n id=\"paint0_linear_2571_1084\"\n x1=\"903\"\n y1=\"3\"\n x2=\"903\"\n y2=\"1789.65\"\n gradientUnits=\"userSpaceOnUse\"\n >\n <stop stopColor=\"#2AABEE\" />\n <stop offset=\"1\" stopColor=\"#229ED9\" />\n </linearGradient>\n </defs>\n </svg>\n );\n}\n","import { useEffect, useRef } from \"react\";\nimport type { TelegramLoginData } from \"./types\";\n\nconst WIDGET_SCRIPT_URL = \"https://telegram.org/js/telegram-widget.js?22\";\nconst CALLBACK_NAME = \"__hypurrConnectTelegramAuth\";\n\nexport interface TelegramLoginWidgetProps {\n botUsername: string;\n onAuth: (data: TelegramLoginData) => void;\n buttonSize?: \"large\" | \"medium\" | \"small\";\n cornerRadius?: number;\n showUserPhoto?: boolean;\n requestAccess?: boolean;\n}\n\nexport function TelegramLoginWidget({\n botUsername,\n onAuth,\n buttonSize = \"large\",\n cornerRadius,\n showUserPhoto = true,\n requestAccess = true,\n}: TelegramLoginWidgetProps) {\n const containerRef = useRef<HTMLDivElement>(null);\n const onAuthRef = useRef(onAuth);\n onAuthRef.current = onAuth;\n\n useEffect(() => {\n const container = containerRef.current;\n if (!container) return;\n\n (window as unknown as Record<string, unknown>)[CALLBACK_NAME] = (\n user: TelegramLoginData,\n ) => {\n onAuthRef.current(user);\n };\n\n const script = document.createElement(\"script\");\n script.src = WIDGET_SCRIPT_URL;\n script.async = true;\n script.setAttribute(\"data-telegram-login\", botUsername);\n script.setAttribute(\"data-size\", buttonSize);\n script.setAttribute(\"data-onauth\", `${CALLBACK_NAME}(user)`);\n script.setAttribute(\"data-userpic\", String(showUserPhoto));\n if (requestAccess) {\n script.setAttribute(\"data-request-access\", \"write\");\n }\n if (cornerRadius !== undefined) {\n script.setAttribute(\"data-radius\", String(cornerRadius));\n }\n\n container.innerHTML = \"\";\n container.appendChild(script);\n\n return () => {\n container.innerHTML = \"\";\n delete (window as unknown as Record<string, unknown>)[CALLBACK_NAME];\n };\n }, [botUsername, buttonSize, cornerRadius, showUserPhoto, requestAccess]);\n\n return <div ref={containerRef} />;\n}\n","import type { ExchangeClient } from \"@hfunlabs/hyperliquid\";\nimport type { StaticClient } from \"hypurr-grpc/ts/hypurr/static/static_service.client\";\nimport type {\n HyperliquidScaleCreateRequest,\n HyperliquidTwapCreateRequest,\n HyperliquidTwapModifyRequest,\n} from \"hypurr-grpc/ts/hypurr/telegram/telegram_service\";\nimport type { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\nimport type {\n HyperliquidWalletScaleSession,\n HyperliquidWalletTwapSession,\n} from \"hypurr-grpc/ts/hypurr/tools\";\nimport type { TelegramChatWalletPack } from \"hypurr-grpc/ts/hypurr/user\";\nimport type { HyperliquidWallet } from \"hypurr-grpc/ts/hypurr/wallet\";\n\n// ─── Config ──────────────────────────────────────────────────────\n\nexport interface HypurrConnectConfig {\n grpcTimeout?: number;\n isTestnet?: boolean;\n /** Polling interval in ms for TWAP/Scale session updates. Default 5000. Set 0 to disable. */\n sessionPollInterval?: number;\n telegram: {\n botUsername: string;\n botId?: string;\n useWidget: boolean;\n };\n}\n\n// ─── Telegram login ──────────────────────────────────────────────\n\nexport interface TelegramLoginData {\n id: number;\n first_name: string;\n last_name?: string;\n username?: string;\n photo_url?: string;\n auth_date: number;\n hash: string;\n}\n\n// ─── Auth ────────────────────────────────────────────────────────\n\nexport type AuthMethod = \"telegram\" | \"eoa\" | null;\n\nexport interface HypurrUser {\n address: string;\n walletId: number;\n displayName: string;\n photoUrl?: string;\n authMethod: AuthMethod;\n telegramId?: string;\n hfunScore?: number;\n reputationScore?: number;\n}\n\n// ─── Agent (EOA flow) ────────────────────────────────────────────\n\nexport interface StoredAgent {\n privateKey: `0x${string}`;\n address: `0x${string}`;\n approvedAt: number;\n /** Epoch ms from the `extraAgents` response; agent is invalid after this time. */\n validUntil: number;\n}\n\nexport type SignTypedDataFn = (params: {\n domain: Record<string, unknown>;\n types: Record<string, { name: string; type: string }[]>;\n primaryType: string;\n message: Record<string, unknown>;\n}) => Promise<`0x${string}`>;\n\n/** Wallet signer provided at EOA connect time for user-signed actions. */\nexport interface EoaSigner {\n signTypedData: SignTypedDataFn;\n chainId: number;\n}\n\n/**\n * Create an {@link EoaSigner} from any EIP-712 signing function.\n *\n * Accepts either a direct function or a `{ current }` ref object so the\n * signer always calls through to the latest function (avoids stale closures\n * with React hooks like wagmi's `useSignTypedData`).\n *\n * @example wagmi v2 — ref pattern (recommended)\n * ```ts\n * const { signTypedDataAsync } = useSignTypedData();\n * const chainId = useChainId();\n * const signerRef = useRef(signTypedDataAsync);\n * signerRef.current = signTypedDataAsync; // stays fresh every render\n *\n * // call once — the ref keeps it up to date\n * connectEoa(address, createEoaSigner(signerRef, chainId));\n * ```\n *\n * @example direct function (e.g. from viem WalletClient)\n * ```ts\n * connectEoa(address, createEoaSigner(client.signTypedData, chainId));\n * ```\n */\nexport function createEoaSigner(\n signTypedDataAsync:\n | ((args: Record<string, unknown>) => Promise<`0x${string}`>)\n | { current: (args: Record<string, unknown>) => Promise<`0x${string}`> },\n chainId: number,\n): EoaSigner {\n const resolve =\n typeof signTypedDataAsync === \"function\"\n ? signTypedDataAsync\n : (args: Record<string, unknown>) => signTypedDataAsync.current(args);\n return {\n signTypedData: (params) => resolve(params),\n chainId,\n };\n}\n\n// ─── TWAP / Scale param types ────────────────────────────────────\n// Omit authData & walletId — auto-filled from context using the selected wallet.\n\nexport type TwapCreateParams = Omit<\n HyperliquidTwapCreateRequest,\n \"authData\" | \"walletId\"\n>;\n\nexport type TwapModifyParams = Omit<\n HyperliquidTwapModifyRequest,\n \"authData\" | \"walletId\"\n>;\n\nexport type ScaleCreateParams = Omit<\n HyperliquidScaleCreateRequest,\n \"authData\" | \"walletId\"\n>;\n\n// ─── Context state ───────────────────────────────────────────────\n\nexport interface HypurrConnectState {\n // Auth\n user: HypurrUser | null;\n isLoggedIn: boolean;\n isLoading: boolean;\n error: string | null;\n authMethod: AuthMethod;\n\n // SDK ExchangeClient — handles both L1 (agent-signed) and user-signed actions.\n // Telegram: backed by GrpcExchangeTransport (HyperliquidCoreAction)\n // EOA: uses a dual wallet — agent key for L1 actions, master wallet for\n // user-signed actions (transfers, withdrawals, etc.) when a signer is provided.\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n exchange: ExchangeClient<any> | null;\n\n // Multi-wallet (Telegram only — EOA has a single wallet)\n wallets: HyperliquidWallet[];\n selectedWalletId: number;\n selectWallet: (walletId: number) => void;\n\n // Wallet management (Telegram only)\n createWallet: (name: string) => Promise<HyperliquidWallet>;\n deleteWallet: (walletId: number) => Promise<void>;\n refreshWallets: () => void;\n\n // Wallet packs & labels (Telegram only)\n packs: TelegramChatWalletPack[];\n createWalletPack: (name: string) => Promise<number>;\n addPackLabel: (params: {\n walletAddress: string;\n walletLabel: string;\n packId: number;\n }) => Promise<void>;\n modifyPackLabel: (params: {\n walletLabelOld: string;\n walletLabelNew: string;\n packId: number;\n }) => Promise<void>;\n removePackLabel: (params: {\n walletLabel: string;\n packId: number;\n }) => Promise<void>;\n\n // TWAP sessions (Telegram only)\n createTwap: (params: TwapCreateParams) => Promise<HyperliquidWalletTwapSession>;\n modifyTwap: (params: TwapModifyParams) => Promise<HyperliquidWalletTwapSession>;\n cancelTwap: (sessionId: number) => Promise<void>;\n\n // Scale sessions (Telegram only)\n createScale: (params: ScaleCreateParams) => Promise<HyperliquidWalletScaleSession>;\n cancelScale: (sessionId: number) => Promise<void>;\n\n // Login modal\n loginModalOpen: boolean;\n openLoginModal: () => void;\n closeLoginModal: () => void;\n\n // Auth actions\n connectEoa: (address: `0x${string}`, signer?: EoaSigner) => void;\n approveAgent: (\n signTypedDataAsync: SignTypedDataFn,\n chainId: number,\n ) => Promise<void>;\n logout: () => void;\n\n // EOA agent management\n agent: StoredAgent | null;\n agentReady: boolean;\n clearAgent: () => void;\n\n // Telegram config\n botId: string;\n\n // Low-level access\n authDataMap: Record<string, string>;\n telegramClient: TelegramClient;\n staticClient: StaticClient;\n}\n"],"mappings":"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","names":["PrivateKeySigner","user","isTestnet","useCallback","useEffect","jsx","jsx","jsxs","useEffect","useRef","jsx","jsx","jsxs","useCallback","useEffect"]}
|
|
1
|
+
{"version":3,"sources":["../src/HypurrConnectProvider.tsx","../src/agent.ts","../src/grpc.ts","../src/GrpcExchangeTransport.ts","../src/LoginModal.tsx","../src/icons/MetaMaskColorIcon.tsx","../src/icons/TelegramColorIcon.tsx","../src/types.ts"],"sourcesContent":["import {\n ExchangeClient,\n HttpTransport,\n type IRequestTransport,\n} from \"@hfunlabs/hyperliquid\";\nimport {\n PrivateKeySigner,\n signUserSignedAction,\n} from \"@hfunlabs/hyperliquid/signing\";\nimport type { RpcOptions } from \"@protobuf-ts/runtime-rpc\";\nimport type { TelegramUserResponse } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service\";\nimport type {\n TelegramUser as HypurrTelegramUser,\n TelegramChatWalletPack,\n} from \"hypurr-grpc/ts/hypurr/user\";\nimport type { HyperliquidWallet } from \"hypurr-grpc/ts/hypurr/wallet\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n type ReactNode,\n} from \"react\";\nimport {\n AGENT_NAME,\n clearAgent as clearStoredAgent,\n fetchActiveAgent,\n generateAgentKey,\n isAgentValid,\n isDeadAgentError,\n loadAgent,\n saveAgent,\n} from \"./agent\";\nimport { createStaticClient, createTelegramClient } from \"./grpc\";\nimport { GrpcExchangeTransport } from \"./GrpcExchangeTransport\";\nimport type {\n AuthMethod,\n EoaSigner,\n HypurrConnectConfig,\n HypurrConnectState,\n HypurrUser,\n SignTypedDataFn,\n StoredAgent,\n} from \"./types\";\n\n/** @internal context value — extends the public type with fields used only by library internals */\ninterface InternalConnectState extends HypurrConnectState {\n loginTelegram: () => void;\n botUsername: string;\n useWidget: boolean;\n}\n\nconst TELEGRAM_STORAGE_KEY = \"hypurr-connect-tg-jwt\";\nconst LEGACY_TELEGRAM_STORAGE_KEY = \"hypurr-connect-tg-user\";\nconst TELEGRAM_AUTH_STATE_KEY = \"hypurr-connect-auth-state\";\nconst TELEGRAM_AUTH_MESSAGE = \"hypurr-connect:telegram-auth\";\nconst DEFAULT_AUTH_HUB_URL = \"https://127.0.0.1:443/login\";\nconst DEFAULT_TELEGRAM_SCOPES = [\n \"telegram:user:read\",\n \"telegram:wallet:read\",\n \"telegram:wallet:write\",\n \"telegram:trade:read\",\n \"telegram:trade:write\",\n \"telegram:cabal:read\",\n \"telegram:cabal:write\",\n \"telegram:agent:write\",\n];\n\nfunction isInvalidTelegramAuthError(err: unknown): boolean {\n const msg =\n err instanceof Error\n ? err.message\n : typeof err === \"object\" && err !== null && \"message\" in err\n ? String((err as { message: unknown }).message)\n : String(err);\n return /invalid telegram auth data|invalid auth token|missing authorization token/i.test(\n msg,\n );\n}\n\nfunction currentReturnTo(): string {\n const url = new URL(window.location.href);\n for (const param of [\n \"token\",\n \"token_type\",\n \"token_source\",\n \"state\",\n \"scope\",\n ]) {\n url.searchParams.delete(param);\n }\n return url.toString();\n}\n\nfunction randomState(): string {\n const bytes = new Uint8Array(16);\n crypto.getRandomValues(bytes);\n return Array.from(bytes, (byte) => byte.toString(16).padStart(2, \"0\")).join(\n \"\",\n );\n}\n\nfunction normalizeScopes(scope?: string | string[]): string {\n if (Array.isArray(scope)) return scope.join(\" \");\n return scope?.trim() || DEFAULT_TELEGRAM_SCOPES.join(\" \");\n}\n\nfunction isTelegramAuthMessage(\n data: unknown,\n): data is {\n type: typeof TELEGRAM_AUTH_MESSAGE;\n token: string;\n state: string;\n} {\n return (\n typeof data === \"object\" &&\n data !== null &&\n \"type\" in data &&\n \"token\" in data &&\n \"state\" in data &&\n (data as { type: unknown }).type === TELEGRAM_AUTH_MESSAGE &&\n typeof (data as { token: unknown }).token === \"string\" &&\n typeof (data as { state: unknown }).state === \"string\"\n );\n}\n\nconst HypurrConnectContext = createContext<InternalConnectState | null>(null);\n\nexport function useHypurrConnect(): HypurrConnectState {\n const ctx = useContext(HypurrConnectContext);\n if (!ctx)\n throw new Error(\n \"useHypurrConnect must be used within <HypurrConnectProvider>\",\n );\n return ctx;\n}\n\n/** @internal — gives library components access to fields not on the public API */\nexport function useHypurrConnectInternal(): InternalConnectState {\n const ctx = useContext(HypurrConnectContext);\n if (!ctx)\n throw new Error(\n \"useHypurrConnectInternal must be used within <HypurrConnectProvider>\",\n );\n return ctx;\n}\n\nexport function HypurrConnectProvider({\n config,\n children,\n}: {\n config: HypurrConnectConfig;\n children: ReactNode;\n}) {\n const tgClient = useMemo(() => createTelegramClient(config), [config]);\n const staticClient = useMemo(() => createStaticClient(config), [config]);\n\n // ── Telegram auth state ──────────────────────────────────────\n const [tgAuthToken, setTgAuthToken] = useState<string | null>(() => {\n try {\n return localStorage.getItem(TELEGRAM_STORAGE_KEY);\n } catch {\n return null;\n }\n });\n const [tgUser, setTgUser] = useState<HypurrTelegramUser | null>(null);\n const [tgLoading, setTgLoading] = useState(false);\n const [tgError, setTgError] = useState<string | null>(null);\n\n const authDataMap = useMemo(() => ({}), []);\n const telegramRpcOptions = useMemo<RpcOptions | undefined>(\n () =>\n tgAuthToken\n ? { meta: { authorization: `Bearer ${tgAuthToken}` } }\n : undefined,\n [tgAuthToken],\n );\n\n const [tgUserTick, setTgUserTick] = useState(0);\n\n // Auto-disconnect when the server rejects telegram auth data.\n const onInvalidAuthRef = useRef<(() => void) | null>(null);\n onInvalidAuthRef.current = () => {\n console.warn(\n \"[HypurrConnect] Invalid Telegram auth token — disconnecting.\",\n );\n setTgAuthToken(null);\n setTgUser(null);\n setTgError(null);\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n localStorage.removeItem(LEGACY_TELEGRAM_STORAGE_KEY);\n };\n\n const acceptTelegramToken = useCallback((token: string) => {\n setTgAuthToken(token);\n setTgError(null);\n localStorage.setItem(TELEGRAM_STORAGE_KEY, token);\n localStorage.removeItem(LEGACY_TELEGRAM_STORAGE_KEY);\n }, []);\n\n useEffect(() => {\n const params = new URLSearchParams(window.location.search);\n const token = params.get(\"token\");\n if (!token) {\n localStorage.removeItem(LEGACY_TELEGRAM_STORAGE_KEY);\n return;\n }\n\n const callbackState = params.get(\"state\") ?? \"\";\n\n if (window.opener && window.opener !== window) {\n window.opener.postMessage(\n {\n type: TELEGRAM_AUTH_MESSAGE,\n token,\n state: callbackState,\n },\n window.location.origin,\n );\n window.close();\n return;\n }\n\n const expectedState = sessionStorage.getItem(TELEGRAM_AUTH_STATE_KEY);\n sessionStorage.removeItem(TELEGRAM_AUTH_STATE_KEY);\n if (!expectedState || callbackState !== expectedState) {\n setTgError(\"Invalid auth callback state.\");\n return;\n }\n\n acceptTelegramToken(token);\n\n const cleanUrl = new URL(window.location.href);\n for (const param of [\n \"token\",\n \"token_type\",\n \"token_source\",\n \"state\",\n \"scope\",\n ]) {\n cleanUrl.searchParams.delete(param);\n }\n window.history.replaceState({}, document.title, cleanUrl.toString());\n }, [acceptTelegramToken]);\n\n useEffect(() => {\n function onMessage(event: MessageEvent) {\n if (event.origin !== window.location.origin) return;\n if (!isTelegramAuthMessage(event.data)) return;\n\n const expectedState = sessionStorage.getItem(TELEGRAM_AUTH_STATE_KEY);\n sessionStorage.removeItem(TELEGRAM_AUTH_STATE_KEY);\n if (!expectedState || event.data.state !== expectedState) {\n setTgError(\"Invalid auth callback state.\");\n return;\n }\n\n acceptTelegramToken(event.data.token);\n }\n\n window.addEventListener(\"message\", onMessage);\n return () => window.removeEventListener(\"message\", onMessage);\n }, [acceptTelegramToken]);\n\n useEffect(() => {\n if (!tgAuthToken || !telegramRpcOptions) return;\n let cancelled = false;\n setTgLoading(true);\n setTgError(null);\n\n (async () => {\n try {\n const [{ response: userResp }, { response: walletsResp }] =\n await Promise.all([\n tgClient.telegramUser({ authData: {} }, telegramRpcOptions),\n tgClient.telegramUserWallets({ authData: {} }, telegramRpcOptions),\n ]);\n if (cancelled) return;\n const user = (userResp as TelegramUserResponse).user ?? null;\n if (user) {\n // TelegramUser.wallets lacks twap/scale sessions; replace with\n // the full wallets from TelegramUserWallets which populates them.\n user.wallets = walletsResp.wallets;\n }\n setTgUser(user);\n } catch (err) {\n if (cancelled) return;\n if (isInvalidTelegramAuthError(err)) {\n onInvalidAuthRef.current?.();\n return;\n }\n console.error(\"[HypurrConnect] gRPC TelegramUser failed:\", err);\n setTgError(err instanceof Error ? err.message : String(err));\n } finally {\n if (!cancelled) setTgLoading(false);\n }\n })();\n\n return () => {\n cancelled = true;\n };\n }, [tgAuthToken, telegramRpcOptions, tgClient, tgUserTick]);\n\n // ── EOA auth state ───────────────────────────────────────────\n const [eoaAddress, setEoaAddress] = useState<`0x${string}` | null>(null);\n const [agent, setAgent] = useState<StoredAgent | null>(null);\n const [eoaLoading, setEoaLoading] = useState(false);\n const [eoaError, setEoaError] = useState<string | null>(null);\n const eoaSignerRef = useRef<EoaSigner | null>(null);\n\n // ── Derived auth ─────────────────────────────────────────────\n const authMethod: AuthMethod = tgAuthToken\n ? \"telegram\"\n : eoaAddress\n ? \"eoa\"\n : null;\n\n // ── Multi-wallet state (Telegram) ─────────────────────────────\n const [wallets, setWallets] = useState<HyperliquidWallet[]>([]);\n const [selectedWalletId, setSelectedWalletId] = useState<number>(0);\n const [packs, setPacks] = useState<TelegramChatWalletPack[]>([]);\n\n const refreshWallets = useCallback(() => setTgUserTick((t) => t + 1), []);\n\n useEffect(() => {\n if (authMethod !== \"telegram\" || !tgUser) {\n setWallets([]);\n setSelectedWalletId(0);\n setPacks([]);\n return;\n }\n\n const userWallets = tgUser.wallets ?? [];\n setWallets(userWallets);\n setPacks(tgUser.packs ?? []);\n\n const defaultId = tgUser.walletId || userWallets[0]?.id || 0;\n setSelectedWalletId((prev) => {\n if (prev && userWallets.some((w) => w.id === prev)) return prev;\n return defaultId;\n });\n }, [authMethod, tgUser]);\n\n const selectedWallet = useMemo(\n () => wallets.find((w) => w.id === selectedWalletId) ?? wallets[0] ?? null,\n [wallets, selectedWalletId],\n );\n\n const selectWallet = useCallback(\n (walletId: number) => {\n if (wallets.some((w) => w.id === walletId)) {\n setSelectedWalletId(walletId);\n }\n },\n [wallets],\n );\n\n // ── Session poller (TWAP + Scale for selected wallet) ──────\n const pollInterval = config.sessionPollInterval ?? 5_000;\n\n useEffect(() => {\n if (authMethod !== \"telegram\" || !selectedWalletId || !pollInterval) return;\n\n let cancelled = false;\n\n const poll = async () => {\n try {\n const [{ response: twapResp }, { response: scaleResp }] =\n await Promise.all([\n tgClient.hyperliquidWalletTwapSessions(\n {\n authData: {},\n walletId: selectedWalletId,\n },\n telegramRpcOptions,\n ),\n tgClient.hyperliquidWalletScaleSessions(\n {\n authData: {},\n walletId: selectedWalletId,\n },\n telegramRpcOptions,\n ),\n ]);\n if (cancelled) return;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: twapResp.sessions,\n scaleSessions: scaleResp.sessions,\n }\n : w,\n ),\n );\n } catch (err) {\n if (isInvalidTelegramAuthError(err)) {\n onInvalidAuthRef.current?.();\n return;\n }\n // Silently ignore poll errors — next tick will retry.\n }\n };\n\n // Fire immediately on mount / wallet switch, then on interval.\n poll();\n const id = setInterval(poll, pollInterval);\n return () => {\n cancelled = true;\n clearInterval(id);\n };\n }, [\n authMethod,\n selectedWalletId,\n pollInterval,\n tgClient,\n telegramRpcOptions,\n ]);\n\n const user = useMemo<HypurrUser | null>(() => {\n if (tgAuthToken && authMethod === \"telegram\" && selectedWallet && tgUser) {\n return {\n address: selectedWallet.ethereumAddress,\n walletId: selectedWallet.id,\n displayName: tgUser.telegramUsername\n ? `@${tgUser.telegramUsername}`\n : `Telegram ${tgUser.telegramId}`,\n authMethod: \"telegram\",\n telegramId: String(tgUser.telegramId),\n hfunScore: tgUser?.reputation?.hfunScore,\n reputationScore: tgUser?.reputation?.reputationScore,\n };\n }\n if (eoaAddress && authMethod === \"eoa\") {\n return {\n address: eoaAddress,\n walletId: 0,\n displayName: `${eoaAddress.slice(0, 6)}...${eoaAddress.slice(-4)}`,\n authMethod: \"eoa\",\n };\n }\n return null;\n }, [tgAuthToken, selectedWallet, eoaAddress, authMethod, tgUser]);\n\n // ── Exchange client ──────────────────────────────────────────\n // Telegram: GrpcExchangeTransport → HyperliquidCoreAction (server signs)\n // EOA: dual wallet — agent key for L1 actions, master signer for user-signed\n // actions (transfers, withdrawals, etc.). The dual wallet inspects the\n // EIP-712 domain name to decide which key signs each request.\n // When a signer is available but no agent exists yet, the dual wallet\n // auto-provisions an agent on the first L1 action (triggers one extra\n // wallet popup for the approveAgent user-signed action).\n\n const onDeadAgentRef = useRef<((address: `0x${string}`) => void) | null>(\n null,\n );\n onDeadAgentRef.current = (addr: `0x${string}`) => {\n clearStoredAgent(addr);\n setAgent(null);\n setEoaError(\"Agent expired or was deregistered. Please reconnect.\");\n };\n\n // Mutable slot for the agent signer — the dual wallet reads this so it can\n // pick up a newly provisioned agent without waiting for a React re-render.\n const agentSignerRef = useRef<PrivateKeySigner | null>(\n agent ? new PrivateKeySigner(agent.privateKey) : null,\n );\n useEffect(() => {\n agentSignerRef.current = agent\n ? new PrivateKeySigner(agent.privateKey)\n : null;\n }, [agent]);\n\n // Lock to prevent concurrent auto-provisioning attempts\n const provisioningRef = useRef<Promise<PrivateKeySigner> | null>(null);\n\n const agentReady =\n authMethod === \"telegram\" || (authMethod === \"eoa\" && !!agent);\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const exchange = useMemo<ExchangeClient<any> | null>(() => {\n if (authMethod === \"telegram\" && user?.address) {\n const transport = new GrpcExchangeTransport({\n isTestnet: config.isTestnet ?? false,\n telegramClient: tgClient,\n rpcOptions: telegramRpcOptions,\n walletId: user.walletId,\n onAuthError: () => onInvalidAuthRef.current?.(),\n });\n return new ExchangeClient({\n transport,\n externalSigning: true,\n userAddress: user.address as `0x${string}`,\n });\n }\n\n if (authMethod === \"eoa\" && eoaAddress) {\n const hasSigner = !!eoaSignerRef.current;\n\n if (!agent && !hasSigner) {\n const noAgentTransport: IRequestTransport = {\n isTestnet: config.isTestnet ?? false,\n request(): Promise<never> {\n throw new Error(\n \"[HypurrConnect] No agent key approved and no wallet signer available. \" +\n \"Either call approveAgent(signTypedDataAsync) or pass a signer to \" +\n \"connectEoa(address, { signTypedData, chainId }).\",\n );\n },\n };\n return new ExchangeClient({\n transport: noAgentTransport,\n externalSigning: true,\n userAddress: eoaAddress,\n });\n }\n\n const isTestnet = config.isTestnet ?? false;\n const inner = new HttpTransport({ isTestnet });\n const deadAgentAddr = eoaAddress;\n const guardedTransport: IRequestTransport = {\n isTestnet: inner.isTestnet,\n async request<T>(\n endpoint: \"info\" | \"exchange\" | \"explorer\",\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n try {\n return await inner.request<T>(endpoint, payload, signal);\n } catch (err) {\n if (endpoint === \"exchange\" && isDeadAgentError(err)) {\n onDeadAgentRef.current?.(deadAgentAddr);\n }\n throw err;\n }\n },\n };\n\n const signerRef = eoaSignerRef;\n const agentRef = agentSignerRef;\n const provRef = provisioningRef;\n const ownerAddress = eoaAddress;\n\n /**\n * Auto-provision an agent key when one doesn't exist yet.\n *\n * Bypasses the SDK's `executeUserSignedAction` (and its per-address\n * semaphore) to avoid deadlocking when called from inside\n * `dualWallet.signTypedData`, which is already inside the SDK's\n * `executeL1Action` lock for the same address.\n */\n const ensureAgent = async (): Promise<PrivateKeySigner> => {\n const existing = agentRef.current;\n if (existing) return existing;\n\n if (provRef.current) return provRef.current;\n\n const signer = signerRef.current;\n if (!signer) {\n throw new Error(\n \"[HypurrConnect] No wallet signer available to auto-provision agent. \" +\n \"Pass a signer to connectEoa(address, { signTypedData, chainId }).\",\n );\n }\n\n provRef.current = (async () => {\n try {\n const { privateKey, address: agentAddress } =\n await generateAgentKey();\n\n const chainIdHex =\n `0x${signer.chainId.toString(16)}` as `0x${string}`;\n const nonce = Date.now();\n const action = {\n type: \"approveAgent\" as const,\n signatureChainId: chainIdHex,\n hyperliquidChain: (isTestnet ? \"Testnet\" : \"Mainnet\") as\n | \"Testnet\"\n | \"Mainnet\",\n agentAddress: agentAddress.toLowerCase() as `0x${string}`,\n agentName: AGENT_NAME,\n nonce,\n };\n\n const approveAgentTypes = {\n \"HyperliquidTransaction:ApproveAgent\": [\n { name: \"hyperliquidChain\", type: \"string\" },\n { name: \"agentAddress\", type: \"address\" },\n { name: \"agentName\", type: \"string\" },\n { name: \"nonce\", type: \"uint64\" },\n ],\n };\n\n const wallet = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n signTypedData(params: any) {\n return signer.signTypedData(params);\n },\n getAddresses: async () => [ownerAddress] as `0x${string}`[],\n getChainId: async () => signer.chainId,\n };\n\n const signature = await signUserSignedAction({\n wallet,\n action,\n types: approveAgentTypes,\n });\n\n const apiUrl = isTestnet\n ? \"https://api-ui.hyperliquid-testnet.xyz/exchange\"\n : \"https://api.hyperliquid.xyz/exchange\";\n\n const res = await fetch(apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ action, signature, nonce }),\n });\n\n const body = await res.json();\n if (body?.status === \"err\") {\n throw new Error(\n `approveAgent API error: ${body.response ?? JSON.stringify(body)}`,\n );\n }\n\n const remote = await fetchActiveAgent(ownerAddress, isTestnet);\n const validUntil =\n remote?.validUntil ?? Date.now() + 7 * 24 * 60 * 60 * 1000;\n\n const stored: StoredAgent = {\n privateKey,\n address: agentAddress,\n approvedAt: Date.now(),\n validUntil,\n };\n saveAgent(ownerAddress, stored);\n\n const newSigner = new PrivateKeySigner(privateKey);\n agentRef.current = newSigner;\n setAgent(stored);\n\n return newSigner;\n } finally {\n provRef.current = null;\n }\n })();\n\n return provRef.current;\n };\n\n // Dual wallet: routes signing based on the EIP-712 domain.\n // \"Exchange\" domain → L1 action → agent key signs (auto-provisions if needed).\n // \"HyperliquidSignTransaction\" domain → user-signed → master wallet (popup).\n const dualWallet = {\n address: ownerAddress,\n async signTypedData(params: {\n domain: {\n name?: string;\n version?: string;\n chainId?: number;\n verifyingContract?: `0x${string}`;\n salt?: `0x${string}`;\n };\n types: Record<string, { name: string; type: string }[]>;\n primaryType: string;\n message: Record<string, unknown>;\n }): Promise<`0x${string}`> {\n if (params.domain.name === \"HyperliquidSignTransaction\") {\n const signer = signerRef.current;\n if (!signer) {\n throw new Error(\n \"[HypurrConnect] No wallet signer available for user-signed actions. \" +\n \"Pass a signer to connectEoa(address, { signTypedData, chainId }).\",\n );\n }\n return signer.signTypedData(\n params as Parameters<typeof signer.signTypedData>[0],\n );\n }\n\n const agentSigner = await ensureAgent();\n return agentSigner.signTypedData(params);\n },\n };\n\n return new ExchangeClient({\n transport: guardedTransport,\n wallet: dualWallet,\n signatureChainId: () => {\n const id = signerRef.current?.chainId ?? 42161;\n return `0x${id.toString(16)}` as `0x${string}`;\n },\n });\n }\n\n return null;\n }, [\n authMethod,\n user,\n agent,\n eoaAddress,\n config.isTestnet,\n tgClient,\n telegramRpcOptions,\n ]);\n\n const handleClearAgent = useCallback(() => {\n if (eoaAddress) {\n clearStoredAgent(eoaAddress);\n setAgent(null);\n }\n }, [eoaAddress]);\n\n // ── Wallet management (Telegram only) ───────────────────────\n const createWallet = useCallback(\n async (name: string): Promise<HyperliquidWallet> => {\n const { response } = await tgClient.hyperliquidWalletCreate(\n {\n authData: {},\n name,\n },\n telegramRpcOptions,\n );\n refreshWallets();\n if (!response.wallet)\n throw new Error(\"Wallet creation returned no wallet\");\n return response.wallet;\n },\n [tgClient, telegramRpcOptions, refreshWallets],\n );\n\n const deleteWallet = useCallback(\n async (walletId: number): Promise<void> => {\n await tgClient.hyperliquidWalletDelete(\n {\n authData: {},\n walletId,\n },\n telegramRpcOptions,\n );\n if (walletId === selectedWalletId) {\n const remaining = wallets.filter((w) => w.id !== walletId);\n setSelectedWalletId(remaining[0]?.id ?? 0);\n }\n refreshWallets();\n },\n [tgClient, telegramRpcOptions, selectedWalletId, wallets, refreshWallets],\n );\n\n const createWalletPack = useCallback(\n async (name: string): Promise<number> => {\n const { response } = await tgClient.telegramChatWalletPackCreate(\n {\n authData: {},\n name,\n },\n telegramRpcOptions,\n );\n refreshWallets();\n return response.packId;\n },\n [tgClient, telegramRpcOptions, refreshWallets],\n );\n\n const addPackLabel = useCallback(\n async (params: {\n walletAddress: string;\n walletLabel: string;\n packId: number;\n }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelAdd(\n {\n authData: {},\n ...params,\n },\n telegramRpcOptions,\n );\n refreshWallets();\n },\n [tgClient, telegramRpcOptions, refreshWallets],\n );\n\n const modifyPackLabel = useCallback(\n async (params: {\n walletLabelOld: string;\n walletLabelNew: string;\n packId: number;\n }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelModify(\n {\n authData: {},\n ...params,\n },\n telegramRpcOptions,\n );\n refreshWallets();\n },\n [tgClient, telegramRpcOptions, refreshWallets],\n );\n\n const removePackLabel = useCallback(\n async (params: { walletLabel: string; packId: number }): Promise<void> => {\n await tgClient.telegramChatWalletPackLabelRemove(\n {\n authData: {},\n ...params,\n },\n telegramRpcOptions,\n );\n refreshWallets();\n },\n [tgClient, telegramRpcOptions, refreshWallets],\n );\n\n // ── TWAP session management (Telegram only) ─────────────────\n const createTwap = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidTwapCreate>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidTwapCreate(\n {\n authData: {},\n walletId: selectedWalletId,\n ...params,\n },\n telegramRpcOptions,\n );\n if (!response.session)\n throw new Error(\"TWAP creation returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? { ...w, twapSessions: [...w.twapSessions, session] }\n : w,\n ),\n );\n return session;\n },\n [tgClient, telegramRpcOptions, selectedWalletId],\n );\n\n const modifyTwap = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidTwapModify>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidTwapModify(\n {\n authData: {},\n walletId: selectedWalletId,\n ...params,\n },\n telegramRpcOptions,\n );\n if (!response.session) throw new Error(\"TWAP modify returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: w.twapSessions.map((s) =>\n s.id === session.id ? session : s,\n ),\n }\n : w,\n ),\n );\n return session;\n },\n [tgClient, telegramRpcOptions, selectedWalletId],\n );\n\n const cancelTwap = useCallback(\n async (sessionId: number) => {\n await tgClient.hyperliquidTwapCancel(\n {\n authData: {},\n walletId: selectedWalletId,\n twapSessionId: sessionId,\n },\n telegramRpcOptions,\n );\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n twapSessions: w.twapSessions.filter((s) => s.id !== sessionId),\n }\n : w,\n ),\n );\n },\n [tgClient, telegramRpcOptions, selectedWalletId],\n );\n\n // ── Scale session management (Telegram only) ───────────────\n const createScale = useCallback(\n async (\n params: Omit<\n Parameters<typeof tgClient.hyperliquidScaleCreate>[0],\n \"authData\" | \"walletId\"\n >,\n ) => {\n const { response } = await tgClient.hyperliquidScaleCreate(\n {\n authData: {},\n walletId: selectedWalletId,\n ...params,\n },\n telegramRpcOptions,\n );\n if (!response.session)\n throw new Error(\"Scale creation returned no session\");\n const session = response.session;\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? { ...w, scaleSessions: [...w.scaleSessions, session] }\n : w,\n ),\n );\n return session;\n },\n [tgClient, telegramRpcOptions, selectedWalletId],\n );\n\n const cancelScale = useCallback(\n async (sessionId: number) => {\n await tgClient.hyperliquidScaleCancel(\n {\n authData: {},\n walletId: selectedWalletId,\n scaleSessionId: sessionId,\n },\n telegramRpcOptions,\n );\n setWallets((prev) =>\n prev.map((w) =>\n w.id === selectedWalletId\n ? {\n ...w,\n scaleSessions: w.scaleSessions.filter(\n (s) => s.id !== sessionId,\n ),\n }\n : w,\n ),\n );\n },\n [tgClient, telegramRpcOptions, selectedWalletId],\n );\n\n // ── Login modal state ────────────────────────────────────────\n const [loginModalOpen, setLoginModalOpen] = useState(false);\n const openLoginModal = useCallback(() => setLoginModalOpen(true), []);\n const closeLoginModal = useCallback(() => setLoginModalOpen(false), []);\n\n // ── Auth actions ─────────────────────────────────────────────\n const loginTelegram = useCallback(() => {\n const state = randomState();\n sessionStorage.setItem(TELEGRAM_AUTH_STATE_KEY, state);\n\n const configuredReturnTo = config.telegram.returnTo;\n const returnTo =\n typeof configuredReturnTo === \"function\"\n ? configuredReturnTo()\n : configuredReturnTo || currentReturnTo();\n\n const authUrl = new URL(config.telegram.authHubUrl || DEFAULT_AUTH_HUB_URL);\n authUrl.searchParams.set(\"return_to\", returnTo);\n authUrl.searchParams.set(\"state\", state);\n authUrl.searchParams.set(\"scope\", normalizeScopes(config.telegram.scope));\n\n const width = 520;\n const height = 720;\n const left = window.screenX + Math.max(0, (window.outerWidth - width) / 2);\n const top = window.screenY + Math.max(0, (window.outerHeight - height) / 2);\n const popup = window.open(\n authUrl.toString(),\n \"hypurr_telegram_auth\",\n [\n `width=${width}`,\n `height=${height}`,\n `left=${Math.round(left)}`,\n `top=${Math.round(top)}`,\n \"resizable=yes\",\n \"scrollbars=yes\",\n ].join(\",\"),\n );\n\n if (popup) {\n popup.focus();\n return;\n }\n\n window.location.assign(authUrl.toString());\n }, [\n config.telegram.authHubUrl,\n config.telegram.returnTo,\n config.telegram.scope,\n ]);\n\n const connectEoa = useCallback(\n (address: `0x${string}`, signer?: EoaSigner) => {\n eoaSignerRef.current = signer ?? null;\n setEoaAddress(address);\n setTgAuthToken(null);\n setTgUser(null);\n setTgError(null);\n setEoaError(null);\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n localStorage.removeItem(LEGACY_TELEGRAM_STORAGE_KEY);\n\n const existing = loadAgent(address);\n if (existing && existing.validUntil > Date.now()) {\n setAgent(existing);\n } else {\n if (existing) clearStoredAgent(address);\n setAgent(null);\n }\n },\n [],\n );\n\n const approveAgentFn = useCallback(\n async (signTypedDataAsync: SignTypedDataFn, chainId: number) => {\n if (!eoaAddress) {\n throw new Error(\n \"[HypurrConnect] Cannot approve agent: no EOA wallet connected. Call connectEoa(address) first.\",\n );\n }\n\n eoaSignerRef.current = { signTypedData: signTypedDataAsync, chainId };\n\n setEoaLoading(true);\n setEoaError(null);\n try {\n const existing = loadAgent(eoaAddress);\n if (existing) {\n const isTestnet = config.isTestnet ?? false;\n const valid = await isAgentValid(existing, eoaAddress, isTestnet);\n if (valid) {\n setAgent(existing);\n return;\n }\n clearStoredAgent(eoaAddress);\n }\n\n const { privateKey, address: agentAddress } = await generateAgentKey();\n const isTestnet = config.isTestnet ?? false;\n\n const chainIdHex = `0x${chainId.toString(16)}` as `0x${string}`;\n const nonce = Date.now();\n const action = {\n type: \"approveAgent\" as const,\n signatureChainId: chainIdHex,\n hyperliquidChain: (isTestnet ? \"Testnet\" : \"Mainnet\") as\n | \"Testnet\"\n | \"Mainnet\",\n agentAddress: agentAddress.toLowerCase() as `0x${string}`,\n agentName: AGENT_NAME,\n nonce,\n };\n\n const approveAgentTypes = {\n \"HyperliquidTransaction:ApproveAgent\": [\n { name: \"hyperliquidChain\", type: \"string\" },\n { name: \"agentAddress\", type: \"address\" },\n { name: \"agentName\", type: \"string\" },\n { name: \"nonce\", type: \"uint64\" },\n ],\n };\n\n const wallet = {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n signTypedData(params: any) {\n return signTypedDataAsync(params);\n },\n getAddresses: async () => [eoaAddress] as `0x${string}`[],\n getChainId: async () => chainId,\n };\n\n const signature = await signUserSignedAction({\n wallet,\n action,\n types: approveAgentTypes,\n });\n\n const apiUrl = isTestnet\n ? \"https://api.hyperliquid-testnet.xyz/exchange\"\n : \"https://api.hyperliquid.xyz/exchange\";\n\n const res = await fetch(apiUrl, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ action, signature, nonce }),\n });\n\n const body = await res.json();\n if (body?.status === \"err\") {\n throw new Error(\n `approveAgent API error: ${body.response ?? JSON.stringify(body)}`,\n );\n }\n\n const remote = await fetchActiveAgent(eoaAddress, isTestnet);\n const validUntil =\n remote?.validUntil ?? Date.now() + 7 * 24 * 60 * 60 * 1000;\n\n const stored: StoredAgent = {\n privateKey,\n address: agentAddress,\n approvedAt: Date.now(),\n validUntil,\n };\n saveAgent(eoaAddress, stored);\n setAgent(stored);\n } catch (err) {\n console.error(\"[HypurrConnect] EOA agent approval failed:\", err);\n setEoaError(err instanceof Error ? err.message : String(err));\n setAgent(null);\n } finally {\n setEoaLoading(false);\n }\n },\n [eoaAddress, config.isTestnet],\n );\n\n const logout = useCallback(() => {\n setTgUser(null);\n setTgError(null);\n setTgAuthToken(null);\n setEoaAddress(null);\n setAgent(null);\n setEoaError(null);\n eoaSignerRef.current = null;\n localStorage.removeItem(TELEGRAM_STORAGE_KEY);\n localStorage.removeItem(LEGACY_TELEGRAM_STORAGE_KEY);\n }, []);\n\n // ── Context value ────────────────────────────────────────────\n const value = useMemo<InternalConnectState>(\n () => ({\n user,\n isLoggedIn: !!user,\n isLoading: tgLoading || eoaLoading,\n error: tgError ?? eoaError,\n authMethod,\n exchange,\n\n wallets,\n selectedWalletId,\n selectWallet,\n\n createWallet,\n deleteWallet,\n refreshWallets,\n\n packs,\n createWalletPack,\n addPackLabel,\n modifyPackLabel,\n removePackLabel,\n\n createTwap,\n modifyTwap,\n cancelTwap,\n\n createScale,\n cancelScale,\n\n loginModalOpen,\n openLoginModal,\n closeLoginModal,\n\n loginTelegram,\n connectEoa,\n approveAgent: approveAgentFn,\n logout,\n\n agent,\n agentReady,\n clearAgent: handleClearAgent,\n\n botId: config.telegram?.botId ?? \"\",\n botUsername: config.telegram?.botUsername ?? \"\",\n useWidget: config.telegram?.useWidget ?? false,\n\n authDataMap,\n authToken: tgAuthToken,\n telegramRpcOptions,\n telegramClient: tgClient,\n staticClient,\n }),\n [\n user,\n tgLoading,\n eoaLoading,\n tgError,\n eoaError,\n authMethod,\n exchange,\n wallets,\n selectedWalletId,\n selectWallet,\n createWallet,\n deleteWallet,\n refreshWallets,\n packs,\n createWalletPack,\n addPackLabel,\n modifyPackLabel,\n removePackLabel,\n createTwap,\n modifyTwap,\n cancelTwap,\n createScale,\n cancelScale,\n loginModalOpen,\n openLoginModal,\n closeLoginModal,\n loginTelegram,\n connectEoa,\n approveAgentFn,\n logout,\n agent,\n agentReady,\n handleClearAgent,\n config.telegram?.botId,\n config.telegram?.botUsername,\n config.telegram?.useWidget,\n authDataMap,\n tgAuthToken,\n telegramRpcOptions,\n tgClient,\n staticClient,\n ],\n );\n\n return (\n <HypurrConnectContext.Provider value={value}>\n {children}\n </HypurrConnectContext.Provider>\n );\n}\n","import type { StoredAgent } from \"./types\";\n\nexport const AGENT_NAME = \"hypurr-connect\";\n\nconst AGENT_STORAGE_PREFIX = \"hypurr-connect-agent\";\n\nfunction storageKey(masterAddress: string): string {\n return `${AGENT_STORAGE_PREFIX}:${masterAddress.toLowerCase()}`;\n}\n\nexport function loadAgent(masterAddress: string): StoredAgent | null {\n try {\n const raw = localStorage.getItem(storageKey(masterAddress));\n return raw ? JSON.parse(raw) : null;\n } catch {\n return null;\n }\n}\n\nexport function saveAgent(masterAddress: string, agent: StoredAgent): void {\n localStorage.setItem(storageKey(masterAddress), JSON.stringify(agent));\n}\n\nexport function clearAgent(masterAddress: string): void {\n localStorage.removeItem(storageKey(masterAddress));\n}\n\n/**\n * Generate a random 32-byte private key and derive its address using the\n * SDK's PrivateKeySigner (no viem dependency needed).\n */\nexport async function generateAgentKey(): Promise<{\n privateKey: `0x${string}`;\n address: `0x${string}`;\n}> {\n const bytes = crypto.getRandomValues(new Uint8Array(32));\n const hex = Array.from(bytes)\n .map((b) => b.toString(16).padStart(2, \"0\"))\n .join(\"\");\n const privateKey = `0x${hex}` as `0x${string}`;\n\n const { PrivateKeySigner } = await import(\"@hfunlabs/hyperliquid/signing\");\n const signer = new PrivateKeySigner(privateKey);\n return { privateKey, address: signer.address };\n}\n\ninterface ExtraAgent {\n address: string;\n name: string;\n validUntil: number;\n}\n\n/**\n * Query the Hyperliquid info API for the named agents registered to a user.\n * Returns the matching entry for AGENT_NAME if it exists and is still valid.\n */\nexport async function fetchActiveAgent(\n userAddress: string,\n isTestnet: boolean,\n): Promise<ExtraAgent | null> {\n const url = isTestnet\n ? \"https://api.hyperliquid-testnet.xyz/info\"\n : \"https://api.hyperliquid.xyz/info\";\n\n const res = await fetch(url, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ type: \"extraAgents\", user: userAddress }),\n });\n\n if (!res.ok) return null;\n\n const agents: unknown = await res.json();\n if (!Array.isArray(agents)) return null;\n\n const nowMs = Date.now();\n const match = (agents as ExtraAgent[]).find(\n (a) => a.name === AGENT_NAME && a.validUntil * 1000 > nowMs,\n );\n if (!match) return null;\n return { ...match, validUntil: match.validUntil * 1000 };\n}\n\n/**\n * Checks whether a stored agent is still valid: the address must appear in the\n * on-chain `extraAgents` list and not be expired.\n */\nexport async function isAgentValid(\n stored: StoredAgent,\n userAddress: string,\n isTestnet: boolean,\n): Promise<boolean> {\n if (stored.validUntil <= Date.now()) return false;\n\n const remote = await fetchActiveAgent(userAddress, isTestnet);\n if (!remote) return false;\n\n return (\n remote.address.toLowerCase() === stored.address.toLowerCase() &&\n remote.validUntil > Date.now()\n );\n}\n\nconst DEAD_AGENT_PATTERNS = [\n /agent address .+ is not valid/i,\n /unknown signer/i,\n /not authorized/i,\n /not an agent/i,\n];\n\n/**\n * Returns true if the error indicates the agent has been pruned, expired,\n * or is otherwise no longer registered on-chain.\n */\nexport function isDeadAgentError(err: unknown): boolean {\n const msg =\n err instanceof Error\n ? err.message\n : typeof err === \"object\" && err !== null && \"message\" in err\n ? String((err as { message: unknown }).message)\n : String(err);\n return DEAD_AGENT_PATTERNS.some((p) => p.test(msg));\n}\n","import { GrpcWebFetchTransport } from \"@protobuf-ts/grpcweb-transport\";\nimport { StaticClient } from \"hypurr-grpc/ts/hypurr/static/static_service.client\";\nimport { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\nimport type { HypurrConnectConfig } from \"./types\";\n\nconst DEFAULT_GRPC_URL = \"https://grpc.hypurr.fun\";\n\nfunction createTransport(config: HypurrConnectConfig) {\n return new GrpcWebFetchTransport({\n baseUrl: config.grpcUrl ?? DEFAULT_GRPC_URL,\n timeout: config.grpcTimeout ?? 15_000,\n });\n}\n\nexport function createTelegramClient(\n config: HypurrConnectConfig,\n): TelegramClient {\n return new TelegramClient(createTransport(config));\n}\n\nexport function createStaticClient(config: HypurrConnectConfig): StaticClient {\n return new StaticClient(createTransport(config));\n}\n","import type { IRequestTransport } from \"@hfunlabs/hyperliquid\";\nimport type { RpcOptions } from \"@protobuf-ts/runtime-rpc\";\nimport type { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\n\nexport interface GrpcExchangeTransportConfig {\n isTestnet?: boolean;\n telegramClient: TelegramClient;\n rpcOptions?: RpcOptions;\n walletId: number;\n onAuthError?: () => void;\n}\n\ninterface ExchangePayload {\n action: { type: string; [key: string]: unknown };\n nonce: number;\n signature?: { r: string; s: string; v: number };\n vaultAddress?: string | null;\n}\n\n/**\n * Routes exchange requests through the Hypurr gRPC backend (HyperliquidCoreAction)\n * for server-side signing. The backend handles signature generation.\n *\n * The SDK-generated nonce is forwarded; the action payload is JSON-encoded as bytes.\n * Info/explorer requests are proxied directly to the Hyperliquid HTTP API.\n */\nexport class GrpcExchangeTransport implements IRequestTransport {\n isTestnet: boolean;\n private telegramClient: TelegramClient;\n private rpcOptions?: RpcOptions;\n private walletId: number;\n private infoUrl: string;\n private onAuthError?: () => void;\n\n constructor(config: GrpcExchangeTransportConfig) {\n this.isTestnet = config.isTestnet ?? false;\n this.telegramClient = config.telegramClient;\n this.rpcOptions = config.rpcOptions;\n this.walletId = config.walletId;\n this.onAuthError = config.onAuthError;\n this.infoUrl = this.isTestnet\n ? \"https://api.hyperliquid-testnet.xyz\"\n : \"https://api.hyperliquid.xyz\";\n }\n\n async request<T>(\n endpoint: \"info\" | \"exchange\" | \"explorer\",\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n if (endpoint === \"exchange\") {\n return this.exchangeViaGrpc<T>(payload as ExchangePayload, signal);\n }\n return this.directRequest<T>(endpoint, payload, signal);\n }\n\n private async exchangeViaGrpc<T>(\n payload: ExchangePayload,\n signal?: AbortSignal,\n ): Promise<T> {\n if (signal?.aborted) {\n throw new DOMException(\"Request aborted\", \"AbortError\");\n }\n\n const actionBytes = new TextEncoder().encode(\n JSON.stringify(payload.action),\n );\n\n console.debug(\"[GrpcExchangeTransport] sending action:\", payload.action);\n\n let response;\n try {\n ({ response } = await this.telegramClient.hyperliquidCoreAction(\n {\n authData: {},\n walletId: this.walletId,\n action: actionBytes,\n nonce: payload.nonce || Date.now(),\n },\n this.rpcOptions,\n ));\n } catch (err) {\n if (\n this.onAuthError &&\n err instanceof Error &&\n /invalid telegram auth data|invalid auth token|missing authorization token/i.test(\n err.message,\n )\n ) {\n this.onAuthError();\n }\n throw err;\n }\n\n if (signal?.aborted) {\n throw new DOMException(\"Request aborted\", \"AbortError\");\n }\n\n const { status, result, error } = response;\n\n console.debug(\n \"[GrpcExchangeTransport] gRPC status:\",\n status,\n \"error:\",\n error,\n );\n\n if (error) {\n throw new Error(`GrpcExchangeTransport: ${error}`);\n }\n\n // The backend's `result` contains the raw Hyperliquid API JSON response.\n // The SDK's assertSuccessResponse expects the same shape as the HTTP API,\n // e.g. { status: \"ok\", response: { type: \"order\", data: { statuses: [...] } } }\n if (result && result.length > 0) {\n const parsed = JSON.parse(new TextDecoder().decode(result));\n console.debug(\"[GrpcExchangeTransport] parsed result:\", parsed);\n return parsed as T;\n }\n\n // Fallback: return a minimal success shape if no result bytes\n return { status: \"ok\", response: status } as T;\n }\n\n private async directRequest<T>(\n endpoint: string,\n payload: unknown,\n signal?: AbortSignal,\n ): Promise<T> {\n const res = await fetch(`${this.infoUrl}/${endpoint}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(payload),\n signal,\n });\n\n if (!res.ok) {\n throw new Error(`HTTP ${res.status}: ${await res.text()}`);\n }\n\n return (await res.json()) as T;\n }\n}\n","import {\n AnimatePresence,\n motion,\n useAnimationControls,\n type PanInfo,\n} from \"framer-motion\";\nimport {\n useCallback,\n useSyncExternalStore,\n type CSSProperties,\n type ReactNode,\n} from \"react\";\nimport { useHypurrConnectInternal } from \"./HypurrConnectProvider\";\nimport { MetaMaskColorIcon } from \"./icons/MetaMaskColorIcon\";\nimport { TelegramColorIcon } from \"./icons/TelegramColorIcon\";\n\nexport interface LoginModalProps {\n onConnectWallet: () => void;\n walletIcon?: ReactNode;\n}\n\nconst MOBILE_BREAKPOINT = 640;\n\nconst btnStyle: CSSProperties = {\n display: \"flex\",\n height: 53,\n width: \"100%\",\n alignItems: \"center\",\n gap: 12,\n overflow: \"hidden\",\n borderRadius: 6,\n background: \"rgba(255,255,255,0.05)\",\n padding: \"0 24px\",\n fontSize: 14,\n fontWeight: 600,\n letterSpacing: \"-0.01em\",\n color: \"#fff\",\n cursor: \"pointer\",\n border: \"none\",\n transition: \"background 150ms\",\n};\n\nconst btnHoverBg = { background: \"rgba(255,255,255,0.1)\" };\n\nconst backdropStyle: CSSProperties = {\n position: \"fixed\",\n inset: 0,\n zIndex: 100,\n background: \"rgba(0,0,0,0.6)\",\n backdropFilter: \"blur(2px)\",\n WebkitBackdropFilter: \"blur(2px)\",\n};\n\nconst modalWrapperStyle: CSSProperties = {\n position: \"fixed\",\n inset: 0,\n zIndex: 101,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n padding: 16,\n};\n\nconst modalBoxStyle: CSSProperties = {\n display: \"flex\",\n width: 400,\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 16,\n overflow: \"hidden\",\n borderRadius: 12,\n border: \"1px solid rgba(255,255,255,0.1)\",\n background: \"#282828\",\n padding: 24,\n};\n\nconst headingStyle: CSSProperties = {\n fontSize: 16,\n fontWeight: 700,\n letterSpacing: \"-0.025em\",\n color: \"#fff\",\n margin: 0,\n};\n\nconst dividerStyle: CSSProperties = {\n height: 1,\n width: \"100%\",\n background: \"rgba(255,255,255,0.05)\",\n};\n\nconst iconSize: CSSProperties = { width: 20, height: 20 };\n\nconst mobileQuery =\n typeof window !== \"undefined\"\n ? window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT}px)`)\n : null;\n\nfunction subscribeMobile(cb: () => void) {\n mobileQuery?.addEventListener(\"change\", cb);\n return () => mobileQuery?.removeEventListener(\"change\", cb);\n}\n\nfunction getSnapshotMobile() {\n return mobileQuery?.matches ?? false;\n}\n\nfunction useIsMobile() {\n return useSyncExternalStore(subscribeMobile, getSnapshotMobile, () => false);\n}\n\nfunction HoverButton({\n onClick,\n children,\n}: {\n onClick: () => void;\n children: ReactNode;\n}) {\n return (\n <motion.button\n type=\"button\"\n onClick={onClick}\n style={btnStyle}\n whileHover={btnHoverBg}\n >\n {children}\n </motion.button>\n );\n}\n\nexport function LoginModal({ onConnectWallet, walletIcon }: LoginModalProps) {\n const { loginTelegram, loginModalOpen, closeLoginModal } =\n useHypurrConnectInternal();\n\n const handleTelegramAuth = useCallback(() => {\n closeLoginModal();\n loginTelegram();\n }, [loginTelegram, closeLoginModal]);\n\n const isMobile = useIsMobile();\n\n const modalContent = (\n <>\n <div\n style={{\n display: \"flex\",\n width: \"100%\",\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 8,\n overflow: \"hidden\",\n }}\n >\n <HoverButton onClick={handleTelegramAuth}>\n <TelegramColorIcon style={iconSize} />\n Telegram\n </HoverButton>\n </div>\n\n <div style={dividerStyle} />\n\n <HoverButton\n onClick={() => {\n closeLoginModal();\n onConnectWallet();\n }}\n >\n {walletIcon ?? <MetaMaskColorIcon style={iconSize} />}\n Wallet\n </HoverButton>\n </>\n );\n\n return (\n <AnimatePresence>\n {loginModalOpen &&\n (isMobile ? (\n <MobileDrawer key=\"drawer\" onClose={closeLoginModal}>\n {modalContent}\n </MobileDrawer>\n ) : (\n <>\n <motion.div\n key=\"backdrop\"\n style={backdropStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.15 }}\n onClick={closeLoginModal}\n />\n <motion.div\n key=\"modal-wrapper\"\n style={modalWrapperStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.05 }}\n onClick={closeLoginModal}\n >\n <motion.div\n style={modalBoxStyle}\n initial={{ scale: 0.96, opacity: 0, y: 8 }}\n animate={{ scale: 1, opacity: 1, y: 0 }}\n exit={{ scale: 0.96, opacity: 0, y: 8 }}\n transition={{ duration: 0.12 }}\n onClick={(e) => e.stopPropagation()}\n >\n <h2 style={headingStyle}>Connect</h2>\n {modalContent}\n </motion.div>\n </motion.div>\n </>\n ))}\n </AnimatePresence>\n );\n}\n\nconst drawerSheetStyle: CSSProperties = {\n position: \"fixed\",\n left: 0,\n right: 0,\n bottom: 0,\n zIndex: 101,\n display: \"flex\",\n flexDirection: \"column\",\n alignItems: \"center\",\n gap: 16,\n borderTopLeftRadius: 12,\n borderTopRightRadius: 12,\n borderLeft: \"1px solid rgba(255,255,255,0.1)\",\n borderRight: \"1px solid rgba(255,255,255,0.1)\",\n borderTop: \"1px solid rgba(255,255,255,0.1)\",\n background: \"#282828\",\n padding: \"12px 24px max(24px, env(safe-area-inset-bottom))\",\n};\n\nconst drawerBgStyle: CSSProperties = {\n position: \"absolute\",\n left: 0,\n right: 0,\n top: 0,\n bottom: \"-100vh\",\n zIndex: -1,\n background: \"#282828\",\n borderTopLeftRadius: 12,\n borderTopRightRadius: 12,\n};\n\nconst grabHandleAreaStyle: CSSProperties = {\n width: \"100%\",\n cursor: \"grab\",\n paddingBottom: 4,\n};\n\nconst grabHandleStyle: CSSProperties = {\n margin: \"0 auto\",\n height: 4,\n width: 100,\n borderRadius: 9999,\n background: \"rgba(255,255,255,0.05)\",\n};\n\nfunction MobileDrawer({\n children,\n onClose,\n}: {\n children: ReactNode;\n onClose: () => void;\n}) {\n const controls = useAnimationControls();\n\n const handleDragEnd = useCallback(\n (_: unknown, info: PanInfo) => {\n if (info.offset.y > 100 || info.velocity.y > 500) {\n onClose();\n } else {\n controls.start({ y: 0 });\n }\n },\n [onClose, controls],\n );\n\n return (\n <>\n <motion.div\n key=\"drawer-backdrop\"\n style={backdropStyle}\n initial={{ opacity: 0 }}\n animate={{ opacity: 1 }}\n exit={{ opacity: 0 }}\n transition={{ duration: 0.15 }}\n onClick={onClose}\n />\n\n <motion.div\n key=\"drawer-sheet\"\n style={drawerSheetStyle}\n initial={{ y: \"100%\" }}\n animate={{ y: 0 }}\n exit={{ y: \"100%\" }}\n transition={{ type: \"tween\", duration: 0.3, ease: [0.32, 0.72, 0, 1] }}\n drag=\"y\"\n dragConstraints={{ top: 0, bottom: 0 }}\n dragElastic={{ top: 0, bottom: 0.4 }}\n onDragEnd={handleDragEnd}\n >\n <div style={drawerBgStyle} />\n\n <div style={grabHandleAreaStyle}>\n <div style={grabHandleStyle} />\n </div>\n\n <p style={headingStyle}>Connect</p>\n\n {children}\n </motion.div>\n </>\n );\n}\n","import type { CSSProperties } from \"react\";\n\nexport function MetaMaskColorIcon({ style }: { style?: CSSProperties }) {\n return (\n <svg\n width=\"24\"\n height=\"24\"\n viewBox=\"0 0 24 24\"\n style={style}\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <g clipPath=\"url(#clip0_2567_1088)\">\n <path\n d=\"M19.8188 19.418L15.9421 18.2871L13.0186 19.9994L10.9788 19.9985L8.05356 18.2871L4.17862 19.418L3 15.5193L4.17875 11.1924L3 7.5341L4.17875 3L10.2336 6.54437H13.7639L19.8188 3L20.9976 7.5341L19.8188 11.1924L20.9976 15.5193L19.8188 19.418Z\"\n fill=\"#FF5C16\"\n />\n <path\n d=\"M4.17969 3L10.2347 6.54685L9.99394 8.98101L4.17969 3ZM8.05476 15.5209L10.7189 17.5093L8.05476 18.2869V15.5209ZM10.5059 12.2335L9.99394 8.98275L6.71642 11.1934L6.71464 11.1925V11.1941L6.72479 13.4695L8.05387 12.2336L10.5059 12.2335ZM19.819 3L13.7641 6.54685L14.004 8.98101L19.819 3ZM15.9441 15.5209L13.2798 17.5093L15.9441 18.2869V15.5209ZM17.2833 11.194V11.1924L17.2825 11.1932L14.0049 8.98275L13.4929 12.2335H15.944L17.2739 13.4693L17.2833 11.194Z\"\n fill=\"#FF5C16\"\n />\n <path\n d=\"M8.05369 18.2867L4.17875 19.4177L3 15.5207H8.05369V18.2867ZM10.5049 12.2324L11.245 16.9321L10.2191 14.319L6.72296 13.4691L8.0528 12.2325L10.5049 12.2324ZM15.9438 18.2867L19.8188 19.4177L20.9976 15.5206H15.9438C15.9438 15.5207 15.9438 18.2867 15.9438 18.2867ZM13.4927 12.2324L12.7526 16.9321L13.7783 14.319L17.2748 13.4691L15.944 12.2325L13.4927 12.2324Z\"\n fill=\"#E34807\"\n />\n <path\n d=\"M3 15.5194L4.17875 11.1924H6.71358L6.72283 13.4686L10.2194 14.3185L11.2451 16.9315L10.7178 17.5069L8.05369 15.5185H3V15.5194ZM20.9976 15.5194L19.8188 11.1924H17.2839L17.2746 13.4686L13.7783 14.3185L12.7524 16.9315L13.2796 17.5069L15.9439 15.5185H20.9976V15.5194ZM13.7639 6.54443H10.2336L9.99389 8.97859L11.2453 16.9289H12.7526L14.0047 8.97859L13.7639 6.54443Z\"\n fill=\"#FF8D5D\"\n />\n <path\n d=\"M4.17875 3L3 7.5341L4.17875 11.1924H6.71358L9.99287 8.98114L4.17875 3ZM9.77231 13.1766H8.62399L7.9988 13.7771L10.2202 14.3166L9.77231 13.1757V13.1766ZM19.8188 3L20.9976 7.5341L19.8188 11.1924H17.2839L14.0047 8.98114L19.8188 3ZM14.2269 13.1766H15.3769L16.0021 13.7778L13.7782 14.3184L14.2269 13.1757V13.1766ZM13.0178 18.4484L13.2798 17.5086L12.7524 16.9332H11.244L10.7168 17.5086L10.9787 18.4484\"\n fill=\"#661800\"\n />\n <path\n d=\"M13.0173 18.4482V20.0001H10.9785V18.4482H13.0173Z\"\n fill=\"#C0C4CD\"\n />\n <path\n d=\"M8.05469 18.2854L10.9807 19.9994V18.4475L10.7187 17.5078L8.05469 18.2854ZM15.944 18.2854L13.0179 19.9994V18.4475L13.2799 17.5078L15.944 18.2854Z\"\n fill=\"#E7EBF6\"\n />\n </g>\n <defs>\n <clipPath id=\"clip0_2567_1088\">\n <rect\n width=\"18\"\n height=\"17\"\n fill=\"white\"\n transform=\"translate(3 3)\"\n />\n </clipPath>\n </defs>\n </svg>\n );\n}\n","import type { CSSProperties } from \"react\";\n\nexport function TelegramColorIcon({ style }: { style?: CSSProperties }) {\n return (\n <svg\n width=\"24\"\n height=\"24\"\n viewBox=\"0 0 24 24\"\n style={style}\n fill=\"none\"\n xmlns=\"http://www.w3.org/2000/svg\"\n >\n <path\n d=\"M12 21C16.9706 21 21 16.9706 21 12C21 7.02944 16.9706 3 12 3C7.02944 3 3 7.02944 3 12C3 16.9706 7.02944 21 12 21Z\"\n fill=\"url(#paint0_linear_2571_1084)\"\n />\n <path\n fillRule=\"evenodd\"\n clipRule=\"evenodd\"\n d=\"M7.07426 11.905C9.69794 10.7619 11.4475 10.0083 12.3229 9.64417C14.8222 8.60458 15.3416 8.424 15.6801 8.41803C15.7546 8.41672 15.921 8.43517 16.0289 8.52267C16.1199 8.59655 16.145 8.69635 16.1569 8.7664C16.1689 8.83645 16.1839 8.99602 16.172 9.1207C16.0366 10.5438 15.4505 13.9973 15.1523 15.5912C15.0262 16.2657 14.7778 16.4918 14.5373 16.514C14.0146 16.562 13.6178 16.1686 13.1115 15.8367C12.3194 15.3175 11.8719 14.9943 11.103 14.4876C10.2145 13.902 10.7905 13.5802 11.2969 13.0542C11.4294 12.9166 13.7322 10.822 13.7768 10.632C13.7824 10.6082 13.7875 10.5196 13.7349 10.4729C13.6823 10.4261 13.6046 10.4421 13.5486 10.4548C13.4691 10.4728 12.2037 11.3092 9.75232 12.964C9.39313 13.2106 9.06779 13.3308 8.7763 13.3245C8.45496 13.3176 7.83681 13.1428 7.37729 12.9934C6.81366 12.8102 6.3657 12.7134 6.40471 12.4022C6.42503 12.2401 6.64821 12.0744 7.07426 11.905Z\"\n fill=\"white\"\n />\n <defs>\n <linearGradient\n id=\"paint0_linear_2571_1084\"\n x1=\"903\"\n y1=\"3\"\n x2=\"903\"\n y2=\"1789.65\"\n gradientUnits=\"userSpaceOnUse\"\n >\n <stop stopColor=\"#2AABEE\" />\n <stop offset=\"1\" stopColor=\"#229ED9\" />\n </linearGradient>\n </defs>\n </svg>\n );\n}\n","import type { ExchangeClient } from \"@hfunlabs/hyperliquid\";\nimport type { RpcOptions } from \"@protobuf-ts/runtime-rpc\";\nimport type { StaticClient } from \"hypurr-grpc/ts/hypurr/static/static_service.client\";\nimport type {\n HyperliquidScaleCreateRequest,\n HyperliquidTwapCreateRequest,\n HyperliquidTwapModifyRequest,\n} from \"hypurr-grpc/ts/hypurr/telegram/telegram_service\";\nimport type { TelegramClient } from \"hypurr-grpc/ts/hypurr/telegram/telegram_service.client\";\nimport type {\n HyperliquidWalletScaleSession,\n HyperliquidWalletTwapSession,\n} from \"hypurr-grpc/ts/hypurr/tools\";\nimport type { TelegramChatWalletPack } from \"hypurr-grpc/ts/hypurr/user\";\nimport type { HyperliquidWallet } from \"hypurr-grpc/ts/hypurr/wallet\";\n\n// ─── Config ──────────────────────────────────────────────────────\n\nexport interface HypurrConnectConfig {\n /** gRPC-web base URL. Defaults to https://grpc.hypurr.fun. */\n grpcUrl?: string;\n grpcTimeout?: number;\n isTestnet?: boolean;\n /** Polling interval in ms for TWAP/Scale session updates. Default 5000. Set 0 to disable. */\n sessionPollInterval?: number;\n telegram: {\n /** Deprecated for the hub flow; retained for older consumers. */\n botUsername?: string;\n /** Deprecated for the hub flow; retained for older consumers. */\n botId?: string;\n /** Deprecated: Telegram login is now handled by the auth hub. */\n useWidget?: boolean;\n /** Auth hub login URL. Defaults to https://auth.hypurr.fun/login. */\n authHubUrl?: string;\n /** Optional callback URL. Defaults to the current page without auth query params. */\n returnTo?: string | (() => string);\n /** Requested hub scopes. Defaults to the scopes required by this SDK. */\n scope?: string | string[];\n };\n}\n\n// ─── Telegram login ──────────────────────────────────────────────\n\n/** @deprecated Telegram login is handled by the auth hub; raw Telegram data is no longer used by the provider. */\nexport interface TelegramLoginData {\n id: number;\n first_name: string;\n last_name?: string;\n username?: string;\n photo_url?: string;\n auth_date: number;\n hash: string;\n}\n\n// ─── Auth ────────────────────────────────────────────────────────\n\nexport type AuthMethod = \"telegram\" | \"eoa\" | null;\n\nexport interface HypurrUser {\n address: string;\n walletId: number;\n displayName: string;\n photoUrl?: string;\n authMethod: AuthMethod;\n telegramId?: string;\n hfunScore?: number;\n reputationScore?: number;\n}\n\n// ─── Agent (EOA flow) ────────────────────────────────────────────\n\nexport interface StoredAgent {\n privateKey: `0x${string}`;\n address: `0x${string}`;\n approvedAt: number;\n /** Epoch ms from the `extraAgents` response; agent is invalid after this time. */\n validUntil: number;\n}\n\nexport type SignTypedDataFn = (params: {\n domain: Record<string, unknown>;\n types: Record<string, { name: string; type: string }[]>;\n primaryType: string;\n message: Record<string, unknown>;\n}) => Promise<`0x${string}`>;\n\n/** Wallet signer provided at EOA connect time for user-signed actions. */\nexport interface EoaSigner {\n signTypedData: SignTypedDataFn;\n chainId: number;\n}\n\n/**\n * Create an {@link EoaSigner} from any EIP-712 signing function.\n *\n * Accepts either a direct function or a `{ current }` ref object so the\n * signer always calls through to the latest function (avoids stale closures\n * with React hooks like wagmi's `useSignTypedData`).\n *\n * @example wagmi v2 — ref pattern (recommended)\n * ```ts\n * const { signTypedDataAsync } = useSignTypedData();\n * const chainId = useChainId();\n * const signerRef = useRef(signTypedDataAsync);\n * signerRef.current = signTypedDataAsync; // stays fresh every render\n *\n * // call once — the ref keeps it up to date\n * connectEoa(address, createEoaSigner(signerRef, chainId));\n * ```\n *\n * @example direct function (e.g. from viem WalletClient)\n * ```ts\n * connectEoa(address, createEoaSigner(client.signTypedData, chainId));\n * ```\n */\nexport function createEoaSigner(\n signTypedDataAsync:\n | ((args: Record<string, unknown>) => Promise<`0x${string}`>)\n | { current: (args: Record<string, unknown>) => Promise<`0x${string}`> },\n chainId: number,\n): EoaSigner {\n const resolve =\n typeof signTypedDataAsync === \"function\"\n ? signTypedDataAsync\n : (args: Record<string, unknown>) => signTypedDataAsync.current(args);\n return {\n signTypedData: (params) => resolve(params),\n chainId,\n };\n}\n\n// ─── TWAP / Scale param types ────────────────────────────────────\n// Omit authData & walletId — auto-filled from context using the selected wallet.\n\nexport type TwapCreateParams = Omit<\n HyperliquidTwapCreateRequest,\n \"authData\" | \"walletId\"\n>;\n\nexport type TwapModifyParams = Omit<\n HyperliquidTwapModifyRequest,\n \"authData\" | \"walletId\"\n>;\n\nexport type ScaleCreateParams = Omit<\n HyperliquidScaleCreateRequest,\n \"authData\" | \"walletId\"\n>;\n\n// ─── Context state ───────────────────────────────────────────────\n\nexport interface HypurrConnectState {\n // Auth\n user: HypurrUser | null;\n isLoggedIn: boolean;\n isLoading: boolean;\n error: string | null;\n authMethod: AuthMethod;\n\n // SDK ExchangeClient — handles both L1 (agent-signed) and user-signed actions.\n // Telegram: backed by GrpcExchangeTransport (HyperliquidCoreAction)\n // EOA: uses a dual wallet — agent key for L1 actions, master wallet for\n // user-signed actions (transfers, withdrawals, etc.) when a signer is provided.\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n exchange: ExchangeClient<any> | null;\n\n // Multi-wallet (Telegram only — EOA has a single wallet)\n wallets: HyperliquidWallet[];\n selectedWalletId: number;\n selectWallet: (walletId: number) => void;\n\n // Wallet management (Telegram only)\n createWallet: (name: string) => Promise<HyperliquidWallet>;\n deleteWallet: (walletId: number) => Promise<void>;\n refreshWallets: () => void;\n\n // Wallet packs & labels (Telegram only)\n packs: TelegramChatWalletPack[];\n createWalletPack: (name: string) => Promise<number>;\n addPackLabel: (params: {\n walletAddress: string;\n walletLabel: string;\n packId: number;\n }) => Promise<void>;\n modifyPackLabel: (params: {\n walletLabelOld: string;\n walletLabelNew: string;\n packId: number;\n }) => Promise<void>;\n removePackLabel: (params: {\n walletLabel: string;\n packId: number;\n }) => Promise<void>;\n\n // TWAP sessions (Telegram only)\n createTwap: (\n params: TwapCreateParams,\n ) => Promise<HyperliquidWalletTwapSession>;\n modifyTwap: (\n params: TwapModifyParams,\n ) => Promise<HyperliquidWalletTwapSession>;\n cancelTwap: (sessionId: number) => Promise<void>;\n\n // Scale sessions (Telegram only)\n createScale: (\n params: ScaleCreateParams,\n ) => Promise<HyperliquidWalletScaleSession>;\n cancelScale: (sessionId: number) => Promise<void>;\n\n // Login modal\n loginModalOpen: boolean;\n openLoginModal: () => void;\n closeLoginModal: () => void;\n\n // Auth actions\n connectEoa: (address: `0x${string}`, signer?: EoaSigner) => void;\n approveAgent: (\n signTypedDataAsync: SignTypedDataFn,\n chainId: number,\n ) => Promise<void>;\n logout: () => void;\n\n // EOA agent management\n agent: StoredAgent | null;\n agentReady: boolean;\n clearAgent: () => void;\n\n // Telegram config\n botId: string;\n\n // Low-level access\n /** Deprecated: JWT auth leaves authData empty; use `telegramRpcOptions` for low-level calls. */\n authDataMap: Record<string, string>;\n authToken: string | null;\n telegramRpcOptions?: RpcOptions;\n telegramClient: TelegramClient;\n staticClient: StaticClient;\n}\n"],"mappings":";AAAA;AAAA,EACE;AAAA,EACA;AAAA,OAEK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AAQP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OAEK;;;ACvBA,IAAM,aAAa;AAE1B,IAAM,uBAAuB;AAE7B,SAAS,WAAW,eAA+B;AACjD,SAAO,GAAG,oBAAoB,IAAI,cAAc,YAAY,CAAC;AAC/D;AAEO,SAAS,UAAU,eAA2C;AACnE,MAAI;AACF,UAAM,MAAM,aAAa,QAAQ,WAAW,aAAa,CAAC;AAC1D,WAAO,MAAM,KAAK,MAAM,GAAG,IAAI;AAAA,EACjC,QAAQ;AACN,WAAO;AAAA,EACT;AACF;AAEO,SAAS,UAAU,eAAuB,OAA0B;AACzE,eAAa,QAAQ,WAAW,aAAa,GAAG,KAAK,UAAU,KAAK,CAAC;AACvE;AAEO,SAAS,WAAW,eAA6B;AACtD,eAAa,WAAW,WAAW,aAAa,CAAC;AACnD;AAMA,eAAsB,mBAGnB;AACD,QAAM,QAAQ,OAAO,gBAAgB,IAAI,WAAW,EAAE,CAAC;AACvD,QAAM,MAAM,MAAM,KAAK,KAAK,EACzB,IAAI,CAAC,MAAM,EAAE,SAAS,EAAE,EAAE,SAAS,GAAG,GAAG,CAAC,EAC1C,KAAK,EAAE;AACV,QAAM,aAAa,KAAK,GAAG;AAE3B,QAAM,EAAE,kBAAAA,kBAAiB,IAAI,MAAM,OAAO,+BAA+B;AACzE,QAAM,SAAS,IAAIA,kBAAiB,UAAU;AAC9C,SAAO,EAAE,YAAY,SAAS,OAAO,QAAQ;AAC/C;AAYA,eAAsB,iBACpB,aACA,WAC4B;AAC5B,QAAM,MAAM,YACR,6CACA;AAEJ,QAAM,MAAM,MAAM,MAAM,KAAK;AAAA,IAC3B,QAAQ;AAAA,IACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,IAC9C,MAAM,KAAK,UAAU,EAAE,MAAM,eAAe,MAAM,YAAY,CAAC;AAAA,EACjE,CAAC;AAED,MAAI,CAAC,IAAI,GAAI,QAAO;AAEpB,QAAM,SAAkB,MAAM,IAAI,KAAK;AACvC,MAAI,CAAC,MAAM,QAAQ,MAAM,EAAG,QAAO;AAEnC,QAAM,QAAQ,KAAK,IAAI;AACvB,QAAM,QAAS,OAAwB;AAAA,IACrC,CAAC,MAAM,EAAE,SAAS,cAAc,EAAE,aAAa,MAAO;AAAA,EACxD;AACA,MAAI,CAAC,MAAO,QAAO;AACnB,SAAO,EAAE,GAAG,OAAO,YAAY,MAAM,aAAa,IAAK;AACzD;AAMA,eAAsB,aACpB,QACA,aACA,WACkB;AAClB,MAAI,OAAO,cAAc,KAAK,IAAI,EAAG,QAAO;AAE5C,QAAM,SAAS,MAAM,iBAAiB,aAAa,SAAS;AAC5D,MAAI,CAAC,OAAQ,QAAO;AAEpB,SACE,OAAO,QAAQ,YAAY,MAAM,OAAO,QAAQ,YAAY,KAC5D,OAAO,aAAa,KAAK,IAAI;AAEjC;AAEA,IAAM,sBAAsB;AAAA,EAC1B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;AAMO,SAAS,iBAAiB,KAAuB;AACtD,QAAM,MACJ,eAAe,QACX,IAAI,UACJ,OAAO,QAAQ,YAAY,QAAQ,QAAQ,aAAa,MACtD,OAAQ,IAA6B,OAAO,IAC5C,OAAO,GAAG;AAClB,SAAO,oBAAoB,KAAK,CAAC,MAAM,EAAE,KAAK,GAAG,CAAC;AACpD;;;AC1HA,SAAS,6BAA6B;AACtC,SAAS,oBAAoB;AAC7B,SAAS,sBAAsB;AAG/B,IAAM,mBAAmB;AAEzB,SAAS,gBAAgB,QAA6B;AACpD,SAAO,IAAI,sBAAsB;AAAA,IAC/B,SAAS,OAAO,WAAW;AAAA,IAC3B,SAAS,OAAO,eAAe;AAAA,EACjC,CAAC;AACH;AAEO,SAAS,qBACd,QACgB;AAChB,SAAO,IAAI,eAAe,gBAAgB,MAAM,CAAC;AACnD;AAEO,SAAS,mBAAmB,QAA2C;AAC5E,SAAO,IAAI,aAAa,gBAAgB,MAAM,CAAC;AACjD;;;ACIO,IAAM,wBAAN,MAAyD;AAAA,EAC9D;AAAA,EACQ;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EAER,YAAY,QAAqC;AAC/C,SAAK,YAAY,OAAO,aAAa;AACrC,SAAK,iBAAiB,OAAO;AAC7B,SAAK,aAAa,OAAO;AACzB,SAAK,WAAW,OAAO;AACvB,SAAK,cAAc,OAAO;AAC1B,SAAK,UAAU,KAAK,YAChB,wCACA;AAAA,EACN;AAAA,EAEA,MAAM,QACJ,UACA,SACA,QACY;AACZ,QAAI,aAAa,YAAY;AAC3B,aAAO,KAAK,gBAAmB,SAA4B,MAAM;AAAA,IACnE;AACA,WAAO,KAAK,cAAiB,UAAU,SAAS,MAAM;AAAA,EACxD;AAAA,EAEA,MAAc,gBACZ,SACA,QACY;AACZ,QAAI,QAAQ,SAAS;AACnB,YAAM,IAAI,aAAa,mBAAmB,YAAY;AAAA,IACxD;AAEA,UAAM,cAAc,IAAI,YAAY,EAAE;AAAA,MACpC,KAAK,UAAU,QAAQ,MAAM;AAAA,IAC/B;AAEA,YAAQ,MAAM,2CAA2C,QAAQ,MAAM;AAEvE,QAAI;AACJ,QAAI;AACF,OAAC,EAAE,SAAS,IAAI,MAAM,KAAK,eAAe;AAAA,QACxC;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU,KAAK;AAAA,UACf,QAAQ;AAAA,UACR,OAAO,QAAQ,SAAS,KAAK,IAAI;AAAA,QACnC;AAAA,QACA,KAAK;AAAA,MACP;AAAA,IACF,SAAS,KAAK;AACZ,UACE,KAAK,eACL,eAAe,SACf,6EAA6E;AAAA,QAC3E,IAAI;AAAA,MACN,GACA;AACA,aAAK,YAAY;AAAA,MACnB;AACA,YAAM;AAAA,IACR;AAEA,QAAI,QAAQ,SAAS;AACnB,YAAM,IAAI,aAAa,mBAAmB,YAAY;AAAA,IACxD;AAEA,UAAM,EAAE,QAAQ,QAAQ,MAAM,IAAI;AAElC,YAAQ;AAAA,MACN;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,QAAI,OAAO;AACT,YAAM,IAAI,MAAM,0BAA0B,KAAK,EAAE;AAAA,IACnD;AAKA,QAAI,UAAU,OAAO,SAAS,GAAG;AAC/B,YAAM,SAAS,KAAK,MAAM,IAAI,YAAY,EAAE,OAAO,MAAM,CAAC;AAC1D,cAAQ,MAAM,0CAA0C,MAAM;AAC9D,aAAO;AAAA,IACT;AAGA,WAAO,EAAE,QAAQ,MAAM,UAAU,OAAO;AAAA,EAC1C;AAAA,EAEA,MAAc,cACZ,UACA,SACA,QACY;AACZ,UAAM,MAAM,MAAM,MAAM,GAAG,KAAK,OAAO,IAAI,QAAQ,IAAI;AAAA,MACrD,QAAQ;AAAA,MACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,MAC9C,MAAM,KAAK,UAAU,OAAO;AAAA,MAC5B;AAAA,IACF,CAAC;AAED,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,IAAI,MAAM,QAAQ,IAAI,MAAM,KAAK,MAAM,IAAI,KAAK,CAAC,EAAE;AAAA,IAC3D;AAEA,WAAQ,MAAM,IAAI,KAAK;AAAA,EACzB;AACF;;;AHslCI;AA7qCJ,IAAM,uBAAuB;AAC7B,IAAM,8BAA8B;AACpC,IAAM,0BAA0B;AAChC,IAAM,wBAAwB;AAC9B,IAAM,uBAAuB;AAC7B,IAAM,0BAA0B;AAAA,EAC9B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;AAEA,SAAS,2BAA2B,KAAuB;AACzD,QAAM,MACJ,eAAe,QACX,IAAI,UACJ,OAAO,QAAQ,YAAY,QAAQ,QAAQ,aAAa,MACtD,OAAQ,IAA6B,OAAO,IAC5C,OAAO,GAAG;AAClB,SAAO,6EAA6E;AAAA,IAClF;AAAA,EACF;AACF;AAEA,SAAS,kBAA0B;AACjC,QAAM,MAAM,IAAI,IAAI,OAAO,SAAS,IAAI;AACxC,aAAW,SAAS;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,GAAG;AACD,QAAI,aAAa,OAAO,KAAK;AAAA,EAC/B;AACA,SAAO,IAAI,SAAS;AACtB;AAEA,SAAS,cAAsB;AAC7B,QAAM,QAAQ,IAAI,WAAW,EAAE;AAC/B,SAAO,gBAAgB,KAAK;AAC5B,SAAO,MAAM,KAAK,OAAO,CAAC,SAAS,KAAK,SAAS,EAAE,EAAE,SAAS,GAAG,GAAG,CAAC,EAAE;AAAA,IACrE;AAAA,EACF;AACF;AAEA,SAAS,gBAAgB,OAAmC;AAC1D,MAAI,MAAM,QAAQ,KAAK,EAAG,QAAO,MAAM,KAAK,GAAG;AAC/C,SAAO,OAAO,KAAK,KAAK,wBAAwB,KAAK,GAAG;AAC1D;AAEA,SAAS,sBACP,MAKA;AACA,SACE,OAAO,SAAS,YAChB,SAAS,QACT,UAAU,QACV,WAAW,QACX,WAAW,QACV,KAA2B,SAAS,yBACrC,OAAQ,KAA4B,UAAU,YAC9C,OAAQ,KAA4B,UAAU;AAElD;AAEA,IAAM,uBAAuB,cAA2C,IAAI;AAErE,SAAS,mBAAuC;AACrD,QAAM,MAAM,WAAW,oBAAoB;AAC3C,MAAI,CAAC;AACH,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AACF,SAAO;AACT;AAGO,SAAS,2BAAiD;AAC/D,QAAM,MAAM,WAAW,oBAAoB;AAC3C,MAAI,CAAC;AACH,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AACF,SAAO;AACT;AAEO,SAAS,sBAAsB;AAAA,EACpC;AAAA,EACA;AACF,GAGG;AACD,QAAM,WAAW,QAAQ,MAAM,qBAAqB,MAAM,GAAG,CAAC,MAAM,CAAC;AACrE,QAAM,eAAe,QAAQ,MAAM,mBAAmB,MAAM,GAAG,CAAC,MAAM,CAAC;AAGvE,QAAM,CAAC,aAAa,cAAc,IAAI,SAAwB,MAAM;AAClE,QAAI;AACF,aAAO,aAAa,QAAQ,oBAAoB;AAAA,IAClD,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF,CAAC;AACD,QAAM,CAAC,QAAQ,SAAS,IAAI,SAAoC,IAAI;AACpE,QAAM,CAAC,WAAW,YAAY,IAAI,SAAS,KAAK;AAChD,QAAM,CAAC,SAAS,UAAU,IAAI,SAAwB,IAAI;AAE1D,QAAM,cAAc,QAAQ,OAAO,CAAC,IAAI,CAAC,CAAC;AAC1C,QAAM,qBAAqB;AAAA,IACzB,MACE,cACI,EAAE,MAAM,EAAE,eAAe,UAAU,WAAW,GAAG,EAAE,IACnD;AAAA,IACN,CAAC,WAAW;AAAA,EACd;AAEA,QAAM,CAAC,YAAY,aAAa,IAAI,SAAS,CAAC;AAG9C,QAAM,mBAAmB,OAA4B,IAAI;AACzD,mBAAiB,UAAU,MAAM;AAC/B,YAAQ;AAAA,MACN;AAAA,IACF;AACA,mBAAe,IAAI;AACnB,cAAU,IAAI;AACd,eAAW,IAAI;AACf,iBAAa,WAAW,oBAAoB;AAC5C,iBAAa,WAAW,2BAA2B;AAAA,EACrD;AAEA,QAAM,sBAAsB,YAAY,CAAC,UAAkB;AACzD,mBAAe,KAAK;AACpB,eAAW,IAAI;AACf,iBAAa,QAAQ,sBAAsB,KAAK;AAChD,iBAAa,WAAW,2BAA2B;AAAA,EACrD,GAAG,CAAC,CAAC;AAEL,YAAU,MAAM;AACd,UAAM,SAAS,IAAI,gBAAgB,OAAO,SAAS,MAAM;AACzD,UAAM,QAAQ,OAAO,IAAI,OAAO;AAChC,QAAI,CAAC,OAAO;AACV,mBAAa,WAAW,2BAA2B;AACnD;AAAA,IACF;AAEA,UAAM,gBAAgB,OAAO,IAAI,OAAO,KAAK;AAE7C,QAAI,OAAO,UAAU,OAAO,WAAW,QAAQ;AAC7C,aAAO,OAAO;AAAA,QACZ;AAAA,UACE,MAAM;AAAA,UACN;AAAA,UACA,OAAO;AAAA,QACT;AAAA,QACA,OAAO,SAAS;AAAA,MAClB;AACA,aAAO,MAAM;AACb;AAAA,IACF;AAEA,UAAM,gBAAgB,eAAe,QAAQ,uBAAuB;AACpE,mBAAe,WAAW,uBAAuB;AACjD,QAAI,CAAC,iBAAiB,kBAAkB,eAAe;AACrD,iBAAW,8BAA8B;AACzC;AAAA,IACF;AAEA,wBAAoB,KAAK;AAEzB,UAAM,WAAW,IAAI,IAAI,OAAO,SAAS,IAAI;AAC7C,eAAW,SAAS;AAAA,MAClB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF,GAAG;AACD,eAAS,aAAa,OAAO,KAAK;AAAA,IACpC;AACA,WAAO,QAAQ,aAAa,CAAC,GAAG,SAAS,OAAO,SAAS,SAAS,CAAC;AAAA,EACrE,GAAG,CAAC,mBAAmB,CAAC;AAExB,YAAU,MAAM;AACd,aAAS,UAAU,OAAqB;AACtC,UAAI,MAAM,WAAW,OAAO,SAAS,OAAQ;AAC7C,UAAI,CAAC,sBAAsB,MAAM,IAAI,EAAG;AAExC,YAAM,gBAAgB,eAAe,QAAQ,uBAAuB;AACpE,qBAAe,WAAW,uBAAuB;AACjD,UAAI,CAAC,iBAAiB,MAAM,KAAK,UAAU,eAAe;AACxD,mBAAW,8BAA8B;AACzC;AAAA,MACF;AAEA,0BAAoB,MAAM,KAAK,KAAK;AAAA,IACtC;AAEA,WAAO,iBAAiB,WAAW,SAAS;AAC5C,WAAO,MAAM,OAAO,oBAAoB,WAAW,SAAS;AAAA,EAC9D,GAAG,CAAC,mBAAmB,CAAC;AAExB,YAAU,MAAM;AACd,QAAI,CAAC,eAAe,CAAC,mBAAoB;AACzC,QAAI,YAAY;AAChB,iBAAa,IAAI;AACjB,eAAW,IAAI;AAEf,KAAC,YAAY;AACX,UAAI;AACF,cAAM,CAAC,EAAE,UAAU,SAAS,GAAG,EAAE,UAAU,YAAY,CAAC,IACtD,MAAM,QAAQ,IAAI;AAAA,UAChB,SAAS,aAAa,EAAE,UAAU,CAAC,EAAE,GAAG,kBAAkB;AAAA,UAC1D,SAAS,oBAAoB,EAAE,UAAU,CAAC,EAAE,GAAG,kBAAkB;AAAA,QACnE,CAAC;AACH,YAAI,UAAW;AACf,cAAMC,QAAQ,SAAkC,QAAQ;AACxD,YAAIA,OAAM;AAGR,UAAAA,MAAK,UAAU,YAAY;AAAA,QAC7B;AACA,kBAAUA,KAAI;AAAA,MAChB,SAAS,KAAK;AACZ,YAAI,UAAW;AACf,YAAI,2BAA2B,GAAG,GAAG;AACnC,2BAAiB,UAAU;AAC3B;AAAA,QACF;AACA,gBAAQ,MAAM,6CAA6C,GAAG;AAC9D,mBAAW,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC;AAAA,MAC7D,UAAE;AACA,YAAI,CAAC,UAAW,cAAa,KAAK;AAAA,MACpC;AAAA,IACF,GAAG;AAEH,WAAO,MAAM;AACX,kBAAY;AAAA,IACd;AAAA,EACF,GAAG,CAAC,aAAa,oBAAoB,UAAU,UAAU,CAAC;AAG1D,QAAM,CAAC,YAAY,aAAa,IAAI,SAA+B,IAAI;AACvE,QAAM,CAAC,OAAO,QAAQ,IAAI,SAA6B,IAAI;AAC3D,QAAM,CAAC,YAAY,aAAa,IAAI,SAAS,KAAK;AAClD,QAAM,CAAC,UAAU,WAAW,IAAI,SAAwB,IAAI;AAC5D,QAAM,eAAe,OAAyB,IAAI;AAGlD,QAAM,aAAyB,cAC3B,aACA,aACE,QACA;AAGN,QAAM,CAAC,SAAS,UAAU,IAAI,SAA8B,CAAC,CAAC;AAC9D,QAAM,CAAC,kBAAkB,mBAAmB,IAAI,SAAiB,CAAC;AAClE,QAAM,CAAC,OAAO,QAAQ,IAAI,SAAmC,CAAC,CAAC;AAE/D,QAAM,iBAAiB,YAAY,MAAM,cAAc,CAAC,MAAM,IAAI,CAAC,GAAG,CAAC,CAAC;AAExE,YAAU,MAAM;AACd,QAAI,eAAe,cAAc,CAAC,QAAQ;AACxC,iBAAW,CAAC,CAAC;AACb,0BAAoB,CAAC;AACrB,eAAS,CAAC,CAAC;AACX;AAAA,IACF;AAEA,UAAM,cAAc,OAAO,WAAW,CAAC;AACvC,eAAW,WAAW;AACtB,aAAS,OAAO,SAAS,CAAC,CAAC;AAE3B,UAAM,YAAY,OAAO,YAAY,YAAY,CAAC,GAAG,MAAM;AAC3D,wBAAoB,CAAC,SAAS;AAC5B,UAAI,QAAQ,YAAY,KAAK,CAAC,MAAM,EAAE,OAAO,IAAI,EAAG,QAAO;AAC3D,aAAO;AAAA,IACT,CAAC;AAAA,EACH,GAAG,CAAC,YAAY,MAAM,CAAC;AAEvB,QAAM,iBAAiB;AAAA,IACrB,MAAM,QAAQ,KAAK,CAAC,MAAM,EAAE,OAAO,gBAAgB,KAAK,QAAQ,CAAC,KAAK;AAAA,IACtE,CAAC,SAAS,gBAAgB;AAAA,EAC5B;AAEA,QAAM,eAAe;AAAA,IACnB,CAAC,aAAqB;AACpB,UAAI,QAAQ,KAAK,CAAC,MAAM,EAAE,OAAO,QAAQ,GAAG;AAC1C,4BAAoB,QAAQ;AAAA,MAC9B;AAAA,IACF;AAAA,IACA,CAAC,OAAO;AAAA,EACV;AAGA,QAAM,eAAe,OAAO,uBAAuB;AAEnD,YAAU,MAAM;AACd,QAAI,eAAe,cAAc,CAAC,oBAAoB,CAAC,aAAc;AAErE,QAAI,YAAY;AAEhB,UAAM,OAAO,YAAY;AACvB,UAAI;AACF,cAAM,CAAC,EAAE,UAAU,SAAS,GAAG,EAAE,UAAU,UAAU,CAAC,IACpD,MAAM,QAAQ,IAAI;AAAA,UAChB,SAAS;AAAA,YACP;AAAA,cACE,UAAU,CAAC;AAAA,cACX,UAAU;AAAA,YACZ;AAAA,YACA;AAAA,UACF;AAAA,UACA,SAAS;AAAA,YACP;AAAA,cACE,UAAU,CAAC;AAAA,cACX,UAAU;AAAA,YACZ;AAAA,YACA;AAAA,UACF;AAAA,QACF,CAAC;AACH,YAAI,UAAW;AACf;AAAA,UAAW,CAAC,SACV,KAAK;AAAA,YAAI,CAAC,MACR,EAAE,OAAO,mBACL;AAAA,cACE,GAAG;AAAA,cACH,cAAc,SAAS;AAAA,cACvB,eAAe,UAAU;AAAA,YAC3B,IACA;AAAA,UACN;AAAA,QACF;AAAA,MACF,SAAS,KAAK;AACZ,YAAI,2BAA2B,GAAG,GAAG;AACnC,2BAAiB,UAAU;AAC3B;AAAA,QACF;AAAA,MAEF;AAAA,IACF;AAGA,SAAK;AACL,UAAM,KAAK,YAAY,MAAM,YAAY;AACzC,WAAO,MAAM;AACX,kBAAY;AACZ,oBAAc,EAAE;AAAA,IAClB;AAAA,EACF,GAAG;AAAA,IACD;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,CAAC;AAED,QAAM,OAAO,QAA2B,MAAM;AAC5C,QAAI,eAAe,eAAe,cAAc,kBAAkB,QAAQ;AACxE,aAAO;AAAA,QACL,SAAS,eAAe;AAAA,QACxB,UAAU,eAAe;AAAA,QACzB,aAAa,OAAO,mBAChB,IAAI,OAAO,gBAAgB,KAC3B,YAAY,OAAO,UAAU;AAAA,QACjC,YAAY;AAAA,QACZ,YAAY,OAAO,OAAO,UAAU;AAAA,QACpC,WAAW,QAAQ,YAAY;AAAA,QAC/B,iBAAiB,QAAQ,YAAY;AAAA,MACvC;AAAA,IACF;AACA,QAAI,cAAc,eAAe,OAAO;AACtC,aAAO;AAAA,QACL,SAAS;AAAA,QACT,UAAU;AAAA,QACV,aAAa,GAAG,WAAW,MAAM,GAAG,CAAC,CAAC,MAAM,WAAW,MAAM,EAAE,CAAC;AAAA,QAChE,YAAY;AAAA,MACd;AAAA,IACF;AACA,WAAO;AAAA,EACT,GAAG,CAAC,aAAa,gBAAgB,YAAY,YAAY,MAAM,CAAC;AAWhE,QAAM,iBAAiB;AAAA,IACrB;AAAA,EACF;AACA,iBAAe,UAAU,CAAC,SAAwB;AAChD,eAAiB,IAAI;AACrB,aAAS,IAAI;AACb,gBAAY,sDAAsD;AAAA,EACpE;AAIA,QAAM,iBAAiB;AAAA,IACrB,QAAQ,IAAI,iBAAiB,MAAM,UAAU,IAAI;AAAA,EACnD;AACA,YAAU,MAAM;AACd,mBAAe,UAAU,QACrB,IAAI,iBAAiB,MAAM,UAAU,IACrC;AAAA,EACN,GAAG,CAAC,KAAK,CAAC;AAGV,QAAM,kBAAkB,OAAyC,IAAI;AAErE,QAAM,aACJ,eAAe,cAAe,eAAe,SAAS,CAAC,CAAC;AAG1D,QAAM,WAAW,QAAoC,MAAM;AACzD,QAAI,eAAe,cAAc,MAAM,SAAS;AAC9C,YAAM,YAAY,IAAI,sBAAsB;AAAA,QAC1C,WAAW,OAAO,aAAa;AAAA,QAC/B,gBAAgB;AAAA,QAChB,YAAY;AAAA,QACZ,UAAU,KAAK;AAAA,QACf,aAAa,MAAM,iBAAiB,UAAU;AAAA,MAChD,CAAC;AACD,aAAO,IAAI,eAAe;AAAA,QACxB;AAAA,QACA,iBAAiB;AAAA,QACjB,aAAa,KAAK;AAAA,MACpB,CAAC;AAAA,IACH;AAEA,QAAI,eAAe,SAAS,YAAY;AACtC,YAAM,YAAY,CAAC,CAAC,aAAa;AAEjC,UAAI,CAAC,SAAS,CAAC,WAAW;AACxB,cAAM,mBAAsC;AAAA,UAC1C,WAAW,OAAO,aAAa;AAAA,UAC/B,UAA0B;AACxB,kBAAM,IAAI;AAAA,cACR;AAAA,YAGF;AAAA,UACF;AAAA,QACF;AACA,eAAO,IAAI,eAAe;AAAA,UACxB,WAAW;AAAA,UACX,iBAAiB;AAAA,UACjB,aAAa;AAAA,QACf,CAAC;AAAA,MACH;AAEA,YAAM,YAAY,OAAO,aAAa;AACtC,YAAM,QAAQ,IAAI,cAAc,EAAE,UAAU,CAAC;AAC7C,YAAM,gBAAgB;AACtB,YAAM,mBAAsC;AAAA,QAC1C,WAAW,MAAM;AAAA,QACjB,MAAM,QACJ,UACA,SACA,QACY;AACZ,cAAI;AACF,mBAAO,MAAM,MAAM,QAAW,UAAU,SAAS,MAAM;AAAA,UACzD,SAAS,KAAK;AACZ,gBAAI,aAAa,cAAc,iBAAiB,GAAG,GAAG;AACpD,6BAAe,UAAU,aAAa;AAAA,YACxC;AACA,kBAAM;AAAA,UACR;AAAA,QACF;AAAA,MACF;AAEA,YAAM,YAAY;AAClB,YAAM,WAAW;AACjB,YAAM,UAAU;AAChB,YAAM,eAAe;AAUrB,YAAM,cAAc,YAAuC;AACzD,cAAM,WAAW,SAAS;AAC1B,YAAI,SAAU,QAAO;AAErB,YAAI,QAAQ,QAAS,QAAO,QAAQ;AAEpC,cAAM,SAAS,UAAU;AACzB,YAAI,CAAC,QAAQ;AACX,gBAAM,IAAI;AAAA,YACR;AAAA,UAEF;AAAA,QACF;AAEA,gBAAQ,WAAW,YAAY;AAC7B,cAAI;AACF,kBAAM,EAAE,YAAY,SAAS,aAAa,IACxC,MAAM,iBAAiB;AAEzB,kBAAM,aACJ,KAAK,OAAO,QAAQ,SAAS,EAAE,CAAC;AAClC,kBAAM,QAAQ,KAAK,IAAI;AACvB,kBAAM,SAAS;AAAA,cACb,MAAM;AAAA,cACN,kBAAkB;AAAA,cAClB,kBAAmB,YAAY,YAAY;AAAA,cAG3C,cAAc,aAAa,YAAY;AAAA,cACvC,WAAW;AAAA,cACX;AAAA,YACF;AAEA,kBAAM,oBAAoB;AAAA,cACxB,uCAAuC;AAAA,gBACrC,EAAE,MAAM,oBAAoB,MAAM,SAAS;AAAA,gBAC3C,EAAE,MAAM,gBAAgB,MAAM,UAAU;AAAA,gBACxC,EAAE,MAAM,aAAa,MAAM,SAAS;AAAA,gBACpC,EAAE,MAAM,SAAS,MAAM,SAAS;AAAA,cAClC;AAAA,YACF;AAEA,kBAAM,SAAS;AAAA;AAAA,cAEb,cAAc,QAAa;AACzB,uBAAO,OAAO,cAAc,MAAM;AAAA,cACpC;AAAA,cACA,cAAc,YAAY,CAAC,YAAY;AAAA,cACvC,YAAY,YAAY,OAAO;AAAA,YACjC;AAEA,kBAAM,YAAY,MAAM,qBAAqB;AAAA,cAC3C;AAAA,cACA;AAAA,cACA,OAAO;AAAA,YACT,CAAC;AAED,kBAAM,SAAS,YACX,oDACA;AAEJ,kBAAM,MAAM,MAAM,MAAM,QAAQ;AAAA,cAC9B,QAAQ;AAAA,cACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,cAC9C,MAAM,KAAK,UAAU,EAAE,QAAQ,WAAW,MAAM,CAAC;AAAA,YACnD,CAAC;AAED,kBAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,gBAAI,MAAM,WAAW,OAAO;AAC1B,oBAAM,IAAI;AAAA,gBACR,2BAA2B,KAAK,YAAY,KAAK,UAAU,IAAI,CAAC;AAAA,cAClE;AAAA,YACF;AAEA,kBAAM,SAAS,MAAM,iBAAiB,cAAc,SAAS;AAC7D,kBAAM,aACJ,QAAQ,cAAc,KAAK,IAAI,IAAI,IAAI,KAAK,KAAK,KAAK;AAExD,kBAAM,SAAsB;AAAA,cAC1B;AAAA,cACA,SAAS;AAAA,cACT,YAAY,KAAK,IAAI;AAAA,cACrB;AAAA,YACF;AACA,sBAAU,cAAc,MAAM;AAE9B,kBAAM,YAAY,IAAI,iBAAiB,UAAU;AACjD,qBAAS,UAAU;AACnB,qBAAS,MAAM;AAEf,mBAAO;AAAA,UACT,UAAE;AACA,oBAAQ,UAAU;AAAA,UACpB;AAAA,QACF,GAAG;AAEH,eAAO,QAAQ;AAAA,MACjB;AAKA,YAAM,aAAa;AAAA,QACjB,SAAS;AAAA,QACT,MAAM,cAAc,QAWO;AACzB,cAAI,OAAO,OAAO,SAAS,8BAA8B;AACvD,kBAAM,SAAS,UAAU;AACzB,gBAAI,CAAC,QAAQ;AACX,oBAAM,IAAI;AAAA,gBACR;AAAA,cAEF;AAAA,YACF;AACA,mBAAO,OAAO;AAAA,cACZ;AAAA,YACF;AAAA,UACF;AAEA,gBAAM,cAAc,MAAM,YAAY;AACtC,iBAAO,YAAY,cAAc,MAAM;AAAA,QACzC;AAAA,MACF;AAEA,aAAO,IAAI,eAAe;AAAA,QACxB,WAAW;AAAA,QACX,QAAQ;AAAA,QACR,kBAAkB,MAAM;AACtB,gBAAM,KAAK,UAAU,SAAS,WAAW;AACzC,iBAAO,KAAK,GAAG,SAAS,EAAE,CAAC;AAAA,QAC7B;AAAA,MACF,CAAC;AAAA,IACH;AAEA,WAAO;AAAA,EACT,GAAG;AAAA,IACD;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,OAAO;AAAA,IACP;AAAA,IACA;AAAA,EACF,CAAC;AAED,QAAM,mBAAmB,YAAY,MAAM;AACzC,QAAI,YAAY;AACd,iBAAiB,UAAU;AAC3B,eAAS,IAAI;AAAA,IACf;AAAA,EACF,GAAG,CAAC,UAAU,CAAC;AAGf,QAAM,eAAe;AAAA,IACnB,OAAO,SAA6C;AAClD,YAAM,EAAE,SAAS,IAAI,MAAM,SAAS;AAAA,QAClC;AAAA,UACE,UAAU,CAAC;AAAA,UACX;AAAA,QACF;AAAA,QACA;AAAA,MACF;AACA,qBAAe;AACf,UAAI,CAAC,SAAS;AACZ,cAAM,IAAI,MAAM,oCAAoC;AACtD,aAAO,SAAS;AAAA,IAClB;AAAA,IACA,CAAC,UAAU,oBAAoB,cAAc;AAAA,EAC/C;AAEA,QAAM,eAAe;AAAA,IACnB,OAAO,aAAoC;AACzC,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX;AAAA,QACF;AAAA,QACA;AAAA,MACF;AACA,UAAI,aAAa,kBAAkB;AACjC,cAAM,YAAY,QAAQ,OAAO,CAAC,MAAM,EAAE,OAAO,QAAQ;AACzD,4BAAoB,UAAU,CAAC,GAAG,MAAM,CAAC;AAAA,MAC3C;AACA,qBAAe;AAAA,IACjB;AAAA,IACA,CAAC,UAAU,oBAAoB,kBAAkB,SAAS,cAAc;AAAA,EAC1E;AAEA,QAAM,mBAAmB;AAAA,IACvB,OAAO,SAAkC;AACvC,YAAM,EAAE,SAAS,IAAI,MAAM,SAAS;AAAA,QAClC;AAAA,UACE,UAAU,CAAC;AAAA,UACX;AAAA,QACF;AAAA,QACA;AAAA,MACF;AACA,qBAAe;AACf,aAAO,SAAS;AAAA,IAClB;AAAA,IACA,CAAC,UAAU,oBAAoB,cAAc;AAAA,EAC/C;AAEA,QAAM,eAAe;AAAA,IACnB,OAAO,WAIc;AACnB,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,qBAAe;AAAA,IACjB;AAAA,IACA,CAAC,UAAU,oBAAoB,cAAc;AAAA,EAC/C;AAEA,QAAM,kBAAkB;AAAA,IACtB,OAAO,WAIc;AACnB,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,qBAAe;AAAA,IACjB;AAAA,IACA,CAAC,UAAU,oBAAoB,cAAc;AAAA,EAC/C;AAEA,QAAM,kBAAkB;AAAA,IACtB,OAAO,WAAmE;AACxE,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,qBAAe;AAAA,IACjB;AAAA,IACA,CAAC,UAAU,oBAAoB,cAAc;AAAA,EAC/C;AAGA,QAAM,aAAa;AAAA,IACjB,OACE,WAIG;AACH,YAAM,EAAE,SAAS,IAAI,MAAM,SAAS;AAAA,QAClC;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU;AAAA,UACV,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,UAAI,CAAC,SAAS;AACZ,cAAM,IAAI,MAAM,mCAAmC;AACrD,YAAM,UAAU,SAAS;AACzB;AAAA,QAAW,CAAC,SACV,KAAK;AAAA,UAAI,CAAC,MACR,EAAE,OAAO,mBACL,EAAE,GAAG,GAAG,cAAc,CAAC,GAAG,EAAE,cAAc,OAAO,EAAE,IACnD;AAAA,QACN;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAAA,IACA,CAAC,UAAU,oBAAoB,gBAAgB;AAAA,EACjD;AAEA,QAAM,aAAa;AAAA,IACjB,OACE,WAIG;AACH,YAAM,EAAE,SAAS,IAAI,MAAM,SAAS;AAAA,QAClC;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU;AAAA,UACV,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,UAAI,CAAC,SAAS,QAAS,OAAM,IAAI,MAAM,iCAAiC;AACxE,YAAM,UAAU,SAAS;AACzB;AAAA,QAAW,CAAC,SACV,KAAK;AAAA,UAAI,CAAC,MACR,EAAE,OAAO,mBACL;AAAA,YACE,GAAG;AAAA,YACH,cAAc,EAAE,aAAa;AAAA,cAAI,CAAC,MAChC,EAAE,OAAO,QAAQ,KAAK,UAAU;AAAA,YAClC;AAAA,UACF,IACA;AAAA,QACN;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAAA,IACA,CAAC,UAAU,oBAAoB,gBAAgB;AAAA,EACjD;AAEA,QAAM,aAAa;AAAA,IACjB,OAAO,cAAsB;AAC3B,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU;AAAA,UACV,eAAe;AAAA,QACjB;AAAA,QACA;AAAA,MACF;AACA;AAAA,QAAW,CAAC,SACV,KAAK;AAAA,UAAI,CAAC,MACR,EAAE,OAAO,mBACL;AAAA,YACE,GAAG;AAAA,YACH,cAAc,EAAE,aAAa,OAAO,CAAC,MAAM,EAAE,OAAO,SAAS;AAAA,UAC/D,IACA;AAAA,QACN;AAAA,MACF;AAAA,IACF;AAAA,IACA,CAAC,UAAU,oBAAoB,gBAAgB;AAAA,EACjD;AAGA,QAAM,cAAc;AAAA,IAClB,OACE,WAIG;AACH,YAAM,EAAE,SAAS,IAAI,MAAM,SAAS;AAAA,QAClC;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU;AAAA,UACV,GAAG;AAAA,QACL;AAAA,QACA;AAAA,MACF;AACA,UAAI,CAAC,SAAS;AACZ,cAAM,IAAI,MAAM,oCAAoC;AACtD,YAAM,UAAU,SAAS;AACzB;AAAA,QAAW,CAAC,SACV,KAAK;AAAA,UAAI,CAAC,MACR,EAAE,OAAO,mBACL,EAAE,GAAG,GAAG,eAAe,CAAC,GAAG,EAAE,eAAe,OAAO,EAAE,IACrD;AAAA,QACN;AAAA,MACF;AACA,aAAO;AAAA,IACT;AAAA,IACA,CAAC,UAAU,oBAAoB,gBAAgB;AAAA,EACjD;AAEA,QAAM,cAAc;AAAA,IAClB,OAAO,cAAsB;AAC3B,YAAM,SAAS;AAAA,QACb;AAAA,UACE,UAAU,CAAC;AAAA,UACX,UAAU;AAAA,UACV,gBAAgB;AAAA,QAClB;AAAA,QACA;AAAA,MACF;AACA;AAAA,QAAW,CAAC,SACV,KAAK;AAAA,UAAI,CAAC,MACR,EAAE,OAAO,mBACL;AAAA,YACE,GAAG;AAAA,YACH,eAAe,EAAE,cAAc;AAAA,cAC7B,CAAC,MAAM,EAAE,OAAO;AAAA,YAClB;AAAA,UACF,IACA;AAAA,QACN;AAAA,MACF;AAAA,IACF;AAAA,IACA,CAAC,UAAU,oBAAoB,gBAAgB;AAAA,EACjD;AAGA,QAAM,CAAC,gBAAgB,iBAAiB,IAAI,SAAS,KAAK;AAC1D,QAAM,iBAAiB,YAAY,MAAM,kBAAkB,IAAI,GAAG,CAAC,CAAC;AACpE,QAAM,kBAAkB,YAAY,MAAM,kBAAkB,KAAK,GAAG,CAAC,CAAC;AAGtE,QAAM,gBAAgB,YAAY,MAAM;AACtC,UAAM,QAAQ,YAAY;AAC1B,mBAAe,QAAQ,yBAAyB,KAAK;AAErD,UAAM,qBAAqB,OAAO,SAAS;AAC3C,UAAM,WACJ,OAAO,uBAAuB,aAC1B,mBAAmB,IACnB,sBAAsB,gBAAgB;AAE5C,UAAM,UAAU,IAAI,IAAI,OAAO,SAAS,cAAc,oBAAoB;AAC1E,YAAQ,aAAa,IAAI,aAAa,QAAQ;AAC9C,YAAQ,aAAa,IAAI,SAAS,KAAK;AACvC,YAAQ,aAAa,IAAI,SAAS,gBAAgB,OAAO,SAAS,KAAK,CAAC;AAExE,UAAM,QAAQ;AACd,UAAM,SAAS;AACf,UAAM,OAAO,OAAO,UAAU,KAAK,IAAI,IAAI,OAAO,aAAa,SAAS,CAAC;AACzE,UAAM,MAAM,OAAO,UAAU,KAAK,IAAI,IAAI,OAAO,cAAc,UAAU,CAAC;AAC1E,UAAM,QAAQ,OAAO;AAAA,MACnB,QAAQ,SAAS;AAAA,MACjB;AAAA,MACA;AAAA,QACE,SAAS,KAAK;AAAA,QACd,UAAU,MAAM;AAAA,QAChB,QAAQ,KAAK,MAAM,IAAI,CAAC;AAAA,QACxB,OAAO,KAAK,MAAM,GAAG,CAAC;AAAA,QACtB;AAAA,QACA;AAAA,MACF,EAAE,KAAK,GAAG;AAAA,IACZ;AAEA,QAAI,OAAO;AACT,YAAM,MAAM;AACZ;AAAA,IACF;AAEA,WAAO,SAAS,OAAO,QAAQ,SAAS,CAAC;AAAA,EAC3C,GAAG;AAAA,IACD,OAAO,SAAS;AAAA,IAChB,OAAO,SAAS;AAAA,IAChB,OAAO,SAAS;AAAA,EAClB,CAAC;AAED,QAAM,aAAa;AAAA,IACjB,CAAC,SAAwB,WAAuB;AAC9C,mBAAa,UAAU,UAAU;AACjC,oBAAc,OAAO;AACrB,qBAAe,IAAI;AACnB,gBAAU,IAAI;AACd,iBAAW,IAAI;AACf,kBAAY,IAAI;AAChB,mBAAa,WAAW,oBAAoB;AAC5C,mBAAa,WAAW,2BAA2B;AAEnD,YAAM,WAAW,UAAU,OAAO;AAClC,UAAI,YAAY,SAAS,aAAa,KAAK,IAAI,GAAG;AAChD,iBAAS,QAAQ;AAAA,MACnB,OAAO;AACL,YAAI,SAAU,YAAiB,OAAO;AACtC,iBAAS,IAAI;AAAA,MACf;AAAA,IACF;AAAA,IACA,CAAC;AAAA,EACH;AAEA,QAAM,iBAAiB;AAAA,IACrB,OAAO,oBAAqC,YAAoB;AAC9D,UAAI,CAAC,YAAY;AACf,cAAM,IAAI;AAAA,UACR;AAAA,QACF;AAAA,MACF;AAEA,mBAAa,UAAU,EAAE,eAAe,oBAAoB,QAAQ;AAEpE,oBAAc,IAAI;AAClB,kBAAY,IAAI;AAChB,UAAI;AACF,cAAM,WAAW,UAAU,UAAU;AACrC,YAAI,UAAU;AACZ,gBAAMC,aAAY,OAAO,aAAa;AACtC,gBAAM,QAAQ,MAAM,aAAa,UAAU,YAAYA,UAAS;AAChE,cAAI,OAAO;AACT,qBAAS,QAAQ;AACjB;AAAA,UACF;AACA,qBAAiB,UAAU;AAAA,QAC7B;AAEA,cAAM,EAAE,YAAY,SAAS,aAAa,IAAI,MAAM,iBAAiB;AACrE,cAAM,YAAY,OAAO,aAAa;AAEtC,cAAM,aAAa,KAAK,QAAQ,SAAS,EAAE,CAAC;AAC5C,cAAM,QAAQ,KAAK,IAAI;AACvB,cAAM,SAAS;AAAA,UACb,MAAM;AAAA,UACN,kBAAkB;AAAA,UAClB,kBAAmB,YAAY,YAAY;AAAA,UAG3C,cAAc,aAAa,YAAY;AAAA,UACvC,WAAW;AAAA,UACX;AAAA,QACF;AAEA,cAAM,oBAAoB;AAAA,UACxB,uCAAuC;AAAA,YACrC,EAAE,MAAM,oBAAoB,MAAM,SAAS;AAAA,YAC3C,EAAE,MAAM,gBAAgB,MAAM,UAAU;AAAA,YACxC,EAAE,MAAM,aAAa,MAAM,SAAS;AAAA,YACpC,EAAE,MAAM,SAAS,MAAM,SAAS;AAAA,UAClC;AAAA,QACF;AAEA,cAAM,SAAS;AAAA;AAAA,UAEb,cAAc,QAAa;AACzB,mBAAO,mBAAmB,MAAM;AAAA,UAClC;AAAA,UACA,cAAc,YAAY,CAAC,UAAU;AAAA,UACrC,YAAY,YAAY;AAAA,QAC1B;AAEA,cAAM,YAAY,MAAM,qBAAqB;AAAA,UAC3C;AAAA,UACA;AAAA,UACA,OAAO;AAAA,QACT,CAAC;AAED,cAAM,SAAS,YACX,iDACA;AAEJ,cAAM,MAAM,MAAM,MAAM,QAAQ;AAAA,UAC9B,QAAQ;AAAA,UACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,UAC9C,MAAM,KAAK,UAAU,EAAE,QAAQ,WAAW,MAAM,CAAC;AAAA,QACnD,CAAC;AAED,cAAM,OAAO,MAAM,IAAI,KAAK;AAC5B,YAAI,MAAM,WAAW,OAAO;AAC1B,gBAAM,IAAI;AAAA,YACR,2BAA2B,KAAK,YAAY,KAAK,UAAU,IAAI,CAAC;AAAA,UAClE;AAAA,QACF;AAEA,cAAM,SAAS,MAAM,iBAAiB,YAAY,SAAS;AAC3D,cAAM,aACJ,QAAQ,cAAc,KAAK,IAAI,IAAI,IAAI,KAAK,KAAK,KAAK;AAExD,cAAM,SAAsB;AAAA,UAC1B;AAAA,UACA,SAAS;AAAA,UACT,YAAY,KAAK,IAAI;AAAA,UACrB;AAAA,QACF;AACA,kBAAU,YAAY,MAAM;AAC5B,iBAAS,MAAM;AAAA,MACjB,SAAS,KAAK;AACZ,gBAAQ,MAAM,8CAA8C,GAAG;AAC/D,oBAAY,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC;AAC5D,iBAAS,IAAI;AAAA,MACf,UAAE;AACA,sBAAc,KAAK;AAAA,MACrB;AAAA,IACF;AAAA,IACA,CAAC,YAAY,OAAO,SAAS;AAAA,EAC/B;AAEA,QAAM,SAAS,YAAY,MAAM;AAC/B,cAAU,IAAI;AACd,eAAW,IAAI;AACf,mBAAe,IAAI;AACnB,kBAAc,IAAI;AAClB,aAAS,IAAI;AACb,gBAAY,IAAI;AAChB,iBAAa,UAAU;AACvB,iBAAa,WAAW,oBAAoB;AAC5C,iBAAa,WAAW,2BAA2B;AAAA,EACrD,GAAG,CAAC,CAAC;AAGL,QAAM,QAAQ;AAAA,IACZ,OAAO;AAAA,MACL;AAAA,MACA,YAAY,CAAC,CAAC;AAAA,MACd,WAAW,aAAa;AAAA,MACxB,OAAO,WAAW;AAAA,MAClB;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA;AAAA,MAEA;AAAA,MACA;AAAA,MACA,cAAc;AAAA,MACd;AAAA,MAEA;AAAA,MACA;AAAA,MACA,YAAY;AAAA,MAEZ,OAAO,OAAO,UAAU,SAAS;AAAA,MACjC,aAAa,OAAO,UAAU,eAAe;AAAA,MAC7C,WAAW,OAAO,UAAU,aAAa;AAAA,MAEzC;AAAA,MACA,WAAW;AAAA,MACX;AAAA,MACA,gBAAgB;AAAA,MAChB;AAAA,IACF;AAAA,IACA;AAAA,MACE;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA,OAAO,UAAU;AAAA,MACjB,OAAO,UAAU;AAAA,MACjB,OAAO,UAAU;AAAA,MACjB;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAAA,EACF;AAEA,SACE,oBAAC,qBAAqB,UAArB,EAA8B,OAC5B,UACH;AAEJ;;;AIxuCA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,OAEK;AACP;AAAA,EACE,eAAAC;AAAA,EACA;AAAA,OAGK;;;ACCD,SACE,OAAAC,MADF;AAVC,SAAS,kBAAkB,EAAE,MAAM,GAA8B;AACtE,SACE;AAAA,IAAC;AAAA;AAAA,MACC,OAAM;AAAA,MACN,QAAO;AAAA,MACP,SAAQ;AAAA,MACR;AAAA,MACA,MAAK;AAAA,MACL,OAAM;AAAA,MAEN;AAAA,6BAAC,OAAE,UAAS,yBACV;AAAA,0BAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,UACA,gBAAAA;AAAA,YAAC;AAAA;AAAA,cACC,GAAE;AAAA,cACF,MAAK;AAAA;AAAA,UACP;AAAA,WACF;AAAA,QACA,gBAAAA,KAAC,UACC,0BAAAA,KAAC,cAAS,IAAG,mBACX,0BAAAA;AAAA,UAAC;AAAA;AAAA,YACC,OAAM;AAAA,YACN,QAAO;AAAA,YACP,MAAK;AAAA,YACL,WAAU;AAAA;AAAA,QACZ,GACF,GACF;AAAA;AAAA;AAAA,EACF;AAEJ;;;AC1CM,gBAAAC,MAWE,QAAAC,aAXF;AAVC,SAAS,kBAAkB,EAAE,MAAM,GAA8B;AACtE,SACE,gBAAAA;AAAA,IAAC;AAAA;AAAA,MACC,OAAM;AAAA,MACN,QAAO;AAAA,MACP,SAAQ;AAAA,MACR;AAAA,MACA,MAAK;AAAA,MACL,OAAM;AAAA,MAEN;AAAA,wBAAAD;AAAA,UAAC;AAAA;AAAA,YACC,GAAE;AAAA,YACF,MAAK;AAAA;AAAA,QACP;AAAA,QACA,gBAAAA;AAAA,UAAC;AAAA;AAAA,YACC,UAAS;AAAA,YACT,UAAS;AAAA,YACT,GAAE;AAAA,YACF,MAAK;AAAA;AAAA,QACP;AAAA,QACA,gBAAAA,KAAC,UACC,0BAAAC;AAAA,UAAC;AAAA;AAAA,YACC,IAAG;AAAA,YACH,IAAG;AAAA,YACH,IAAG;AAAA,YACH,IAAG;AAAA,YACH,IAAG;AAAA,YACH,eAAc;AAAA,YAEd;AAAA,8BAAAD,KAAC,UAAK,WAAU,WAAU;AAAA,cAC1B,gBAAAA,KAAC,UAAK,QAAO,KAAI,WAAU,WAAU;AAAA;AAAA;AAAA,QACvC,GACF;AAAA;AAAA;AAAA,EACF;AAEJ;;;AFiFI,SAuBA,UAvBA,OAAAE,MAkCI,QAAAC,aAlCJ;AAjGJ,IAAM,oBAAoB;AAE1B,IAAM,WAA0B;AAAA,EAC9B,SAAS;AAAA,EACT,QAAQ;AAAA,EACR,OAAO;AAAA,EACP,YAAY;AAAA,EACZ,KAAK;AAAA,EACL,UAAU;AAAA,EACV,cAAc;AAAA,EACd,YAAY;AAAA,EACZ,SAAS;AAAA,EACT,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,eAAe;AAAA,EACf,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,YAAY;AACd;AAEA,IAAM,aAAa,EAAE,YAAY,wBAAwB;AAEzD,IAAM,gBAA+B;AAAA,EACnC,UAAU;AAAA,EACV,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,YAAY;AAAA,EACZ,gBAAgB;AAAA,EAChB,sBAAsB;AACxB;AAEA,IAAM,oBAAmC;AAAA,EACvC,UAAU;AAAA,EACV,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,YAAY;AAAA,EACZ,gBAAgB;AAAA,EAChB,SAAS;AACX;AAEA,IAAM,gBAA+B;AAAA,EACnC,SAAS;AAAA,EACT,OAAO;AAAA,EACP,eAAe;AAAA,EACf,YAAY;AAAA,EACZ,KAAK;AAAA,EACL,UAAU;AAAA,EACV,cAAc;AAAA,EACd,QAAQ;AAAA,EACR,YAAY;AAAA,EACZ,SAAS;AACX;AAEA,IAAM,eAA8B;AAAA,EAClC,UAAU;AAAA,EACV,YAAY;AAAA,EACZ,eAAe;AAAA,EACf,OAAO;AAAA,EACP,QAAQ;AACV;AAEA,IAAM,eAA8B;AAAA,EAClC,QAAQ;AAAA,EACR,OAAO;AAAA,EACP,YAAY;AACd;AAEA,IAAM,WAA0B,EAAE,OAAO,IAAI,QAAQ,GAAG;AAExD,IAAM,cACJ,OAAO,WAAW,cACd,OAAO,WAAW,eAAe,iBAAiB,KAAK,IACvD;AAEN,SAAS,gBAAgB,IAAgB;AACvC,eAAa,iBAAiB,UAAU,EAAE;AAC1C,SAAO,MAAM,aAAa,oBAAoB,UAAU,EAAE;AAC5D;AAEA,SAAS,oBAAoB;AAC3B,SAAO,aAAa,WAAW;AACjC;AAEA,SAAS,cAAc;AACrB,SAAO,qBAAqB,iBAAiB,mBAAmB,MAAM,KAAK;AAC7E;AAEA,SAAS,YAAY;AAAA,EACnB;AAAA,EACA;AACF,GAGG;AACD,SACE,gBAAAD;AAAA,IAAC,OAAO;AAAA,IAAP;AAAA,MACC,MAAK;AAAA,MACL;AAAA,MACA,OAAO;AAAA,MACP,YAAY;AAAA,MAEX;AAAA;AAAA,EACH;AAEJ;AAEO,SAAS,WAAW,EAAE,iBAAiB,WAAW,GAAoB;AAC3E,QAAM,EAAE,eAAe,gBAAgB,gBAAgB,IACrD,yBAAyB;AAE3B,QAAM,qBAAqBE,aAAY,MAAM;AAC3C,oBAAgB;AAChB,kBAAc;AAAA,EAChB,GAAG,CAAC,eAAe,eAAe,CAAC;AAEnC,QAAM,WAAW,YAAY;AAE7B,QAAM,eACJ,gBAAAD,MAAA,YACE;AAAA,oBAAAD;AAAA,MAAC;AAAA;AAAA,QACC,OAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO;AAAA,UACP,eAAe;AAAA,UACf,YAAY;AAAA,UACZ,KAAK;AAAA,UACL,UAAU;AAAA,QACZ;AAAA,QAEA,0BAAAC,MAAC,eAAY,SAAS,oBACpB;AAAA,0BAAAD,KAAC,qBAAkB,OAAO,UAAU;AAAA,UAAE;AAAA,WAExC;AAAA;AAAA,IACF;AAAA,IAEA,gBAAAA,KAAC,SAAI,OAAO,cAAc;AAAA,IAE1B,gBAAAC;AAAA,MAAC;AAAA;AAAA,QACC,SAAS,MAAM;AACb,0BAAgB;AAChB,0BAAgB;AAAA,QAClB;AAAA,QAEC;AAAA,wBAAc,gBAAAD,KAAC,qBAAkB,OAAO,UAAU;AAAA,UAAG;AAAA;AAAA;AAAA,IAExD;AAAA,KACF;AAGF,SACE,gBAAAA,KAAC,mBACE,6BACE,WACC,gBAAAA,KAAC,gBAA0B,SAAS,iBACjC,0BADe,QAElB,IAEA,gBAAAC,MAAA,YACE;AAAA,oBAAAD;AAAA,MAAC,OAAO;AAAA,MAAP;AAAA,QAEC,OAAO;AAAA,QACP,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,MAAM,EAAE,SAAS,EAAE;AAAA,QACnB,YAAY,EAAE,UAAU,KAAK;AAAA,QAC7B,SAAS;AAAA;AAAA,MANL;AAAA,IAON;AAAA,IACA,gBAAAA;AAAA,MAAC,OAAO;AAAA,MAAP;AAAA,QAEC,OAAO;AAAA,QACP,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,MAAM,EAAE,SAAS,EAAE;AAAA,QACnB,YAAY,EAAE,UAAU,KAAK;AAAA,QAC7B,SAAS;AAAA,QAET,0BAAAC;AAAA,UAAC,OAAO;AAAA,UAAP;AAAA,YACC,OAAO;AAAA,YACP,SAAS,EAAE,OAAO,MAAM,SAAS,GAAG,GAAG,EAAE;AAAA,YACzC,SAAS,EAAE,OAAO,GAAG,SAAS,GAAG,GAAG,EAAE;AAAA,YACtC,MAAM,EAAE,OAAO,MAAM,SAAS,GAAG,GAAG,EAAE;AAAA,YACtC,YAAY,EAAE,UAAU,KAAK;AAAA,YAC7B,SAAS,CAAC,MAAM,EAAE,gBAAgB;AAAA,YAElC;AAAA,8BAAAD,KAAC,QAAG,OAAO,cAAc,qBAAO;AAAA,cAC/B;AAAA;AAAA;AAAA,QACH;AAAA;AAAA,MAlBI;AAAA,IAmBN;AAAA,KACF,IAEN;AAEJ;AAEA,IAAM,mBAAkC;AAAA,EACtC,UAAU;AAAA,EACV,MAAM;AAAA,EACN,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,eAAe;AAAA,EACf,YAAY;AAAA,EACZ,KAAK;AAAA,EACL,qBAAqB;AAAA,EACrB,sBAAsB;AAAA,EACtB,YAAY;AAAA,EACZ,aAAa;AAAA,EACb,WAAW;AAAA,EACX,YAAY;AAAA,EACZ,SAAS;AACX;AAEA,IAAM,gBAA+B;AAAA,EACnC,UAAU;AAAA,EACV,MAAM;AAAA,EACN,OAAO;AAAA,EACP,KAAK;AAAA,EACL,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,YAAY;AAAA,EACZ,qBAAqB;AAAA,EACrB,sBAAsB;AACxB;AAEA,IAAM,sBAAqC;AAAA,EACzC,OAAO;AAAA,EACP,QAAQ;AAAA,EACR,eAAe;AACjB;AAEA,IAAM,kBAAiC;AAAA,EACrC,QAAQ;AAAA,EACR,QAAQ;AAAA,EACR,OAAO;AAAA,EACP,cAAc;AAAA,EACd,YAAY;AACd;AAEA,SAAS,aAAa;AAAA,EACpB;AAAA,EACA;AACF,GAGG;AACD,QAAM,WAAW,qBAAqB;AAEtC,QAAM,gBAAgBE;AAAA,IACpB,CAAC,GAAY,SAAkB;AAC7B,UAAI,KAAK,OAAO,IAAI,OAAO,KAAK,SAAS,IAAI,KAAK;AAChD,gBAAQ;AAAA,MACV,OAAO;AACL,iBAAS,MAAM,EAAE,GAAG,EAAE,CAAC;AAAA,MACzB;AAAA,IACF;AAAA,IACA,CAAC,SAAS,QAAQ;AAAA,EACpB;AAEA,SACE,gBAAAD,MAAA,YACE;AAAA,oBAAAD;AAAA,MAAC,OAAO;AAAA,MAAP;AAAA,QAEC,OAAO;AAAA,QACP,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,SAAS,EAAE,SAAS,EAAE;AAAA,QACtB,MAAM,EAAE,SAAS,EAAE;AAAA,QACnB,YAAY,EAAE,UAAU,KAAK;AAAA,QAC7B,SAAS;AAAA;AAAA,MANL;AAAA,IAON;AAAA,IAEA,gBAAAC;AAAA,MAAC,OAAO;AAAA,MAAP;AAAA,QAEC,OAAO;AAAA,QACP,SAAS,EAAE,GAAG,OAAO;AAAA,QACrB,SAAS,EAAE,GAAG,EAAE;AAAA,QAChB,MAAM,EAAE,GAAG,OAAO;AAAA,QAClB,YAAY,EAAE,MAAM,SAAS,UAAU,KAAK,MAAM,CAAC,MAAM,MAAM,GAAG,CAAC,EAAE;AAAA,QACrE,MAAK;AAAA,QACL,iBAAiB,EAAE,KAAK,GAAG,QAAQ,EAAE;AAAA,QACrC,aAAa,EAAE,KAAK,GAAG,QAAQ,IAAI;AAAA,QACnC,WAAW;AAAA,QAEX;AAAA,0BAAAD,KAAC,SAAI,OAAO,eAAe;AAAA,UAE3B,gBAAAA,KAAC,SAAI,OAAO,qBACV,0BAAAA,KAAC,SAAI,OAAO,iBAAiB,GAC/B;AAAA,UAEA,gBAAAA,KAAC,OAAE,OAAO,cAAc,qBAAO;AAAA,UAE9B;AAAA;AAAA;AAAA,MAnBG;AAAA,IAoBN;AAAA,KACF;AAEJ;;;AG3MO,SAAS,gBACd,oBAGA,SACW;AACX,QAAM,UACJ,OAAO,uBAAuB,aAC1B,qBACA,CAAC,SAAkC,mBAAmB,QAAQ,IAAI;AACxE,SAAO;AAAA,IACL,eAAe,CAAC,WAAW,QAAQ,MAAM;AAAA,IACzC;AAAA,EACF;AACF;","names":["PrivateKeySigner","user","isTestnet","useCallback","jsx","jsx","jsxs","jsx","jsxs","useCallback"]}
|
package/package.json
CHANGED
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
import type { IRequestTransport } from "@hfunlabs/hyperliquid";
|
|
2
|
+
import type { RpcOptions } from "@protobuf-ts/runtime-rpc";
|
|
2
3
|
import type { TelegramClient } from "hypurr-grpc/ts/hypurr/telegram/telegram_service.client";
|
|
3
4
|
|
|
4
5
|
export interface GrpcExchangeTransportConfig {
|
|
5
6
|
isTestnet?: boolean;
|
|
6
7
|
telegramClient: TelegramClient;
|
|
7
|
-
|
|
8
|
+
rpcOptions?: RpcOptions;
|
|
8
9
|
walletId: number;
|
|
9
10
|
onAuthError?: () => void;
|
|
10
11
|
}
|
|
@@ -26,7 +27,7 @@ interface ExchangePayload {
|
|
|
26
27
|
export class GrpcExchangeTransport implements IRequestTransport {
|
|
27
28
|
isTestnet: boolean;
|
|
28
29
|
private telegramClient: TelegramClient;
|
|
29
|
-
private
|
|
30
|
+
private rpcOptions?: RpcOptions;
|
|
30
31
|
private walletId: number;
|
|
31
32
|
private infoUrl: string;
|
|
32
33
|
private onAuthError?: () => void;
|
|
@@ -34,7 +35,7 @@ export class GrpcExchangeTransport implements IRequestTransport {
|
|
|
34
35
|
constructor(config: GrpcExchangeTransportConfig) {
|
|
35
36
|
this.isTestnet = config.isTestnet ?? false;
|
|
36
37
|
this.telegramClient = config.telegramClient;
|
|
37
|
-
this.
|
|
38
|
+
this.rpcOptions = config.rpcOptions;
|
|
38
39
|
this.walletId = config.walletId;
|
|
39
40
|
this.onAuthError = config.onAuthError;
|
|
40
41
|
this.infoUrl = this.isTestnet
|
|
@@ -69,17 +70,22 @@ export class GrpcExchangeTransport implements IRequestTransport {
|
|
|
69
70
|
|
|
70
71
|
let response;
|
|
71
72
|
try {
|
|
72
|
-
({ response } = await this.telegramClient.hyperliquidCoreAction(
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
73
|
+
({ response } = await this.telegramClient.hyperliquidCoreAction(
|
|
74
|
+
{
|
|
75
|
+
authData: {},
|
|
76
|
+
walletId: this.walletId,
|
|
77
|
+
action: actionBytes,
|
|
78
|
+
nonce: payload.nonce || Date.now(),
|
|
79
|
+
},
|
|
80
|
+
this.rpcOptions,
|
|
81
|
+
));
|
|
78
82
|
} catch (err) {
|
|
79
83
|
if (
|
|
80
84
|
this.onAuthError &&
|
|
81
85
|
err instanceof Error &&
|
|
82
|
-
/invalid telegram auth data/i.test(
|
|
86
|
+
/invalid telegram auth data|invalid auth token|missing authorization token/i.test(
|
|
87
|
+
err.message,
|
|
88
|
+
)
|
|
83
89
|
) {
|
|
84
90
|
this.onAuthError();
|
|
85
91
|
}
|